> I think we would want to be slightly more precise here, as > "conflicts in the submodules" could be understood to mean: > 1) conflicting submodule pointers in the superproject being merged > 2) 1. + also content conflicts in the submodule merge > Here we are just talking about 1. Will clarify > Also, the merge does not automatically fail, it only fails if > a fast-forward *of the submodule pointer* is not possible, which > might be what you meant above; but to me "a non-fast-forward merge in a project > with conflict in the submodules" read like the non-fast-forwardness being talked about > was in the superproject, not in the submodule(s). It was what I meant above, but I do agree that I can reword this section as well. > I agree with Elijah here, the submodule conflcit resolution might be to: > 1) just choose one of the existing submodule commits on either side of > the superproject branches being merged > 2) choose an exisiting merge commit in the submodule repository (maybe after fetching it first) > 3) create such a merge commit (what you are talking about here) > I also agree that it is highly repository- and workflow-dependent what > the "right" resolution is. I replied to Elijah's message with some ideas I had about updating my first step, but to summarize I do agree that my current message doesn't account for the fact that a merge resolution may already exist, and the test cases you mention serve as proof that my current message is insufficient. > I think moving it to merge_finalize / merge_switch_to_result is indeed a > good suggestion, then we might be improving the UX across the board and not just > for 'git merge'. I'll spend some time looking into moving the suggestion code into these functions. My main concern is knowing which tests I would have to update to test these changes. On Sat, Jun 11, 2022 at 10:08 AM Philippe Blain <levraiphilippeblain@xxxxxxxxx> wrote: > > Hi Calvin and Elijah, > > Le 2022-06-11 à 00:53, Elijah Newren a écrit : > > On Fri, Jun 10, 2022 at 4:29 PM Calvin Wan <calvinwan@xxxxxxxxxx> wrote: > >> > >> When attempting to do a non-fast-forward merge in a project with > >> conflicts in the submodules, the merge fails and git prints the > >> following error: > > I think we would want to be slightly more precise here, as > "conflicts in the submodules" could be understood to mean: > > 1) conflicting submodule pointers in the superproject being merged > 2) 1. + also content conflicts in the submodule merge > > Here we are just talking about 1. > > Also, the merge does not automatically fail, it only fails if > a fast-forward *of the submodule pointer* is not possible, which > might be what you meant above; but to me "a non-fast-forward merge in a project > with conflict in the submodules" read like the non-fast-forwardness being talked about > was in the superproject, not in the submodule(s). > > [message 0] > >> > >> Failed to merge submodule <submodules> > >> CONFLICT (submodule):Merge conflict in <submodule> > >> Automatic merge failed; fix conflicts and then commit the result. > > Aside: the first <submodules> should be singular. > > This is indeed the output you get with the ort strategy if no existing merge commit > exist in the submodule repository that merges the submodule pointers recorded > in the superproject branches being merged. With the older "recursive" strategy, > this message is: > > [message 1] > Failed to merge submodule sub (merge following commits not found) > Auto-merging sub > CONFLICT (submodule): Merge conflict in sub > Automatic merge failed; fix conflicts and then commit the result > > c73cda76b1 (merge-ort: copy and adapt merge_submodule() > from merge-recursive.c, 2021-01-01) does not mention why that error message > was changed, but perhaps it is just because it is slightly confusing > to the user (they might not be expecting Git to look for an existing > merge and so they don't know what merge the message is talking about). > > Maybe something like "failed to find existing commit merging <hash1> and <hash2>" > would be clearer... > > >> > >> Git is left in a conflicted state, which requires the user to: > >> 1. merge submodules > >> 2. add submodules changes to the superproject > >> 3. merge superproject > > > > I think we may need to tweak these steps a bit: > > > > 1. merge submodules OR update submodules to an already existing > > commit that reflects the merge (i.e. as submodules they may well be > > independently actively developed. Someone may have already merged the > > appropriate branch/commit and the already extant merges should be used > > in preference to creating new ones.) > > 2. <just as you said> > > 3. FINISH merging the superproject (i.e. don't redo the merge) > > > > I might be off on step 1; I have only used submodules extremely > > lightly and usually only for a limited time, so I'm not really sure > > what the expected workflow is. I could also imagine it potentially > > being repository-dependent whether you would want to merge or select > > an appropriate commit to update to. > > I agree with Elijah here, the submodule conflcit resolution might be to: > > 1) just choose one of the existing submodule commits on either side of > the superproject branches being merged > 2) choose an exisiting merge commit in the submodule repository (maybe after fetching it first) > 3) create such a merge commit (what you are talking about here) > > I also agree that it is highly repository- and workflow-dependent what > the "right" resolution is. > > Note that the code does try to find an existing merge commit in the submodule > repository, in this case the error message is different. If such a merge commit > exists: > > [message 2] > Failed to merge submodule sub, but a possible merge resolution exists: > aafcfa2 Merge branch 'sub-c' into sub-d > > > If this is correct simply add it to the index for example > by using: > > git update-index --cacheinfo 160000 aafcfa2a62764282ab848d5d6bea86ba217c1b24 "sub" > > which will accept this suggestion. > > CONFLICT (submodule): Merge conflict in sub > Automatic merge failed; fix conflicts and then commit the result. > > if multiple merge exist: > > [message 3] > Failed to merge submodule sub, but multiple possible merges exist: > 2729a0c Merge branch 'sub-c' into ambiguous > aafcfa2 Merge branch 'sub-c' into sub-d > > CONFLICT (submodule): Merge conflict in sub > Automatic merge failed; fix conflicts and then commit the result. > > Another aside, I really don't think we should instruct users to run > plumbing like 'git update-index --cacheinfo' , they should just cd into > the submodule and checkout the merge commit! > > > > >> These steps are non-obvious for newer submodule users to figure out > >> based on the error message and neither `git submodule status` nor `git > >> status` provide any useful pointers. > >> > >> Update error message to the following when attempting to do a > >> non-fast-forward merge in a project with conflicts in the submodules. > > > > Make sense. > > I agree that more guidance is a very nice addition. > > Regarding 'git status' output, it is downright confusing, since it says: > > Unmerged paths: > (use "git add <file>..." to mark resolution) > both modified: sub > > which is not at all what you want to do most of the time (that would > just stage whatever the currently checked out commit in the submodule is > at the moment!) > > > > >> The error message is based off of what would happen when `merge > >> --recurse-submodules` is eventually supported > >> > >> Failed to merge submodule <submodule> > >> CONFLICT (submodule): Merge conflict in <submodule> > >> Automatic merge failed; recursive merging with submodules is currently > >> not supported. To manually complete the merge: > >> - go to submodule (<submodule>), and merge commit <commit> > >> - come back to superproject, and `git add <submodule>` to record the above merge > >> - resolve any other conflicts in the superproject > >> - commit the resulting index in the superproject > > > > Ah, I see you've fixed step 3 here; that's good. > > > > However, these steps miss out on the merge-or-update submodule > > possibility...and since you mention these steps are potentially the > > basis for some future work, I think it's worth calling that out again. > > I'm slightly worried that the 'update' part of merge-or-update may > > throw a wrench in the plans for `merge --recurse-submodules`. > > > > Slightly off topic here, but for me the most important improvement that > 'git merge --recurse-submodules' would bring is when there is *no submodule > conflicts*, i.e. one side fast-forwards the submodule and the other side > does not touch it, since in that case the worktree of the submodule *is not updated* > by the current code, which is one of the most confusing aspect of using submodules > for new users ("why is "git status" and "git diff" not clean if "git merge" > was fast-forward ?!?"), and the same is true (maybe more even so) for 'git rebase'. > > > (Also, continuing on the `merge --recurse-submodules` talent but > > discussing a different aspect of it, I'm curious if you need to add > > extra dirty-worktree/dirty-index checks for each submodule at the > > start of a merge, whether you need to try to lock N indexes before > > starting, and what other extra details are necessary. But those are > > probably questions to address whenever work on the future series to > > implement this option is underway.) > > > >> Changes since v1: > >> - Removed advice to abort merge > >> - Error message updated to contain more commit/submodule information > >> > >> Signed-off-by: Calvin Wan <calvinwan@xxxxxxxxxx> > >> > >> --- > >> builtin/merge.c | 23 ++++++++++++++++++++++- > >> merge-ort.c | 7 ++++++- > >> merge-recursive.c | 7 ++++++- > >> merge-recursive.h | 4 ++++ > >> t/t6437-submodule-merge.sh | 5 ++++- > >> 5 files changed, 42 insertions(+), 4 deletions(-) > > > > So you're modifying the "git merge" porcelain level (builtin/merge.c), > > the two merges strategies, their common header, and adding some tests. > > No other porcelains are modified... > > > >> diff --git a/builtin/merge.c b/builtin/merge.c > >> index f178f5a3ee..7e2deea7fb 100644 > >> --- a/builtin/merge.c > >> +++ b/builtin/merge.c > >> @@ -88,6 +88,8 @@ static const char *sign_commit; > >> static int autostash; > >> static int no_verify; > >> static char *into_name; > >> +static struct oid_array conflicted_submodule_oids = OID_ARRAY_INIT; > >> +static struct string_list conflicted_submodule_paths = STRING_LIST_INIT_DUP; > >> > >> static struct strategy all_strategy[] = { > >> { "recursive", NO_TRIVIAL }, > >> @@ -734,6 +736,8 @@ static int try_merge_strategy(const char *strategy, struct commit_list *common, > >> } > >> > >> init_merge_options(&o, the_repository); > >> + o.conflicted_submodule_oids = &conflicted_submodule_oids; > >> + o.conflicted_submodule_paths = &conflicted_submodule_paths; > >> if (!strcmp(strategy, "subtree")) > >> o.subtree_shift = ""; > >> > >> @@ -973,8 +977,25 @@ static int suggest_conflicts(void) > >> strbuf_release(&msgbuf); > >> fclose(fp); > >> repo_rerere(the_repository, allow_rerere_auto); > >> - printf(_("Automatic merge failed; " > >> + if (conflicted_submodule_oids.nr > 0) { > >> + int i; > >> + printf(_("Automatic merge failed; recursive merging with submodules is currently\n" > >> + "not supported. To manually complete the merge:\n")); > >> + for (i = 0; i < conflicted_submodule_oids.nr; i++) { > >> + printf(_(" - go to submodule (%s), and merge commit %s\n"), > >> + conflicted_submodule_paths.items[i].string, > >> + oid_to_hex(&conflicted_submodule_oids.oid[i])); > >> + } > >> + printf(_(" - come back to superproject, and `git add")); > >> + for (i = 0; i < conflicted_submodule_paths.nr; i++) > >> + printf(_(" %s"), conflicted_submodule_paths.items[i].string); > >> + printf(_("` to record the above merge \n" > >> + " - resolve any other conflicts in the superproject\n" > >> + " - commit the resulting index in the superproject\n")); > >> + } else { > >> + printf(_("Automatic merge failed; " > >> "fix conflicts and then commit the result.\n")); > >> + } > >> return 1; > >> } > > > > This is kind of nice. I was worried you were going to embed these > > messages in the merge strategies, which could cause problems for other > > users of the merge strategies such as the --remerge-diff options to > > git log and git show (your new messages would be unwanted noise or > > even cause confusion there), and to the merge-tree work. In fact, a > > current submodule-merging message (search for "--cacheinfo") that is > > potentially similar to what you are adding here but which was added at > > the merge strategy level already feels highly problematic to me. I've > > been considering nuking it from the codebase for some time because of > > those issues, though I guess just moving it out elsewhere may also > > work. > > > > Yes, this is the message I copied above. I agree that if we can tweak this > advice to instead mention 'git checkout' and add it to the message > that Calvin is adding in this series, it would make for a really better > UX. > > > However, this implementation does have a drawback: these messages > > won't appear for rebases, cherry-picks, reverts, attempted unstashing > > (git stash apply/pop), or other actions unless you update the relevant > > porcelains for those as well. > > > > A possible alternative here would be to move it to the level of > > merge-recursive and merge-ort that is only called when the working > > tree and index are updated. For example, placing it in > > merge_finalize() in merge-recursive.c and merge_switch_to_result() in > > merge-ort.c -- next to the diff_warn_rename_limit() call in each case. > > However, I'm also fine with keeping it at the porcelain level, it just > > may need to be in a function that is called from several porcelains > > that way. > > I think moving it to merge_finalize / merge_switch_to_result is indeed a > good suggestion, then we might be improving the UX across the board and not just > for 'git merge'. > > > > >> diff --git a/merge-ort.c b/merge-ort.c > >> index 0d3f42592f..c86ee11614 100644 > >> --- a/merge-ort.c > >> +++ b/merge-ort.c > >> @@ -3866,8 +3866,13 @@ static void process_entry(struct merge_options *opt, > >> const char *reason = _("content"); > >> if (ci->filemask == 6) > >> reason = _("add/add"); > >> - if (S_ISGITLINK(merged_file.mode)) > >> + if (S_ISGITLINK(merged_file.mode)) { > >> reason = _("submodule"); > >> + if (opt->conflicted_submodule_oids && opt->conflicted_submodule_paths) { > >> + oid_array_append(opt->conflicted_submodule_oids, &merged_file.oid); > >> + string_list_append(opt->conflicted_submodule_paths, path); > >> + } > >> + } > >> path_msg(opt, path, 0, > >> _("CONFLICT (%s): Merge conflict in %s"), > >> reason, path); > >> diff --git a/merge-recursive.c b/merge-recursive.c > >> index fd1bbde061..ff7cdbefe9 100644 > >> --- a/merge-recursive.c > >> +++ b/merge-recursive.c > >> @@ -3149,8 +3149,13 @@ static int handle_content_merge(struct merge_file_info *mfi, > >> } > >> > >> if (!mfi->clean) { > >> - if (S_ISGITLINK(mfi->blob.mode)) > >> + if (S_ISGITLINK(mfi->blob.mode)) { > >> reason = _("submodule"); > >> + if (opt->conflicted_submodule_oids && opt->conflicted_submodule_paths) { > >> + oid_array_append(opt->conflicted_submodule_oids, &mfi->blob.oid); > >> + string_list_append(opt->conflicted_submodule_paths, path); > >> + } > >> + } > >> output(opt, 1, _("CONFLICT (%s): Merge conflict in %s"), > >> reason, path); > >> if (ci && !df_conflict_remains) > > > > Nice that the changes needed to both the ort and recursive strategies > > are so localized. :-) > > > >> diff --git a/merge-recursive.h b/merge-recursive.h > >> index b88000e3c2..5d267e7a43 100644 > >> --- a/merge-recursive.h > >> +++ b/merge-recursive.h > >> @@ -51,6 +51,10 @@ struct merge_options { > >> > >> /* internal fields used by the implementation */ > >> struct merge_options_internal *priv; > >> + > >> + /* fields that hold submodule conflict information */ > >> + struct oid_array *conflicted_submodule_oids; > >> + struct string_list *conflicted_submodule_paths; > >> }; > > > > Make sense. > > > >> void init_merge_options(struct merge_options *opt, struct repository *repo); > >> diff --git a/t/t6437-submodule-merge.sh b/t/t6437-submodule-merge.sh > >> index 178413c22f..5b384dedc1 100755 > >> --- a/t/t6437-submodule-merge.sh > >> +++ b/t/t6437-submodule-merge.sh > >> @@ -141,6 +141,7 @@ test_expect_success 'merging should conflict for non fast-forward' ' > >> test_must_fail git merge c 2> actual > >> fi && > >> grep $(cat expect) actual > /dev/null && > >> + test_i18ngrep "go to submodule (sub), and merge commit $(git -C sub rev-parse sub-b)" actual && > >> git reset --hard) > >> ' > >> > > In this test, an existing merge does exist, and is suggested to the user with [message 2] above. > So telling users to do a new merge in the submodule is maybe to what we want. > > >> @@ -167,6 +168,7 @@ test_expect_success 'merging should fail for ambiguous common parent' ' > >> fi && > >> grep $(cat expect1) actual > /dev/null && > >> grep $(cat expect2) actual > /dev/null && > >> + test_i18ngrep "go to submodule (sub), and merge commit $(git -C sub rev-parse sub-b)" actual && > >> git reset --hard) > >> ' > > Here, 2 existing merges exist, and they are presented to the users with [message 3] above. > So again we might not want to tell users to do a new merge. > > >> > >> @@ -205,7 +207,8 @@ test_expect_success 'merging should fail for changes that are backwards' ' > >> git commit -a -m "f" && > >> > >> git checkout -b test-backward e && > >> - test_must_fail git merge f) > >> + test_must_fail git merge f >actual && > >> + test_i18ngrep "go to submodule (sub), and merge commit $(git -C sub rev-parse sub-a)" actual) > > > > test_i18ngrep is apparently on the way out: > > > > $ grep -B 3 ^test_i18ngrep t/test-lib-functions.sh > > # Wrapper for grep which used to be used for > > # GIT_TEST_GETTEXT_POISON=false. Only here as a shim for other > > # in-flight changes. Should not be used and will be removed soon. > > test_i18ngrep () { > > > > I think you just want to use grep instead here for each of these hunks. > > > > Here, one side regresses the submodule commit and the other side advances it. > In this case, I really think the right resolution is to choose one side or > the other, and not suggest to do a merge at all. So that means that we might > want to tweak the advice we give based on the type of submodule conflict... > > Also, maybe we would want a new test that reproduces exactly the conditions of > [message 1], i.e. no existing merge exists in the submodule. > > Thanks a lot for wanting to improve the submodule UX! > > Cheers, > > Philippe.