]> git.ipfire.org Git - thirdparty/git.git/commitdiff
rebase -i: check labels and refs when parsing todo list
authorPhillip Wood <phillip.wood@dunelm.org.uk>
Mon, 20 Feb 2023 14:19:34 +0000 (14:19 +0000)
committerJunio C Hamano <gitster@pobox.com>
Tue, 21 Feb 2023 17:18:37 +0000 (09:18 -0800)
Check that the argument to the "label" and "update-ref" commands is a
valid refname when the todo list is parsed rather than waiting until the
command is executed. This means that the user can deal with any errors
at the beginning of the rebase rather than having it stop halfway
through due to a typo in a label name. The "update-ref" command is
changed to reject single level refs as it is all to easy to type
"update-ref branch" which is incorrect rather than "update-ref
refs/heads/branch"

Note that it is not straight forward to check the arguments to "reset"
and "merge" commands as they may be any revision, not just a refname and
we do not have an equivalent of check_refname_format() for revisions.

Helped-by: Derrick Stolee <derrickstolee@github.com>
Signed-off-by: Phillip Wood <phillip.wood@dunelm.org.uk>
Acked-by: Derrick Stolee <derrickstolee@github.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
sequencer.c
t/t3404-rebase-interactive.sh

index fb23f734ade4902f6472d9cf0db1155389f02fa1..c9ca83204b26d73d42c705803d6097dff3f20434 100644 (file)
@@ -2487,6 +2487,34 @@ static int is_command(enum todo_command command, const char **bol)
                 (*bol = p));
 }
 
+static int check_label_or_ref_arg(enum todo_command command, const char *arg)
+{
+       switch (command) {
+       case TODO_LABEL:
+               /*
+                * '#' is not a valid label as the merge command uses it to
+                * separate merge parents from the commit subject.
+                */
+               if (!strcmp(arg, "#") ||
+                   check_refname_format(arg, REFNAME_ALLOW_ONELEVEL))
+                       return error(_("'%s' is not a valid label"), arg);
+               break;
+
+       case TODO_UPDATE_REF:
+               if (check_refname_format(arg, REFNAME_ALLOW_ONELEVEL))
+                       return error(_("'%s' is not a valid refname"), arg);
+               if (check_refname_format(arg, 0))
+                       return error(_("update-ref requires a fully qualified "
+                                      "refname e.g. refs/heads/%s"), arg);
+               break;
+
+       default:
+               BUG("unexpected todo_command");
+       }
+
+       return 0;
+}
+
 static int parse_insn_line(struct repository *r, struct todo_item *item,
                           const char *buf, const char *bol, char *eol)
 {
@@ -2535,10 +2563,19 @@ static int parse_insn_line(struct repository *r, struct todo_item *item,
 
        if (item->command == TODO_EXEC || item->command == TODO_LABEL ||
            item->command == TODO_RESET || item->command == TODO_UPDATE_REF) {
+               int ret = 0;
+
                item->commit = NULL;
                item->arg_offset = bol - buf;
                item->arg_len = (int)(eol - bol);
-               return 0;
+               if (item->command == TODO_LABEL ||
+                   item->command == TODO_UPDATE_REF) {
+                       saved = *eol;
+                       *eol = '\0';
+                       ret = check_label_or_ref_arg(item->command, bol);
+                       *eol = saved;
+               }
+               return ret;
        }
 
        if (item->command == TODO_FIXUP) {
index 462cefd25df3efd51b06023523530b3d97d3d3a9..efeb74ad50a9efb74ec689311efbb9e72dc3141e 100755 (executable)
@@ -2072,6 +2072,7 @@ test_expect_success '--update-refs: --edit-todo with no update-ref lines' '
 '
 
 test_expect_success '--update-refs: check failed ref update' '
+       test_when_finished "test_might_fail git rebase --abort" &&
        git checkout -B update-refs-error no-conflict-branch &&
        git branch -f base HEAD~4 &&
        git branch -f first HEAD~3 &&
@@ -2123,6 +2124,28 @@ test_expect_success '--update-refs: check failed ref update' '
        test_cmp expect err.trimmed
 '
 
+test_expect_success 'bad labels and refs rejected when parsing todo list' '
+       test_when_finished "test_might_fail git rebase --abort" &&
+       cat >todo <<-\EOF &&
+       exec >execed
+       label #
+       label :invalid
+       update-ref :bad
+       update-ref topic
+       EOF
+       rm -f execed &&
+       (
+               set_replace_editor todo &&
+               test_must_fail git rebase -i HEAD 2>err
+       ) &&
+       grep "'\''#'\'' is not a valid label" err &&
+       grep "'\'':invalid'\'' is not a valid label" err &&
+       grep "'\'':bad'\'' is not a valid refname" err &&
+       grep "update-ref requires a fully qualified refname e.g. refs/heads/topic" \
+               err &&
+       test_path_is_missing execed
+'
+
 # This must be the last test in this file
 test_expect_success '$EDITOR and friends are unchanged' '
        test_editor_unchanged