Codepath

Git Rebasing

Introduction

Rebasing is one of Git's most powerful yet misunderstood features. While merging combines branches by creating a new commit that joins two histories, rebasing takes a different approach by reapplying your commits on top of another branch's history. This guide will explain what rebasing is, how to use it effectively, and when to choose it over merging.

🎥 Video: Git Merge vs Rebase (16 mins) - Understanding the differences between rebasing and merging

What is rebasing?

Rebasing is a Git operation that moves or combines a sequence of commits to a new base commit. Unlike merging, which creates a new commit to join histories, rebasing rewrites the commit history by creating new commits for each commit in the original branch.

The term "rebase" literally means to change the base of your branch from one commit to another, making it appear as if you'd created your branch from a different commit.

Git Diagram of Two branches

After rebasing onto main:

Git Diagram of Two branches after rebasing

💡 Tip: Notice how commits D and E have been recreated as D' and E' with new commit hashes after the rebase.

How does rebasing work?

Rebasing works by identifying the common ancestor of the two branches, collecting the changes introduced by the branch you're on, and then replaying those changes on top of the target branch.

The process can be broken down into these steps:

  1. Git temporarily saves the changes made in your current branch as a series of patch files
  2. Git resets your current branch to the same commit as the target branch
  3. Git applies each patch one by one, creating new commits in the process

Diagram of Flow of Rebasing

Example of rebasing

Imagine you have this situation:

# Starting on main branch
$ git checkout -b feature/login  # Create feature branch
# Make some changes
$ git commit -m "Add login form"
$ git commit -m "Add authentication logic"

# Meanwhile, main branch progresses
$ git checkout main
# Some changes by other devs
$ git pull origin main

To rebase your feature branch onto the updated main:

$ git checkout feature/login
$ git rebase main

Why would we use rebase instead of merge?

Rebasing and merging are both designed to integrate changes from one branch into another, but they do it in different ways and serve different purposes.

Advantages of rebasing

  1. Cleaner history: Rebasing produces a linear, straight-line history that's easier to follow
  2. Elimination of unnecessary merge commits: No extra merge commits cluttering the history
  3. Cleaner project history: Makes it look as if your work was created in a sequential order
  4. Easier code reviews: Changes are grouped logically
  5. Fewer conflicts at merge time: If you regularly rebase feature branches on main

git diagram of history from rebasing

Disadvantages of rebasing

  1. Rewrites history: Can be problematic for collaboration if not used carefully
  2. Loss of context: The context of when changes were originally made is lost
  3. Potential for repeated conflict resolution: May need to resolve the same conflicts multiple times
  4. More complex: Generally requires more Git knowledge to use properly

⚠️ Golden Rule of Rebasing: Never rebase a branch that others have based work on or pulled. Stick to rebasing your own local branches that aren't shared.

🎥 Video: Learn Git Rebase in 6 minutes (6 mins) - Demonstration with animations

Rebasing Workflow

Here's a typical workflow that incorporates rebasing:

Working on a feature branch

# Start a new feature
$ git checkout -b feature/new-payment

# Make changes and commit
$ git add payment.js
$ git commit -m "Add payment form"

# Make more changes
$ git add payment-processing.js
$ git commit -m "Add payment processing"

# Keep your branch updated with main
$ git fetch origin
$ git rebase origin/main

Before creating a pull request

# Ensure your branch is up-to-date with main
$ git checkout main
$ git pull
$ git checkout feature/new-payment
$ git rebase main

# Push your changes (force push needed after rebase)
$ git push origin feature/new-payment --force-with-lease

💡 Tip: Forcing is necessary after rebasing because the commit history has changed. Using --force-with-lease is safer than --force as it prevents overwriting others' work.

The above workflow can be visualized as follows:

Visualization of Rebasing Workflow

Handling rebase conflicts

Conflicts during a rebase happen for the same reasons as merge conflicts: Git can't automatically determine how to combine changes. However, they're handled slightly differently.

When conflicts happen during a rebase

During a rebase, Git applies each of your commits one by one on top of the target branch. If a conflict occurs, the rebase pauses at the problematic commit and lets you resolve the conflict.

$ git rebase main
Auto-merging payment.js
CONFLICT (content): Merge conflict in payment.js
error: could not apply a2b3c4d... Add payment validation
Resolve all conflicts manually, mark them as resolved with
"git add/rm <conflicted_files>", then run "git rebase --continue".
You can instead skip this commit: run "git rebase --skip".
To abort and get back to the state before "git rebase", run "git rebase --abort".

Resolving rebase conflicts

  1. Edit the files to resolve conflicts (same markers as merge conflicts)
  2. Add the resolved files to the staging area
  3. Continue the rebase process
  4. Repeat for any additional conflicts
# After manually resolving conflicts in your editor
$ git add payment.js
$ git rebase --continue

If you encounter multiple conflicts, you may need to repeat this process several times, once for each commit that causes conflicts.

Flowchart of Rebasing multiple times

⚠️ Note: Unlike merge, rebase makes you resolve conflicts on a commit-by-commit basis, which can mean resolving the same conflict multiple times if several of your commits modify the same part of a file.

Best practices for rebasing

1. Rebase local branches only

Follow the golden rule: never rebase branches that others have based work on. Stick to rebasing your private feature branches.

2. Communicate with your team

Make sure your team has a shared understanding of when rebasing is appropriate and when it's not.

3. Use interactive rebasing for cleanup

Before sharing your work, use interactive rebasing to clean up your commit history:

$ git rebase -i HEAD~3  # Rebase the last 3 commits

This opens an editor with options for each commit. To make rebasing simpler, you can squash commits together into one, so that you only have to resolve conflicts once.

pick a2b3c4d Add payment form
pick b3c4d5e Fix typo in payment form
pick c4d5e6f Add payment processing

# Commands:
# p, pick <commit> = use commit
# r, reword <commit> = use commit, but edit the commit message
# e, edit <commit> = use commit, but stop for amending
# s, squash <commit> = use commit, but meld into previous commit
# f, fixup <commit> = like "squash", but discard this commit's log message
# ...

4. Use fixup commits during development

When fixing a previous commit, use the --fixup option:

$ git commit --fixup a2b3c4d

Later, use autosquash to automatically organize and combine these during an interactive rebase:

$ git rebase -i --autosquash main

5. Test after rebasing

Always test your code after a rebase to ensure nothing broke in the process.

6. Use force-with-lease when pushing

After rebasing, you'll need to force push, but use --force-with-lease for safety:

$ git push --force-with-lease

Common commands used when rebasing

Basic rebase

# Rebase current branch onto main
$ git rebase main

# Rebase feature branch onto main
$ git checkout feature/branch
$ git rebase main

Interactive rebase

# Start an interactive rebase of the last 5 commits
$ git rebase -i HEAD~5

# Interactively rebase all commits different from main
$ git rebase -i main

Handling conflicts

# Continue rebase after resolving conflicts
$ git rebase --continue

# Skip the current commit in the rebase
$ git rebase --skip

# Abort the rebase and return to the original state
$ git rebase --abort

Advanced rebasing

# Automatically squash fixup commits
$ git rebase -i --autosquash main

# Rebase and run tests after each commit
$ git rebase -x "npm test" main

Further Reading

Fork me on GitHub