I\'m aware that git bisect is branch-aware by design, so that if between good commit, G, and bad commit, B, you merged in a branch, it needs to take those changes into consi
So, the assumption that the first parent of a merge commit is always the same branch isn't always correct. For instance, if you are off on a topic branch and merge master to it to get up to date (so for this merge commit, the first parent is the topic branch) and then checkout master and merge topic back to it, you get a fast forward merge, which just moves master to the merge commit that had first parent as your topic branch. This may seem contrived, but its actually a pretty normal workflow - I always merge master into my branch so that my merge back to master will be a trivial merge (i.e., fast forward-able) (Sorry James, always forget to rebase it).
There is one way that I've found to help figure out which parent is your branch - the merge commit comment itself. By default, git composes a merge commit comment that says which branch was merged and you can use this to deduce which parent is the branch you are interested in, so long as the person doing the merge commit didn't overwrite this merge commit comment.
So I tried this out and it seems to work for me. I wrote a Python script to help do this on github. If you run this script, it will try to trace backwards and follow your branch and emit a list of commit ids that are the tips of the branches that are merged into your branch. With this list, you can give these to "git bisect good" and bisect will then omit all of the commits on the merged branches from your bisection, achieving the desired result.
If the history looks like:
A - B - C - H - I - J - K - L \ / D - E - F - G
where L is bad, B is good, and you want to ignore the DEFG branch, then running
$ git bisect start $ git bisect skip $( git rev-list G ^C ) $ git bisect bad L $ git bisect good B
where B,C,G,and L are the respective shas seems to do what you want.
You can make git treat you history as linear using grafts. To linearize the whole first parent history you can use:
git rev-list --first-parent --merges --parents HEAD | cut -d' ' -f1,2 > .git/info/grafts
Just drop the grafts file when you're done with the bisection.
Björn Steinbrink's answer works great, but recently started printing this:
hint: Support for /info/grafts is deprecated hint: and will be removed in a future Git version. hint: hint: Please use "git replace --convert-graft-file" hint: to convert the grafts into replace refs. hint: hint: Turn this message off by running hint: "git config advice.graftFileDeprecated false"
Here's a more modern version of his solution using "git replace" instead of grafts:
git rev-list --first-parent --merges --parents HEAD | \ while read COMMIT PARENT1 PARENT2; do git replace --graft $COMMIT $PARENT1; done
Unfortunately, it is much slower for large repos (around 3 minutes for 150k commits); git replace
doesn't seem to have a bulk-mode yet. You might want to restrict the rev-list to only the commits in scope of your bisect.
To remove the replacements when you are done, you can rm .git/refs/replace/*
.
You might be able to use git bisect start --no-checkout
to avoid having to actually checkout the commit into the working tree. Then I suspect that you can do git checkout BISECT_HEAD
for commits that you actually want to test (i.e. only first parent commits on the main branch). I have not tried this but I hope it would work.
I've been looking for something like this too. As far as I've got is that git rev-list --bisect --first-parent
seems to do what you want to, and the docs for rev-list implies that the --bisect
option is what bisect uses internally - but getting git bisect
to add that flag to its call(s) to rev-list seems less trivial:
The bisect command is implemented by a shell script git-bisect, which in turn uses a builtin command bisect--helper
to actually do the interesting part ("computation, display and checkout" says the comment...), apparently based on a bunch of magic state files in .git/. And it seems to be the rev-list command that is reusing code from bisect--helper rather than the other way around as you might expect.
So, you'd have to extend the bisect--helper code's commit filtering to do it, I think.
As a workaround, something like this might work: after bisect checks something out for you, reset to a different one using git rev-list --bisect --first-parent
, test that and mark it good/bad/skip and continue from there.