| git-subtree(1) | 
 | ============== | 
 |  | 
 | NAME | 
 | ---- | 
 | git-subtree - Merge subtrees together and split repository into subtrees | 
 |  | 
 |  | 
 | SYNOPSIS | 
 | -------- | 
 | [verse] | 
 | 'git subtree' [<options>] -P <prefix> add <local-commit> | 
 | 'git subtree' [<options>] -P <prefix> add <repository> <remote-ref> | 
 | 'git subtree' [<options>] -P <prefix> merge <local-commit> [<repository>] | 
 | 'git subtree' [<options>] -P <prefix> split [<local-commit>] | 
 |  | 
 | [verse] | 
 | 'git subtree' [<options>] -P <prefix> pull <repository> <remote-ref> | 
 | 'git subtree' [<options>] -P <prefix> push <repository> <refspec> | 
 |  | 
 | DESCRIPTION | 
 | ----------- | 
 | Subtrees allow subprojects to be included within a subdirectory | 
 | of the main project, optionally including the subproject's | 
 | entire history. | 
 |  | 
 | For example, you could include the source code for a library | 
 | as a subdirectory of your application. | 
 |  | 
 | Subtrees are not to be confused with submodules, which are meant for | 
 | the same task. Unlike submodules, subtrees do not need any special | 
 | constructions (like '.gitmodules' files or gitlinks) be present in | 
 | your repository, and do not force end-users of your | 
 | repository to do anything special or to understand how subtrees | 
 | work. A subtree is just a subdirectory that can be | 
 | committed to, branched, and merged along with your project in | 
 | any way you want. | 
 |  | 
 | They are also not to be confused with using the subtree merge | 
 | strategy. The main difference is that, besides merging | 
 | the other project as a subdirectory, you can also extract the | 
 | entire history of a subdirectory from your project and make it | 
 | into a standalone project. Unlike the subtree merge strategy | 
 | you can alternate back and forth between these | 
 | two operations. If the standalone library gets updated, you can | 
 | automatically merge the changes into your project; if you | 
 | update the library inside your project, you can "split" the | 
 | changes back out again and merge them back into the library | 
 | project. | 
 |  | 
 | For example, if a library you made for one application ends up being | 
 | useful elsewhere, you can extract its entire history and publish | 
 | that as its own git repository, without accidentally | 
 | intermingling the history of your application project. | 
 |  | 
 | [TIP] | 
 | In order to keep your commit messages clean, we recommend that | 
 | people split their commits between the subtrees and the main | 
 | project as much as possible.  That is, if you make a change that | 
 | affects both the library and the main application, commit it in | 
 | two pieces.  That way, when you split the library commits out | 
 | later, their descriptions will still make sense.  But if this | 
 | isn't important to you, it's not *necessary*.  'git subtree' will | 
 | simply leave out the non-library-related parts of the commit | 
 | when it splits it out into the subproject later. | 
 |  | 
 |  | 
 | COMMANDS | 
 | -------- | 
 | add <local-commit>:: | 
 | add <repository> <remote-ref>:: | 
 | 	Create the <prefix> subtree by importing its contents | 
 | 	from the given <local-commit> or <repository> and <remote-ref>. | 
 | 	A new commit is created	automatically, joining the imported | 
 | 	project's history with your own.  With '--squash', import | 
 | 	only a single commit from the subproject, rather than its | 
 | 	entire history. | 
 |  | 
 | merge <local-commit> [<repository>]:: | 
 | 	Merge recent changes up to <local-commit> into the <prefix> | 
 | 	subtree.  As with normal 'git merge', this doesn't | 
 | 	remove your own local changes; it just merges those | 
 | 	changes into the latest <local-commit>.  With '--squash', | 
 | 	create only one commit that contains all the changes, | 
 | 	rather than merging in the entire history. | 
 | + | 
 | If you use '--squash', the merge direction doesn't always have to be | 
 | forward; you can use this command to go back in time from v2.5 to v2.4, | 
 | for example.  If your merge introduces a conflict, you can resolve it in | 
 | the usual ways. | 
 | + | 
 | When using '--squash', and the previous merge with '--squash' merged an | 
 | annotated tag of the subtree repository, that tag needs to be available locally. | 
 | If <repository> is given, a missing tag will automatically be fetched from that | 
 | repository. | 
 |  | 
 | split [<local-commit>] [<repository>]:: | 
 | 	Extract a new, synthetic project history from the | 
 | 	history of the <prefix> subtree of <local-commit>, or of | 
 | 	HEAD if no <local-commit> is given.  The new history | 
 | 	includes only the commits (including merges) that | 
 | 	affected <prefix>, and each of those commits now has the | 
 | 	contents of <prefix> at the root of the project instead | 
 | 	of in a subdirectory.  Thus, the newly created history | 
 | 	is suitable for export as a separate git repository. | 
 | + | 
 | After splitting successfully, a single commit ID is printed to stdout. | 
 | This corresponds to the HEAD of the newly created tree, which you can | 
 | manipulate however you want. | 
 | + | 
 | Repeated splits of exactly the same history are guaranteed to be | 
 | identical (i.e. to produce the same commit IDs) as long as the | 
 | settings passed to 'split' (such as '--annotate') are the same. | 
 | Because of this, if you add new commits and then re-split, the new | 
 | commits will be attached as commits on top of the history you | 
 | generated last time, so 'git merge' and friends will work as expected. | 
 | + | 
 | When a previous merge with '--squash' merged an annotated tag of the | 
 | subtree repository, that tag needs to be available locally. | 
 | If <repository> is given, a missing tag will automatically be fetched from that | 
 | repository. | 
 |  | 
 | pull <repository> <remote-ref>:: | 
 | 	Exactly like 'merge', but parallels 'git pull' in that | 
 | 	it fetches the given ref from the specified remote | 
 | 	repository. | 
 |  | 
 | push <repository> [+][<local-commit>:]<remote-ref>:: | 
 | 	Does a 'split' using the <prefix> subtree of <local-commit> | 
 | 	and then does a 'git push' to push the result to the | 
 | 	<repository> and <remote-ref>.  This can be used to push your | 
 | 	subtree to different branches of the remote repository.  Just | 
 | 	as with 'split', if no <local-commit> is given, then HEAD is | 
 | 	used.  The optional leading '+' is ignored. | 
 |  | 
 | OPTIONS FOR ALL COMMANDS | 
 | ------------------------ | 
 | -q:: | 
 | --quiet:: | 
 | 	Suppress unnecessary output messages on stderr. | 
 |  | 
 | -d:: | 
 | --debug:: | 
 | 	Produce even more unnecessary output messages on stderr. | 
 |  | 
 | -P <prefix>:: | 
 | --prefix=<prefix>:: | 
 | 	Specify the path in the repository to the subtree you | 
 | 	want to manipulate.  This option is mandatory | 
 | 	for all commands. | 
 |  | 
 | OPTIONS FOR 'add' AND 'merge' (ALSO: 'pull', 'split --rejoin', AND 'push --rejoin') | 
 | ----------------------------------------------------------------------------------- | 
 | These options for 'add' and 'merge' may also be given to 'pull' (which | 
 | wraps 'merge'), 'split --rejoin' (which wraps either 'add' or 'merge' | 
 | as appropriate), and 'push --rejoin' (which wraps 'split --rejoin'). | 
 |  | 
 | --squash:: | 
 | 	Instead of merging the entire history from the subtree project, produce | 
 | 	only a single commit that contains all the differences you want to | 
 | 	merge, and then merge that new commit into your project. | 
 | + | 
 | Using this option helps to reduce log clutter. People rarely want to see | 
 | every change that happened between v1.0 and v1.1 of the library they're | 
 | using, since none of the interim versions were ever included in their | 
 | application. | 
 | + | 
 | Using '--squash' also helps avoid problems when the same subproject is | 
 | included multiple times in the same project, or is removed and then | 
 | re-added.  In such a case, it doesn't make sense to combine the | 
 | histories anyway, since it's unclear which part of the history belongs | 
 | to which subtree. | 
 | + | 
 | Furthermore, with '--squash', you can switch back and forth between | 
 | different versions of a subtree, rather than strictly forward.  'git | 
 | subtree merge --squash' always adjusts the subtree to match the exactly | 
 | specified commit, even if getting to that commit would require undoing | 
 | some changes that were added earlier. | 
 | + | 
 | Whether or not you use '--squash', changes made in your local repository | 
 | remain intact and can be later split and send upstream to the | 
 | subproject. | 
 |  | 
 | -m <message>:: | 
 | --message=<message>:: | 
 | 	Specify <message> as the commit message for the merge commit. | 
 |  | 
 | OPTIONS FOR 'split' (ALSO: 'push') | 
 | ---------------------------------- | 
 | These options for 'split' may also be given to 'push' (which wraps | 
 | 'split'). | 
 |  | 
 | --annotate=<annotation>:: | 
 | 	When generating synthetic history, add <annotation> as a prefix to each | 
 | 	commit message.  Since we're creating new commits with the same commit | 
 | 	message, but possibly different content, from the original commits, this | 
 | 	can help to differentiate them and avoid confusion. | 
 | + | 
 | Whenever you split, you need to use the same <annotation>, or else you | 
 | don't have a guarantee that the new re-created history will be identical | 
 | to the old one.  That will prevent merging from working correctly.  git | 
 | subtree tries to make it work anyway, particularly if you use '--rejoin', | 
 | but it may not always be effective. | 
 |  | 
 | -b <branch>:: | 
 | --branch=<branch>:: | 
 | 	After generating the synthetic history, create a new branch called | 
 | 	<branch> that contains the new history.  This is suitable for immediate | 
 | 	pushing upstream.  <branch> must not already exist. | 
 |  | 
 | --ignore-joins:: | 
 | 	If you use '--rejoin', git subtree attempts to optimize its history | 
 | 	reconstruction to generate only the new commits since the last | 
 | 	'--rejoin'.  '--ignore-joins' disables this behavior, forcing it to | 
 | 	regenerate the entire history.  In a large project, this can take a long | 
 | 	time. | 
 |  | 
 | --onto=<onto>:: | 
 | 	If your subtree was originally imported using something other than git | 
 | 	subtree, its history may not match what git subtree is expecting.  In | 
 | 	that case, you can specify the commit ID <onto> that corresponds to the | 
 | 	first revision of the subproject's history that was imported into your | 
 | 	project, and git subtree will attempt to build its history from there. | 
 | + | 
 | If you used 'git subtree add', you should never need this option. | 
 |  | 
 | --rejoin:: | 
 | 	After splitting, merge the newly created synthetic history back into | 
 | 	your main project.  That way, future splits can search only the part of | 
 | 	history that has been added since the most recent '--rejoin'. | 
 | + | 
 | If your split commits end up merged into the upstream subproject, and | 
 | then you want to get the latest upstream version, this will allow git's | 
 | merge algorithm to more intelligently avoid conflicts (since it knows | 
 | these synthetic commits are already part of the upstream repository). | 
 | + | 
 | Unfortunately, using this option results in 'git log' showing an extra | 
 | copy of every new commit that was created (the original, and the | 
 | synthetic one). | 
 | + | 
 | If you do all your merges with '--squash', make sure you also use | 
 | '--squash' when you 'split --rejoin'. | 
 |  | 
 |  | 
 | EXAMPLE 1. 'add' command | 
 | ------------------------ | 
 | Let's assume that you have a local repository that you would like | 
 | to add an external vendor library to. In this case we will add the | 
 | git-subtree repository as a subdirectory of your already existing | 
 | git-extensions repository in ~/git-extensions/: | 
 |  | 
 | 	$ git subtree add --prefix=git-subtree --squash \ | 
 | 		git://github.com/apenwarr/git-subtree.git master | 
 |  | 
 | 'master' needs to be a valid remote ref and can be a different branch | 
 | name | 
 |  | 
 | You can omit the '--squash' flag, but doing so will increase the number | 
 | of commits that are included in your local repository. | 
 |  | 
 | We now have a ~/git-extensions/git-subtree directory containing code | 
 | from the master branch of git://github.com/apenwarr/git-subtree.git | 
 | in our git-extensions repository. | 
 |  | 
 | EXAMPLE 2. Extract a subtree using 'commit', 'merge' and 'pull' | 
 | --------------------------------------------------------------- | 
 | Let's use the repository for the git source code as an example. | 
 | First, get your own copy of the git.git repository: | 
 |  | 
 | 	$ git clone git://git.kernel.org/pub/scm/git/git.git test-git | 
 | 	$ cd test-git | 
 |  | 
 | gitweb (commit 1130ef3) was merged into git as of commit | 
 | 0a8f4f0, after which it was no longer maintained separately. | 
 | But imagine it had been maintained separately, and we wanted to | 
 | extract git's changes to gitweb since that time, to share with | 
 | the upstream.  You could do this: | 
 |  | 
 | 	$ git subtree split --prefix=gitweb --annotate='(split) ' \ | 
 |         	0a8f4f0^.. --onto=1130ef3 --rejoin \ | 
 |         	--branch gitweb-latest | 
 |         $ gitk gitweb-latest | 
 |         $ git push git@github.com:whatever/gitweb.git gitweb-latest:master | 
 |  | 
 | (We use '0a8f4f0^..' because that means "all the changes from | 
 | 0a8f4f0 to the current version, including 0a8f4f0 itself.") | 
 |  | 
 | If gitweb had originally been merged using 'git subtree add' (or | 
 | a previous split had already been done with '--rejoin' specified) | 
 | then you can do all your splits without having to remember any | 
 | weird commit IDs: | 
 |  | 
 | 	$ git subtree split --prefix=gitweb --annotate='(split) ' --rejoin \ | 
 | 		--branch gitweb-latest2 | 
 |  | 
 | And you can merge changes back in from the upstream project just | 
 | as easily: | 
 |  | 
 | 	$ git subtree pull --prefix=gitweb \ | 
 | 		git@github.com:whatever/gitweb.git master | 
 |  | 
 | Or, using '--squash', you can actually rewind to an earlier | 
 | version of gitweb: | 
 |  | 
 | 	$ git subtree merge --prefix=gitweb --squash gitweb-latest~10 | 
 |  | 
 | Then make some changes: | 
 |  | 
 | 	$ date >gitweb/myfile | 
 | 	$ git add gitweb/myfile | 
 | 	$ git commit -m 'created myfile' | 
 |  | 
 | And fast forward again: | 
 |  | 
 | 	$ git subtree merge --prefix=gitweb --squash gitweb-latest | 
 |  | 
 | And notice that your change is still intact: | 
 |  | 
 | 	$ ls -l gitweb/myfile | 
 |  | 
 | And you can split it out and look at your changes versus | 
 | the standard gitweb: | 
 |  | 
 | 	git log gitweb-latest..$(git subtree split --prefix=gitweb) | 
 |  | 
 | EXAMPLE 3. Extract a subtree using a branch | 
 | ------------------------------------------- | 
 | Suppose you have a source directory with many files and | 
 | subdirectories, and you want to extract the lib directory to its own | 
 | git project. Here's a short way to do it: | 
 |  | 
 | First, make the new repository wherever you want: | 
 |  | 
 | 	$ <go to the new location> | 
 | 	$ git init --bare | 
 |  | 
 | Back in your original directory: | 
 |  | 
 | 	$ git subtree split --prefix=lib --annotate="(split)" -b split | 
 |  | 
 | Then push the new branch onto the new empty repository: | 
 |  | 
 | 	$ git push <new-repo> split:master | 
 |  | 
 |  | 
 | AUTHOR | 
 | ------ | 
 | Written by Avery Pennarun <apenwarr@gmail.com> | 
 |  | 
 |  | 
 | GIT | 
 | --- | 
 | Part of the linkgit:git[1] suite |