]> git.ipfire.org Git - thirdparty/git.git/commitdiff
checkout-index: propagate errors to exit code
authorJeff King <peff@peff.net>
Tue, 27 Oct 2020 07:37:14 +0000 (03:37 -0400)
committerJunio C Hamano <gitster@pobox.com>
Tue, 27 Oct 2020 19:41:56 +0000 (12:41 -0700)
If we encounter an error while checking out an explicit path, we print a
message to stderr but do not actually exit with a non-zero code. While
this is a plumbing command and the behavior goes all the way back to
33db5f4d90 (Add a "checkout-cache" command which does what the name
suggests., 2005-04-09), this is almost certainly an oversight:

  - we _do_ return an exit code from checkout_file(); the caller just
    never reads it

  - errors while checking out all paths (with "-a") do result in a
    non-zero exit code.

  - it would be quite unusual not to use the exit code for an error,
    as otherwise the caller has no idea the command failed except by
    scraping stderr

To keep our tests simple and portable, we can use the most obvious
error: asking to checkout a path which is not in the index at all.

Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
builtin/checkout-index.c
t/t2004-checkout-cache-temp.sh
t/t2006-checkout-index-basic.sh

index 195165d8bd0108236f4ab1ad59f58ec23a056921..4bbfc92dce5a0e71e4389c16fd096f4a22fef4a1 100644 (file)
@@ -167,6 +167,7 @@ int cmd_checkout_index(int argc, const char **argv, const char *prefix)
        int prefix_length;
        int force = 0, quiet = 0, not_new = 0;
        int index_opt = 0;
+       int err = 0;
        struct option builtin_checkout_index_options[] = {
                OPT_BOOL('a', "all", &all,
                        N_("check out all files in the index")),
@@ -231,7 +232,7 @@ int cmd_checkout_index(int argc, const char **argv, const char *prefix)
                if (read_from_stdin)
                        die("git checkout-index: don't mix '--stdin' and explicit filenames");
                p = prefix_path(prefix, prefix_length, arg);
-               checkout_file(p, prefix);
+               err |= checkout_file(p, prefix);
                free(p);
        }
 
@@ -253,13 +254,16 @@ int cmd_checkout_index(int argc, const char **argv, const char *prefix)
                                strbuf_swap(&buf, &unquoted);
                        }
                        p = prefix_path(prefix, prefix_length, buf.buf);
-                       checkout_file(p, prefix);
+                       err |= checkout_file(p, prefix);
                        free(p);
                }
                strbuf_release(&unquoted);
                strbuf_release(&buf);
        }
 
+       if (err)
+               return 1;
+
        if (all)
                checkout_all(prefix, prefix_length);
 
index 4308d698b9291f4bf5bd02332753784d6f934462..a9352b08a8b80e4a58131f31f98eb7481914ecbe 100755 (executable)
@@ -88,7 +88,7 @@ test_expect_success 'checkout all stage 2 to temporary files' '
        done
 '
 
-test_expect_failure 'checkout all stages of unknown path' '
+test_expect_success 'checkout all stages of unknown path' '
        rm -f path* .merge_* actual &&
        test_must_fail git checkout-index --stage=all --temp \
                -- does-not-exist 2>stderr &&
index 57cbdfe9bce93ddd33df08fec506be93e362db58..8e181dbf01c46a324b54471cc62e1c9291852079 100755 (executable)
@@ -21,4 +21,15 @@ test_expect_success 'checkout-index -h in broken repository' '
        test_i18ngrep "[Uu]sage" broken/usage
 '
 
+test_expect_success 'checkout-index reports errors (cmdline)' '
+       test_must_fail git checkout-index -- does-not-exist 2>stderr &&
+       test_i18ngrep not.in.the.cache stderr
+'
+
+test_expect_success 'checkout-index reports errors (stdin)' '
+       echo does-not-exist |
+       test_must_fail git checkout-index --stdin 2>stderr &&
+       test_i18ngrep not.in.the.cache stderr
+'
+
 test_done