fsck_tree(): fix shadowed variable

Commit b2f2039c2b (fsck: accept an oid instead of a "struct tree" for
fsck_tree(), 2019-10-18) introduced a new "oid" parameter to
fsck_tree(), and we pass it to the report() function when we find
problems. However, that is shadowed within the tree-walking loop by the
existing "oid" variable which we use to store the oid of each tree
entry. As a result, we may report the wrong oid for some problems we
detect within the loop (the entry oid, instead of the tree oid).

Our tests didn't catch this because they checked only that we found the
expected fsck problem, not that it was attached to the correct object.

Let's rename both variables in the function to avoid confusion. This
makes the diff a little noisy (e.g., all of the report() calls outside
the loop were already correct but need to be touched), but makes sure we
catch all cases and will avoid similar confusion in the future.

And we can update the test to be a bit more specific and catch this
problem.

Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Jeff King 2021-05-01 11:41:38 -04:00 коммит произвёл Junio C Hamano
Родитель 963d02a24a
Коммит 9e1947cb48
2 изменённых файлов: 25 добавлений и 22 удалений

42
fsck.c
Просмотреть файл

@ -558,7 +558,7 @@ static int verify_ordered(unsigned mode1, const char *name1,
return c1 < c2 ? 0 : TREE_UNORDERED; return c1 < c2 ? 0 : TREE_UNORDERED;
} }
static int fsck_tree(const struct object_id *oid, static int fsck_tree(const struct object_id *tree_oid,
const char *buffer, unsigned long size, const char *buffer, unsigned long size,
struct fsck_options *options) struct fsck_options *options)
{ {
@ -579,7 +579,7 @@ static int fsck_tree(const struct object_id *oid,
struct name_stack df_dup_candidates = { NULL }; struct name_stack df_dup_candidates = { NULL };
if (init_tree_desc_gently(&desc, buffer, size)) { if (init_tree_desc_gently(&desc, buffer, size)) {
retval += report(options, oid, OBJ_TREE, FSCK_MSG_BAD_TREE, "cannot be parsed as a tree"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_BAD_TREE, "cannot be parsed as a tree");
return retval; return retval;
} }
@ -589,11 +589,11 @@ static int fsck_tree(const struct object_id *oid,
while (desc.size) { while (desc.size) {
unsigned short mode; unsigned short mode;
const char *name, *backslash; const char *name, *backslash;
const struct object_id *oid; const struct object_id *entry_oid;
oid = tree_entry_extract(&desc, &name, &mode); entry_oid = tree_entry_extract(&desc, &name, &mode);
has_null_sha1 |= is_null_oid(oid); has_null_sha1 |= is_null_oid(entry_oid);
has_full_path |= !!strchr(name, '/'); has_full_path |= !!strchr(name, '/');
has_empty_name |= !*name; has_empty_name |= !*name;
has_dot |= !strcmp(name, "."); has_dot |= !strcmp(name, ".");
@ -603,10 +603,11 @@ static int fsck_tree(const struct object_id *oid,
if (is_hfs_dotgitmodules(name) || is_ntfs_dotgitmodules(name)) { if (is_hfs_dotgitmodules(name) || is_ntfs_dotgitmodules(name)) {
if (!S_ISLNK(mode)) if (!S_ISLNK(mode))
oidset_insert(&options->gitmodules_found, oid); oidset_insert(&options->gitmodules_found,
entry_oid);
else else
retval += report(options, retval += report(options,
oid, OBJ_TREE, tree_oid, OBJ_TREE,
FSCK_MSG_GITMODULES_SYMLINK, FSCK_MSG_GITMODULES_SYMLINK,
".gitmodules is a symbolic link"); ".gitmodules is a symbolic link");
} }
@ -617,9 +618,10 @@ static int fsck_tree(const struct object_id *oid,
has_dotgit |= is_ntfs_dotgit(backslash); has_dotgit |= is_ntfs_dotgit(backslash);
if (is_ntfs_dotgitmodules(backslash)) { if (is_ntfs_dotgitmodules(backslash)) {
if (!S_ISLNK(mode)) if (!S_ISLNK(mode))
oidset_insert(&options->gitmodules_found, oid); oidset_insert(&options->gitmodules_found,
entry_oid);
else else
retval += report(options, oid, OBJ_TREE, retval += report(options, tree_oid, OBJ_TREE,
FSCK_MSG_GITMODULES_SYMLINK, FSCK_MSG_GITMODULES_SYMLINK,
".gitmodules is a symbolic link"); ".gitmodules is a symbolic link");
} }
@ -628,7 +630,7 @@ static int fsck_tree(const struct object_id *oid,
} }
if (update_tree_entry_gently(&desc)) { if (update_tree_entry_gently(&desc)) {
retval += report(options, oid, OBJ_TREE, FSCK_MSG_BAD_TREE, "cannot be parsed as a tree"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_BAD_TREE, "cannot be parsed as a tree");
break; break;
} }
@ -676,25 +678,25 @@ static int fsck_tree(const struct object_id *oid,
name_stack_clear(&df_dup_candidates); name_stack_clear(&df_dup_candidates);
if (has_null_sha1) if (has_null_sha1)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_NULL_SHA1, "contains entries pointing to null sha1"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_NULL_SHA1, "contains entries pointing to null sha1");
if (has_full_path) if (has_full_path)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_FULL_PATHNAME, "contains full pathnames"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_FULL_PATHNAME, "contains full pathnames");
if (has_empty_name) if (has_empty_name)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_EMPTY_NAME, "contains empty pathname"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_EMPTY_NAME, "contains empty pathname");
if (has_dot) if (has_dot)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_HAS_DOT, "contains '.'"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_HAS_DOT, "contains '.'");
if (has_dotdot) if (has_dotdot)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_HAS_DOTDOT, "contains '..'"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_HAS_DOTDOT, "contains '..'");
if (has_dotgit) if (has_dotgit)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_HAS_DOTGIT, "contains '.git'"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_HAS_DOTGIT, "contains '.git'");
if (has_zero_pad) if (has_zero_pad)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_ZERO_PADDED_FILEMODE, "contains zero-padded file modes"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_ZERO_PADDED_FILEMODE, "contains zero-padded file modes");
if (has_bad_modes) if (has_bad_modes)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_BAD_FILEMODE, "contains bad file modes"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_BAD_FILEMODE, "contains bad file modes");
if (has_dup_entries) if (has_dup_entries)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_DUPLICATE_ENTRIES, "contains duplicate file entries"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_DUPLICATE_ENTRIES, "contains duplicate file entries");
if (not_properly_sorted) if (not_properly_sorted)
retval += report(options, oid, OBJ_TREE, FSCK_MSG_TREE_NOT_SORTED, "not properly sorted"); retval += report(options, tree_oid, OBJ_TREE, FSCK_MSG_TREE_NOT_SORTED, "not properly sorted");
return retval; return retval;
} }

Просмотреть файл

@ -148,12 +148,13 @@ test_expect_success 'fsck detects symlinked .gitmodules file' '
{ {
printf "100644 blob $content\t$tricky\n" && printf "100644 blob $content\t$tricky\n" &&
printf "120000 blob $target\t.gitmodules\n" printf "120000 blob $target\t.gitmodules\n"
} | git mktree && } >bad-tree &&
tree=$(git mktree <bad-tree) &&
# Check not only that we fail, but that it is due to the # Check not only that we fail, but that it is due to the
# symlink detector # symlink detector
test_must_fail git fsck 2>output && test_must_fail git fsck 2>output &&
grep gitmodulesSymlink output grep "tree $tree: gitmodulesSymlink" output
) )
' '