]> git.ipfire.org Git - thirdparty/git.git/commitdiff
read_gitfile(): simplify NOT_A_REPO error message
authorJeff King <peff@peff.net>
Tue, 16 Jun 2026 12:35:16 +0000 (08:35 -0400)
committerJunio C Hamano <gitster@pobox.com>
Tue, 16 Jun 2026 14:19:24 +0000 (07:19 -0700)
If a .git file is well-formed but points to a directory that is not
itself a valid repository, then we say:

  fatal: not a git repository: <pointed-to-repo>

without mentioning the .git file that pointed us there in the first
place. Doing so could better help the user understand the source of the
problem.

In theory the most helpful thing we could do is mention both paths,
like:

  gitfile '<gitfile>' points to invalid repository: <pointed-to-repo>

But there's another catch: when we generate the error, we don't always
know the pointed-to repository! This leads to a potential segfault.

The message comes from read_gitfile_error_die(). Originally we only
called that function from inside read_gitfile_gently(), passing in both
the gitfile path and the pointed-to path. But that changed in 1dd27bfbfd
(setup: improve error diagnosis for invalid .git files, 2026-03-04).
Since then, the caller in setup_git_directory_gently(), even if it wants
to die on error, always passes in the "return_error_code" flag, asking
the function to instead return a numeric error code. And then it calls
read_gitfile_error_die() itself, passing NULL for the pointed-to path.

If we get the READ_GITFILE_ERR_NOT_A_REPO code, we form a message using
that NULL pointer, and either segfault or get garbage like "not a git
repository: (null)", depending on the platform.

We could fix this by having the function pass out both the numeric error
code and the pointed-to path. But that creates a new headache: we have
to allocate that string on the heap and pass ownership back to the
caller. So now every caller has to be aware of it (and either free the
result, or signal that they are not interested by using an extra
parameter).

Instead, let's just drop the pointed-to path from the error message
entirely, and mention only the gitfile. This fixes the NULL dereference
without introducing any more complexity. The user-facing error message
is not as detailed as it could be, but is better than the original.
Since it mentions the gitfile, a user investigating the situation can
look there to find the pointed-to path (whereas you could not go the
other way from the original message).

There's an existing test in t0002 which triggers this case, but we
didn't notice the problem because it checks only that we said "not a
repository", and not the full string. So if we print "(null)" it is
happy. It will probably crash on some non-glibc platforms, but nobody
seems to have reported it yet (the breakage is recent-ish as of v2.54).
I'm also somewhat surprised that building with ASan/UBSan doesn't catch
this, but it doesn't seem to (and I found an open issue with somebody
asking for NULL printf checks to be implemented in the sanitizers).

We'll tweak the test to match the new error, but there's no need to beef
it up further, since we're not showing the pointed-to path at all.

We also racily trigger this in t7450. During parallel cloning we might
see one of several errors, including this one. And so we must update
that message, too (you can otherwise find the failure pretty quickly by
running t7450 with --stress).

Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
setup.c
setup.h
submodule.c
t/t0002-gitfile.sh
t/t7450-bad-git-dotfiles.sh

diff --git a/setup.c b/setup.c
index 7ec4427368a2a756dc7a45905209a08d12cffaba..ae352f572028f9b3d2c2163864dfda67b2cef7d2 100644 (file)
--- a/setup.c
+++ b/setup.c
@@ -917,7 +917,7 @@ int verify_repository_format(const struct repository_format *format,
        return 0;
 }
 
