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 [9124] 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 process.

A Sneak Peek at
The Internet

If you enjoyed this article, you might be interested in my new project, A Sneak Peek at The Internet. What happens when you enter www.facebook.com into your web browser and hit return? A Sneak Peek at The Internet will take you on a deep dive through the network stack, from HTTP, SSL, TCP and IP, all the way down through the data link layer, back up to Facebook's data centres, and then on the return journey back to the browser.

There's more fun, excitement and peril than a Disneyland rollercoaster!