From: Junio C Hamano <gitster@pobox.com>
To: Jonathan del Strother <maillist@steelskies.com>
Cc: git@vger.kernel.org
Subject: Re: [PATCH] Offer to print changes while running git-mergetool
Date: Fri, 06 Feb 2009 16:21:56 -0800 [thread overview]
Message-ID: <7veiyb14gr.fsf@gitster.siamese.dyndns.org> (raw)
In-Reply-To: <57518fd10902061108k6a7691c5r13b2782baf3bfde3@mail.gmail.com> (Jonathan del Strother's message of "Fri, 6 Feb 2009 19:08:47 +0000")
Jonathan del Strother <maillist@steelskies.com> writes:
> On Fri, Feb 6, 2009 at 5:47 PM, Junio C Hamano <gitster@pobox.com> wrote:
>> Jonathan del Strother <maillist@steelskies.com> writes:
>>
>>> On Fri, Feb 6, 2009 at 2:32 PM, Jonathan del Strother
>>> <jon.delStrother@bestbefore.tv> wrote:
>>>> Add a "Show changes" option to each prompt in mergetool. This prints the conflicted changes on the current file, using 'git log -p --merge <file>'
>>>
>>> Just discovered that this doesn't work so well when resolving merges
>>> resulting from "git stash apply" - it produces "fatal: --merge without
>>> MERGE_HEAD". Should git-stash be setting MERGE_HEAD in this case,
>>
>> No no no, please absolutely don't. MERGE_HEAD is an instruction to the
>> eventual commit to create a merge commit and use the commits recorded
>> there as other parents when it does so. You do *NOT* want to end up with
>> a merge with random state after unstashing. None among cherry-pick,
>> rebase, checkout -m (branch switching), nor am -3 should.
>>
>> I'd suggest making the new action conditionally available, by using the
>> presense of MERGE_HEAD as a cue.
>>
>> The thing is, these commands that can potentially end in conflict operate
>> only at the tree level, and not at the level of commit ancestry graph.
>> "log --merge" is all about following the commit ancestry graph, and for
>> conflicts left by these commands it is not a useful way to review.
>
> Maybe I'm misunderstanding the issue,...
There actually are two independent issues. Sorry for being unclear.
My objection is about the issue (1) below. I am pointing out (2) merely
as an issue you need to address if you were to invent a solution that does
not have the issue (1); it is not an objection.
(1) Writing MERGE_HEAD from "git stash apply" (or "git stash pop") is
absolutely a wrong thing to do.
Typically you use stash this way.
... work work work, oops, got interrupted.
$ git stash save
... do some unrelated work.
... perhaps switch branches, perhaps not, it does not matter.
... the important thing is you conclude this and result in
... a clean work tree state.
$ git stash pop
... inspect and resolve issues, perhaps using git log or
... git diff or git mergetool
... continue to work refining what you were doing
$ git commit
Imagine what the last "git commit" does, IF your "stash pop" wrote
any commit object name in MERGE_HEAD. It will record the tree created
from the index in a commit that is a merge between the current HEAD
and whatever commit you wrote in MERGE_HEAD. But you obviously did
not want any merge --- you wanted a straight and honest single parent
commit.
It is Ok if your design is to come up with a marker that is different
from MERGE_HEAD for "git stash pop" to tell "git log --merge" where
the potential conflicts may be coming from, but using MERGE_HEAD as
that marker is unacceptable for this reason.
(2) The set of commits on the left and right side of "git log --merge"
is tied very closely to the topology over the three commits:
$(git merge-base HEAD MERGE_HEAD), which is the base,
HEAD, which is the right-hand side, and
MERGE_HEAD, which is the left-hand side.
During a normal merge resolution, "git log --merge $paths..." looks at
the topology of this graph:
x---x---MERGE_HEAD--??? merge result?
/ /
MB---o---o---o---HEAD
and it simply runs
git log MERGE_HEAD...HEAD -- $paths
Hence, x would appear on the left hand side and o on the right hand
side in the resulting traversal (this makes difference especially if
you did "git log --left-right --merge").
But "stash pop" and "cherry-pick" (they are the same thing) work on a
quite different topology.
If you stashed while on 'master' (whose tip is A), "git stash save" will
create commit S without moving the tip of 'master' (S is saved at the tip
of refs/stash):
x---x---B topic
/
o---o---o---A master
.
...S
When you unstash it on B, "git stash pop" does a three way merge between A
and S and B, but it does *not* use the usual ancestry topology.
It merges S on top of B as if A is the common ancestor (i.e. merge base).
If you actually commited S on the master and cherry-picked it on B,
exactly the same thing happens.
Because of the way the cherry-picking 3-way merge has to happen, the
"virtual ancestory chain" becomes like this:
*---*---*---x---x---B
/ \
A master \
. \
...S..................??? merge result?
where * are "anti-commits" that reverse the effects of the commits o in
the original history that lead to A [*2*].
First of all, "git log A...B -- $path" won't show the virtual topology
depicted above. It will only show commits x and would ignore o, hence it
does not explain the conflicts at all [*1*]. You somehow need to devise a
way to pretend that the topology is like the above one to achieve an
output equivalent to "git log --merge" for a true merge situation.
It is not just the matter of "echo A >.git/MERGE_HEAD" (which is an
absolute no-no for totally unrelated reason, which is (1) above) to tell
"git log --merge" to pretend the topology is like the above when it does
its traversal.
"git log -p A...B -- $path" won't show them without such a change either,
but even if you somehow convinced the revision traversal to pretend the
three commits o before A should be shown, you also would need to teach it
to show them in reverse because they are now anti-commits.
As I said, the longer explanation of issue (2) in this message is not
meant as an objection. I am explaining why "git log --merge [-p]" cannot
be used as-is for what you are trying to do.
My primary objection is "don't mess with MERGE_HEAD", which is the issue
(1) above.
[Footnote]
*1* you handwaved this issue away in your message saying that what you did
is not interesting, but I think that is a cop-out. Why are we showing our
side when inspecting the history in a true merge situation if it is really
uninteresting?
*2* In addition, there is this little problem that you can cherry-pick
(and unstash) between disjoint histories, in which case there won't even
be such a virtual ancestry chain that I obtained by rotating the history a
bit in the above example.
next prev parent reply other threads:[~2009-02-07 0:23 UTC|newest]
Thread overview: 10+ messages / expand[flat|nested] mbox.gz Atom feed top
2009-02-06 14:32 [PATCH] Offer to print changes while running git-mergetool Jonathan del Strother
2009-02-06 14:41 ` Jonathan del Strother
2009-02-06 17:47 ` Junio C Hamano
2009-02-06 19:08 ` Jonathan del Strother
2009-02-07 0:21 ` Junio C Hamano [this message]
2009-02-07 8:11 ` Junio C Hamano
2009-02-07 12:01 ` Jonathan del Strother
2009-02-08 1:24 ` Charles Bailey
2009-02-08 11:43 ` Jonathan del Strother
2009-02-08 12:38 ` Charles Bailey
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=7veiyb14gr.fsf@gitster.siamese.dyndns.org \
--to=gitster@pobox.com \
--cc=git@vger.kernel.org \
--cc=maillist@steelskies.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).