---
title: "Git Submodule Maintenance"
lastmodified: '2012-04-05'
redirect_from:
- /Git_Submodule_Maintenance/
---
Git Submodule Maintenance
=========================
One way of doing work on submodules is described in [GitHub help on submodules](http://help.github.com/submodules/), however the workflow described below is preferred for Mono.
### Git submodules in Mono
These instructions are for developers working on code in this repository. End users do not need to be concerned with the procedures described below. The description applies to each of the submodules used by mono. To list the submodules in use run:
git submodule
#### Adding a submodule
If you have write access to the submodule repository do your work on it in a separate location, do not ever do any work in the mono copy of the submodule.
All submodules should reside under the external/ directory off the top level Mono directory. If you want to add a new submodule, issue the following command from the Mono topmost directory:
git submodule add REPOSITORY_URL external/MODULE_NAME
After that commit and push the changes to .gitmodule and the submodule itself.
### Submodule repository of origin maintenance
The submodule repository of origin (at the REPOSITORY_URL above) must always be modified outside the Mono directory. The repository may be a fork or clone of yet another GIT repository, either on github or elsewhere. If this is the case, you must configure your clone of it by adding a remote reference to the upstream repository:
git remote add upstream UPSTREAM_URL
When there exist upstream changes you need to merge, the following command needs to be used:
git fetch upstream/master
followed by
git merge upstream/BRANCH_NAME
and as soon as all the possible confilits are resolved, push the freshly integrated changes back to our repository
git push origin/BRANCH_NAME
### Submodule maintenance in Mono repository
When the submodule repository of origin is ready to be used by Mono, you need to go to the top level directory of Mono repository clone and make sure the submodules are checked out and up to date:
git submodule init
git submodule update --recursive
Then in order to integrate changes from the submodule repository of origin:
cd external/MODULE_NAME
git pull
you can, of course, use a a specific commit in the 'git pull' above instead of the default HEAD. When the changes are checked out, commit the changes to the Mono repository:
cd ../..
git add external/MODULE_NAME
git commit -m "Module MODULE_NAME updated"
git push
It is **extremely** important not to end the `external/MODULE_NAME` reference above with a / since that will make git remove the submodule reference and commit all its contents as normal files to Mono directory. It is also required that you **always** push the submodule changes **before** any and all work requiring the changes is done in the Mono repository.
Submodule modifications workflow
--------------------------------
If you work both on the submodule itself and Mono, here's a workflow you need to follow in order to maintain a healthy relationship between Mono and the submodule.
Let's assume the following directory hierarchy:
|vc
|----FOO_LIBRARY [your clone of FOO_LIBRARY repository]
|----Mono
|-------external
\----------FOO_LIBRARY [submodule copy of FOO_LIBRARY repository]
We have two scenarios to consider when working with FOO_LIBRARY:
- it is a fork of some upstream repository
- it is the orignal repository
What follows is a workflow which covers both cases above. The only difference is the first step - in the former case you pull changes from upstream, in the latter you make your own changes. You can, of course, combine the two. One thing to keep in mind is that if the submodule is a fork it is a good idea to keep your commits as finely grained as possible. That will make merging easier when you pull changes from the submodule's upstream.
FOO_LIBRARY upstream has made changes to their repository and you need those changes integrated in Mono. For brevity let's assume you make no further changes to the upstream code.
Let's assume our current directory is `vc` in the hierarchy above.
- Pull upstream changes from upstream **master** branch
cd FOO_LIBRARY
git checkout master
git fetch upstream/master
git merge upstream/master
- Resolve merge conflicts, if any
- Push merged changes to our clone of FOO_LIBRARY
git push origin/master
- Test changes with Mono
cd ../Mono
git pull
git submodule update --init --recursive
rsync -a --delete-after ../FOO_LIBRARY external/FOO_LIBRARY
./configure
make
- Run all necessary tests in Mono
- Reset submodule to its original state
rm -rf external/FOO_LIBRARY
git submodule update --init --recursive
- Update submodule with changes in FOO_LIBRARY repository
cd external/FOO_LIBRARY
git pull
- Update Mono reference to the new submodule commit
cd ../../
git add external/FOO_LIBRARY
git commit -m "FOO_LIBRARY submodule updated"
git push
- Do other work which depends on the new code in FOO_LIBRARY