rerere: recalculate conflict ID when unresolved conflict is committed
authorThomas Gummerer <t.gummerer@gmail.com>
Sun, 5 Aug 2018 17:20:37 +0000 (18:20 +0100)
committerJunio C Hamano <gitster@pobox.com>
Mon, 6 Aug 2018 20:22:35 +0000 (13:22 -0700)
Currently when a user doesn't resolve a conflict, commits the results,
and does an operation which creates another conflict, rerere will use
the ID of the previously unresolved conflict for the new conflict.
This is because the conflict is kept in the MERGE_RR file, which
'rerere' reads every time it is invoked.

After the new conflict is solved, rerere will record the resolution
with the ID of the old conflict. So in order to replay the conflict,
both merges would have to be re-done, instead of just the last one, in
order for rerere to be able to automatically resolve the conflict.

Instead of that, assign a new conflict ID if there are still conflicts
in a file and the file had conflicts at a previous step. This ID
matches the conflict we actually resolved at the corresponding step.

Note that there are no backwards compatibility worries here, as rerere
would have failed to even normalize the conflict before this patch
series.

Signed-off-by: Thomas Gummerer <t.gummerer@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
rerere.c
t/t4200-rerere.sh
index f78bef80b1119d72810f4424196cb55f8d30f11d..dd81d09e1915c6581eb4381ff9d8ca5592557aad 100644 (file)
--- a/rerere.c
+++ b/rerere.c
@@ -815,7 +815,7 @@ static int do_plain_rerere(struct string_list *rr, int fd)
                struct rerere_id *id;
                unsigned char sha1[20];
                const char *path = conflict.items[i].string;
                struct rerere_id *id;
                unsigned char sha1[20];
                const char *path = conflict.items[i].string;
-               int ret, has_string;
+               int ret;
 
                /*
                 * Ask handle_file() to scan and assign a
 
                /*
                 * Ask handle_file() to scan and assign a
@@ -823,12 +823,11 @@ static int do_plain_rerere(struct string_list *rr, int fd)
                 * yet.
                 */
                ret = handle_file(path, sha1, NULL);
                 * yet.
                 */
                ret = handle_file(path, sha1, NULL);
-               has_string = string_list_has_string(rr, path);
-               if (ret < 0 && has_string) {
+               if (ret != 0 && string_list_has_string(rr, path)) {
                        remove_variant(string_list_lookup(rr, path)->util);
                        string_list_remove(rr, path, 1);
                }
                        remove_variant(string_list_lookup(rr, path)->util);
                        string_list_remove(rr, path, 1);
                }
-               if (ret < 1 || has_string)
+               if (ret < 1)
                        continue;
 
                id = new_rerere_id(sha1);
                        continue;
 
                id = new_rerere_id(sha1);
index afaf085e4260d38d4608da938b46015974c5530a..819f6dd67291146a8fbc047399573ed151f8f465 100755 (executable)
@@ -635,6 +635,13 @@ test_expect_success 'rerere with inner conflict markers' '
        git commit -q -m "will solve conflicts later" &&
        test_must_fail git merge A &&
        cat test >actual &&
        git commit -q -m "will solve conflicts later" &&
        test_must_fail git merge A &&
        cat test >actual &&
+       test_cmp expect actual &&
+
+       git add test &&
+       git commit -m "rerere solved conflict" &&
+       git reset --hard HEAD~ &&
+       test_must_fail git merge A &&
+       cat test >actual &&
        test_cmp expect actual
 '
 
        test_cmp expect actual
 '