Git submodules are the proper way to use git repositories that are libraries/packages/modules from a main project git repository.
The basic idea is that the main project repository will be connected to one concrete commit of the submodule. This commit can be changed using typical git commands on the submodule repository like "git checkout" as we will see.
Below you can find the most frequent git commands and an explanation, to work with git submodules in a typical scenario between several developers. Code snippets are based on a main project called "gst-project" and a submodule project called "gst-submodule".
##Add a submodule to the main project.
git submodule add https://github.com/alexmorosmarco/gst-submodule.git
(must be executed on main project path)
This command adds a git repository as a submodule of main project and clones it to the current path, inside a folder named as the submodule repository name by default. So in the above example it creates a folder called "gst-submodule" on current path and clones the repository inside it.
Actually it is not same thing as cloning. It clones the code of a concrete commit (last one of master branch by default), and not the code of the head of a concrete branch. So when pulling the main project the submodule code is also pulled but the connected commit is not updated cause the submodule is frozen at that concrete commit by security.
git submodule add https://github.com/alexmorosmarco/gst-submodule.git <path_to_submodule>
Alternatively we can define other name for the folder as an additional parameter.
##Check the status of the submodules
cat .gitmodules
(must be executed on main project path)
The command git submodule add
creates a new file called ".gitmodules" which contains the information of the submodules (local and remote paths and names). Yo can have a look at it to see that information.
git submodule status
(must be executed on main project path)
Prints the pointed commit SHA of every submodule of the main project. That SHA is stored in the folder file of each submodule.
##The typical scenario between several developers.
Step 1: 1st developer adds submodules to the main project using git submodule add ...
command like explained before.
Step 2: 2nd developer prepares his local copy of the main project:
-
Clone the main project using
git clone https://github.com/alexmorosmarco/gst-project.git
(if it is not already on local; otherwisegit pull
to get its last version). -
git submodule init
Initializes the submodules based on the content of the file ".gitmodules" of main project. This command does not clone/download any code.
-
git submodule update
Clones the submodules repositories if their code has not been checked out and points the submodule local repository to the commit that is defined for this submodule in the main project. If the code had already been checked out including that commit, then it just points the submodule local repository to that commit. If the pointed commit has not been checked out, then a
git pull
of the main project is needed before doing agit submodule update
.
Step 3: 1st developer modifies a submodule and pushes it
-
Commit and push the changes on the submodule git repository.
-
Commit and push the changes on main project git repository, that will include the submodule folder which contains the new pointed commit SHA of the submodule.
Note: this works like this cause everytime we change the working copy commit on the submodule git repository, through
git checkout ...
orgit commit ...
for instance, the main project submodule folder will point to that commit.
Step 4: 2nd developer updates to last changes on the submodules
-
git pull
On main project folder. It will pull the main project code and also the submodules code. Instead of this you can do
git fetch
and thengit merge ...
the branches as you need, but take into account that the submodules folders of main project need to be checked out so that git can update the submodules later. -
git submodule update
It updates the submodules to the pointed SHA commit by the main project.
##Credits. Thanks to "svenax" for uploading this screencast from Scott Chacon.