Git rebase to fix your local commits

19 July 2011

Let's say you have the following three commits in your local repository:

1% git log -3 --oneline
2b648f1a Fix propagating errors in findOrInitialize. (28 seconds ago)
38789cd1 Non-destructive filtering for bumblebees. (12 hours ago)
41a35285 Propagate errors the node.js way. (14 hours ago)

The last commit, b648f1a conceptually belongs to the first one, 1a35285. It only came later because say you haven't run the tests before committing it and only realized later you introduced a bug. Or some other misdemeanor. Whatever the background is, it would be great if there was a way to squash the two related commits together. Turns out there is: interactive rebase.

The syntax of the git-rebase is the following:

1git rebase [-i | --interactive] [options] [--onto <newbase>]
2             <upstream> [<branch>]

What happens when you do git rebase is that the commits that are on the current branch but are not in upstream are saved. The current branch is reset to upstream and then the saved commits are replayed on top of this.

It's worth to mention that you should only do this if you have not pushed out these changesets to a remote where others might have pulled from it. Rebase creates new commits and if your collaborators pull the new commits, chaos can ensue. (See "Perils of Rebase" in the ProGit book)

This can be used to achieve what we want:

1% git rebase -i HEAD~3

Since the commits that are on the current branch but not on the commit three commits from here are the last three commits, here is what we get:

1pick 1a35285 Propagate errors the node.js way.
2pick 8789cd1 Non-destructive filtering for bumblebees.
3pick b648f1a Fix propagating errors in findOrInitialize.

We want to meld the "fix" commit into the "propagate" commit since that's how it should have been in the first place. So we move b648f1a up and squash it into the previous commit:

1pick 1a35285 Propagate errors the node.js way.
2squash b648f1a Fix propagating errors in findOrInitialize.
3pick 8789cd1 Non-destructive filtering for scales.

After a successful rebase this is how the new log looks like:

1% git log -3 --oneline
273eed18 Non-destructive filtering for bumblebees. (9 seconds ago)
31e63d17 Propagate errors the node.js way. (39 seconds ago)
41b24891 Minor fixes in Bumblebee buzzing. (16 hours ago)

Note that the three commits we had before have now been nicely compacted into two, and the propagation commit is now consistent and fixed. It can now be pushed.

ps. You might wonder what we use bumblebees for in our project. Actually they are faux. They serve to obfuscate real names in propietary code. I hope I can one day write code where bumblebees will be first-class citizens, though.

Share on Twitter