Rewriting history¶
There are several commands in Git for rewriting history. git rebase -i is
the best known and most flexible: you can reorder, merge, edit and remove
commits. However, this flexibility comes with a degree of complexity: your
working tree and index
are updated, and conflicts may arise that need to be resolved before you can
continue working.
With git commit --fixup and git rebase --autosquash, on the other hand,
you can correct a series of commits relatively easily. Let’s demonstrate this
with an example below:
We have two commits in our
my-featurebranch: one for the actual function, the other for the associated tests:$ git log --oneline my-feature ^origin/main a4587fa (my-feature) Add test for my new feature 56e34e9 Add new feature
During the merge or pull request, we receive feedback on both our function and our tests, which we would like to integrate into our existing commits. To do this, we first create temporary commits:
$ git commit -m "Feedback on the tests from my function" $ git commit -m "Feedback on my function" $ git log --oneline my-feature ^origin/main 556c1e8 (my-feature) Feedback on my function 8780db6 Feedback on the tests from my function a4587fa Add test for my new feature 56e34e9 Add new feature
… with git rebase¶
With
git rebase -iwe can interactively rearrange thepicklines:$ git rebase -i origin/main
This opens our editor:
pick 56e34e9 Add new feature pick a4587fa Add test for my new feature pick 8780db6 Feedback on the tests from my function pick 556c1e8 Feedback on my function
We can then change the lines, for example to:
pick 56e34e9 Add new feature squash 556c1e8 Feedback on my function pick a4587fa Add test for my new feature squash 8780db6 Feedback on the tests from my function
Now we have two commits again:
$ git log --oneline my-feature ^origin/main 31a140a (my-feature) Add test for my new feature 132ae9b Add new feature
The changes can now be pushed to our remote branch using
git push --force-with-lease. The--force-with-leaseoption ensures that any existing changes on the remote branch are not overwritten.
…with git commit --fixup and git rebase --autosquash¶
In Git, however, there is an even easier way to correct a previous commit: with
git commit--fixup and git rebase --autosquash.
We create two temporary commits again, but this time with
git commit--fixup:# Further changes to the tests $ git commit --fixup=31a140a [my-feature dd0c0d1] fixup! Add test for my new feature 1 file changed, 1 insertion(+) # Further changes to my function $ git commit --fixup=132ae9b [my-function bc2298a] fixup! Add new feature 1 file changed, 1 insertion(+) $ git log --oneline my-feature ^origin/main bc2298a (my-feature) fixup! Add new feature dd0c0d1 fixup! Add test for my new feature 31a140a Add test for my new feature 132ae9b Add new feature
For commits with the
--fixup=SHAoption, Git writes a specially formatted commit message that can be read as this commit corrects that commit.Instead of using
git rebase -ito manually specify thePick/Squashlines, we can now simply rungit rebase --autosquash:$ git rebase --autosquash origin/main Successfully rebased and updated refs/heads/my-feature. $ git log --oneline my-feature ^origin/main 694cb48 (my-feature) Add test for my new feature 55cbe9b Add new feature
git rebase --autosquashautomates what we have just done manually withgit rebase -i– but it does not open an editor in which we have to move the commits manually.Tip
The
--fixupoption also contains theamendandrewordoptions to reformulate the commit message, for examplegit commit --fixup:amend=SHA.Further options can be found in the Git commit documentation.
git history¶
Added in version 2.54: Git 2.54 introduces git history on an experimental basis, meaning that
the interface is still subject to further development. git history makes
it easier to correct typos in previous commit messages and to split commits
into two parts:
git history reword SHAopens your editor with the message of the specified commit and rewrites it directly, updating all branches descended from that commit. Unlike Git rebase, it does not access your working tree or your index.
git history split SHAinteractively splits a commit into two parts, allowing you to select which parts should be moved into a new parent commit. The interface is the same as that of
git add --p. After selecting the blocks, Git creates a new commit with these changes as a predecessor to the original commit, which retains all unselected blocks, and rewrites all downstream branches so that they point to the updated history.
Warning
history does not support histories containing merge commits, nor can
it perform operations that would result in a merge conflict.