< Back to Blog

How to Clean Up Fully Merged Feature Branches

Any Git repository may have several outdated branches lingering around. You realize it may be a good idea to clean up your project, but you're not sure which feature branches can be safely removed. Now what?
No worries — you've come to the right place. Let's find out how to delete fully merged branches from a Git repository with confidence.

Most projects have a designated base branch where all feature branches eventually get merged into, such as main or master. Depending on the workflow you are using, there may be more long-running branches; for example, some projects also have a next branch, where the next release is prepared.

Base branches play an important role in determining if branches have been merged, because Git can only tell you that a branch has been merged into some other branch — it cannot determine if a branch has been merged in a way that it can be deleted.

For example, a branch could have been merged into next but not into main yet. How can you be sure?

Introducing git branch --contains and git branch --merged

Git provides two helpers:

  1. git branch --contains <commit>
  2. git branch --merged <commit>

By typing git branch --contains <commit> you will get the list of branches that contain the named commit. Your local list of branches will be filtered by the branches who are a descendant of a certain branch (or arbitrary commit). You could type git branch --no-contains <commit> to get the opposite list.

In other words, the given branch has been merged into the listed branches. This would help you in determining if a branch has been merged into next and main.

With git branch --merged <commit>, your local list of branches will be filtered by all the branches who have been merged into a given branch or commit. Similar to above, you could type git branch --no-merged <commit> and only the branches not merged into the named commit would be listed.

This would help you get a list of branches that have been merged into next. To see which branches have also been merged into main, you would need to run the command again.

To put it another way, contains and merged are like a relationship and its inverse:

Difference between "Contains" and "Merged"
Difference between "Contains" and "Merged"


Note that these commands will only work with your existing local branches. Git doesn't magically detect which branches have been merged into e.g. next for all history backwards — it simply checks your local branches one by one.

However, you can use the -r option to run the checks against remote branches.

Pitfalls

There is a caveat: this only works if you merged branches through a plain git merge. If you squash-merged a feature branch, cherry-picked a small feature branch, or somehow altered the commit hash, Git won't be able to detect if a given branch has been fully merged!

As an alternative, you check each branch individually with git cherry -v, which will compare by diffs instead of hashes. Unfortunately, this isn't foolproof either; if you split up commits, for example, it won't be helpful.

This command requires a specific base branch to be given for comparison of a feature branch.

Let's find out how this can be simplified with Tower, our Git GUI for macOS and Windows.

Not a Tower user yet?

Download our 30-day free trial and experience a better way to work with Git!

Tower app icon


Cleaning Up Merged Branches In Tower

Based on the explanation above, filtering or answering the “Fully merged?” question is more effective if the user specifies a set of base branches that can be compared against.

For this reason, we introduced a new "Branches Review" view in Tower 8 for Mac.

This view compares your local branches to a base branch. You can filter this list of branches by "Fully Merged" to figure out the ones that can safely be deleted:

Cleaning Up Merged Branches in Tower


As you can see, with Tower, a cleaner repository is only a few clicks away!

This is only one of the several advanced features you can find in Tower. Don't forget that you can try it for free for 30 days!

Your Download is in Progress…

Giveaways. Cheat Sheets. eBooks. Discounts. And great content from our blog!