Skip to content

Commit 35682ad

Browse files
vdyegitster
authored andcommitted
checkout-index: integrate with sparse index
Add repository settings to allow usage of the sparse index. When using the `--all` option, sparse directories are ignored by default due to the `skip-worktree` flag, so there is no need to expand the index. If `--ignore-skip-worktree-bits` is specified, the index is expanded in order to check out all files. When checking out individual files, existing behavior in a full index is to exit with an error if a directory is specified (as the directory name will not match an index entry). However, it is possible in a sparse index to match a directory name to a sparse directory index entry, but checking out that sparse directory still results in an error on checkout. To reduce some potential confusion for users, `checkout_file(...)` explicitly exits with an informative error if provided with a sparse directory name. The test corresponding to this scenario verifies the error message, which now differs between sparse index and non-sparse index checkouts. Signed-off-by: Victoria Dye <[email protected]> Reviewed-by: Elijah Newren <[email protected]> Signed-off-by: Junio C Hamano <[email protected]>
1 parent 88078f5 commit 35682ad

File tree

2 files changed

+36
-3
lines changed

2 files changed

+36
-3
lines changed

builtin/checkout-index.c

Lines changed: 26 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -67,6 +67,7 @@ static int checkout_file(const char *name, const char *prefix)
6767
int namelen = strlen(name);
6868
int pos = cache_name_pos(name, namelen);
6969
int has_same_name = 0;
70+
int is_file = 0;
7071
int is_skipped = 1;
7172
int did_checkout = 0;
7273
int errs = 0;
@@ -81,6 +82,9 @@ static int checkout_file(const char *name, const char *prefix)
8182
break;
8283
has_same_name = 1;
8384
pos++;
85+
if (S_ISSPARSEDIR(ce->ce_mode))
86+
break;
87+
is_file = 1;
8488
if (!ignore_skip_worktree && ce_skip_worktree(ce))
8589
break;
8690
is_skipped = 0;
@@ -112,6 +116,8 @@ static int checkout_file(const char *name, const char *prefix)
112116
fprintf(stderr, "git checkout-index: %s ", name);
113117
if (!has_same_name)
114118
fprintf(stderr, "is not in the cache");
119+
else if (!is_file)
120+
fprintf(stderr, "is a sparse directory");
115121
else if (is_skipped)
116122
fprintf(stderr, "has skip-worktree enabled; "
117123
"use '--ignore-skip-worktree-bits' to checkout");
@@ -130,10 +136,25 @@ static int checkout_all(const char *prefix, int prefix_length)
130136
int i, errs = 0;
131137
struct cache_entry *last_ce = NULL;
132138

133-
/* TODO: audit for interaction with sparse-index. */
134-
ensure_full_index(&the_index);
135139
for (i = 0; i < active_nr ; i++) {
136140
struct cache_entry *ce = active_cache[i];
141+
142+
if (S_ISSPARSEDIR(ce->ce_mode)) {
143+
if (!ce_skip_worktree(ce))
144+
BUG("sparse directory '%s' does not have skip-worktree set", ce->name);
145+
146+
/*
147+
* If the current entry is a sparse directory and skip-worktree
148+
* entries are being checked out, expand the index and continue
149+
* the loop on the current index position (now pointing to the
150+
* first entry inside the expanded sparse directory).
151+
*/
152+
if (ignore_skip_worktree) {
153+
ensure_full_index(&the_index);
154+
ce = active_cache[i];
155+
}
156+
}
157+
137158
if (!ignore_skip_worktree && ce_skip_worktree(ce))
138159
continue;
139160
if (ce_stage(ce) != checkout_stage
@@ -225,6 +246,9 @@ int cmd_checkout_index(int argc, const char **argv, const char *prefix)
225246
git_config(git_default_config, NULL);
226247
prefix_length = prefix ? strlen(prefix) : 0;
227248

249+
prepare_repo_settings(the_repository);
250+
the_repository->settings.command_requires_full_index = 0;
251+
228252
if (read_cache() < 0) {
229253
die("invalid cache");
230254
}

t/t1092-sparse-checkout-compatibility.sh

Lines changed: 10 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -801,7 +801,14 @@ test_expect_success 'checkout-index with folders' '
801801
test_all_match test_must_fail git checkout-index -f -- deep/ &&
802802
803803
# Outside checkout definition
804-
test_all_match test_must_fail git checkout-index -f -- folder1/
804+
# Note: although all tests fail (as expected), the messaging differs. For
805+
# non-sparse index checkouts, the error is that the "file" does not appear
806+
# in the index; for sparse checkouts, the error is explicitly that the
807+
# entry is a sparse directory.
808+
run_on_all test_must_fail git checkout-index -f -- folder1/ &&
809+
test_cmp full-checkout-err sparse-checkout-err &&
810+
! test_cmp full-checkout-err sparse-index-err &&
811+
grep "is a sparse directory" sparse-index-err
805812
'
806813

807814
test_expect_success 'checkout-index --all' '
@@ -972,6 +979,8 @@ test_expect_success 'sparse-index is not expanded' '
972979
echo >>sparse-index/untracked.txt &&
973980
ensure_not_expanded add . &&
974981
982+
ensure_not_expanded checkout-index -f a &&
983+
ensure_not_expanded checkout-index -f --all &&
975984
for ref in update-deep update-folder1 update-folder2 update-deep
976985
do
977986
echo >>sparse-index/README.md &&

0 commit comments

Comments
 (0)