354 lines
13 KiB
Plaintext
354 lines
13 KiB
Plaintext
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
|