Subtree2 is a merge strategy for git similiar to the built-in subtree strategy. Additionally, subtree2 has the ability to merge subdirectories of the source branch, too. Like the default subtree strategy it can also merge to subdirectories of the target branch.
The merge strategy is accompanied by a custom git command git subproject, that can be used for pulling a subdirectory of a different branch or repository into a subdirectory of the target/current branch. It merges like git merge --squash
, so that the history does not become cluttered. Source branch and commit are remembered within the merge commit's message to simplify later updates. A subproject can also be merged/pushed back into the source branch.
feature | -s subtree2 |
built-in -s subtree |
built-in -s recursive -Xsubtree=<prefix> |
---|---|---|---|
Merging from a subdirectory of the source branch | -Xtheir-prefix=<prefix> |
- | - |
Merging to a subdirectory of the target branch | -Xmy-prefix=<prefix> |
(tries to auto-detect subdirectory) | -Xsubtree=\<prefix\> |
Merge Algorithm [1] | like built-in resolve |
??? (most likely recursive ) |
recursive |
Merging unrelated histories (--allow-unrelated-histories) | fast forward merge respecting source and target prefix; explicit base can be given using -Xbase=\<base\> |
merges source branch into an automatically chosen subdirectory or the root of the target branch (most likely not the place you expect) | ignores explicit prefix unless the prefix directory is known to git |
Well tested | no | probably | probably |
[1] Some background information on recursive
vs resolve
strategy
-Xtheir-prefix=<prefix>
: Consider only files in subdirectory <prefix> of source branch-Xmy-prefix=<prefix>
: Merge to subdirectory <prefix> of target branch-Xbase=<base-id>
: Use the given commit id as base revision during 3way merge instead (usefull when re-merging aftergit merge --squash
)-Xbase-prefix=<prefix>: The (old) \<prefix\> applicable to the base if different from
their-prefix` (because the directory has been renamed)-Xdiff3
: Show conflicts in "diff3" style, which means that the common ancestor's version is included in confict markers-Xours|-Xtheirs|-Xunion
: resolve conflicts favouring our (or their or both) side of the lines
Add a working copy of this repository to your path.
$ git subproject help
Include another branch or a subdirectory thereof as subdirectory of the
current branch. By refering to remote branches, other repositories can be
included as well. Subproject merges like `git merge --squash`, so that the
history does not become cluttered. Source branch and commit are remembered
within the merge commit's message to simplify later updates. A subproject can
also be merged back into the source branch.
git subproject init <my-prefix> (--their-branch=<their-branch> | <their-branch>) [-m <message>] [--format=<format>] [--their-prefix=<their-prefix>] [--filter=<filter>] [--filter-is-regexp|--filter-is-glob]
git subproject pull <my-prefix> [-m <message>] [--format=<format>] [--their-prefix=<their-prefix>] [--filter=<filter>] [--filter-is-regexp|--filter-is-glob] [--their-branch=<their-branch>] [--base=<base>] [--base-prefix=<base-prefix>] [--base-filter=<base-filter>] [--diff3] [--ours|--theirs|--union]
git subproject push <my-prefix> [-m <message>] [--format=<format>]
Init:
Copy a remote branch's content to a subdirectory <my-prefix> of the current
branch. The command will refuse to run if <my-prefix> already exists.
my-prefix: mandatory - subdirectory in which their branch will be
included
their-branch: mandatory - branch to be merged into a subdirectory of the
current branch
their-prefix: optional - limit merging to this subdirectory of their
branch
filter: optional - limit merging to the set of files matching the
given glob pattern (see --filter-is-glob); the pattern is is
matched against the file path relative to the subproject
directory; `\\` is used as escape character
filter-is-glob: default - interpret --filter as glob pattern, such that
subproject mimics bash's behavior of filename expansion with
options 'globstar' and 'dotglob' enabled, as well as 'extglob'
disabled - see section 3.5.8 of the bash reference manual for
details; the following features are currently not supported,
however: predefined character classes ([:alnum:], ...),
equivalence classes ([=c=], ...) and matching of collating
symbols ([.symbol.]); in constrast to bash, multiple patterns
can be combined using the pipe `|` symbol using an or-logic
filter-is-regexp: optional - interpret --filter as perl style regular expression
- see `man pcre`
message: optional - custom commit message
format: optional - pass to `git log`'s format option when appending
a description of the merged history to the commit message; use
`--format=` to suppress the history
Pull:
Update a subproject by merging changes from the upstream branch.
This command can also be used to change the name and/or prefix of the
upstream branch.
my-prefix: mandatory - subdirectory containing an existing subproject
their-branch: optional - change the name of the upstream branch for now and
the future; you might need to specify a different base for
merging (see below)
base: optonal - overwrite the common-ancestor used for 3way merge;
this might be usefull if the source branch's history has been
rewritten; note - no history can be embedded into the commit
message when this option is used
their-prefix: optional - change their prefix if the directory has been
renamed since the last update
base-prefix: optional - overwrite the prefix applied to the common
ancestor; this might be necessary when specifying a custom
base
filter: optional - limit merging to the set of files matching the
given glob pattern (see --filter-is-glob); the pattern is is
matched against the file path relative to the subproject
directory; `\\` is used as escape character
filter-is-glob: default - interpret --filter as glob pattern, such that
subproject mimics bash's behavior of filename expansion with
options 'globstar' and 'dotglob' enabled, as well as 'extglob'
disabled - see section 3.5.8 of the bash reference manual for
details; the following features are currently not supported,
however: predefined character classes ([:alnum:], ...),
equivalence classes ([=c=], ...) and matching of collating
symbols ([.symbol.]); in constrast to bash, multiple patterns
can be combined using the pipe `|` symbol using an or-logic
filter-is-regexp: optional - interpret --filter as perl style regular expression
- see `man pcre`
base-filter: optional - like filter, but applied to the base revision,
only; this option can be usefull when explicitly specifying
a base revision and file names have changed
message: optional - custom commit message
format: optional - pass to `git log`'s format option when appending
a description of the merged history to the commit message; use
`--format=` to suppress the history
diff3: optional - show conflicts in "diff3" style, that is the common
ancestor's version is included in confict markers
ours:
theirs:
union: optional - resolve conflicts favouring our (or their or both)
side of the lines.
Push:
Push local changes to the upstream branch. Refused if the upstream branch
has changed since the last update.
my-prefix: mandatory - subdirectory containing an existing subproject
message: optional - custom commit message
format: optional - pass to `git log`'s format option when appending
a description of the merged history to the commit message; use
`--format=` to suppress the history
Continue:
Continue a pull/push operation after merge conflicts have been resolved.
- On windows it might be necessary to export
MSYS="noglob"
when dealing with filters containing escaped backslashes (\
) together with any of the characters~?*["'(){}
- see git-for-windows/git#1019 for details