This is an old revision of the document!
Overview
The good and the bad of git subtree (mostly the good).
Links
Synopsis
NAME
       git-subtree - Merge subtrees together and split repository into subtrees
SYNOPSIS
       git subtree add   -P <prefix> <commit>
       git subtree add   -P <prefix> <repository> <ref>
       git subtree pull  -P <prefix> <repository> <ref>
       git subtree push  -P <prefix> <repository> <ref>
       git subtree merge -P <prefix> <commit>
       git subtree split -P <prefix> [OPTIONS] [<commit>]
       
       ... snip ...
       
OPTIONS
       -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.
       -m <message>, --message=<message>
           This option is only valid for add, merge and pull (unsure). Specify <message> as the
           commit message for the merge commit.
The good and the bad
The good:
- Management of a simple workflow is easy.
- Older version of Git are supported (even older than v1.5.2).
- The sub-project’s code is available right after the clone of the super project is done.
- git subtreedoes not require users of your repository to learn anything new. They can ignore the fact that you are using git subtree to manage dependencies.
- git subtreedoes not add new metadata files like git submodule does (i.e., .gitmodule).
- Contents of the module can be modified without having a separate repository copy of the dependency somewhere else.
The bad:
- You must learn about a new merge strategy (i.e. git subtree).
- Contributing code back upstream for the sub-projects is slightly more complicated.
- The responsibility of not mixing super and sub-project code in commits lies with you
Example
Start a new branch
$ git checkout -b subtree
Start a subtree, and examine it
$ git subtree add \ --prefix .vim/bundle/tpope-vim-surround \ https://bitbucket.org/vim-plugins-mirror/vim-surround.git \ master \ --squash git fetch https://bitbucket.org/vim-plugins-mirror/vim-surround.git master warning: no common commits remote: Counting objects: 338, done. remote: Compressing objects: 100% (133/133), done. remote: Total 338 (delta 101), reused 338 (delta 101) Receiving objects: 100% (338/338), 71.46 KiB | 46.00 KiB/s, done. Resolving deltas: 100% (101/101), done. From https://bitbucket.org/vim-plugins-mirror/vim-surround * branch master -> FETCH_HEAD Added dir '.vim/bundle/tpope-vim-surround' $
$ tree .vim
.vim
└── bundle
    └── tpope-vim-surround
        ├── doc
        │   └── surround.txt
        ├── plugin
        │   └── surround.vim
        └── README.markdown
4 directories, 3 files
$
$ git log --oneline cc770c1 (HEAD -> subtree) Merge commit 'bd61f717bcc5d8569846d875952d6aefab987dd1' as '.vim/bundle/tpope-vim-surround' bd61f71 Squashed '.vim/bundle/tpope-vim-surround/' content from commit 02199ea ... snip ... $
To manually update subtree
$ git subtree pull \ --prefix .vim/bundle/tpope-vim-surround \ https://bitbucket.org/vim-plugins-mirror/vim-surround.git \ master \ --squash $