Skip to content

Commit ee4dfee

Browse files
peffgitster
authored andcommitted
rev-list: let traversal die when --missing is not in use
Commit 7c0fe33 (rev-list: handle missing tree objects properly, 2018-10-05) taught the traversal machinery used by git-rev-list to ignore missing trees, so that rev-list could handle them itself. However, it does so only by checking via oid_object_info_extended() that the object exists at all. This can miss several classes of errors that were previously detected by rev-list: - type mismatches (e.g., we expected a tree but got a blob) - failure to read the object data (e.g., due to bitrot on disk) This is especially important because we use "rev-list --objects" as our connectivity check to admit new objects to the repository, and it will now miss these cases (though the bitrot one is less important here, because we'd typically have just hashed and stored the object). There are a few options to fix this: 1. we could check these properties in rev-list when we do the existence check. This is probably too expensive in practice (perhaps even for a type check, but definitely for checking the whole content again, which implies loading each object into memory twice). 2. teach the traversal machinery to differentiate between a missing object, and one that could not be loaded as expected. This probably wouldn't be too hard to detect type mismatches, but detecting bitrot versus a truly missing object would require deep changes to the object-loading code. 3. have the traversal machinery communicate the failure to the caller, so that it can decide how to proceed without re-evaluting the object itself. Of those, I think (3) is probably the best path forward. However, this patch does none of them. In the name of expediently fixing the regression to a normal "rev-list --objects" that we use for connectivity checks, this simply restores the pre-7c0fe330d5 behavior of having the traversal die as soon as it fails to load a tree (when --missing is set to MA_ERROR, which is the default). Note that we can't get rid of the object-existence check in finish_object(), because this also handles blobs (which are not otherwise checked at all by the traversal code). Signed-off-by: Jeff King <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 8348766 commit ee4dfee

File tree

2 files changed

+5
-3
lines changed

2 files changed

+5
-3
lines changed

builtin/rev-list.c

Lines changed: 3 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -379,7 +379,6 @@ int cmd_rev_list(int argc, const char **argv, const char *prefix)
379379
repo_init_revisions(the_repository, &revs, prefix);
380380
revs.abbrev = DEFAULT_ABBREV;
381381
revs.commit_format = CMIT_FMT_UNSPECIFIED;
382-
revs.do_not_die_on_missing_tree = 1;
383382

384383
/*
385384
* Scan the argument list before invoking setup_revisions(), so that we
@@ -409,6 +408,9 @@ int cmd_rev_list(int argc, const char **argv, const char *prefix)
409408
}
410409
}
411410

411+
if (arg_missing_action)
412+
revs.do_not_die_on_missing_tree = 1;
413+
412414
argc = setup_revisions(argc, argv, &revs, &s_r_opt);
413415

414416
memset(&info, 0, sizeof(info));

t/t6102-rev-list-unexpected-objects.sh

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ test_expect_success 'setup unexpected non-tree entry' '
3030
broken_tree="$(git hash-object -w --literally -t tree broken-tree)"
3131
'
3232

33-
test_expect_failure 'traverse unexpected non-tree entry (lone)' '
33+
test_expect_success 'traverse unexpected non-tree entry (lone)' '
3434
test_must_fail git rev-list --objects $broken_tree
3535
'
3636

@@ -63,7 +63,7 @@ test_expect_success 'setup unexpected non-tree root' '
6363
broken-commit)"
6464
'
6565

66-
test_expect_failure 'traverse unexpected non-tree root (lone)' '
66+
test_expect_success 'traverse unexpected non-tree root (lone)' '
6767
test_must_fail git rev-list --objects $broken_commit
6868
'
6969

0 commit comments

Comments
 (0)