> > 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> > > I'm still a little unsure on this. The merge_submodule() logic we > have may not have detected a merge commit that merges the two branches > together, but that doesn't automatically imply a new merge must be > created in the submodule to resolve this conflict. There might be > various reasons that other existing commits in the submodule could be > used as the "correct" update. > > Perhaps that is uncommon enough to not be worth mentioning; I'm just a > little hesitant to treat the merge_submodule() logic as robust and > finding the only choices users might want to use. If we do keep the > wording as-is, it might be nice to at least discuss in the commit > message why we chose that and ignored the or-update-submodule option > in this case. You make a good point about merge_submodule() possibly not finding all of the right choices -- I'll add back the or-update-submodule option back. > > If this is correct simply add it to the index for example > > by using: > > > > git update-index --cacheinfo 160000 <commit> "<submodule>" > > > > which will accept this suggestion. > > The last few lines above will no longer be part of the output once > en/merge-tree is merged; see commit a4040cfa35 (merge-ort: remove > command-line-centric submodule message from merge-ort, 2022-06-18). ack > > CONFLICT (submodule): Merge conflict in <submodule> > > Recursive merging with submodules is currently not supported. > > To manually complete the merge: > > - go to submodule (<submodule>), and either update the submodule to a possible commit above or merge commit <commit> > > Again, I'm hesitant to treat the suggestions from merge_submodule() as > the only possibilities. For example, perhaps the user will want to > pick a commit that contains one of the ones found by merge_submodule() > in its history -- perhaps because the newer commit they want to select > contains an important bugfix for an issue not discovered at the time > of the merge in the submodule. > > Also, I'm worried your sentence may be easy to misunderstand, due to > it being ambiguous whether "merge" is a verb or an adjective. More > precisely, your sentence could be read as "update the submodule to a > possible commit above, or update the submodule to merge commit > <commit>" and then users say, "But <commit> isn't a merge commit; why > does this message claim it is? Do I just update the submodule to that > commit anyway?" Or perhaps users just update it to <commit> without > even checking to find out that it's not a merge commit, with the > deleterious consequences of missing all the important changes > currently found in the submodule. I agree this sentence can be misinterpreted. I also think the main reason my current message is unclear is because there is not enough context for the user to understand why the merge failed. In a standard merge, the reason in "CONFLICT (<reason>)" provides context, whereas in this case, "CONFLICT (submodule)" only tells the user that the submodule is conflicted in some way. The user is unaware that we attempted to fast-forward the submodule, failed, and now require the user to either update the commit or merge the commit. How does this rewording sound? Automatic merge failed; recursive merging with submodules currently only supports fast-forward merges. For each conflicted submodule, if the current commit does not reflect the desired state, either update or merge the commit. This can be accomplished with the following steps: - go to submodule (<submodule>), and either update the commit or merge commit <commit> > > - come back to superproject, and `git add <submodule>` to record the above merge > > "...to record the above merge or update"? will... "update" haha > > If git detects multiple possible merge resolutions, the following is printed: > > > > -------- > > > > Failed to merge submodule sub, but multiple possible merges exist: > > <commit> Merge branch '<branch1>' into <branch2> > > <commit> Merge branch '<branch1>' into <branch3> > > > > CONFLICT (submodule): Merge conflict in <submodule> > > Recursive merging with submodules is currently not supported. > > To manually complete the merge: > > - go to submodule (<submodule>), and either update the submodule to a possible commit above or merge commit <commit> > > Same issues as I mentioned above for the single-merge-resolution-found case. > ditto > > - come back to superproject, and `git add <submodule>` to record the above merge > > "merge or update", right? ditto > > +ret: > > + if (!ret) { > > + if (!csub) { > > + CALLOC_ARRAY(csub, 1); > > + } > > + csub_item.oid = *result; > > Shouldn't this be "*b" rather than "*result"? Yes it should > > Also, are we risking telling the user to "merge commit > 0000000000000000000000000000000000000000" from the submodule, given > some of the early exits that you redirected to come through here? You are correct, but I'm not quite sure what should happen in this case. What does it mean for either o, a, or b to be null? Did a submodule get deleted on one side? > > +void print_submodule_conflict_suggestion(struct conflicted_submodule_list *csub) { > > Maybe just make this function be static? It should be static now that this won't be called in merge-recursive > > + find_unique_abbrev(&id, DEFAULT_ABBREV)); > > Shouldn't this be repo_find_unique_abbrev(), and in particular with > the submodule repository being passed to it? Or perhaps using the > format_commit() function, since merge_submodule() is already using it > for the earlier submodule-related messages? It should and I will go with the format_commit() option so I don't have to pass subrepo information into the print function. > > void merge_switch_to_result(struct merge_options *opt, > > struct tree *head, > > struct merge_result *result, > > @@ -4324,6 +4367,8 @@ void merge_switch_to_result(struct merge_options *opt, > > } > > string_list_clear(&olist, 0); > > > > + print_submodule_conflict_suggestion(opt->conflicted_submodules); > > + > > /* Also include needed rename limit adjustment now */ > > diff_warn_rename_limit("merge.renamelimit", > > opti->renames.needed_limit, 0); > > diff --git a/merge-recursive.c b/merge-recursive.c > > index fd1bbde061..311cc37886 100644 > > --- a/merge-recursive.c > > +++ b/merge-recursive.c > > Is it worth updating merge-recursive.c here? I'd rather only give it > security fix updates until we delete it. Ah wasn't aware that was the status of merge-recursive. Will delete. > > diff --git a/merge-recursive.h b/merge-recursive.h > > index b88000e3c2..b615955fb7 100644 > > --- a/merge-recursive.h > > +++ b/merge-recursive.h > > @@ -9,6 +9,7 @@ struct object_id; > > struct repository; > > struct tree; > > > > +struct conflicted_submodule_list; > > struct merge_options_internal; > > struct merge_options { > > struct repository *repo; > > @@ -51,6 +52,21 @@ struct merge_options { > > > > /* internal fields used by the implementation */ > > struct merge_options_internal *priv; > > + > > + /* field that holds submodule conflict information */ > > + struct conflicted_submodule_list *conflicted_submodules; > > I think this should be added to merge_options_internal rather than to > merge_options. There's no need for an API caller to make use of > these. > > Also, if a clear need arises later for API callers to make use of this > information, then it should be part of merge_result for merge-ort.c, > not part of merge_options. ack > > +}; > > + > > +struct conflicted_submodule_item { > > + struct object_id oid; > > + char *path; > > + int resolution_exists; > > +}; > > + > > +struct conflicted_submodule_list { > > + struct conflicted_submodule_item *items; > > + size_t nr; > > + size_t alloc; > > }; > > Similarly, I think these should be defined in merge-ort.c (and maybe > also merge-recursive.c) near struct merge_options_internal. ack > > void init_merge_options(struct merge_options *opt, struct repository *repo); > > @@ -122,4 +138,6 @@ int merge_recursive_generic(struct merge_options *opt, > > const struct object_id **merge_bases, > > struct commit **result); > > > > +void print_submodule_conflict_suggestion(struct conflicted_submodule_list *csub); > > and I think there's no reason to put this in the header file; it > should just be a static function in merge-ort.c. (And, if we really > want to update merge-recursive.c, then copy the function over there. > *Nothing* in merge-ort.c should call a function in merge-recursive.c > and similarly nothing in merge-recursive.c should call any function in > merge-ort.c. Yes, that implies some duplication. There is a fair > amount already and we've discussed it, and chosen against creating a > shared module as well. I want merge-recursive to not be broken by > merge-ort updates; it should remain stable until the day we finally > get to nuke it. I'm personally of the opinion that merge-recursive > should only get security fixes at this point, though I haven't pinged > to see if other folks agree with that point of view yet or not. I'm > not wasting my time fixing/improving it, though...) Good to know going forward I should only update merge-ort (unless for security fixes). I'll keep this in mind refactoring my patch. > Despite my many nitpicks and whatnot, it looks like your change will > make things nicer for the user, and I think your patch is coming along > nicely. There are some things to fix up, but the overall direction > seems good. Thank you for all of the helpful feedback!