Re: git pull/rebase bug: when "onto" branch has rebasing branch's commits in reflog

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Sun, Jan 3, 2021 at 5:33 PM Johannes Sixt <j6t@xxxxxxxx> wrote:
>
> Am 03.01.21 um 23:02 schrieb Andrew Oates:
> > I've poked at the source code but haven't found exactly what causes
> > the issue --- but if you do a 'git pull --rebase' or 'git rebase' onto
> > a tracking branch that has previously pointed to a commit that the
> > rebasing branch includes, the rebase will be a noop.
> >
> > In practice I've hit this a few times lately when splitting a topic
> > branch into two branches after the fact.
> >
> > Here is a short repro:
> > ```
> > git init
> > touch file1
> > git add file1
> > git commit -a -m "first commit"
> > touch file2
> > git add file2
> > git commit -a -m "second commit"
> > touch file3
> > git add file3
> > git commit -a -m "third commit"
> > git checkout -b branch
> > git branch --set-upstream-to=master
> > git checkout master
> > git reset --hard 'HEAD^1'
> >
> > touch file2.5
> > git add file2.5
> > git commit -a -m "second-and-a-half commit"
> > git --no-pager log --oneline --all --graph
> >
> > #rm .git/logs/refs/heads/master
> >
> > git checkout branch
> > git pull -v --rebase
> > git --no-pager log --oneline --all --graph
> > ```
> >
> > This outputs,
> > * 58432a7 (branch) third commit
> > | * 0e4f775 (HEAD -> master) second-and-a-half commit
> > |/
> > * 37b2e3f second commit
> > * 5e9f0b7 first commit
> > ...
> > Successfully rebased and updated refs/heads/branch.
> > * 0e4f775 (HEAD -> branch, master) second-and-a-half commit
> > * 37b2e3f second commit
> > * 5e9f0b7 first commit
> >
> > showing that "third commit" is lost.  If you execute the "rm ..."
> > line, then the sequence works as expected, and the final state is,
> >
> > Successfully rebased and updated refs/heads/branch.
> > * b636309 (HEAD -> branch) third commit
> > * 410a5dc (master) second-and-a-half commit
> > * 41981d0 second commit
> > * 286398d first commit
> >
> > My best guess is that there's something odd happening in get_fork_point().

Yeah, I guess this is actually WAI, though the outcome was surprising
to me (git silently dropped my commits).  Of course, after poking for
a while, as soon as I sent this I found some clear documentation for
the behavior :)  Under the git rebase docs:
"If your branch was based on <upstream> but <upstream> was rewound and
your branch contains commits which were dropped, this option can be
used with --keep-base in order to drop those commits from your
branch."

In that case, maybe this is a UX suggestion rather than a bug report.
I experienced this via "git pull" rather than "git rebase", which I
generally consider a pretty safe thing to run --- but in this case,
git silently dropped my commits (which luckily I noticed).

Thoughts on how this would be less surprising to me as a user,
 1) somehow make it just DTRT (though per the rebase --fork-point
docs, and your point, this may not be possible/desirable)
 2) document these commits out when checking out the branch[1]
 3) after a pull, log how many such commits were "dropped" for this reason

As it currently stands, the behavior is surprising --- the commits
that get rebased don't match either what is printed when checking out
the branch, or by the summary in the docs ("This is the same set of
commits that would be shown by git log <upstream>..HEAD"), and the
commits are dropped silently.  The fact that the commits are somehow
tied to the old branch via the reflog breaks my mental model of branch
history not being semantically important, or the fact that "git pull"
in two repositories that are otherwise identical would have a
different outcome based on the individual history.

[1] it currently says something along the lines of "...and have X and
Y different commits each, respectively.
  (use "git pull" to merge the remote branch into yours)", which I've
always assumed would mean that (barring stuff like duplicate commits),
I would end up with X+Y commits after pulling.
>
> To me, the outcome looks reasonable:
>
> By the time when the branch was created, it had no commits on top of
> master. Then master was rewound and grew in a different direction, while
> the branch didn't do anything. When you rebase it to its upstream, you
> should expect that no commits are rebased.

FWIW in the cases where I've actually hit this, I had additional
commits on top of the new branch, so only the commits at the "bottom"
were dropped (which is much harder to notice).  I think the difference
is whether you consider the dropped commits part of the original
branch (now discarded), or the new branch --- my mental model was the
latter, in part because I didn't realize that branch history was
actually semantically important.

>
> -- Hannes



[Index of Archives]     [Linux Kernel Development]     [Gcc Help]     [IETF Annouce]     [DCCP]     [Netdev]     [Networking]     [Security]     [V4L]     [Bugtraq]     [Yosemite]     [MIPS Linux]     [ARM Linux]     [Linux Security]     [Linux RAID]     [Linux SCSI]     [Fedora Users]

  Powered by Linux