]> git.ipfire.org Git - thirdparty/git.git/commitdiff
packed-backend: fsck should warn when "packed-refs" file is empty
authorshejialuo <shejialuo@gmail.com>
Wed, 14 May 2025 15:50:26 +0000 (23:50 +0800)
committerJunio C Hamano <gitster@pobox.com>
Wed, 14 May 2025 19:32:58 +0000 (12:32 -0700)
We assume the "packed-refs" won't be empty and instead has at least one
line in it (even when there are no refs packed, there is the file header
line). Because there is no terminating LF in the empty file, we will
report "packedRefEntryNotTerminated(ERROR)" to the user.

However, the runtime code paths would accept an empty "packed-refs"
file, for example, "create_snapshot" would simply return the "snapshot"
without checking the content of "packed-refs". So, we should skip
checking the content of "packed-refs" when it is empty during fsck.

After 694b7a1999 (repack_without_ref(): write peeled refs in the
rewritten file, 2013-04-22), we would always write a header into the
"packed-refs" file. So, versions of Git that are not too ancient never
write such an empty "packed-refs" file.

As an empty file often indicates a sign of a filesystem-level issue, the
way we want to resolve this inconsistency is not make everybody totally
silent but notice and report the anomaly.

Let's create a "FSCK_INFO" message id "EMPTY_PACKED_REFS_FILE" to report
to the users that "packed-refs" is empty.

Signed-off-by: shejialuo <shejialuo@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
Documentation/fsck-msgids.adoc
fsck.h
refs/packed-backend.c
t/t0602-reffiles-fsck.sh

index 9601fff22854b6019f2e0ef52df7452662345b97..0ba4f9a27e4c73b2aeeabec5b6909afd93a4a4cb 100644 (file)
 `emptyName`::
        (WARN) A path contains an empty name.
 
+`emptyPackedRefsFile`::
+       (INFO) "packed-refs" file is empty. Report to the
+       git@vger.kernel.org mailing list if you see this error. As only
+       very early versions of Git would create such an empty
+       "packed_refs" file, we might tighten this rule in the future.
+
 `extraHeaderEntry`::
        (IGNORE) Extra headers found after `tagger`.
 
diff --git a/fsck.h b/fsck.h
index b1deae61eed7b12c53b944bb223466089d663520..0c5869ac34e216807c2d4a88a4207064f193036a 100644 (file)
--- a/fsck.h
+++ b/fsck.h
@@ -84,6 +84,7 @@ enum fsck_msg_type {
        FUNC(LARGE_PATHNAME, WARN) \
        /* infos (reported as warnings, but ignored by default) */ \
        FUNC(BAD_FILEMODE, INFO) \
+       FUNC(EMPTY_PACKED_REFS_FILE, INFO) \
        FUNC(GITMODULES_PARSE, INFO) \
        FUNC(GITIGNORE_SYMLINK, INFO) \
        FUNC(GITATTRIBUTES_SYMLINK, INFO) \
index 3ad1ed0787aadaf69a11892adf3ed4052a257b8c..fb91833e76d9c9cb6e514d9a1dfe21e736004d90 100644 (file)
@@ -2103,6 +2103,15 @@ static int packed_fsck(struct ref_store *ref_store,
                goto cleanup;
        }
 
+       if (!st.st_size) {
+               struct fsck_ref_report report = { 0 };
+               report.path = "packed-refs";
+               ret = fsck_report_ref(o, &report,
+                                     FSCK_MSG_EMPTY_PACKED_REFS_FILE,
+                                     "file is empty");
+               goto cleanup;
+       }
+
        if (strbuf_read(&packed_ref_content, fd, 0) < 0) {
                ret = error_errno(_("unable to read '%s'"), refs->path);
                goto cleanup;
index 9d1dc2144c4b7259bc578e9630923e96c1c98a3e..f671ac4d3aba1a66029bd61fcec006eab32f5077 100755 (executable)
@@ -647,6 +647,23 @@ test_expect_success SYMLINKS 'the filetype of packed-refs should be checked' '
        )
 '
 
+test_expect_success 'empty packed-refs should be reported' '
+       test_when_finished "rm -rf repo" &&
+       git init repo &&
+       (
+               cd repo &&
+               test_commit default &&
+
+               >.git/packed-refs &&
+               git refs verify 2>err &&
+               cat >expect <<-EOF &&
+               warning: packed-refs: emptyPackedRefsFile: file is empty
+               EOF
+               rm .git/packed-refs &&
+               test_cmp expect err
+       )
+'
+
 test_expect_success 'packed-refs header should be checked' '
        test_when_finished "rm -rf repo" &&
        git init repo &&