Git Subtree

Git Subtree

As we mention on our previous page, Git Submodule is useful for only some cases. But there is a preferable way of tracking the history of software dependencies. Many developers prefer Git Subtree to Git Submodule.

What is Git Subtree

What is Git Subtree

Git Subtree is an alternative to Git Submodule. It allows nesting one repository inside another one as a subdirectory. It is one of the ways to track the history of software dependencies. But subtrees should not be confused with submodules. Unlike submodules, subtrees do not need .gitmodules files or gitlinks in the repository. A subtree is just a subdirectory that you can commit to, branch, and merge along with your project.

Why to Use Git Subtree

Why to Use Git Subtree

Pros

git subtree add --prefix .vim/bundle/example https://github.com/Example/vim-example.git master --squash
commit 6d7054b3gcea64e2e31f4d6fb2e3be12e5865e87
Merge: 87fa91e ef86deb
Author: Ann Smith<[email protected]>
Date:   Tue Jun 10 13:37:03 2016 +0200
    Merge commit 'fe67ddf158faccff4082d78a25c45d8cd93e8ba8' as '.vim/bundle/example'
commit fe67ddf158faccff4082d78a25c45d8cd93e8ba8
Author: Ann Smith<[email protected]>
Date:   Tue May 12 13:37:03 2015 +0200
    Squashed '.vim/bundle/example/' content from commit b999b09
    git-subtree-dir: .vim/bundle/example
    git-subtree-split: b999b09cd9d69f359fa5668e81b09dcfde455cca

To update the sub-folder to the latest version of the child repository, run the following:

git subtree pull --prefix .vim/bundle/example https://github.com/Exampel/vim-example.git master --squash

But, git subtree stores subproject commit ids and not references in the meta-data. Find the symbolic name connected with a commit:

git ls-remote https://github.com/Example/vim-example.git | grep <sha-1>

Rebasing After Git Subtree

Rebasing After Git Subtree

Here, you should use the –interactive mode of git rebase and remove the add commits, then execute rebase–continue and re execute the git subtree add command after the rebase process is done.

OPTIONS

OPTIONS

类目类目
-q, –quietSuppresses the unnecessary result messages on stderr.
-d, –debugProduces more unnecessary result messages on stderr.
-P <prefix>, –prefix=<prefix>Defines the path in the repository to the subtree you want to manipulate. It is mandatory for all commands.
-m <message>, –message=<message>Specifies <message> as the commit message for the merge commit.
It is only valid for add, merge and pull.

Using Git Subtree Without Remote Tracking

Using Git Subtree Without Remote Tracking

Add the git subtree at a specified prefix folder. Use the –squash flag to preserve the whole subproject history in your main repository:

git subtree add --prefix .vim/bundle/vim-double-upon https:/hostname.org/example/vim-plugins.git master --squash

The result will be the following:

git fetch https:/hostname.org/example/vim-plugins.git  master
warning: no common commits
remote: Counting objects: 325, done.
remote: Compressing objects: 100% (145/145), done.
remote: Total 325 (delta 101), reused 313 (delta 89)
Receiving objects: 100% (325/325), 61.47 KiB, done.
Resolving deltas: 100% (110/110), done.
From https:/hostname.org/vim-plugins.git
* branch master -} FETCH_HEAD
Added dir '.vim/bundle/vim-double-upon '

This creates a merge commit by squashing the entire history of the vim-surround repository into a single one:

3bca0ad [4 minutes ago] (HEAD, stree) Merge commit 'fa2f5dc4f1b94356bca8a440c786a94f75dc0a45' as '.vim/bundle/vim-double-upon' [John Brown]
fa2f5dc [4 minutes ago] Squashed '.vim/bundle/vim-double-upon/' content from commit 13189ec [John Brown]

For updating the code of the plugin from the upstream repository, do a git subtree pull:

git subtree pull --prefix .vim/bundle/vim-double-upon https:/hostname.org/example/vim-plugins.git master --squash

To make the commands shorter, add the sub-project as a remote.

Adding Sub-project as a Remote

Adding Sub-project as a Remote

Adding as a remote shortens the process:

git remote add -f vim-double-upon https:/hostname.org/example/vim-plugins.git

Add the subtree:

git subtree add --prefix .vim/bundle/vim-double-upon vim-double-upon master --squash

Update the sub-project like this:

git fetch vim-double-upon master
git subtree pull --prefix .vim/bundle/vim-double-upon vim-double-upon master --squash

Git subtree is the alternative from submodules, but if submodules are aimed at putting another project in a directory of your repository and keeping the remote repository in sync, git subtree allows keeping a subproject separate and allow bidirectional collaboration between your main repo and the subprojects.



请遵守《互联网环境法规》文明发言,欢迎讨论问题
扫码反馈

扫一扫,反馈当前页面

咨询反馈
扫码关注
返回顶部