-void read_gitfile_error_die(int error_code, const char *path, const char *dir)
+void read_gitfile_error_die(int error_code, const char *path)
 {
        switch (error_code) {
        case READ_GITFILE_ERR_NOT_A_FILE:
@@ -937,7 +937,8 @@ void read_gitfile_error_die(int error_code, const char *path, const char *dir)
        case READ_GITFILE_ERR_NO_PATH:
                die(_("no path in gitfile: %s"), path);
        case READ_GITFILE_ERR_NOT_A_REPO:
-               die(_("not a git repository: %s"), dir);
+               die(_("gitfile does not point to a valid repository: %s"),
+                   path);
        default:
                BUG("unknown error code");
        }
@@ -1028,7 +1029,7 @@ cleanup_return:
        if (return_error_code)
                *return_error_code = error_code;
        else if (error_code)
-               read_gitfile_error_die(error_code, path, dir);
+               read_gitfile_error_die(error_code, path);
 
        free(buf);
        return error_code ? NULL : path;
@@ -1633,7 +1634,7 @@ static enum discovery_result setup_git_directory_gently_1(struct strbuf *dir,
                                        return GIT_DIR_INVALID_GITFILE;
                        default:
                                if (die_on_error)
-                                       read_gitfile_error_die(error_code, dir->buf, NULL);
+                                       read_gitfile_error_die(error_code, dir->buf);
                                else
                                        return GIT_DIR_INVALID_GITFILE;
                        }
diff --git a/setup.h b/setup.h
index 80bc6e5f078af8a94aa2488be90d0201ace2f305..2630ed196b68bdd248a4a40bf34749d9a29ffdf4 100644 (file)
--- a/setup.h
+++ b/setup.h
@@ -38,7 +38,7 @@ int is_nonbare_repository_dir(struct strbuf *path);
 #define READ_GITFILE_ERR_TOO_LARGE 8
 #define READ_GITFILE_ERR_MISSING 9
 #define READ_GITFILE_ERR_IS_A_DIR 10
-void read_gitfile_error_die(int error_code, const char *path, const char *dir);
+void read_gitfile_error_die(int error_code, const char *path);
 const char *read_gitfile_gently(const char *path, int *return_error_code);
 #define read_gitfile(path) read_gitfile_gently((path), NULL)
 const char *resolve_gitdir_gently(const char *suspect, int *return_error_code);
index b1a0363f9d2a96a1084953bf46f9c788d425741f..9ab7fec8dc7abc18cdb0d16f189bb52deb3e1dd1 100644 (file)
@@ -2578,7 +2578,7 @@ void absorb_git_dir_into_superproject(const char *path,
 
                if (err_code != READ_GITFILE_ERR_NOT_A_REPO)
                        /* We don't know what broke here. */
-                       read_gitfile_error_die(err_code, path, NULL);
+                       read_gitfile_error_die(err_code, path);
 
                /*
                * Maybe populated, but no git directory was found?
index dfbcdddbcc1f0fd5cf632524b3dfa5a8337a7f02..6356e9ec7272a2cd9eda0e0f209a4ec4f11fc840 100755 (executable)
@@ -27,7 +27,7 @@ test_expect_success 'bad setup: invalid .git file format' '
 test_expect_success 'bad setup: invalid .git file path' '
        echo "gitdir: $REAL.not" >.git &&
        test_must_fail git rev-parse 2>.err &&
-       test_grep "not a git repository" .err
+       test_grep "gitfile does not point to a valid repository" .err
 '
 
 test_expect_success 'final setup + check rev-parse --git-dir' '
index f512eed278c46b60f824be2865a7a56859188c9c..7fdbf085c91f11695b7a061139c7b18c3d238059 100755 (executable)
@@ -348,7 +348,7 @@ test_expect_success 'git dirs of sibling submodules must not be nested' '
 test_expect_success 'submodule git dir nesting detection must work with parallel cloning' '
        test_must_fail git clone --recurse-submodules --jobs=2 nested clone_parallel 2>err &&
        cat err &&
-       grep -E "(already exists|is inside git dir|not a git repository)" err &&
+       grep -E "(already exists|is inside git dir|does not point to a valid repository)" err &&
        {
                test_path_is_missing .git/modules/hippo/HEAD ||
                test_path_is_missing .git/modules/hippo/hooks/HEAD