In a previous post, Using git submodules to track plugins I introduced the idea of using git submodules as part of your workflow in developing Rails applications. At the time, Rails itself wasn’t using git, but that has finally happened. You can find the official Ruby on Rails source code repository at http://github.com/rails/rails. So, how to we track Rails with git submodules?
Let’s start from our books application of the previous post. And we’ll add in the submodule for Rails:
mathie@tullibardine:books$ git submodule add git://github.com/rails/rails.git vendor/rails Initialized empty Git repository in /Users/mathie/tmp/src/books/vendor/rails/.git/ remote: Generating pack... remote: Done counting 67937 objects. remote: Deltifying 67937 objects... remote: 100% (67937/67937) done remote: Total 67937 (delta 52081), reused 67937 (delta 52081) Receiving objects: 100% (67937/67937), 9.88 MiB | 20 KiB/s, done. Resolving deltas: 100% (52081/52081), done.
and let’s see what that’s done:
mathie@tullibardine:books$ git stat # On branch master # Changes to be committed: # (use "git reset HEAD <file>..." to unstage) # # modified: .gitmodules # new file: vendor/rails # mathie@tullibardine:books$ git diff --cached diff --git a/.gitmodules b/.gitmodules index 228959d..64ce630 100644 --- a/.gitmodules +++ b/.gitmodules @@ -4,3 +4,6 @@ [submodule "vendor/plugins/timestamped_booleans"] path = vendor/plugins/timestamped_booleans url = git://github.com/rubaidh/timestamped_booleans +[submodule "vendor/rails"] + path = vendor/rails + url = git://github.com/rails/rails.git diff --git a/vendor/rails b/vendor/rails new file mode 160000 index 0000000..f46fd6f --- /dev/null +++ b/vendor/rails @@ -0,0 +1 @@ +Subproject commit f46fd6f2fceb22f00669f066fc98f92a18e5875f
So we’ve successfully added in a new submodule for Rails, for the current HEAD. Let’s commit that and push it upstream:
mathie@tullibardine:books$ git commit -m "Use git submodules to track HEAD of rails repository." Created commit 4bb82c9: Use git submodules to track HEAD of rails repository. 2 files changed, 4 insertions(+), 0 deletions(-) create mode 160000 vendor/rails mathie@tullibardine:books$ git push [ ... ]
That’s all well and good, but we don’t always want to track edge Rails. Sometimes we want to track the stable branch or pin ourselves to a particular version. Let’s first of all pin to version 2.0.2:
mathie@tullibardine:books$ cd vendor/rails/ mathie@tullibardine:rails$ git tag [ ... ] v2.0.2 [ ... ] mathie@tullibardine:rails$ git co v2.0.2 Note: moving to "v2.0.2" which isn't a local branch [ ... ] HEAD is now at c8da518... Tagged Rails 2.0.2 mathie@tullibardine:rails$ cd ../.. mathie@tullibardine:books$ git status # On branch master # Changed but not updated: # (use "git add <file>..." to update what will be committed) # # modified: vendor/rails # no changes added to commit (use "git add" and/or "git commit -a") mathie@tullibardine:books$ git add vendor/rails mathie@tullibardine:books$ git commit -m "Pin ourselves to Rails v2.0.2." Created commit 5e66474: Pin ourselves to Rails v2.0.2. 1 files changed, 1 insertions(+), 1 deletions(-) mathie@tullibardine:books$ git push [ ... ]
What we’ve done here is look at the Rails repository’s list of tags to see
which one tags version 2.0.2. We find that it’s called “v2.0.2” so we will
check out that particular tag. Since we’re not making any changes, just
wanting to check out the tree at that particular state, we don’t need to worry
that we’re not on a branch. We then shift back up to our project root and do a
git status. This notes that we have made a change to
vendor/rails in that
we’re tracking a different commit id. Do a
git diff and you’ll see what I
mean. We’re happy to add that to the index, commit and push it upstream. Now
everybody we’re sharing with is also pinning their clone to Rails 2.0.2.
How about we want to track the stable branch instead? Just as easy:
mathie@tullibardine:books$ cd vendor/rails mathie@tullibardine:rails$ git co origin/2-0-stable Note: moving to "origin/2-0-stable" which isn't a local branch If you want to create a new branch from this checkout, you may do so (now or later) by using -b with the checkout command again. Example: git checkout -b <new_branch_name> HEAD is now at 2c96f50... Merge  from trunk: Avoid remote_ip spoofing.
This time we’re just checking out a copy of the remote branch
origin/2-0-stable. The rest is exactly the same as before – add the fact
that you’re tracking a different commit to the index of your main project,
commit it and push. Every time you want to update to the latest version of the
2-0-stable branch (just like when you did
piston update), you repeat this