Message ID | c6eb286b60808bc9e69ce9b09fe4389db15db492.1660892256.git.gitgitgadget@gmail.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Series | Show original filenames in merge tree | expand |
Hi all (and I guess in particular Elijah), On Fri, 19 Aug 2022, Johannes Schindelin via GitGitGadget wrote: > diff --git a/t/t4069-remerge-diff.sh b/t/t4069-remerge-diff.sh > index 35f94957fce..bc580a242ac 100755 > --- a/t/t4069-remerge-diff.sh > +++ b/t/t4069-remerge-diff.sh > @@ -131,11 +131,12 @@ test_expect_success 'setup non-content conflicts' ' > test_expect_success 'remerge-diff with non-content conflicts' ' > git log -1 --oneline resolution >tmp && > cat <<-EOF >>tmp && > + diff --git a/file_or_directory b/file_or_directory > + remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > diff --git a/file_or_directory~HASH (side1) b/wanted_content > similarity index 100% > rename from file_or_directory~HASH (side1) > rename to wanted_content > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > diff --git a/letters b/letters > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > diff --git a/letters_side2 b/letters_side2 I would have liked to avoid touching this file altogether, of course, but when I investigated, I came to the conclusion that while this patch still does not produce the best possible outcome for remerge diffs, it does improve upon the current situation. The thing is, when mentioning that we had to rename `file_or_directory` to `file_or_directory~HASH (side1)` to be able to write the file because a directory of the same name already was in the way, I would actually have expected this to come under the diff header diff --git a/file_or_directory b/file_or_directory~HASH (side1) Previously, it was under the header diff --git a/file_or_directory~HASH (side1) b/wanted_content and with this patch it is under diff --git a/file_or_directory b/file_or_directory which is still not ideal: It mentions only the original file name, not the munged one. When I looked into the implementation details I figured out that the information is not quite at our fingertips. In [`create_filepairs_for_header_only_notifications()`][create_filepairs], we have access to the primary filename (in the above example, `file_or_directory`) as the key of the entry in the strmap `o->additional_path_headers`, which is [populated from `path_messages`][additional-headers], which in turn [points to the `conflicts` strmap][path_messages]. Note: this code has changed between v2.37.2 and the current main branch, in v2.37.2 the information is still copied from `conflicts` to `path_messages`. So basically, when we generate that diff header, we know the primary path (great!) and we have access to a strmap that points to a string list of conflict messages. So from where do we get that munged path name? The string list of conflict messages [has `util` pointers to `struct logical_conflict_info` data][conflicts-strmap], which does contain the list of involved paths. This should be enough, right? But: - `struct logical_conflict_info` is declared locally in `merge-ort.c`, we do not have access to that information in `diff.c` (nor would we really _want_ to let that `merge-ort` implementation detail spill over into `diff.c`), and even if we _could_ declare it in `merge-ort.h`, there is still this problem: - The path list in `struct logical_conflict_info` is arbitrary-sized, i.e. we could even have _three_ paths in there, and that does not fit into a standard diff header. - Additionally, we generate _one_ diff header for _all_ of the conflict messages listed for a particular primary path in `additional_path_headers`. But each of these conflict messages relates to its very own list of non-primary paths. For example, one conflict message could talk about a file/directory conflict, another one could talk about conflicting renames to two different filenames (in a recursive merge, both is possible in the same merge). It is simply impossible to combine all of those conflict messages under a single diff header that matches all of the involved paths. - Even aside from that, the remerge diff code seems to have a usability wart (or even bug) where restricting the diff via a pathspec that matches a secondary path but not the primary path of a conflict (think: renames) would _not_ show the conflict message to the user, it would be shown only if the pathspec also includes the primary path. In general, I found it hard to think of a _really_ ideal design where to present that remerge conflict information. After all, in a remerge diff, we do not even write out any files, therefore that conflict message suggesting that we renamed the file to a munged name is somewhat misleading. Sure, we should mention the conflict between the file and the directory, but since nothing was written to disk, there is not actually any _need_ to mention a munged filename at all. And that goes even for the part of the remerge diff shown in the patch quoted above, where it talks about `file_or_directory~HASH (side1)` being renamed... That file never had that filename... So I am not quite sure where we want the remerge design to go from here. In any case, this remerge aspect is safely outside the scope of this here patch series, which means that this here patch series should not be concerned with it. In this email, I just wanted to mention _why_ I did not include any patch to address the remerge issues any further. Ciao, Dscho [create_filepairs]: https://github.com/git/git/blob/v2.37.2/diff.c#L6337-L6383 [additional-headers]: https://github.com/git/git/blob/v2.37.2/log-tree.c#L992 [path_messages]: https://github.com/git/git/blob/795ea8776be/merge-ort.c#L4846 [conflicts-strmap]: https://github.com/git/git/blob/795ea8776be/merge-ort.c#L351-L360
Hi Dscho, On Thu, Aug 18, 2022 at 11:57 PM Johannes Schindelin via GitGitGadget <gitgitgadget@gmail.com> wrote: > > From: Johannes Schindelin <johannes.schindelin@gmx.de> > > When printing the messages in a machine-readable format (i.e. in `-z` > mode), the intention is to support third-party software that wants to > present the conflicts in a "pretty" way to human readers. > > To that end, `git merge-tree -z` prefixes the conflict messages by a > variable-size list of paths and a conflict identifier. This list and > this identifier are intended to be machine-parseable, the conflict > message is sort of free-form and not intended to be parsed. > > Keeping in mind that the intended use case is to have third-party > software use `git merge-tree` to perform worktree-less merges and then > present the conflicts (if any) to human readers, it makes much more > sense to show the original file names of the involved files instead of > the ones we internally munged to allow for creating a tree object that > contains entries for both sides of the conflict (which requires them to > have different names). I can see how this would be useful to you if it could be achieved...well, assuming you replaced "instead of" with "in addition to". It'd be a huge undertaking to achieve, though; far beyond what this series attempts. I think your attempt here may be based on the presumption that any hash in a higher stage in the index maps directly to the hash of some <revision>:<filename> pair from one of the commits being merged. That assumption holds in simple cases, but does not hold generally. merge-ort has a very strong "(at most) 3 stages" assumption for any conflict, driven by the need to ultimately represent conflicts in the index, which is pervasive in the code. Trying to create such a direct mapping between higher stage conflicts and <revision>:<filename> pairs from the commits being merged would require breaking that assumption. That would mean getting rid of many of the hardcoded array sizes of 3, as well as modifying who knows how many portions of logic built around this assumption. Your attempt also seems to be based on the presumption that the original value of "path" as passed to process_entry(), represents a path from one of the two sides of history. In simple cases, it does. But the assumption does not generally hold. Further... > So let's mention the original file names prominently, as first item in > that variable-size list of paths. This is in conflict with the needs of other uses of merge-ort: * When the user is running an active merge with conflicts written to the index and working tree, the filenames where the conflicts are recorded may not match the original filenames, and the filenames where things are recorded is more useful and thus primary. * For remerge-diff, the filename where the conflict would be placed is important because that's where stuff ends up. Any changes the user made for the final merge is relative to that munged path, so that munged path is the important bit, not the original path. Adding additional paths to path_msg() to cover other paths relevant to the conflict make sense, but the primary path really needs to be left as-is for these other cases. > Note: For the modify/delete conflict type, we used to mention _only_ the > munged name in that path list. To allow for tools to read the tree > object produced by `git merge-tree -z` into a Git index in order to > resolve the conflicts, it is necessary to list not only the original > name but _also_ the munged name so that the item with the munged file > name can be removed from that Git index. Therefore, this patch teaches > `git merge-tree` to show both the original _and_ the munged name in that > list. This makes sense. Passing additional valid and relevant information to callers of merge-tree seems beneficial. > Also note: This patch changes the output of the remerge diff slightly: > whereas before, we printed the notice about a file/directory conflict > under the diff header mentioning the munged file name, we now print it > under a separate diff header that mentions the original file name. > That is the explanation why t4301 is touched by this patch. Sounds problematic. > Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de> > --- > merge-ort.c | 14 +++++++------- > t/t4069-remerge-diff.sh | 8 ++++---- > t/t4301-merge-tree-write-tree.sh | 4 ++-- > 3 files changed, 13 insertions(+), 13 deletions(-) > > diff --git a/merge-ort.c b/merge-ort.c > index 01f150ef3b5..211f6823e1d 100644 > --- a/merge-ort.c > +++ b/merge-ort.c > @@ -3741,6 +3741,7 @@ static void process_entry(struct merge_options *opt, > struct conflict_info *ci, > struct directory_versions *dir_metadata) > { > + const char *orig_path = path; > int df_file_index = 0; > > VERIFY_CI(ci); > @@ -3787,7 +3788,6 @@ static void process_entry(struct merge_options *opt, > */ > struct conflict_info *new_ci; > const char *branch; > - const char *old_path = path; > int i; > > assert(ci->merged.result.mode == S_IFDIR); This looks like you are trying to make sure you always have the "original" path available, meaning a path corresponding to a file from one of the commits being merged. This doesn't quite accomplish that, though. Some questions: * What about cases where "path" is a location that exists only in the merge base? (rename/rename(1to2) is an example where this will come up) * What about cases where "path" is a location that does not exist anywhere in history? (directory renames yield such a result. e.g. side A renames olddir/ -> newdir/, and side B adds olddir/foo, then "path" will be newdir/foo) > @@ -3838,10 +3838,10 @@ static void process_entry(struct merge_options *opt, > strmap_put(&opt->priv->paths, path, new_ci); > > path_msg(opt, CONFLICT_FILE_DIRECTORY, 0, > - path, old_path, NULL, NULL, > + orig_path, path, NULL, NULL, > _("CONFLICT (file/directory): directory in the way " > "of %s from %s; moving it to %s instead."), > - old_path, branch, path); > + orig_path, branch, path); > /* > * Zero out the filemask for the old ci. At this point, ci > @@ -3921,7 +3921,7 @@ static void process_entry(struct merge_options *opt, > > if (rename_a && rename_b) { > path_msg(opt, CONFLICT_DISTINCT_MODES, 0, > - path, a_path, b_path, NULL, > + orig_path, a_path, b_path, NULL, > _("CONFLICT (distinct types): %s had " > "different types on each side; " > "renamed both of them so each can " > @@ -3929,7 +3929,7 @@ static void process_entry(struct merge_options *opt, > path); > } else { > path_msg(opt, CONFLICT_DISTINCT_MODES, 0, > - path, rename_a ? a_path : b_path, > + orig_path, rename_a ? a_path : b_path, > NULL, NULL, > _("CONFLICT (distinct types): %s had " > "different types on each side; " I think for both of these last two cases, path == orig_path, so you haven't actually made an effective change here. (And, as above, I want path to always be the primary_path passed to path_msg() as other code was written with that assumption in mind.) > @@ -4022,7 +4022,7 @@ static void process_entry(struct merge_options *opt, > if (S_ISGITLINK(merged_file.mode)) > reason = _("submodule"); > path_msg(opt, CONFLICT_CONTENTS, 0, > - path, NULL, NULL, NULL, > + orig_path, NULL, NULL, NULL, > _("CONFLICT (%s): Merge conflict in %s"), > reason, path); > } Here's another case where path == orig_path, so you haven't made an effective change. But this one highlights something interesting... In addition to the fact that path/orig_path may be a location that didn't exist on either side of history, there might actually be two relevant paths from the two different sides of history which are involved in the content merge, with neither of them being path or orig_path. Let me break it down, starting with a simpler two path case: If we have a standard rename, e.g. foo -> bar, and both sides modified the file but did so in a conflicting manner, then we will end up in this chunk of code. The conflict info emitted by merge-tree -z which is always of the form <number-of-paths>NUL<path1>NUL<path2>NUL<pathN>NUL<stable-short-type-description>NUL<message>NUL will in this particular case be 1<NUL>bar<NUL>Auto-merging<NUL>Auto-merging bar<newline><NUL> 1<NUL>bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): Merge conflict in bar<newline><NUL> Neither of these messages has any mention of "foo", despite the fact that "foo" was the name of the file in both the merge base and one side, and "bar" was only the name of the file on one side. Now, let's make it more interesting. side A modifies foo, and renames olddir/->newdir/. side B modifies foo in a conflicting manner, and renames foo->olddir/bar. Our `merge-tree -z` conflict information emitted will be of the form 1<NUL>newdir/bar<NUL>Auto-merging<NUL>Auto-merging newdir/bar<newline><NUL> 1<NUL>newdir/bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): Merge conflict in newdir/bar<newline><NUL> For this more interesting case, the only files that existed were "foo" and "olddir/bar", neither of which are mentioned in the conflict info. The conflict info only reports on "newdir/bar". And for both of these examples, your patch doesn't change the existing behavior at all since path == orig_path for this hunk of the patch. > @@ -4067,7 +4067,7 @@ static void process_entry(struct merge_options *opt, > */ > } else { > path_msg(opt, CONFLICT_MODIFY_DELETE, 0, > - path, NULL, NULL, NULL, > + orig_path, path, NULL, NULL, > _("CONFLICT (modify/delete): %s deleted in %s " > "and modified in %s. Version %s of %s left " > "in tree."), Adding orig_path here after path would make sense, though it's still incomplete. For example if a path was modified and renamed on one side, and deleted on the other, then we'd get to this hunk, and the modify/delete would omit information about the other path. Further, directory renames could be at play, such that neither orig_path nor path would correspond to a filename in either of the commits being merged. There are also other path_msg() calls that should probably be informed of additional filenames due to simple renames: CONFLICT_CONTENTS (as noted above), CONFLICT_BINARY, and INFO_AUTO_MERGING. Going beyond simple renames, if you want "fully original" filenames, then we'd probably have to audit nearly all the path_msg() calls for when directory renames are possible, and modify the directory renaming code to plumb the original name through to this layer. And even if we do all that, then we still aren't matching what I think you are expecting where every conflicted higher order entry we report maps to a single <revision>:<filename> pairing. We may have a higher stage conflict entry that represents merged contents of three different filecontents from three different filenames. For example, if someone has the following setup: Commit O: one file named "original" Commit A: modify original + rename to primordial Commit B: add file named primordial, modify original differently + rename it to primary where A & B both have O as a parent or ancestor, and O is the unique merge base of A & B. Now if we try to merge A & B, then the index will have four higher order entries (using <mode> <hash> <stage-number> <filename>, as we'd see from either `ls-files -u` style output or from the first part of merge-tree -z output): 100644 <hash of O:original> 1 original 100644 <hash of auto-merge of O:original, A:primordial, & B:primary> 3 primary 100644 <hash of auto-merge of O:original, A:primordial, & B:primary> 2 primordial 100644 <hash of B:primordial> 3 primordial Note that the two middle entries, despite having one name attached to them, actually come from a merge of three different files. Further, that hash for those two entries will not have yet appeared as the contents for any blob in any revision of history, and will in fact represent a file with conflict markers (even if none of the original files -- O:original, A:primodial, B:primary -- had any conflicts). And, the version of "primodial" in the merge-result-tree would likely have nested conflict markers, despite O being a unique merge base (meaning we get nested conflict markers without merging merge bases). Trying to have all the conflict output from merge-tree map to some original file the user had from one of their sides of history is thus a goal that merge-ort is nowhere close to satisfying. I'm not even sure it's tractable. However, there is a much smaller scoped goal that we could easily achieve: providing additional filename information in the path_msg() calls, much like you start to do with the modify/delete case. > diff --git a/t/t4069-remerge-diff.sh b/t/t4069-remerge-diff.sh > index 35f94957fce..bc580a242ac 100755 > --- a/t/t4069-remerge-diff.sh > +++ b/t/t4069-remerge-diff.sh > @@ -131,11 +131,12 @@ test_expect_success 'setup non-content conflicts' ' > test_expect_success 'remerge-diff with non-content conflicts' ' > git log -1 --oneline resolution >tmp && > cat <<-EOF >>tmp && > + diff --git a/file_or_directory b/file_or_directory > + remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > diff --git a/file_or_directory~HASH (side1) b/wanted_content > similarity index 100% > rename from file_or_directory~HASH (side1) > rename to wanted_content > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > diff --git a/letters b/letters > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > diff --git a/letters_side2 b/letters_side2 > @@ -168,7 +169,7 @@ test_expect_success 'remerge-diff with non-content conflicts' ' > test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no diff content' ' > git log -1 --oneline resolution >tmp && > cat <<-EOF >>tmp && > - diff --git a/file_or_directory~HASH (side1) b/file_or_directory~HASH (side1) > + diff --git a/file_or_directory b/file_or_directory > remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > diff --git a/letters b/letters > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > @@ -184,14 +185,13 @@ test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no dif > test_cmp expect actual > ' > > -test_expect_success 'remerge-diff w/ diff-filter=R: relevant file + conflict header' ' > +test_expect_success 'remerge-diff w/ diff-filter=R: relevant file' ' > git log -1 --oneline resolution >tmp && > cat <<-EOF >>tmp && > diff --git a/file_or_directory~HASH (side1) b/wanted_content > similarity index 100% > rename from file_or_directory~HASH (side1) > rename to wanted_content > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > EOF > # We still have some sha1 hashes above; rip them out so test works > # with sha256 All of these look like regressions to me; you discussed these in a separate email so I'll go into detail in response to that one.
On Fri, Aug 19, 2022 at 2:17 AM Johannes Schindelin <Johannes.Schindelin@gmx.de> wrote: > > Hi all (and I guess in particular Elijah), > > On Fri, 19 Aug 2022, Johannes Schindelin via GitGitGadget wrote: > > > diff --git a/t/t4069-remerge-diff.sh b/t/t4069-remerge-diff.sh > > index 35f94957fce..bc580a242ac 100755 > > --- a/t/t4069-remerge-diff.sh > > +++ b/t/t4069-remerge-diff.sh > > @@ -131,11 +131,12 @@ test_expect_success 'setup non-content conflicts' ' > > test_expect_success 'remerge-diff with non-content conflicts' ' > > git log -1 --oneline resolution >tmp && > > cat <<-EOF >>tmp && > > + diff --git a/file_or_directory b/file_or_directory > > + remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > diff --git a/file_or_directory~HASH (side1) b/wanted_content > > similarity index 100% > > rename from file_or_directory~HASH (side1) > > rename to wanted_content > > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > diff --git a/letters b/letters > > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > > diff --git a/letters_side2 b/letters_side2 > > I would have liked to avoid touching this file altogether, of course, but > when I investigated, I came to the conclusion that while this patch still > does not produce the best possible outcome for remerge diffs, it does > improve upon the current situation. No, this is a strict regression. Those file headers corresponding to the file/directory conflict (similarity index, rename from/to, and CONFLICT) are all supposed to be grouped together in the same diff header. As to the names... > The thing is, when mentioning that we had to rename `file_or_directory` to > `file_or_directory~HASH (side1)` to be able to write the file because a > directory of the same name already was in the way, I would actually have > expected this to come under the diff header > > diff --git a/file_or_directory b/file_or_directory~HASH (side1) > > Previously, it was under the header > > diff --git a/file_or_directory~HASH (side1) b/wanted_content > > and with this patch it is under > > diff --git a/file_or_directory b/file_or_directory > > which is still not ideal: It mentions only the original file name, not the > munged one. The point of remerge-diff is to show how a user modifies from the auto-merged state to the final merge commit. As such, the filenames from before the merge starts are irrelevant; the only two relevant things are the filename at the end of the automatic merge, and the filename recorded in the commit. In particular, for the case here, upon "git merge $BRANCH" the merge would have stopped with a file named "file_or_directory~HASH\ (side1)" present in the working copy[1]. The user resolved it via renaming that file to "wanted_content". As such, the first and third options you list are incorrect, and the second one -- from the behavior previous to your patch -- is correct. [1] Technically, "git merge $BRANCH" would have used "HEAD" rather than "HASH (side1)", but that's only because at the time of invocation of git merge, HEAD would have been a very specific commit. The same isn't true of HEAD when the user runs `git log --remerge-diff` at some very different time, so we opt to replace "HEAD" with something more meaningful for readers of the diff when running --remerge-diff.
On Sat, Aug 20, 2022 at 4:17 PM Elijah Newren <newren@gmail.com> wrote: > On Thu, Aug 18, 2022 at 11:57 PM Johannes Schindelin via GitGitGadget > <gitgitgadget@gmail.com> wrote: [...] > > @@ -4022,7 +4022,7 @@ static void process_entry(struct merge_options *opt, > > if (S_ISGITLINK(merged_file.mode)) > > reason = _("submodule"); > > path_msg(opt, CONFLICT_CONTENTS, 0, > > - path, NULL, NULL, NULL, > > + orig_path, NULL, NULL, NULL, > > _("CONFLICT (%s): Merge conflict in %s"), > > reason, path); > > } > > Here's another case where path == orig_path, so you haven't made an > effective change. But this one highlights something interesting... > > In addition to the fact that path/orig_path may be a location that > didn't exist on either side of history, there might actually be two > relevant paths from the two different sides of history which are > involved in the content merge, with neither of them being path or > orig_path. Let me break it down, starting with a simpler two path > case: > > If we have a standard rename, e.g. foo -> bar, and both sides modified > the file but did so in a conflicting manner, then we will end up in > this chunk of code. The conflict info emitted by merge-tree -z which > is always of the form > <number-of-paths>NUL<path1>NUL<path2>NUL<pathN>NUL<stable-short-type-description>NUL<message>NUL > will in this particular case be > 1<NUL>bar<NUL>Auto-merging<NUL>Auto-merging bar<newline><NUL> > 1<NUL>bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): Merge > conflict in bar<newline><NUL> > Neither of these messages has any mention of "foo", despite the fact > that "foo" was the name of the file in both the merge base and one > side, and "bar" was only the name of the file on one side. > > Now, let's make it more interesting. side A modifies foo, and renames > olddir/->newdir/. side B modifies foo in a conflicting manner, and > renames foo->olddir/bar. Our `merge-tree -z` conflict information > emitted will be of the form > 1<NUL>newdir/bar<NUL>Auto-merging<NUL>Auto-merging newdir/bar<newline><NUL> > 1<NUL>newdir/bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): > Merge conflict in newdir/bar<newline><NUL> > For this more interesting case, the only files that existed were "foo" > and "olddir/bar", neither of which are mentioned in the conflict info. > The conflict info only reports on "newdir/bar". > > And for both of these examples, your patch doesn't change the existing > behavior at all since path == orig_path for this hunk of the patch. But, actually, trying to create some more testcases for the testsuite, maybe this isn't so bad. For example, with this interesting testcase: Commit O: foo, olddir/{a,b,c} Commit A: delete foo, rename olddir/ -> newdir/, add newdir/bar/file Commit B: modify foo & rename foo -> olddir/bar Which involves four different types of conflicts: * a directory rename (which further modifies foo->olddir/bar to end up at newdir/bar) * a rename/delete (delete foo vs. rename to olddir/bar) * a modify/delete (since foo was modified as well on one side) * a directory/file conflict (newdir/bar vs newdir/bar/file, forcing newdir/bar to be further renamed to newdir/bar~B^0) The <Conflicted file info> section will look like 100644 <oldhash> 1 newdir/bar~B^0 100644 <newhash> 3 newdir/bar~B^0 While that only includes the name "newdir/bar~B^0" and not "newdir/bar", or "olddir/bar", or "foo", the <Informational messages> has all necessary information to tie it together. Replacing null characters with newlines, the <Informational messages> section is: 2 newdir/bar olddir/bar CONFLICT (directory rename suggested) CONFLICT (file location): foo renamed to olddir/bar in B^0, inside a directory that was renamed in A^0, suggesting it should perhaps be moved to newdir/bar. 2 newdir/bar foo CONFLICT (rename/delete) CONFLICT (rename/delete): foo renamed to newdir/bar in B^0, but deleted in A^0. 2 newdir/bar~B^0 newdir/bar CONFLICT (file/directory) CONFLICT (file/directory): directory in the way of newdir/bar from B^0; moving it to newdir/bar~B^0 instead. 1 newdir/bar~B^0 CONFLICT (modify/delete) CONFLICT (modify/delete): newdir/bar~B^0 deleted in A^0 and modified in B^0. Version B^0 of newdir/bar~B^0 left in tree. This provides all the mappings to tie together foo, olddir/bar, newdir/bar, and newdir/bar~B^0, and shows the four conflict types present. So, all the information you need is present, it just can't be parsed out of a single line like you want. (But adding the "newdir/bar" name to the modify/delete conflict at least does seem like it'd be a little nicer.) And trying to parse it out of a single line is probably tantamount to trying to enforce a 1-to-1 mapping between paths and conflicts, which will paint us into a corner, as I've pointed out before[1,2] [1] https://lore.kernel.org/git/CABPp-BGCL0onSmpgKuO1k2spYCkx=v27ed9TSSxFib=OdDcLbw@mail.gmail.com/ [2] https://lore.kernel.org/git/CABPp-BGnqXdFBNAyKRXgvCHv+aUZTMg-CgcQf95dKAR-e1zSjQ@mail.gmail.com/
Hi Elijah, On Sat, 20 Aug 2022, Elijah Newren wrote: > On Thu, Aug 18, 2022 at 11:57 PM Johannes Schindelin via GitGitGadget > <gitgitgadget@gmail.com> wrote: > > > > From: Johannes Schindelin <johannes.schindelin@gmx.de> > > > > When printing the messages in a machine-readable format (i.e. in `-z` > > mode), the intention is to support third-party software that wants to > > present the conflicts in a "pretty" way to human readers. > > > > To that end, `git merge-tree -z` prefixes the conflict messages by a > > variable-size list of paths and a conflict identifier. This list and > > this identifier are intended to be machine-parseable, the conflict > > message is sort of free-form and not intended to be parsed. > > > > Keeping in mind that the intended use case is to have third-party > > software use `git merge-tree` to perform worktree-less merges and then > > present the conflicts (if any) to human readers, it makes much more > > sense to show the original file names of the involved files instead of > > the ones we internally munged to allow for creating a tree object that > > contains entries for both sides of the conflict (which requires them to > > have different names). > > I can see how this would be useful to you if it could be > achieved...well, assuming you replaced "instead of" with "in addition > to". It'd be a huge undertaking to achieve, though; far beyond what > this series attempts. I feared you would say that :-) And I realize that I should have lead with the context that is relevant to understand what I am driving at. For now, all I am trying to do is to verify that we can achieve identical results between a new code path that uses `git merge-tree` (with options) and an existing code path that uses libgit2 to perform a 3-way merge. To that end, I have added some patches to optionally disable rename detection, to allow specifying a merge base instead of performing a full recursive merge, to optionally detect whether the `MERGE_HEAD` is already reachable from `HEAD` and in that case exit with a specific exit status, among other things. And yes, I realize that I "only" benefit from the worktree-less nature of `merge-ort` so far, and not from any other aspects `merge-ort` offers. One thing I need from the result is a list of conflicts in the same format as we would have received if writing out an index with conflicts and then parsing just the entries with non-zero stages. So for now, I do not care _so_ much about rename conflicts, or conflicts resulting from merging merge base candidates. I don't want to slam the door shut, of course, but due to a teeny tiny bit of time pressure, I want to focus very, very much on the libgit2-like operation for now :-) What I do need, however, is the unmunged filenames as well as the corresponding file modes and the object hashes. > I think your attempt here may be based on the presumption that any > hash in a higher stage in the index maps directly to the hash of some > <revision>:<filename> pair from one of the commits being merged. That > assumption holds in simple cases, but does not hold generally. > merge-ort has a very strong "(at most) 3 stages" assumption for any > conflict, driven by the need to ultimately represent conflicts in the > index, which is pervasive in the code. This is actually very helpful information, this had not sunk in yet. So when encountering a "stage 3 before stages 1 & 2" situation arises, like in the test case I added, you're saying that I could simply reorder those and be done and not have to expect that the `stage 1 & 2` part is followed by _another_ stage 3 part for the same name? Except, of course, that I need the original file names, and the invariant of at most 3 stages holds true only when taking munged file names into account, right? > Trying to create such a direct mapping between higher stage conflicts > and <revision>:<filename> pairs from the commits being merged would > require breaking that assumption. That would mean getting rid of many of > the hardcoded array sizes of 3, as well as modifying who knows how many > portions of logic built around this assumption. > > Your attempt also seems to be based on the presumption that the > original value of "path" as passed to process_entry(), represents a > path from one of the two sides of history. In simple cases, it does. > But the assumption does not generally hold. Heh, in my case it always holds because I disable rename detection. The only time when multiple file names are involved is in case of file/directory conflicts and the like. In which case I still need the original filename. But I would contend that we need to know the original file name(s) even in the case of renames, so that we can present the users with all the information pertinent to the conflict, to allow them to resolve it (so that they have the full context). > Further... > > > So let's mention the original file names prominently, as first item in > > that variable-size list of paths. > > This is in conflict with the needs of other uses of merge-ort: > * When the user is running an active merge with conflicts written to > the index and working tree, the filenames where the conflicts are > recorded may not match the original filenames, and the filenames where > things are recorded is more useful and thus primary. > * For remerge-diff, the filename where the conflict would be placed > is important because that's where stuff ends up. Any changes the user > made for the final merge is relative to that munged path, so that > munged path is the important bit, not the original path. Okay, I understand now. So basically, if I need different/additional information, I will _have_ to introduce a new flag to trigger code paths that produce that. And my changes to the remerge test cases merely demonstrate precisely that need. > Adding additional paths to path_msg() to cover other paths relevant to > the conflict make sense, but the primary path really needs to be left > as-is for these other cases. I fear that I will need much, much more than just the additional paths ;-) So far, I am parsing the `ls-files -u`-like part so that I get mode and hash information, but eventually I want to parse that out of the messages part and that only produces lists of paths so far... And a simple, single list of paths is not enough to convey the information that I need. Concretely, I need: - the original file name(s) involved in the conflict, - the modes and object hashes corresponding to the different stages - maybe some provenance information in the future, in case of complex rename/merge recursions In particular when presenting a file/directory conflict, it will be important to know what stage refers to the file and what stage to the directory (so as to know whether the respective object hash refers to a file or even symlink and needs to be read as a blob). > > Note: For the modify/delete conflict type, we used to mention _only_ the > > munged name in that path list. To allow for tools to read the tree > > object produced by `git merge-tree -z` into a Git index in order to > > resolve the conflicts, it is necessary to list not only the original > > name but _also_ the munged name so that the item with the munged file > > name can be removed from that Git index. Therefore, this patch teaches > > `git merge-tree` to show both the original _and_ the munged name in that > > list. > > This makes sense. Passing additional valid and relevant information > to callers of merge-tree seems beneficial. And now that you mention this, I realize that this should be pulled out into its own patch. > > Also note: This patch changes the output of the remerge diff slightly: > > whereas before, we printed the notice about a file/directory conflict > > under the diff header mentioning the munged file name, we now print it > > under a separate diff header that mentions the original file name. > > That is the explanation why t4301 is touched by this patch. > > Sounds problematic. Fair enough. > > Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de> > > --- > > merge-ort.c | 14 +++++++------- > > t/t4069-remerge-diff.sh | 8 ++++---- > > t/t4301-merge-tree-write-tree.sh | 4 ++-- > > 3 files changed, 13 insertions(+), 13 deletions(-) > > > > diff --git a/merge-ort.c b/merge-ort.c > > index 01f150ef3b5..211f6823e1d 100644 > > --- a/merge-ort.c > > +++ b/merge-ort.c > > @@ -3741,6 +3741,7 @@ static void process_entry(struct merge_options *opt, > > struct conflict_info *ci, > > struct directory_versions *dir_metadata) > > { > > + const char *orig_path = path; > > int df_file_index = 0; > > > > VERIFY_CI(ci); > > @@ -3787,7 +3788,6 @@ static void process_entry(struct merge_options *opt, > > */ > > struct conflict_info *new_ci; > > const char *branch; > > - const char *old_path = path; > > int i; > > > > assert(ci->merged.result.mode == S_IFDIR); > > This looks like you are trying to make sure you always have the > "original" path available, meaning a path corresponding to a file from > one of the commits being merged. This doesn't quite accomplish that, > though. Some questions: > > * What about cases where "path" is a location that exists only in > the merge base? (rename/rename(1to2) is an example where this will > come up) That's a good point. I will need to invent some way to convey all three original file names in that case, not just one. > * What about cases where "path" is a location that does not exist > anywhere in history? (directory renames yield such a result. e.g. > side A renames olddir/ -> newdir/, and side B adds olddir/foo, then > "path" will be newdir/foo) Hmm. As mentioned, I want to focus on the scenario with disabled renames, but not shut the door on rename scenarios like this one. So maybe I need to collect the original names in this instance, too, with some (machine-parseable) indicator talking about the nature of that origin. > > @@ -3838,10 +3838,10 @@ static void process_entry(struct merge_options *opt, > > strmap_put(&opt->priv->paths, path, new_ci); > > > > path_msg(opt, CONFLICT_FILE_DIRECTORY, 0, > > - path, old_path, NULL, NULL, > > + orig_path, path, NULL, NULL, > > _("CONFLICT (file/directory): directory in the way " > > "of %s from %s; moving it to %s instead."), > > - old_path, branch, path); > > + orig_path, branch, path); > > /* > > * Zero out the filemask for the old ci. At this point, ci > > @@ -3921,7 +3921,7 @@ static void process_entry(struct merge_options *opt, > > > > if (rename_a && rename_b) { > > path_msg(opt, CONFLICT_DISTINCT_MODES, 0, > > - path, a_path, b_path, NULL, > > + orig_path, a_path, b_path, NULL, > > _("CONFLICT (distinct types): %s had " > > "different types on each side; " > > "renamed both of them so each can " > > @@ -3929,7 +3929,7 @@ static void process_entry(struct merge_options *opt, > > path); > > } else { > > path_msg(opt, CONFLICT_DISTINCT_MODES, 0, > > - path, rename_a ? a_path : b_path, > > + orig_path, rename_a ? a_path : b_path, > > NULL, NULL, > > _("CONFLICT (distinct types): %s had " > > "different types on each side; " > > I think for both of these last two cases, path == orig_path, so you > haven't actually made an effective change here. Not quite. If `ci->df_conflict && ci->merged.result.mode != 0`, then we assign `path = unique_path(opt, path, branch);` and `path` no longer refers to the same string as `orig_path`. It is possible that there is no actual code path that hits that clause and then also hits the `CONFLICT_DISTINC_MODES` calls, but I think there is: if a file has changed to a symlink in one branch _and_ was replaced by a directory in the other branch. > (And, as above, I want path to always be the primary_path passed to > path_msg() as other code was written with that assumption in mind.) > > > @@ -4022,7 +4022,7 @@ static void process_entry(struct merge_options *opt, > > if (S_ISGITLINK(merged_file.mode)) > > reason = _("submodule"); > > path_msg(opt, CONFLICT_CONTENTS, 0, > > - path, NULL, NULL, NULL, > > + orig_path, NULL, NULL, NULL, > > _("CONFLICT (%s): Merge conflict in %s"), > > reason, path); > > } > > Here's another case where path == orig_path, so you haven't made an > effective change. I cannot think of a case where a file/directory conflict can also result in a content conflict, so at least in the regular 3-way merge, I think you're right and this changed line does the same as before in all cases. > But this one highlights something interesting... > > In addition to the fact that path/orig_path may be a location that > didn't exist on either side of history, there might actually be two > relevant paths from the two different sides of history which are > involved in the content merge, with neither of them being path or > orig_path. Let me break it down, starting with a simpler two path > case: > > If we have a standard rename, e.g. foo -> bar, and both sides modified > the file but did so in a conflicting manner, then we will end up in > this chunk of code. The conflict info emitted by merge-tree -z which > is always of the form > <number-of-paths>NUL<path1>NUL<path2>NUL<pathN>NUL<stable-short-type-description>NUL<message>NUL > will in this particular case be > 1<NUL>bar<NUL>Auto-merging<NUL>Auto-merging bar<newline><NUL> > 1<NUL>bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): Merge > conflict in bar<newline><NUL> > Neither of these messages has any mention of "foo", despite the fact > that "foo" was the name of the file in both the merge base and one > side, and "bar" was only the name of the file on one side. I guess so, as no conflict message will mention the rename: it does not conflict. > Now, let's make it more interesting. side A modifies foo, and renames > olddir/->newdir/. side B modifies foo in a conflicting manner, and > renames foo->olddir/bar. Our `merge-tree -z` conflict information > emitted will be of the form > 1<NUL>newdir/bar<NUL>Auto-merging<NUL>Auto-merging newdir/bar<newline><NUL> > 1<NUL>newdir/bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): > Merge conflict in newdir/bar<newline><NUL> > For this more interesting case, the only files that existed were "foo" > and "olddir/bar", neither of which are mentioned in the conflict info. > The conflict info only reports on "newdir/bar". > > And for both of these examples, your patch doesn't change the existing > behavior at all since path == orig_path for this hunk of the patch. That is true, too. Which makes me think that the consuming code I have to work with will need some elaborate changes when it wants to start supporting renames. But that's not the stage where I'm at right now. > > @@ -4067,7 +4067,7 @@ static void process_entry(struct merge_options *opt, > > */ > > } else { > > path_msg(opt, CONFLICT_MODIFY_DELETE, 0, > > - path, NULL, NULL, NULL, > > + orig_path, path, NULL, NULL, > > _("CONFLICT (modify/delete): %s deleted in %s " > > "and modified in %s. Version %s of %s left " > > "in tree."), > > Adding orig_path here after path would make sense, though it's still > incomplete. For example if a path was modified and renamed on one > side, and deleted on the other, then we'd get to this hunk, and the > modify/delete would omit information about the other path. Further, > directory renames could be at play, such that neither orig_path nor > path would correspond to a filename in either of the commits being > merged. > > There are also other path_msg() calls that should probably be informed > of additional filenames due to simple renames: CONFLICT_CONTENTS (as > noted above), CONFLICT_BINARY, and INFO_AUTO_MERGING. > > Going beyond simple renames, if you want "fully original" filenames, > then we'd probably have to audit nearly all the path_msg() calls for > when directory renames are possible, and modify the directory renaming > code to plumb the original name through to this layer. > > And even if we do all that, then we still aren't matching what I think > you are expecting where every conflicted higher order entry we report > maps to a single <revision>:<filename> pairing. We may have a higher > stage conflict entry that represents merged contents of three > different filecontents from three different filenames. For example, > if someone has the following setup: > > Commit O: one file named "original" > Commit A: modify original + rename to primordial > Commit B: add file named primordial, modify original differently + > rename it to primary > > where A & B both have O as a parent or ancestor, and O is the unique > merge base of A & B. Now if we try to merge A & B, then the index > will have four higher order entries (using <mode> <hash> > <stage-number> <filename>, as we'd see from either `ls-files -u` style > output or from the first part of merge-tree -z output): > > 100644 <hash of O:original> 1 original > 100644 <hash of auto-merge of O:original, A:primordial, & > B:primary> 3 primary > 100644 <hash of auto-merge of O:original, A:primordial, & > B:primary> 2 primordial > 100644 <hash of B:primordial> 3 primordial > > Note that the two middle entries, despite having one name attached to > them, actually come from a merge of three different files. Further, > that hash for those two entries will not have yet appeared as the > contents for any blob in any revision of history, and will in fact > represent a file with conflict markers (even if none of the original > files -- O:original, A:primodial, B:primary -- had any conflicts). > And, the version of "primodial" in the merge-result-tree would likely > have nested conflict markers, despite O being a unique merge base > (meaning we get nested conflict markers without merging merge bases). > > Trying to have all the conflict output from merge-tree map to some > original file the user had from one of their sides of history is thus > a goal that merge-ort is nowhere close to satisfying. I'm not even > sure it's tractable. Sure, but I am getting more and more convinced that we will need to provide "sort of" the original name. Somehow. That is, if there were successful (directory) renames happening before the 3-way merge that resulted in a conflict, then I do not really care if the "original original" filename isn't shown. But I do care that the filename before munging is shown (and yes, we should probably also show the munged filename). And since recursive merges can result in multiple levels of munging, it gets harder to explain what I want, but I think you'll get the idea that the "pre-munged" filename of one conflict could be the "post-munged" filename of a conflict that happened at a lower level of that recursion. > However, there is a much smaller scoped goal that we could easily > achieve: providing additional filename information in the path_msg() > calls, much like you start to do with the modify/delete case. Sure, but we also need an indicator somehow what purpose each of those additional filenames serve. I guess we could make it implicit, just like we make it implicit that the first path in a file/directory conflict refers to the file's name and the second to the directory's. But it makes me a bit uneasy and I wish we had more like a structure instead of just a string that we could print out. That would make it possible to convey from where (as in "which branch?") the file or directory came. > > diff --git a/t/t4069-remerge-diff.sh b/t/t4069-remerge-diff.sh > > index 35f94957fce..bc580a242ac 100755 > > --- a/t/t4069-remerge-diff.sh > > +++ b/t/t4069-remerge-diff.sh > > @@ -131,11 +131,12 @@ test_expect_success 'setup non-content conflicts' ' > > test_expect_success 'remerge-diff with non-content conflicts' ' > > git log -1 --oneline resolution >tmp && > > cat <<-EOF >>tmp && > > + diff --git a/file_or_directory b/file_or_directory > > + remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > diff --git a/file_or_directory~HASH (side1) b/wanted_content > > similarity index 100% > > rename from file_or_directory~HASH (side1) > > rename to wanted_content > > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > diff --git a/letters b/letters > > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > > diff --git a/letters_side2 b/letters_side2 > > @@ -168,7 +169,7 @@ test_expect_success 'remerge-diff with non-content conflicts' ' > > test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no diff content' ' > > git log -1 --oneline resolution >tmp && > > cat <<-EOF >>tmp && > > - diff --git a/file_or_directory~HASH (side1) b/file_or_directory~HASH (side1) > > + diff --git a/file_or_directory b/file_or_directory > > remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > diff --git a/letters b/letters > > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > > @@ -184,14 +185,13 @@ test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no dif > > test_cmp expect actual > > ' > > > > -test_expect_success 'remerge-diff w/ diff-filter=R: relevant file + conflict header' ' > > +test_expect_success 'remerge-diff w/ diff-filter=R: relevant file' ' > > git log -1 --oneline resolution >tmp && > > cat <<-EOF >>tmp && > > diff --git a/file_or_directory~HASH (side1) b/wanted_content > > similarity index 100% > > rename from file_or_directory~HASH (side1) > > rename to wanted_content > > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > EOF > > # We still have some sha1 hashes above; rip them out so test works > > # with sha256 > > All of these look like regressions to me; you discussed these in a > separate email so I'll go into detail in response to that one. Thank you for being so diligent. I always appreciate it when I see a well-written, obviously well-edited mail. I find it very considerate. You did a good job at convincing me that my approach is sub-optimal, and that we need at least one new flag to get the information that I need to produce in my use case because the caller requires it to present a nice conflict resolution UI to the users. Back to the drawing table for me, I guess. Thank you! Dscho
On Mon, Aug 22, 2022 at 1:12 PM Johannes Schindelin <Johannes.Schindelin@gmx.de> wrote: > > Hi Elijah, > > On Sat, 20 Aug 2022, Elijah Newren wrote: > > > On Thu, Aug 18, 2022 at 11:57 PM Johannes Schindelin via GitGitGadget > > <gitgitgadget@gmail.com> wrote: > > > > > > From: Johannes Schindelin <johannes.schindelin@gmx.de> > > > > > > When printing the messages in a machine-readable format (i.e. in `-z` > > > mode), the intention is to support third-party software that wants to > > > present the conflicts in a "pretty" way to human readers. > > > > > > To that end, `git merge-tree -z` prefixes the conflict messages by a > > > variable-size list of paths and a conflict identifier. This list and > > > this identifier are intended to be machine-parseable, the conflict > > > message is sort of free-form and not intended to be parsed. > > > > > > Keeping in mind that the intended use case is to have third-party > > > software use `git merge-tree` to perform worktree-less merges and then > > > present the conflicts (if any) to human readers, it makes much more > > > sense to show the original file names of the involved files instead of > > > the ones we internally munged to allow for creating a tree object that > > > contains entries for both sides of the conflict (which requires them to > > > have different names). > > > > I can see how this would be useful to you if it could be > > achieved...well, assuming you replaced "instead of" with "in addition > > to". It'd be a huge undertaking to achieve, though; far beyond what > > this series attempts. > > I feared you would say that :-) > > And I realize that I should have lead with the context that is relevant to > understand what I am driving at. > > For now, all I am trying to do is to verify that we can achieve identical > results between a new code path that uses `git merge-tree` (with options) > and an existing code path that uses libgit2 to perform a 3-way merge. > > To that end, I have added some patches to optionally disable rename > detection, to allow specifying a merge base instead of performing a full > recursive merge, to optionally detect whether the `MERGE_HEAD` is already > reachable from `HEAD` and in that case exit with a specific exit status, > among other things. > > And yes, I realize that I "only" benefit from the worktree-less nature of > `merge-ort` so far, and not from any other aspects `merge-ort` offers. > > One thing I need from the result is a list of conflicts in the same format > as we would have received if writing out an index with conflicts and then > parsing just the entries with non-zero stages. Thanks, this is good context. Especially this last paragraph, where you say you want the non-zero stage entries from what would be written to the index. I'll return to this many times. > So for now, I do not care _so_ much about rename conflicts, or conflicts > resulting from merging merge base candidates. I don't want to slam the > door shut, of course, but due to a teeny tiny bit of time pressure, I want > to focus very, very much on the libgit2-like operation for now :-) > > What I do need, however, is the unmunged filenames as well as the > corresponding file modes and the object hashes. If you're under time pressure...don't you already have all the information you need? Why can't you just parse the <Conflicted file info> and <Informational messages> sections to get it? I'll come back to this again with the testcase you highlighted in your series, at the very end of the email. > > I think your attempt here may be based on the presumption that any > > hash in a higher stage in the index maps directly to the hash of some > > <revision>:<filename> pair from one of the commits being merged. That > > assumption holds in simple cases, but does not hold generally. > > merge-ort has a very strong "(at most) 3 stages" assumption for any > > conflict, driven by the need to ultimately represent conflicts in the > > index, which is pervasive in the code. > > This is actually very helpful information, this had not sunk in yet. > > So when encountering a "stage 3 before stages 1 & 2" situation arises, > like in the test case I added, you're saying that I could simply reorder > those and be done and not have to expect that the `stage 1 & 2` part is > followed by _another_ stage 3 part for the same name? > > Except, of course, that I need the original file names, and the invariant > of at most 3 stages holds true only when taking munged file names into > account, right? You are correct that ort is never going to present two "stage 3" entries for any path it reports. stages are very much an index-related concept, and always associated with potentially munged filenames in ort. This importantly means that there is no such thing as "the unmunged filename" for some index entries. In order to achieve the "at most 3 stages" assumption plastered all over ort, it has to compress information from more than 3 <revision>:<filename> pairs down to at most 3. So some of those non-zero stage index entries do *not* correspond to a file from history. Any request to get "the original filename" would somehow have to respond with up to three filenames instead. (And likewise, any request to get "the original file contents from one of the sides of history" would have to respond with the content of up to three files instead.) I know, I know, you don't care because you're not dealing with renames right now. But if you're modifying the code, it has to be in a way that doesn't paint us into a corner that doesn't work for renames. > > Trying to create such a direct mapping between higher stage conflicts > > and <revision>:<filename> pairs from the commits being merged would > > require breaking that assumption. That would mean getting rid of many of > > the hardcoded array sizes of 3, as well as modifying who knows how many > > portions of logic built around this assumption. > > > > Your attempt also seems to be based on the presumption that the > > original value of "path" as passed to process_entry(), represents a > > path from one of the two sides of history. In simple cases, it does. > > But the assumption does not generally hold. > > Heh, in my case it always holds because I disable rename detection. The > only time when multiple file names are involved is in case of > file/directory conflicts and the like. In which case I still need the > original filename. > > But I would contend that we need to know the original file name(s) even in > the case of renames, so that we can present the users with all the > information pertinent to the conflict, to allow them to resolve it (so > that they have the full context). To reiterate, "the original file name(s)" is an ill-posed problem. It only exists for some entries. (Same for "the original file contents") ...well, unless you rewire ort completely and toss out the "at most 3 stages" assumption. > > Further... > > > > > So let's mention the original file names prominently, as first item in > > > that variable-size list of paths. > > > > This is in conflict with the needs of other uses of merge-ort: > > * When the user is running an active merge with conflicts written to > > the index and working tree, the filenames where the conflicts are > > recorded may not match the original filenames, and the filenames where > > things are recorded is more useful and thus primary. > > * For remerge-diff, the filename where the conflict would be placed > > is important because that's where stuff ends up. Any changes the user > > made for the final merge is relative to that munged path, so that > > munged path is the important bit, not the original path. > > Okay, I understand now. > > So basically, if I need different/additional information, I will _have_ to > introduce a new flag to trigger code paths that produce that. > > And my changes to the remerge test cases merely demonstrate precisely that > need. > > > Adding additional paths to path_msg() to cover other paths relevant to > > the conflict make sense, but the primary path really needs to be left > > as-is for these other cases. > > I fear that I will need much, much more than just the additional paths ;-) > So far, I am parsing the `ls-files -u`-like part so that I get mode and > hash information, but eventually I want to parse that out of the messages > part and that only produces lists of paths so far... List of paths + conflict type + detailed conflict message, right? > And a simple, single list of paths is not enough to convey the information > that I need. Concretely, I need: > > - the original file name(s) involved in the conflict, Trying to get the original file name(s) may well cause you to need to include additional conflicts, since they were the cause of the name munging. So, are you displaying multiple conflicts at once, or just gathering the extra names and printing them without the extra context? If you do display multiple conflicts at once, are you focusing on one conflict and printing the necessary extra conflicts and paths to understand that one, and then potentially again listing some of those paths and conflicts again separately? And, to make it worse, what about cases where there is not a clear line between where one conflict starts and another ends? Do you start just agglomerating arbitrarily many conflicts together? And get the original names for all of them or some subset? And then do you print all the "original file name(s)" once for the whole set, or once per conflict within it? I'll list some examples of what I mean below with some different possibilities... > - the modes and object hashes corresponding to the different stages Which you can get by parsing the <Conflict info> section, and comparing to the names from your list of paths, right? > - maybe some provenance information in the future, in case of complex > rename/merge recursions > > In particular when presenting a file/directory conflict, it will be > important to know what stage refers to the file and what stage to the > directory (so as to know whether the respective object hash refers to a > file or even symlink and needs to be read as a blob). Is this follow-on paragraph meant to be an example of provenance information? Or a separate thought? I'm struggling a bit to follow. The stage and mode and hash information all come from the <Conflict info> section (i.e. the section with `ls-files -u`-like output). You already have that available. And the mode gives you the file/directory/symlink/submodule type. No conflict entries for a "tree" are presented (because it'd be kind of confusing to have a conflict entry for a tree as well as for several files under the tree, and allow the user to resolve them independently). So, when there's a file/directory conflict in particular, there will not be a stage provided to represent the directory. But I'm not sure how relevant that all is; it feels like I'm missing what you mean here. > > > Note: For the modify/delete conflict type, we used to mention _only_ the > > > munged name in that path list. To allow for tools to read the tree > > > object produced by `git merge-tree -z` into a Git index in order to > > > resolve the conflicts, it is necessary to list not only the original > > > name but _also_ the munged name so that the item with the munged file > > > name can be removed from that Git index. Therefore, this patch teaches > > > `git merge-tree` to show both the original _and_ the munged name in that > > > list. > > > > This makes sense. Passing additional valid and relevant information > > to callers of merge-tree seems beneficial. > > And now that you mention this, I realize that this should be pulled out > into its own patch. > > > > Also note: This patch changes the output of the remerge diff slightly: > > > whereas before, we printed the notice about a file/directory conflict > > > under the diff header mentioning the munged file name, we now print it > > > under a separate diff header that mentions the original file name. > > > That is the explanation why t4301 is touched by this patch. > > > > Sounds problematic. > > Fair enough. > > > > Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de> > > > --- > > > merge-ort.c | 14 +++++++------- > > > t/t4069-remerge-diff.sh | 8 ++++---- > > > t/t4301-merge-tree-write-tree.sh | 4 ++-- > > > 3 files changed, 13 insertions(+), 13 deletions(-) > > > > > > diff --git a/merge-ort.c b/merge-ort.c > > > index 01f150ef3b5..211f6823e1d 100644 > > > --- a/merge-ort.c > > > +++ b/merge-ort.c > > > @@ -3741,6 +3741,7 @@ static void process_entry(struct merge_options *opt, > > > struct conflict_info *ci, > > > struct directory_versions *dir_metadata) > > > { > > > + const char *orig_path = path; > > > int df_file_index = 0; > > > > > > VERIFY_CI(ci); > > > @@ -3787,7 +3788,6 @@ static void process_entry(struct merge_options *opt, > > > */ > > > struct conflict_info *new_ci; > > > const char *branch; > > > - const char *old_path = path; > > > int i; > > > > > > assert(ci->merged.result.mode == S_IFDIR); > > > > This looks like you are trying to make sure you always have the > > "original" path available, meaning a path corresponding to a file from > > one of the commits being merged. This doesn't quite accomplish that, > > though. Some questions: > > > > * What about cases where "path" is a location that exists only in > > the merge base? (rename/rename(1to2) is an example where this will > > come up) > > That's a good point. I will need to invent some way to convey all three > original file names in that case, not just one. Do you? The CONFLICT_RENAME_RENAME output in the <Informational Messages> section will have 3 path-from-merge-base path-from-side1 path-from-side2 CONFLICT (rename/rename) <detailed conflict message> So the user of merge-tree -z already has the information to tie these three filenames together. Why does process_entry() also need to have this information and provide it to subsequent calls? (Note that for a rename/rename(1to2) without other conflicts involved, process_entry() will be called for each path involved, and you'll get a <Conflict info> entry for each. One of those is from the merge base. So you'll have 3 <Conflict info> entries, as well as the above <Informational Message> to be able to tie them together.) However, my comments in the last 2 paragraphs gets us further away from my original question from my last email. In my previous email, I was trying to ask whether you were concentrating on being able to say which "side" or "branch" a file was from. Index entries at stage 1 don't come from either side. Is that problematic to you? > > * What about cases where "path" is a location that does not exist > > anywhere in history? (directory renames yield such a result. e.g. > > side A renames olddir/ -> newdir/, and side B adds olddir/foo, then > > "path" will be newdir/foo) > > Hmm. As mentioned, I want to focus on the scenario with disabled renames, > but not shut the door on rename scenarios like this one. > > So maybe I need to collect the original names in this instance, too, with > some (machine-parseable) indicator talking about the nature of that > origin. Which I think you can get from the <Informational Messages> that already exists... (again, example at the end) > > > @@ -3838,10 +3838,10 @@ static void process_entry(struct merge_options *opt, > > > strmap_put(&opt->priv->paths, path, new_ci); > > > > > > path_msg(opt, CONFLICT_FILE_DIRECTORY, 0, > > > - path, old_path, NULL, NULL, > > > + orig_path, path, NULL, NULL, > > > _("CONFLICT (file/directory): directory in the way " > > > "of %s from %s; moving it to %s instead."), > > > - old_path, branch, path); > > > + orig_path, branch, path); > > > /* > > > * Zero out the filemask for the old ci. At this point, ci > > > @@ -3921,7 +3921,7 @@ static void process_entry(struct merge_options *opt, > > > > > > if (rename_a && rename_b) { > > > path_msg(opt, CONFLICT_DISTINCT_MODES, 0, > > > - path, a_path, b_path, NULL, > > > + orig_path, a_path, b_path, NULL, > > > _("CONFLICT (distinct types): %s had " > > > "different types on each side; " > > > "renamed both of them so each can " > > > @@ -3929,7 +3929,7 @@ static void process_entry(struct merge_options *opt, > > > path); > > > } else { > > > path_msg(opt, CONFLICT_DISTINCT_MODES, 0, > > > - path, rename_a ? a_path : b_path, > > > + orig_path, rename_a ? a_path : b_path, > > > NULL, NULL, > > > _("CONFLICT (distinct types): %s had " > > > "different types on each side; " > > > > I think for both of these last two cases, path == orig_path, so you > > haven't actually made an effective change here. > > Not quite. If `ci->df_conflict && ci->merged.result.mode != 0`, then we > assign `path = unique_path(opt, path, branch);` and `path` no longer > refers to the same string as `orig_path`. > > It is possible that there is no actual code path that hits that clause and > then also hits the `CONFLICT_DISTINC_MODES` calls, but I think there is: > if a file has changed to a symlink in one branch _and_ was replaced by a > directory in the other branch. The "ci->filemask >= 6" check that guards this code prevents any such possibility. In particular, from your example, if it was changed to a symlink in one branch and replaced by a directory in the other branch, then ci->filemask is either 3 or 5, which would prevent the code from getting here. > > (And, as above, I want path to always be the primary_path passed to > > path_msg() as other code was written with that assumption in mind.) > > > > > @@ -4022,7 +4022,7 @@ static void process_entry(struct merge_options *opt, > > > if (S_ISGITLINK(merged_file.mode)) > > > reason = _("submodule"); > > > path_msg(opt, CONFLICT_CONTENTS, 0, > > > - path, NULL, NULL, NULL, > > > + orig_path, NULL, NULL, NULL, > > > _("CONFLICT (%s): Merge conflict in %s"), > > > reason, path); > > > } > > > > Here's another case where path == orig_path, so you haven't made an > > effective change. > > I cannot think of a case where a file/directory conflict can also result > in a content conflict, so at least in the regular 3-way merge, I think > you're right and this changed line does the same as before in all cases. > > > But this one highlights something interesting... > > > > In addition to the fact that path/orig_path may be a location that > > didn't exist on either side of history, there might actually be two > > relevant paths from the two different sides of history which are > > involved in the content merge, with neither of them being path or > > orig_path. Let me break it down, starting with a simpler two path > > case: > > > > If we have a standard rename, e.g. foo -> bar, and both sides modified > > the file but did so in a conflicting manner, then we will end up in > > this chunk of code. The conflict info emitted by merge-tree -z which > > is always of the form > > <number-of-paths>NUL<path1>NUL<path2>NUL<pathN>NUL<stable-short-type-description>NUL<message>NUL > > will in this particular case be > > 1<NUL>bar<NUL>Auto-merging<NUL>Auto-merging bar<newline><NUL> > > 1<NUL>bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): Merge > > conflict in bar<newline><NUL> > > Neither of these messages has any mention of "foo", despite the fact > > that "foo" was the name of the file in both the merge base and one > > side, and "bar" was only the name of the file on one side. > > I guess so, as no conflict message will mention the rename: it does not > conflict. Right, so we should probably also produce informational messages for normal renames; adding a path_msg() call to the "/* normal rename */" code path should do it. Though, I think we might only want such messages when merge-tree -z is being used. (Or perhaps if some kind of higher verbosity is requested?) > > Now, let's make it more interesting. side A modifies foo, and renames > > olddir/->newdir/. side B modifies foo in a conflicting manner, and > > renames foo->olddir/bar. Our `merge-tree -z` conflict information > > emitted will be of the form > > 1<NUL>newdir/bar<NUL>Auto-merging<NUL>Auto-merging newdir/bar<newline><NUL> > > 1<NUL>newdir/bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): > > Merge conflict in newdir/bar<newline><NUL> > > For this more interesting case, the only files that existed were "foo" > > and "olddir/bar", neither of which are mentioned in the conflict info. > > The conflict info only reports on "newdir/bar". > > > > And for both of these examples, your patch doesn't change the existing > > behavior at all since path == orig_path for this hunk of the patch. > > That is true, too. > > Which makes me think that the consuming code I have to work with will need > some elaborate changes when it wants to start supporting renames. > > But that's not the stage where I'm at right now. > > > > @@ -4067,7 +4067,7 @@ static void process_entry(struct merge_options *opt, > > > */ > > > } else { > > > path_msg(opt, CONFLICT_MODIFY_DELETE, 0, > > > - path, NULL, NULL, NULL, > > > + orig_path, path, NULL, NULL, > > > _("CONFLICT (modify/delete): %s deleted in %s " > > > "and modified in %s. Version %s of %s left " > > > "in tree."), > > > > Adding orig_path here after path would make sense, though it's still > > incomplete. For example if a path was modified and renamed on one > > side, and deleted on the other, then we'd get to this hunk, and the > > modify/delete would omit information about the other path. Further, > > directory renames could be at play, such that neither orig_path nor > > path would correspond to a filename in either of the commits being > > merged. > > > > There are also other path_msg() calls that should probably be informed > > of additional filenames due to simple renames: CONFLICT_CONTENTS (as > > noted above), CONFLICT_BINARY, and INFO_AUTO_MERGING. > > > > Going beyond simple renames, if you want "fully original" filenames, > > then we'd probably have to audit nearly all the path_msg() calls for > > when directory renames are possible, and modify the directory renaming > > code to plumb the original name through to this layer. > > > > And even if we do all that, then we still aren't matching what I think > > you are expecting where every conflicted higher order entry we report > > maps to a single <revision>:<filename> pairing. We may have a higher > > stage conflict entry that represents merged contents of three > > different filecontents from three different filenames. For example, > > if someone has the following setup: > > > > Commit O: one file named "original" > > Commit A: modify original + rename to primordial > > Commit B: add file named primordial, modify original differently + > > rename it to primary > > > > where A & B both have O as a parent or ancestor, and O is the unique > > merge base of A & B. Now if we try to merge A & B, then the index > > will have four higher order entries (using <mode> <hash> > > <stage-number> <filename>, as we'd see from either `ls-files -u` style > > output or from the first part of merge-tree -z output): > > > > 100644 <hash of O:original> 1 original > > 100644 <hash of auto-merge of O:original, A:primordial, & > > B:primary> 3 primary > > 100644 <hash of auto-merge of O:original, A:primordial, & > > B:primary> 2 primordial > > 100644 <hash of B:primordial> 3 primordial > > > > Note that the two middle entries, despite having one name attached to > > them, actually come from a merge of three different files. Further, > > that hash for those two entries will not have yet appeared as the > > contents for any blob in any revision of history, and will in fact > > represent a file with conflict markers (even if none of the original > > files -- O:original, A:primodial, B:primary -- had any conflicts). > > And, the version of "primodial" in the merge-result-tree would likely > > have nested conflict markers, despite O being a unique merge base > > (meaning we get nested conflict markers without merging merge bases). > > > > Trying to have all the conflict output from merge-tree map to some > > original file the user had from one of their sides of history is thus > > a goal that merge-ort is nowhere close to satisfying. I'm not even > > sure it's tractable. > > Sure, but I am getting more and more convinced that we will need to > provide "sort of" the original name. Somehow. Can I ask in more detail what you want to present to a user? Maybe it'd help frame the conversation more. Let me provide two examples, each with lots of alternatives, to see if any are the kind of output you want to be able to present to a user (and assume in each case it's not just a block message but you have some kind of programmatic knowledge of the components referenced in the message): Example 1. Are you thinking of something like "The following files were involved in a modify/delete conflict: foo, olddir/bar, newdir/bar, newdir/bar~B", followed by three other conflict messages that also involve those same files? Or, more like: "The following files were involved in a directory rename, and in a rename/delete conflict, and in a file/directory conflict, and in a modify/delete conflict: foo, olddir/bar, newdir/bar, newdir/bar~B" Or, more like: "The following files were involved in a directory rename, and in a rename/delete conflict, and in a file/directory conflict, and in a modify/delete conflict: foo (from side A), olddir/bar (from side B), newdir/bar (munged name), newdir/bar~B (munged name)" Or, more like: "The file newdir/bar~B has a modify/delete conflict. It was renamed to newdir/bar~B from newdir/bar because of a file/directory conflict. It was renamed to newdir/bar from olddir/bar because of a directory rename. It was renamed from foo to olddir/bar on side B. foo was deleted on side A." Or, more like: "foo was deleted on side A, and renamed to olddir/bar on side B. olddir/bar was renamed to newdir/bar because of a directory rename. newdir/bar was renamed to newdir/bar~B because of a file/directory conflict. newdir/bar~B has a modify/delete conflict." Or something completely different? Example 2. Are you thinking of something like: "The following files were involved in an add/add conflict: file-one, file-two, file-three, file-four, and file-five" ? Or, is it more like: "The following files were involved in an add/add conflict: file-one (from branch A), file-two (from merge base), file-three (from branch A), file-three (from branch B), file-four (from merge base), and file-five (from branch B)" ? Or, is it more like: "The following files were involved in an add/add conflict: file-three (from [see details]) and file-three (from [see details]). Note that file-three was part of a rename/rename(1to2) conflict, which involved file-one (from branch A), file-two (from merge base), and file-three (from branch B). Note that file-three was part of a rename/rename(1to2) conflict, which involved file-three (from branch A), file-four (from merge base), and file-five (from branch B)" Or, is it more like: "file-three was part of a rename/rename(1to2) conflict, which involved file-one (from branch A), file-two (from merge base), and file-three (from branch B). file-three was part of a rename/rename(1to2) conflict, which involved file-three (from branch A), file-four (from merge base), and file-five (from branch B). file-three and file-three are involved in an add/add conflict" (Of course, all of this example is assuming we're just focusing on the ultimate file-three vs file-three add/add conflict, not the additional conflicts that file-one and file-five may be involved in.) Or are you wanting to display something completely different? I think we already have enough info to do most of the above. The main shortcoming is just that path_msg() calls for various rename-oriented conflicts might also need to be augmented with branches (though perhaps that's natural to do if we consider the list passed to it to be a list of names rather than just a list of pathnames; then we can just add the branch(s) to the list) > That is, if there were successful (directory) renames happening before the > 3-way merge that resulted in a conflict, then I do not really care if the > "original original" filename isn't shown. But I do care that the filename > before munging is shown (and yes, we should probably also show the munged > filename). No, directory renames are part of the filename munging done *by* the three-way merge (not before it). That munging, though, applies before process_entry() is called. Here's an example (from Example 1 above): Commit O: foo, olddir/{a,b,c} Commit A: delete foo, rename olddir/ -> newdir/, add newdir/bar/file Commit B: modify foo & rename foo -> olddir/bar Here, A had a file named "foo". B had a file named "olddir/bar" (renamed from foo). The merge notices the olddir/ -> newdir/ directory rename in A and munges the filename further to "newdir/bar". The merge also notices that there is a directory in the way of "newdir/bar", and thus munges the filename even further to "newdir/bar~B". This would result in two higher order stages in the `ls-files -u`-style output, both for newdir/bar~B. The "original filename" for one of those is "foo" and for the other is "olddir/bar". However, the value for "path" passed to the lone "process_entry()" call will be "newdir/bar". The value of "path" after the call to unique_path() will be "newdir/bar~B". Neither "newdir/bar" or "newdir/bar~B" is an original filename. > And since recursive merges can result in multiple levels of munging, it > gets harder to explain what I want, but I think you'll get the idea that > the "pre-munged" filename of one conflict could be the "post-munged" > filename of a conflict that happened at a lower level of that recursion. Yes, recursing makes things more interesting. But even if you assume a unique merge-base, you still cannot assume that there is any such thing as "the original filename" for an entry in the `ls-files -u` output. > > However, there is a much smaller scoped goal that we could easily > > achieve: providing additional filename information in the path_msg() > > calls, much like you start to do with the modify/delete case. > > Sure, but we also need an indicator somehow what purpose each of those > additional filenames serve. I guess we could make it implicit, just like > we make it implicit that the first path in a file/directory conflict > refers to the file's name and the second to the directory's. Yes, I mean the point of the <Informational messages> was to provide parsable information and this was what I had in mind when I created it. Very minor nitpick, though: the first path refers to the *munged* file's name, and the second path refers to the directory's name. (If the file and directory had different names before our additional munging, we wouldn't have needed that additional munging). > But it makes > me a bit uneasy and I wish we had more like a structure instead of just a > string that we could print out. That would make it possible to convey > from where (as in "which branch?") the file or directory came. And what about other cases where there is no "the branch" that a file came from? For example, doing rename + add/add: Commit O: foo Commit A: modify foo + rename foo->bar Commit B: modify foo, add unrelated bar In this case, we'll have a "bar" at stage 3 representing B:bar. We'll also get a three-way merge of O:foo, A:bar, and B:foo, stuck in a stage 2 entry under the filename bar. The stage 2 entry for bar did not come from one branch, it represents merged contents that came from all three commits. > > > diff --git a/t/t4069-remerge-diff.sh b/t/t4069-remerge-diff.sh > > > index 35f94957fce..bc580a242ac 100755 > > > --- a/t/t4069-remerge-diff.sh > > > +++ b/t/t4069-remerge-diff.sh > > > @@ -131,11 +131,12 @@ test_expect_success 'setup non-content conflicts' ' > > > test_expect_success 'remerge-diff with non-content conflicts' ' > > > git log -1 --oneline resolution >tmp && > > > cat <<-EOF >>tmp && > > > + diff --git a/file_or_directory b/file_or_directory > > > + remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > > diff --git a/file_or_directory~HASH (side1) b/wanted_content > > > similarity index 100% > > > rename from file_or_directory~HASH (side1) > > > rename to wanted_content > > > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > > diff --git a/letters b/letters > > > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > > > diff --git a/letters_side2 b/letters_side2 > > > @@ -168,7 +169,7 @@ test_expect_success 'remerge-diff with non-content conflicts' ' > > > test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no diff content' ' > > > git log -1 --oneline resolution >tmp && > > > cat <<-EOF >>tmp && > > > - diff --git a/file_or_directory~HASH (side1) b/file_or_directory~HASH (side1) > > > + diff --git a/file_or_directory b/file_or_directory > > > remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > > diff --git a/letters b/letters > > > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > > > @@ -184,14 +185,13 @@ test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no dif > > > test_cmp expect actual > > > ' > > > > > > -test_expect_success 'remerge-diff w/ diff-filter=R: relevant file + conflict header' ' > > > +test_expect_success 'remerge-diff w/ diff-filter=R: relevant file' ' > > > git log -1 --oneline resolution >tmp && > > > cat <<-EOF >>tmp && > > > diff --git a/file_or_directory~HASH (side1) b/wanted_content > > > similarity index 100% > > > rename from file_or_directory~HASH (side1) > > > rename to wanted_content > > > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > > EOF > > > # We still have some sha1 hashes above; rip them out so test works > > > # with sha256 > > > > All of these look like regressions to me; you discussed these in a > > separate email so I'll go into detail in response to that one. > > Thank you for being so diligent. I always appreciate it when I see a > well-written, obviously well-edited mail. I find it very considerate. > > You did a good job at convincing me that my approach is sub-optimal, and > that we need at least one new flag to get the information that I need to > produce in my use case because the caller requires it to present a nice > conflict resolution UI to the users. I'm not so sure you need additional information. Don't the <Conflicted file info> and <Informational messages> provide enough info already, for the usecases you are considering? In particular, you were trying to add a testcase in your series for a simple directory/file conflict, but for such a case you can already determine original filenames. The <Conflicted file info> and <Informational messages> sections of the `merge-tree -z` output for these, after replacing NUL characters with newlines, would be: 120000 <HASH> 3 origfile 100644 <HASH> 1 origfile~branch 100644 <HASH> 2 origfile~branch 2 origfile origfile~branch CONFLICT (distinct modes) CONFLICT (distinct types): whatever had different types on each side; renamed one of them so each can be recorded somewhere. Now, the "distinct modes" conflict always results in filename munging, with both names listed. So, from this output, you know the original name is "origfile" whenever you see "origfile~branch". So now you can present all the `ls-files -u` output from the first section with remapping munged filenames back to original filenames. Doesn't that solve your immediate usecase already, without even needing any merge-ort changes?
Hi Elijah, I'll top-post because I will still need much more time to wrap my head around everything you said in your very thoughtful reply. By the way, have I mentioned before how much I appreciate well-edited, well-crafted mails? It must have taken quite a bit of care and editing to write this, and I just want to thank you for that. As to the problem we are discussing: I am fully on board with you on the necessity to keep the door open for rename detection and recursive merges and the zoo of conflict types that comes with it. After thinking about this for a couple of nights, I think I want to tackle the design from the "other" side, namely from the UI. At this stage, I _think_ it would do well for me to give a more high-level context as to my motivating factors. The thing is, these worktree-less merges do not live in a vacuum. They are backing the Pull Request UI. There exists a relatively simple UI to resolve simple merge conflicts, which factors into what I want from `git merge-tree`. You've pointed out in the past, very rightfully, that the simple way to "pre-resolve" conflicts that I have implemented over here is just not good enough except for a very, very limited manner of driving `git merge-tree`. These "pre-resolved conflicts" are currently essentially a map of <path> -> <OID> pairs. Now, to design a much better way to present conflicts as well as allowing users to resolve some of them (the simpler ones, like content conflicts) in a web UI, we need `git merge-tree` to give us not an `ls-files -u`-style output, that only works reliably for very simple cases. In general, I think the design would work best if we parsed the `messages` part exclusively on the server side (ignoring the `ls-files`-like output, probably even suppressing it) and if we received enough information to recreate the conflicts from there. Which means that we need the involved paths (whether munged or not, I am less and less concerned) together with the modes and OIDs. It would probably be helpful even to provide a tree OID for the directory part of any file/directory conflict, where the web UI could then present the contents in a tree view to the user. This way, we could present multiple conflicts, even multiple conflicts for the very same file, and allow resolving them individually. And we could associate resolutions with the exact list of paths, modes and OIDs involved, then teach `merge-tree` to optionally take such resolutions into account, to produce merges with resolved conflicts. Doing it this way will require substantial changes on the server side, in particular in the UI, which currently does not have any support for handling conflicts involving renames. It will be a lot of work, and I am certain that I will have to pivot many times regarding the design of `git merge-tree` features supporting that UI. I'm excited to start that journey. What do you think? Ciao, Dscho On Mon, 22 Aug 2022, Elijah Newren wrote: > On Mon, Aug 22, 2022 at 1:12 PM Johannes Schindelin > <Johannes.Schindelin@gmx.de> wrote: > > > > Hi Elijah, > > > > On Sat, 20 Aug 2022, Elijah Newren wrote: > > > > > On Thu, Aug 18, 2022 at 11:57 PM Johannes Schindelin via GitGitGadget > > > <gitgitgadget@gmail.com> wrote: > > > > > > > > From: Johannes Schindelin <johannes.schindelin@gmx.de> > > > > > > > > When printing the messages in a machine-readable format (i.e. in `-z` > > > > mode), the intention is to support third-party software that wants to > > > > present the conflicts in a "pretty" way to human readers. > > > > > > > > To that end, `git merge-tree -z` prefixes the conflict messages by a > > > > variable-size list of paths and a conflict identifier. This list and > > > > this identifier are intended to be machine-parseable, the conflict > > > > message is sort of free-form and not intended to be parsed. > > > > > > > > Keeping in mind that the intended use case is to have third-party > > > > software use `git merge-tree` to perform worktree-less merges and then > > > > present the conflicts (if any) to human readers, it makes much more > > > > sense to show the original file names of the involved files instead of > > > > the ones we internally munged to allow for creating a tree object that > > > > contains entries for both sides of the conflict (which requires them to > > > > have different names). > > > > > > I can see how this would be useful to you if it could be > > > achieved...well, assuming you replaced "instead of" with "in addition > > > to". It'd be a huge undertaking to achieve, though; far beyond what > > > this series attempts. > > > > I feared you would say that :-) > > > > And I realize that I should have lead with the context that is relevant to > > understand what I am driving at. > > > > For now, all I am trying to do is to verify that we can achieve identical > > results between a new code path that uses `git merge-tree` (with options) > > and an existing code path that uses libgit2 to perform a 3-way merge. > > > > To that end, I have added some patches to optionally disable rename > > detection, to allow specifying a merge base instead of performing a full > > recursive merge, to optionally detect whether the `MERGE_HEAD` is already > > reachable from `HEAD` and in that case exit with a specific exit status, > > among other things. > > > > And yes, I realize that I "only" benefit from the worktree-less nature of > > `merge-ort` so far, and not from any other aspects `merge-ort` offers. > > > > One thing I need from the result is a list of conflicts in the same format > > as we would have received if writing out an index with conflicts and then > > parsing just the entries with non-zero stages. > > Thanks, this is good context. Especially this last paragraph, where > you say you want the non-zero stage entries from what would be written > to the index. I'll return to this many times. > > > So for now, I do not care _so_ much about rename conflicts, or conflicts > > resulting from merging merge base candidates. I don't want to slam the > > door shut, of course, but due to a teeny tiny bit of time pressure, I want > > to focus very, very much on the libgit2-like operation for now :-) > > > > What I do need, however, is the unmunged filenames as well as the > > corresponding file modes and the object hashes. > > If you're under time pressure...don't you already have all the > information you need? Why can't you just parse the <Conflicted file > info> and <Informational messages> sections to get it? I'll come back > to this again with the testcase you highlighted in your series, at the > very end of the email. > > > > I think your attempt here may be based on the presumption that any > > > hash in a higher stage in the index maps directly to the hash of some > > > <revision>:<filename> pair from one of the commits being merged. That > > > assumption holds in simple cases, but does not hold generally. > > > merge-ort has a very strong "(at most) 3 stages" assumption for any > > > conflict, driven by the need to ultimately represent conflicts in the > > > index, which is pervasive in the code. > > > > This is actually very helpful information, this had not sunk in yet. > > > > So when encountering a "stage 3 before stages 1 & 2" situation arises, > > like in the test case I added, you're saying that I could simply reorder > > those and be done and not have to expect that the `stage 1 & 2` part is > > followed by _another_ stage 3 part for the same name? > > > > Except, of course, that I need the original file names, and the invariant > > of at most 3 stages holds true only when taking munged file names into > > account, right? > > You are correct that ort is never going to present two "stage 3" > entries for any path it reports. > > stages are very much an index-related concept, and always associated > with potentially munged filenames in ort. This importantly means that > there is no such thing as "the unmunged filename" for some index > entries. > > In order to achieve the "at most 3 stages" assumption plastered all > over ort, it has to compress information from more than 3 > <revision>:<filename> pairs down to at most 3. So some of those > non-zero stage index entries do *not* correspond to a file from > history. Any request to get "the original filename" would somehow > have to respond with up to three filenames instead. (And likewise, > any request to get "the original file contents from one of the sides > of history" would have to respond with the content of up to three > files instead.) > > > I know, I know, you don't care because you're not dealing with renames > right now. But if you're modifying the code, it has to be in a way > that doesn't paint us into a corner that doesn't work for renames. > > > > Trying to create such a direct mapping between higher stage conflicts > > > and <revision>:<filename> pairs from the commits being merged would > > > require breaking that assumption. That would mean getting rid of many of > > > the hardcoded array sizes of 3, as well as modifying who knows how many > > > portions of logic built around this assumption. > > > > > > Your attempt also seems to be based on the presumption that the > > > original value of "path" as passed to process_entry(), represents a > > > path from one of the two sides of history. In simple cases, it does. > > > But the assumption does not generally hold. > > > > Heh, in my case it always holds because I disable rename detection. The > > only time when multiple file names are involved is in case of > > file/directory conflicts and the like. In which case I still need the > > original filename. > > > > But I would contend that we need to know the original file name(s) even in > > the case of renames, so that we can present the users with all the > > information pertinent to the conflict, to allow them to resolve it (so > > that they have the full context). > > To reiterate, "the original file name(s)" is an ill-posed problem. It > only exists for some entries. (Same for "the original file contents") > ...well, unless you rewire ort completely and toss out the "at most 3 > stages" assumption. > > > > Further... > > > > > > > So let's mention the original file names prominently, as first item in > > > > that variable-size list of paths. > > > > > > This is in conflict with the needs of other uses of merge-ort: > > > * When the user is running an active merge with conflicts written to > > > the index and working tree, the filenames where the conflicts are > > > recorded may not match the original filenames, and the filenames where > > > things are recorded is more useful and thus primary. > > > * For remerge-diff, the filename where the conflict would be placed > > > is important because that's where stuff ends up. Any changes the user > > > made for the final merge is relative to that munged path, so that > > > munged path is the important bit, not the original path. > > > > Okay, I understand now. > > > > So basically, if I need different/additional information, I will _have_ to > > introduce a new flag to trigger code paths that produce that. > > > > And my changes to the remerge test cases merely demonstrate precisely that > > need. > > > > > Adding additional paths to path_msg() to cover other paths relevant to > > > the conflict make sense, but the primary path really needs to be left > > > as-is for these other cases. > > > > I fear that I will need much, much more than just the additional paths ;-) > > So far, I am parsing the `ls-files -u`-like part so that I get mode and > > hash information, but eventually I want to parse that out of the messages > > part and that only produces lists of paths so far... > > List of paths + conflict type + detailed conflict message, right? > > > And a simple, single list of paths is not enough to convey the information > > that I need. Concretely, I need: > > > > - the original file name(s) involved in the conflict, > > Trying to get the original file name(s) may well cause you to need to > include additional conflicts, since they were the cause of the name > munging. So, are you displaying multiple conflicts at once, or just > gathering the extra names and printing them without the extra context? > If you do display multiple conflicts at once, are you focusing on one > conflict and printing the necessary extra conflicts and paths to > understand that one, and then potentially again listing some of those > paths and conflicts again separately? > > And, to make it worse, what about cases where there is not a clear > line between where one conflict starts and another ends? Do you start > just agglomerating arbitrarily many conflicts together? And get the > original names for all of them or some subset? And then do you print > all the "original file name(s)" once for the whole set, or once per > conflict within it? > > I'll list some examples of what I mean below with some different > possibilities... > > > - the modes and object hashes corresponding to the different stages > > Which you can get by parsing the <Conflict info> section, and > comparing to the names from your list of paths, right? > > > - maybe some provenance information in the future, in case of complex > > rename/merge recursions > > > > In particular when presenting a file/directory conflict, it will be > > important to know what stage refers to the file and what stage to the > > directory (so as to know whether the respective object hash refers to a > > file or even symlink and needs to be read as a blob). > > Is this follow-on paragraph meant to be an example of provenance > information? Or a separate thought? I'm struggling a bit to follow. > > The stage and mode and hash information all come from the <Conflict > info> section (i.e. the section with `ls-files -u`-like output). You > already have that available. And the mode gives you the > file/directory/symlink/submodule type. > > No conflict entries for a "tree" are presented (because it'd be kind > of confusing to have a conflict entry for a tree as well as for > several files under the tree, and allow the user to resolve them > independently). So, when there's a file/directory conflict in > particular, there will not be a stage provided to represent the > directory. > > But I'm not sure how relevant that all is; it feels like I'm missing > what you mean here. > > > > > Note: For the modify/delete conflict type, we used to mention _only_ the > > > > munged name in that path list. To allow for tools to read the tree > > > > object produced by `git merge-tree -z` into a Git index in order to > > > > resolve the conflicts, it is necessary to list not only the original > > > > name but _also_ the munged name so that the item with the munged file > > > > name can be removed from that Git index. Therefore, this patch teaches > > > > `git merge-tree` to show both the original _and_ the munged name in that > > > > list. > > > > > > This makes sense. Passing additional valid and relevant information > > > to callers of merge-tree seems beneficial. > > > > And now that you mention this, I realize that this should be pulled out > > into its own patch. > > > > > > Also note: This patch changes the output of the remerge diff slightly: > > > > whereas before, we printed the notice about a file/directory conflict > > > > under the diff header mentioning the munged file name, we now print it > > > > under a separate diff header that mentions the original file name. > > > > That is the explanation why t4301 is touched by this patch. > > > > > > Sounds problematic. > > > > Fair enough. > > > > > > Signed-off-by: Johannes Schindelin <johannes.schindelin@gmx.de> > > > > --- > > > > merge-ort.c | 14 +++++++------- > > > > t/t4069-remerge-diff.sh | 8 ++++---- > > > > t/t4301-merge-tree-write-tree.sh | 4 ++-- > > > > 3 files changed, 13 insertions(+), 13 deletions(-) > > > > > > > > diff --git a/merge-ort.c b/merge-ort.c > > > > index 01f150ef3b5..211f6823e1d 100644 > > > > --- a/merge-ort.c > > > > +++ b/merge-ort.c > > > > @@ -3741,6 +3741,7 @@ static void process_entry(struct merge_options *opt, > > > > struct conflict_info *ci, > > > > struct directory_versions *dir_metadata) > > > > { > > > > + const char *orig_path = path; > > > > int df_file_index = 0; > > > > > > > > VERIFY_CI(ci); > > > > @@ -3787,7 +3788,6 @@ static void process_entry(struct merge_options *opt, > > > > */ > > > > struct conflict_info *new_ci; > > > > const char *branch; > > > > - const char *old_path = path; > > > > int i; > > > > > > > > assert(ci->merged.result.mode == S_IFDIR); > > > > > > This looks like you are trying to make sure you always have the > > > "original" path available, meaning a path corresponding to a file from > > > one of the commits being merged. This doesn't quite accomplish that, > > > though. Some questions: > > > > > > * What about cases where "path" is a location that exists only in > > > the merge base? (rename/rename(1to2) is an example where this will > > > come up) > > > > That's a good point. I will need to invent some way to convey all three > > original file names in that case, not just one. > > Do you? The CONFLICT_RENAME_RENAME output in the <Informational > Messages> section will have > 3 > path-from-merge-base > path-from-side1 > path-from-side2 > CONFLICT (rename/rename) > <detailed conflict message> > So the user of merge-tree -z already has the information to tie these > three filenames together. Why does process_entry() also need to have > this information and provide it to subsequent calls? > > (Note that for a rename/rename(1to2) without other conflicts involved, > process_entry() will be called for each path involved, and you'll get > a <Conflict info> entry for each. One of those is from the merge > base. So you'll have 3 <Conflict info> entries, as well as the above > <Informational Message> to be able to tie them together.) > > However, my comments in the last 2 paragraphs gets us further away > from my original question from my last email. > > In my previous email, I was trying to ask whether you were > concentrating on being able to say which "side" or "branch" a file was > from. Index entries at stage 1 don't come from either side. Is that > problematic to you? > > > > * What about cases where "path" is a location that does not exist > > > anywhere in history? (directory renames yield such a result. e.g. > > > side A renames olddir/ -> newdir/, and side B adds olddir/foo, then > > > "path" will be newdir/foo) > > > > Hmm. As mentioned, I want to focus on the scenario with disabled renames, > > but not shut the door on rename scenarios like this one. > > > > So maybe I need to collect the original names in this instance, too, with > > some (machine-parseable) indicator talking about the nature of that > > origin. > > Which I think you can get from the <Informational Messages> that > already exists... (again, example at the end) > > > > > @@ -3838,10 +3838,10 @@ static void process_entry(struct merge_options *opt, > > > > strmap_put(&opt->priv->paths, path, new_ci); > > > > > > > > path_msg(opt, CONFLICT_FILE_DIRECTORY, 0, > > > > - path, old_path, NULL, NULL, > > > > + orig_path, path, NULL, NULL, > > > > _("CONFLICT (file/directory): directory in the way " > > > > "of %s from %s; moving it to %s instead."), > > > > - old_path, branch, path); > > > > + orig_path, branch, path); > > > > /* > > > > * Zero out the filemask for the old ci. At this point, ci > > > > @@ -3921,7 +3921,7 @@ static void process_entry(struct merge_options *opt, > > > > > > > > if (rename_a && rename_b) { > > > > path_msg(opt, CONFLICT_DISTINCT_MODES, 0, > > > > - path, a_path, b_path, NULL, > > > > + orig_path, a_path, b_path, NULL, > > > > _("CONFLICT (distinct types): %s had " > > > > "different types on each side; " > > > > "renamed both of them so each can " > > > > @@ -3929,7 +3929,7 @@ static void process_entry(struct merge_options *opt, > > > > path); > > > > } else { > > > > path_msg(opt, CONFLICT_DISTINCT_MODES, 0, > > > > - path, rename_a ? a_path : b_path, > > > > + orig_path, rename_a ? a_path : b_path, > > > > NULL, NULL, > > > > _("CONFLICT (distinct types): %s had " > > > > "different types on each side; " > > > > > > I think for both of these last two cases, path == orig_path, so you > > > haven't actually made an effective change here. > > > > Not quite. If `ci->df_conflict && ci->merged.result.mode != 0`, then we > > assign `path = unique_path(opt, path, branch);` and `path` no longer > > refers to the same string as `orig_path`. > > > > It is possible that there is no actual code path that hits that clause and > > then also hits the `CONFLICT_DISTINC_MODES` calls, but I think there is: > > if a file has changed to a symlink in one branch _and_ was replaced by a > > directory in the other branch. > > The "ci->filemask >= 6" check that guards this code prevents any such > possibility. In particular, from your example, if it was changed to a > symlink in one branch and replaced by a directory in the other branch, > then ci->filemask is either 3 or 5, which would prevent the code from > getting here. > > > > (And, as above, I want path to always be the primary_path passed to > > > path_msg() as other code was written with that assumption in mind.) > > > > > > > @@ -4022,7 +4022,7 @@ static void process_entry(struct merge_options *opt, > > > > if (S_ISGITLINK(merged_file.mode)) > > > > reason = _("submodule"); > > > > path_msg(opt, CONFLICT_CONTENTS, 0, > > > > - path, NULL, NULL, NULL, > > > > + orig_path, NULL, NULL, NULL, > > > > _("CONFLICT (%s): Merge conflict in %s"), > > > > reason, path); > > > > } > > > > > > Here's another case where path == orig_path, so you haven't made an > > > effective change. > > > > I cannot think of a case where a file/directory conflict can also result > > in a content conflict, so at least in the regular 3-way merge, I think > > you're right and this changed line does the same as before in all cases. > > > > > But this one highlights something interesting... > > > > > > In addition to the fact that path/orig_path may be a location that > > > didn't exist on either side of history, there might actually be two > > > relevant paths from the two different sides of history which are > > > involved in the content merge, with neither of them being path or > > > orig_path. Let me break it down, starting with a simpler two path > > > case: > > > > > > If we have a standard rename, e.g. foo -> bar, and both sides modified > > > the file but did so in a conflicting manner, then we will end up in > > > this chunk of code. The conflict info emitted by merge-tree -z which > > > is always of the form > > > <number-of-paths>NUL<path1>NUL<path2>NUL<pathN>NUL<stable-short-type-description>NUL<message>NUL > > > will in this particular case be > > > 1<NUL>bar<NUL>Auto-merging<NUL>Auto-merging bar<newline><NUL> > > > 1<NUL>bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): Merge > > > conflict in bar<newline><NUL> > > > Neither of these messages has any mention of "foo", despite the fact > > > that "foo" was the name of the file in both the merge base and one > > > side, and "bar" was only the name of the file on one side. > > > > I guess so, as no conflict message will mention the rename: it does not > > conflict. > > Right, so we should probably also produce informational messages for > normal renames; adding a path_msg() call to the "/* normal rename */" > code path should do it. Though, I think we might only want such > messages when merge-tree -z is being used. (Or perhaps if some kind > of higher verbosity is requested?) > > > > Now, let's make it more interesting. side A modifies foo, and renames > > > olddir/->newdir/. side B modifies foo in a conflicting manner, and > > > renames foo->olddir/bar. Our `merge-tree -z` conflict information > > > emitted will be of the form > > > 1<NUL>newdir/bar<NUL>Auto-merging<NUL>Auto-merging newdir/bar<newline><NUL> > > > 1<NUL>newdir/bar<NUL>CONFLICT (contents)<NUL>CONFLICT (content): > > > Merge conflict in newdir/bar<newline><NUL> > > > For this more interesting case, the only files that existed were "foo" > > > and "olddir/bar", neither of which are mentioned in the conflict info. > > > The conflict info only reports on "newdir/bar". > > > > > > And for both of these examples, your patch doesn't change the existing > > > behavior at all since path == orig_path for this hunk of the patch. > > > > That is true, too. > > > > Which makes me think that the consuming code I have to work with will need > > some elaborate changes when it wants to start supporting renames. > > > > But that's not the stage where I'm at right now. > > > > > > @@ -4067,7 +4067,7 @@ static void process_entry(struct merge_options *opt, > > > > */ > > > > } else { > > > > path_msg(opt, CONFLICT_MODIFY_DELETE, 0, > > > > - path, NULL, NULL, NULL, > > > > + orig_path, path, NULL, NULL, > > > > _("CONFLICT (modify/delete): %s deleted in %s " > > > > "and modified in %s. Version %s of %s left " > > > > "in tree."), > > > > > > Adding orig_path here after path would make sense, though it's still > > > incomplete. For example if a path was modified and renamed on one > > > side, and deleted on the other, then we'd get to this hunk, and the > > > modify/delete would omit information about the other path. Further, > > > directory renames could be at play, such that neither orig_path nor > > > path would correspond to a filename in either of the commits being > > > merged. > > > > > > There are also other path_msg() calls that should probably be informed > > > of additional filenames due to simple renames: CONFLICT_CONTENTS (as > > > noted above), CONFLICT_BINARY, and INFO_AUTO_MERGING. > > > > > > Going beyond simple renames, if you want "fully original" filenames, > > > then we'd probably have to audit nearly all the path_msg() calls for > > > when directory renames are possible, and modify the directory renaming > > > code to plumb the original name through to this layer. > > > > > > And even if we do all that, then we still aren't matching what I think > > > you are expecting where every conflicted higher order entry we report > > > maps to a single <revision>:<filename> pairing. We may have a higher > > > stage conflict entry that represents merged contents of three > > > different filecontents from three different filenames. For example, > > > if someone has the following setup: > > > > > > Commit O: one file named "original" > > > Commit A: modify original + rename to primordial > > > Commit B: add file named primordial, modify original differently + > > > rename it to primary > > > > > > where A & B both have O as a parent or ancestor, and O is the unique > > > merge base of A & B. Now if we try to merge A & B, then the index > > > will have four higher order entries (using <mode> <hash> > > > <stage-number> <filename>, as we'd see from either `ls-files -u` style > > > output or from the first part of merge-tree -z output): > > > > > > 100644 <hash of O:original> 1 original > > > 100644 <hash of auto-merge of O:original, A:primordial, & > > > B:primary> 3 primary > > > 100644 <hash of auto-merge of O:original, A:primordial, & > > > B:primary> 2 primordial > > > 100644 <hash of B:primordial> 3 primordial > > > > > > Note that the two middle entries, despite having one name attached to > > > them, actually come from a merge of three different files. Further, > > > that hash for those two entries will not have yet appeared as the > > > contents for any blob in any revision of history, and will in fact > > > represent a file with conflict markers (even if none of the original > > > files -- O:original, A:primodial, B:primary -- had any conflicts). > > > And, the version of "primodial" in the merge-result-tree would likely > > > have nested conflict markers, despite O being a unique merge base > > > (meaning we get nested conflict markers without merging merge bases). > > > > > > Trying to have all the conflict output from merge-tree map to some > > > original file the user had from one of their sides of history is thus > > > a goal that merge-ort is nowhere close to satisfying. I'm not even > > > sure it's tractable. > > > > Sure, but I am getting more and more convinced that we will need to > > provide "sort of" the original name. Somehow. > > Can I ask in more detail what you want to present to a user? Maybe > it'd help frame the conversation more. > > Let me provide two examples, each with lots of alternatives, to see if > any are the kind of output you want to be able to present to a user > (and assume in each case it's not just a block message but you have > some kind of programmatic knowledge of the components referenced in > the message): > > Example 1. > > Are you thinking of something like "The following files were involved > in a modify/delete conflict: foo, olddir/bar, newdir/bar, > newdir/bar~B", followed by three other conflict messages that also involve > those same files? > > Or, more like: "The following files were involved in a directory > rename, and in a rename/delete conflict, and in a file/directory > conflict, and in a modify/delete conflict: foo, olddir/bar, > newdir/bar, newdir/bar~B" > > Or, more like: "The following files were involved in a directory > rename, and in a rename/delete conflict, and in a file/directory > conflict, and in a modify/delete conflict: foo (from side A), > olddir/bar (from side B), newdir/bar (munged name), newdir/bar~B > (munged name)" > > Or, more like: "The file newdir/bar~B has a modify/delete conflict. > It was renamed to newdir/bar~B from newdir/bar because of a > file/directory conflict. It was renamed to newdir/bar from olddir/bar > because of a directory rename. It was renamed from foo to olddir/bar > on side B. foo was deleted on side A." > > Or, more like: "foo was deleted on side A, and renamed to olddir/bar > on side B. olddir/bar was renamed to newdir/bar because of a > directory rename. newdir/bar was renamed to newdir/bar~B because of a > file/directory conflict. newdir/bar~B has a modify/delete conflict." > > Or something completely different? > > > Example 2. > > Are you thinking of something like: "The following files were involved > in an add/add conflict: file-one, file-two, file-three, file-four, and > file-five" ? > > Or, is it more like: "The following files were involved in an add/add > conflict: file-one (from branch A), file-two (from merge base), > file-three (from branch A), file-three (from branch B), file-four > (from merge base), and file-five (from branch B)" ? > > Or, is it more like: "The following files were involved in an add/add > conflict: file-three (from [see details]) and file-three (from [see > details]). Note that file-three was part of a rename/rename(1to2) > conflict, which involved file-one (from branch A), file-two (from > merge base), and file-three (from branch B). Note that file-three was > part of a rename/rename(1to2) conflict, which involved file-three > (from branch A), file-four (from merge base), and file-five (from > branch B)" > > Or, is it more like: "file-three was part of a rename/rename(1to2) > conflict, which involved file-one (from branch A), file-two (from > merge base), and file-three (from branch B). file-three was part of a > rename/rename(1to2) conflict, which involved file-three (from branch > A), file-four (from merge base), and file-five (from branch B). > file-three and file-three are involved in an add/add conflict" > > (Of course, all of this example is assuming we're just focusing on the > ultimate file-three vs file-three add/add conflict, not the additional > conflicts that file-one and file-five may be involved in.) > > Or are you wanting to display something completely different? > > > > I think we already have enough info to do most of the above. > > The main shortcoming is just that path_msg() calls for various > rename-oriented conflicts might also need to be augmented with > branches (though perhaps that's natural to do if we consider the list > passed to it to be a list of names rather than just a list of > pathnames; then we can just add the branch(s) to the list) > > > That is, if there were successful (directory) renames happening before the > > 3-way merge that resulted in a conflict, then I do not really care if the > > "original original" filename isn't shown. But I do care that the filename > > before munging is shown (and yes, we should probably also show the munged > > filename). > > No, directory renames are part of the filename munging done *by* the > three-way merge (not before it). That munging, though, applies before > process_entry() is called. Here's an example (from Example 1 above): > > Commit O: foo, olddir/{a,b,c} > Commit A: delete foo, rename olddir/ -> newdir/, add newdir/bar/file > Commit B: modify foo & rename foo -> olddir/bar > > Here, A had a file named "foo". B had a file named "olddir/bar" > (renamed from foo). The merge notices the olddir/ -> newdir/ > directory rename in A and munges the filename further to "newdir/bar". > The merge also notices that there is a directory in the way of > "newdir/bar", and thus munges the filename even further to > "newdir/bar~B". This would result in two higher order stages in the > `ls-files -u`-style output, both for newdir/bar~B. > > The "original filename" for one of those is "foo" and for the other is > "olddir/bar". However, the value for "path" passed to the lone > "process_entry()" call will be "newdir/bar". The value of "path" > after the call to unique_path() will be "newdir/bar~B". Neither > "newdir/bar" or "newdir/bar~B" is an original filename. > > > And since recursive merges can result in multiple levels of munging, it > > gets harder to explain what I want, but I think you'll get the idea that > > the "pre-munged" filename of one conflict could be the "post-munged" > > filename of a conflict that happened at a lower level of that recursion. > > Yes, recursing makes things more interesting. But even if you assume > a unique merge-base, you still cannot assume that there is any such > thing as "the original filename" for an entry in the `ls-files -u` output. > > > > However, there is a much smaller scoped goal that we could easily > > > achieve: providing additional filename information in the path_msg() > > > calls, much like you start to do with the modify/delete case. > > > > Sure, but we also need an indicator somehow what purpose each of those > > additional filenames serve. I guess we could make it implicit, just like > > we make it implicit that the first path in a file/directory conflict > > refers to the file's name and the second to the directory's. > > Yes, I mean the point of the <Informational messages> was to provide > parsable information and this was what I had in mind when I created > it. > > Very minor nitpick, though: the first path refers to the *munged* > file's name, and the second path refers to the directory's name. (If > the file and directory had different names before our additional > munging, we wouldn't have needed that additional munging). > > > But it makes > > me a bit uneasy and I wish we had more like a structure instead of just a > > string that we could print out. That would make it possible to convey > > from where (as in "which branch?") the file or directory came. > > And what about other cases where there is no "the branch" that a file > came from? For example, doing rename + add/add: > > Commit O: foo > Commit A: modify foo + rename foo->bar > Commit B: modify foo, add unrelated bar > > In this case, we'll have a "bar" at stage 3 representing B:bar. We'll > also get a three-way merge of O:foo, A:bar, and B:foo, stuck in a > stage 2 entry under the filename bar. The stage 2 entry for bar did > not come from one branch, it represents merged contents that came from > all three commits. > > > > > diff --git a/t/t4069-remerge-diff.sh b/t/t4069-remerge-diff.sh > > > > index 35f94957fce..bc580a242ac 100755 > > > > --- a/t/t4069-remerge-diff.sh > > > > +++ b/t/t4069-remerge-diff.sh > > > > @@ -131,11 +131,12 @@ test_expect_success 'setup non-content conflicts' ' > > > > test_expect_success 'remerge-diff with non-content conflicts' ' > > > > git log -1 --oneline resolution >tmp && > > > > cat <<-EOF >>tmp && > > > > + diff --git a/file_or_directory b/file_or_directory > > > > + remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > > > diff --git a/file_or_directory~HASH (side1) b/wanted_content > > > > similarity index 100% > > > > rename from file_or_directory~HASH (side1) > > > > rename to wanted_content > > > > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > > > diff --git a/letters b/letters > > > > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > > > > diff --git a/letters_side2 b/letters_side2 > > > > @@ -168,7 +169,7 @@ test_expect_success 'remerge-diff with non-content conflicts' ' > > > > test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no diff content' ' > > > > git log -1 --oneline resolution >tmp && > > > > cat <<-EOF >>tmp && > > > > - diff --git a/file_or_directory~HASH (side1) b/file_or_directory~HASH (side1) > > > > + diff --git a/file_or_directory b/file_or_directory > > > > remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > > > diff --git a/letters b/letters > > > > remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). > > > > @@ -184,14 +185,13 @@ test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no dif > > > > test_cmp expect actual > > > > ' > > > > > > > > -test_expect_success 'remerge-diff w/ diff-filter=R: relevant file + conflict header' ' > > > > +test_expect_success 'remerge-diff w/ diff-filter=R: relevant file' ' > > > > git log -1 --oneline resolution >tmp && > > > > cat <<-EOF >>tmp && > > > > diff --git a/file_or_directory~HASH (side1) b/wanted_content > > > > similarity index 100% > > > > rename from file_or_directory~HASH (side1) > > > > rename to wanted_content > > > > - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. > > > > EOF > > > > # We still have some sha1 hashes above; rip them out so test works > > > > # with sha256 > > > > > > All of these look like regressions to me; you discussed these in a > > > separate email so I'll go into detail in response to that one. > > > > Thank you for being so diligent. I always appreciate it when I see a > > well-written, obviously well-edited mail. I find it very considerate. > > > > You did a good job at convincing me that my approach is sub-optimal, and > > that we need at least one new flag to get the information that I need to > > produce in my use case because the caller requires it to present a nice > > conflict resolution UI to the users. > > I'm not so sure you need additional information. Don't the > <Conflicted file info> and <Informational messages> provide enough > info already, for the usecases you are considering? In particular, > you were trying to add a testcase in your series for a simple > directory/file conflict, but for such a case you can already determine > original filenames. The <Conflicted file info> and <Informational > messages> sections of the `merge-tree -z` output for these, after > replacing NUL characters with newlines, would be: > > 120000 <HASH> 3 origfile > 100644 <HASH> 1 origfile~branch > 100644 <HASH> 2 origfile~branch > > 2 > origfile > origfile~branch > CONFLICT (distinct modes) > CONFLICT (distinct types): whatever had different types on each > side; renamed one of them so each can be recorded somewhere. > > Now, the "distinct modes" conflict always results in filename munging, > with both names listed. So, from this output, you know the original > name is "origfile" whenever you see "origfile~branch". So now you can > present all the `ls-files -u` output from the first section with > remapping munged filenames back to original filenames. > > Doesn't that solve your immediate usecase already, without even > needing any merge-ort changes? >
On Fri, Aug 26, 2022 at 8:35 AM Johannes Schindelin <Johannes.Schindelin@gmx.de> wrote: > > Hi Elijah, > > I'll top-post because I will still need much more time to wrap my head > around everything you said in your very thoughtful reply. > > By the way, have I mentioned before how much I appreciate well-edited, > well-crafted mails? It must have taken quite a bit of care and editing to > write this, and I just want to thank you for that. > > As to the problem we are discussing: I am fully on board with you on the > necessity to keep the door open for rename detection and recursive merges > and the zoo of conflict types that comes with it. > > After thinking about this for a couple of nights, I think I want to tackle > the design from the "other" side, namely from the UI. > > At this stage, I _think_ it would do well for me to give a more high-level > context as to my motivating factors. This is helpful, but I would really like a few concrete examples too. I tried to provide some in my last email with alternative guesses at what you might want. It'd be great to have an answer on those. I'll highlight them again at the end. > The thing is, these worktree-less merges do not live in a vacuum. They are > backing the Pull Request UI. There exists a relatively simple UI to > resolve simple merge conflicts, which factors into what I want from `git > merge-tree`. > > You've pointed out in the past, very rightfully, that the simple way to > "pre-resolve" conflicts that I have implemented over here is just not good > enough except for a very, very limited manner of driving `git merge-tree`. > These "pre-resolved conflicts" are currently essentially a map of <path> > -> <OID> pairs. > > Now, to design a much better way to present conflicts as well as allowing > users to resolve some of them (the simpler ones, like content conflicts) > in a web UI, we need `git merge-tree` to give us not an `ls-files > -u`-style output, that only works reliably for very simple cases. I agree we should not _only_ pay attention to `ls-files -u` info; that's too simplistic. And, in fact, that's the reason for having merge-tree provide another section of information. If we used both sets of information, we can handle more complex cases... > In general, I think the design would work best if we parsed the `messages` > part exclusively on the server side (ignoring the `ls-files`-like output, > probably even suppressing it) and if we received enough information to > recreate the conflicts from there. Which means that we need the involved > paths (whether munged or not, I am less and less concerned) together with > the modes and OIDs. It would probably be helpful even to provide a tree > OID for the directory part of any file/directory conflict, where the web > UI could then present the contents in a tree view to the user. By the `messages` part, do you mean the existing <Informational messages> section, which consists of multiple entries of the form <number-of-paths>NUL<path1>NUL<path2>NUL<pathN>NUL<stable-short-type-description>NUL<message>NUL ? Or do you mean the <message> part of each entry from that section, or do you perhaps mean something else? I'll assume for now you mean the full <Informational messages> section rather than some sub-component or something else, but let me know if I'm wrong. Also, why ignore or suppress the `ls-files -u`-like output? Doesn't it provide the modes and OIDs that you want? > This way, we could present multiple conflicts, even multiple conflicts for > the very same file, and allow resolving them individually. You seem to still be assuming separable conflicts? I tried to ask about two concrete examples in the last email with lots of different possibilities for what you might want to show the user (search for "Example 1" and "Example 2"; I'll also include them at the end of this email and snip out the other parts of the old email). I'm not sure either example is one where it makes sense for a user to attempt to resolve conflicts individually. Or is this more of a limited UI thing, where you want to search through the conflicts and affected paths, and if the conflicts aren't separable then you want to tell users to resolve the conflicts some other way, but proceed and allow users to use your UI if the conflicts are simple enough that they are separable? > And we could associate resolutions with the exact list of paths, modes and > OIDs involved, then teach `merge-tree` to optionally take such resolutions > into account, to produce merges with resolved conflicts. This seems like future work we could address another time...but perhaps I could follow the tangent for just a second. Why would it make sense to add conflict-resolution information to `merge-tree`? When you ran `merge-tree` the first time, it already provided you with a tree, and a list of conflicted files. If you have updates for those conflicted files, you can update the tree yourself without needing to redo any merges...right? Why would extending merge-tree and calling it again and having it repeat a merge even help? > Doing it this way will require substantial changes on the server side, in > particular in the UI, which currently does not have any support for > handling conflicts involving renames. It will be a lot of work, and I am > certain that I will have to pivot many times regarding the design of `git > merge-tree` features supporting that UI. I'm excited to start that > journey. > > What do you think? I'm open to going down new paths and even doing a lot of refactoring to get what we need, I'd just like to understand why they are needed first. In particular, I'm confused why you feel you should only ever use part of the "git merge-tree -z" output. Previously it seemed you were trying to use only `ls-files -u`-style output, now you're trying to use only the <Informational messages> section, and I'm trying to advocate for using both. I think you can already handle many of the cases you are concerned about if you do so, without any further merge-ort.c changes, and that the remaining cases should be addressable with just a few small tweaks to some rename-related path_msg() calls. But maybe I'm still not understanding something about the usecase, or there's something I'm missing about why the information provided really isn't sufficient to solve the cases you're currently facing. Could you look over the concrete examples I provided and give some feedback on what you want to do with them? That might point out what I'm missing. Let me quote the two relevant sections of my last email with these examples, in reverse order: > > I'm not so sure you need additional information. Don't the > > <Conflicted file info> and <Informational messages> provide enough > > info already, for the usecases you are considering? In particular, > > you were trying to add a testcase in your series for a simple > > directory/file conflict, but for such a case you can already determine > > original filenames. The <Conflicted file info> and <Informational > > messages> sections of the `merge-tree -z` output for these, after > > replacing NUL characters with newlines, would be: > > > > 120000 <HASH> 3 origfile > > 100644 <HASH> 1 origfile~branch > > 100644 <HASH> 2 origfile~branch > > > > 2 > > origfile > > origfile~branch > > CONFLICT (distinct modes) > > CONFLICT (distinct types): whatever had different types on each > > side; renamed one of them so each can be recorded somewhere. > > > > Now, the "distinct modes" conflict always results in filename munging, > > with both names listed. So, from this output, you know the original > > name is "origfile" whenever you see "origfile~branch". So now you can > > present all the `ls-files -u` output from the first section with > > remapping munged filenames back to original filenames. > > > > Doesn't that solve your immediate usecase already, without even > > needing any merge-ort changes? And, also the section of my previous email with the two more interesting/complicated examples: > > Can I ask in more detail what you want to present to a user? Maybe > > it'd help frame the conversation more. > > > > Let me provide two examples, each with lots of alternatives, to see if > > any are the kind of output you want to be able to present to a user > > (and assume in each case it's not just a block message but you have > > some kind of programmatic knowledge of the components referenced in > > the message): > > > > Example 1. (Updated note: if you want to see Example 1 created in test code, go to the en/t4301-more-merge-tree-tests topic, look at t4301, and find the "directory rename + rename/delete + modify/delete + directory/file conflict" testcase.) > > Are you thinking of something like "The following files were involved > > in a modify/delete conflict: foo, olddir/bar, newdir/bar, > > newdir/bar~B", followed by three other conflict messages that also involve > > those same files? > > > > Or, more like: "The following files were involved in a directory > > rename, and in a rename/delete conflict, and in a file/directory > > conflict, and in a modify/delete conflict: foo, olddir/bar, > > newdir/bar, newdir/bar~B" > > > > Or, more like: "The following files were involved in a directory > > rename, and in a rename/delete conflict, and in a file/directory > > conflict, and in a modify/delete conflict: foo (from side A), > > olddir/bar (from side B), newdir/bar (munged name), newdir/bar~B > > (munged name)" > > > > Or, more like: "The file newdir/bar~B has a modify/delete conflict. > > It was renamed to newdir/bar~B from newdir/bar because of a > > file/directory conflict. It was renamed to newdir/bar from olddir/bar > > because of a directory rename. It was renamed from foo to olddir/bar > > on side B. foo was deleted on side A." > > > > Or, more like: "foo was deleted on side A, and renamed to olddir/bar > > on side B. olddir/bar was renamed to newdir/bar because of a > > directory rename. newdir/bar was renamed to newdir/bar~B because of a > > file/directory conflict. newdir/bar~B has a modify/delete conflict." > > > > Or something completely different? > > > > > > Example 2. (Updated note: if you want to see a close neighbor to Example 2 in test code, go to the en/t4301-more-merge-tree-tests topic, look at t4301, and find the "mod6: chains of rename/rename(1to2) and add/add via colliding renames" testcase. That testcase is actually a bit more complex, having three intertwined sets of each of these three already tightly bound conflicts, reusing them in a way to end up with 6 total intertwined conflicts. But if you want to pick just one of the three intertwined sets, instead of "file-one" through "file-five", look at the files "two" through "six" from that testcase.) > > Are you thinking of something like: "The following files were involved > > in an add/add conflict: file-one, file-two, file-three, file-four, and > > file-five" ? > > > > Or, is it more like: "The following files were involved in an add/add > > conflict: file-one (from branch A), file-two (from merge base), > > file-three (from branch A), file-three (from branch B), file-four > > (from merge base), and file-five (from branch B)" ? > > > > Or, is it more like: "The following files were involved in an add/add > > conflict: file-three (from [see details]) and file-three (from [see > > details]). Note that file-three was part of a rename/rename(1to2) > > conflict, which involved file-one (from branch A), file-two (from > > merge base), and file-three (from branch B). Note that file-three was > > part of a rename/rename(1to2) conflict, which involved file-three > > (from branch A), file-four (from merge base), and file-five (from > > branch B)" > > > > Or, is it more like: "file-three was part of a rename/rename(1to2) > > conflict, which involved file-one (from branch A), file-two (from > > merge base), and file-three (from branch B). file-three was part of a > > rename/rename(1to2) conflict, which involved file-three (from branch > > A), file-four (from merge base), and file-five (from branch B). > > file-three and file-three are involved in an add/add conflict" > > > > (Of course, all of this example is assuming we're just focusing on the > > ultimate file-three vs file-three add/add conflict, not the additional > > conflicts that file-one and file-five may be involved in.) > > > > Or are you wanting to display something completely different?
diff --git a/merge-ort.c b/merge-ort.c index 01f150ef3b5..211f6823e1d 100644 --- a/merge-ort.c +++ b/merge-ort.c @@ -3741,6 +3741,7 @@ static void process_entry(struct merge_options *opt, struct conflict_info *ci, struct directory_versions *dir_metadata) { + const char *orig_path = path; int df_file_index = 0; VERIFY_CI(ci); @@ -3787,7 +3788,6 @@ static void process_entry(struct merge_options *opt, */ struct conflict_info *new_ci; const char *branch; - const char *old_path = path; int i; assert(ci->merged.result.mode == S_IFDIR); @@ -3838,10 +3838,10 @@ static void process_entry(struct merge_options *opt, strmap_put(&opt->priv->paths, path, new_ci); path_msg(opt, CONFLICT_FILE_DIRECTORY, 0, - path, old_path, NULL, NULL, + orig_path, path, NULL, NULL, _("CONFLICT (file/directory): directory in the way " "of %s from %s; moving it to %s instead."), - old_path, branch, path); + orig_path, branch, path); /* * Zero out the filemask for the old ci. At this point, ci @@ -3921,7 +3921,7 @@ static void process_entry(struct merge_options *opt, if (rename_a && rename_b) { path_msg(opt, CONFLICT_DISTINCT_MODES, 0, - path, a_path, b_path, NULL, + orig_path, a_path, b_path, NULL, _("CONFLICT (distinct types): %s had " "different types on each side; " "renamed both of them so each can " @@ -3929,7 +3929,7 @@ static void process_entry(struct merge_options *opt, path); } else { path_msg(opt, CONFLICT_DISTINCT_MODES, 0, - path, rename_a ? a_path : b_path, + orig_path, rename_a ? a_path : b_path, NULL, NULL, _("CONFLICT (distinct types): %s had " "different types on each side; " @@ -4022,7 +4022,7 @@ static void process_entry(struct merge_options *opt, if (S_ISGITLINK(merged_file.mode)) reason = _("submodule"); path_msg(opt, CONFLICT_CONTENTS, 0, - path, NULL, NULL, NULL, + orig_path, NULL, NULL, NULL, _("CONFLICT (%s): Merge conflict in %s"), reason, path); } @@ -4067,7 +4067,7 @@ static void process_entry(struct merge_options *opt, */ } else { path_msg(opt, CONFLICT_MODIFY_DELETE, 0, - path, NULL, NULL, NULL, + orig_path, path, NULL, NULL, _("CONFLICT (modify/delete): %s deleted in %s " "and modified in %s. Version %s of %s left " "in tree."), diff --git a/t/t4069-remerge-diff.sh b/t/t4069-remerge-diff.sh index 35f94957fce..bc580a242ac 100755 --- a/t/t4069-remerge-diff.sh +++ b/t/t4069-remerge-diff.sh @@ -131,11 +131,12 @@ test_expect_success 'setup non-content conflicts' ' test_expect_success 'remerge-diff with non-content conflicts' ' git log -1 --oneline resolution >tmp && cat <<-EOF >>tmp && + diff --git a/file_or_directory b/file_or_directory + remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. diff --git a/file_or_directory~HASH (side1) b/wanted_content similarity index 100% rename from file_or_directory~HASH (side1) rename to wanted_content - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. diff --git a/letters b/letters remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). diff --git a/letters_side2 b/letters_side2 @@ -168,7 +169,7 @@ test_expect_success 'remerge-diff with non-content conflicts' ' test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no diff content' ' git log -1 --oneline resolution >tmp && cat <<-EOF >>tmp && - diff --git a/file_or_directory~HASH (side1) b/file_or_directory~HASH (side1) + diff --git a/file_or_directory b/file_or_directory remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. diff --git a/letters b/letters remerge CONFLICT (rename/rename): letters renamed to letters_side1 in HASH (side1) and to letters_side2 in HASH (side2). @@ -184,14 +185,13 @@ test_expect_success 'remerge-diff w/ diff-filter=U: all conflict headers, no dif test_cmp expect actual ' -test_expect_success 'remerge-diff w/ diff-filter=R: relevant file + conflict header' ' +test_expect_success 'remerge-diff w/ diff-filter=R: relevant file' ' git log -1 --oneline resolution >tmp && cat <<-EOF >>tmp && diff --git a/file_or_directory~HASH (side1) b/wanted_content similarity index 100% rename from file_or_directory~HASH (side1) rename to wanted_content - remerge CONFLICT (file/directory): directory in the way of file_or_directory from HASH (side1); moving it to file_or_directory~HASH (side1) instead. EOF # We still have some sha1 hashes above; rip them out so test works # with sha256 diff --git a/t/t4301-merge-tree-write-tree.sh b/t/t4301-merge-tree-write-tree.sh index f091259a55e..e0ef9724b2b 100755 --- a/t/t4301-merge-tree-write-tree.sh +++ b/t/t4301-merge-tree-write-tree.sh @@ -211,8 +211,8 @@ test_expect_success 'NUL terminated conflicted file "lines"' ' q_to_nul <<-EOF >>expect && 1QgreetingQAuto-mergingQAuto-merging greeting Q1QgreetingQCONFLICT (contents)QCONFLICT (content): Merge conflict in greeting - Q2Qwhatever~tweak1QwhateverQCONFLICT (file/directory)QCONFLICT (file/directory): directory in the way of whatever from tweak1; moving it to whatever~tweak1 instead. - Q1Qwhatever~tweak1QCONFLICT (modify/delete)QCONFLICT (modify/delete): whatever~tweak1 deleted in side2 and modified in tweak1. Version tweak1 of whatever~tweak1 left in tree. + Q2QwhateverQwhatever~tweak1QCONFLICT (file/directory)QCONFLICT (file/directory): directory in the way of whatever from tweak1; moving it to whatever~tweak1 instead. + Q2QwhateverQwhatever~tweak1QCONFLICT (modify/delete)QCONFLICT (modify/delete): whatever~tweak1 deleted in side2 and modified in tweak1. Version tweak1 of whatever~tweak1 left in tree. Q1QΑυτά μου φαίνονται κινέζικαQAuto-mergingQAuto-merging Αυτά μου φαίνονται κινέζικα Q1QΑυτά μου φαίνονται κινέζικαQCONFLICT (contents)QCONFLICT (content): Merge conflict in Αυτά μου φαίνονται κινέζικα Q