3 test_description='split index mode tests'
7 # We need total control of index splitting here
8 sane_unset GIT_TEST_SPLIT_INDEX
10 # Testing a hard coded SHA against an index with an extension
11 # that can vary from run to run is problematic so we disable
13 sane_unset GIT_TEST_FSMONITOR
14 sane_unset GIT_TEST_INDEX_THREADS
16 # Create a file named as $1 with content read from stdin.
17 # Set the file's mtime to a few seconds in the past to avoid racy situations.
18 create_non_racy_file () {
20 test-tool chmtime =-5 "$1"
23 test_expect_success 'enable split index' '
24 git config splitIndex.maxPercentChange 100 &&
25 git update-index --split-index &&
26 test-tool dump-split-index .git/index >actual &&
27 indexversion=$(test-tool index-version <.git/index) &&
28 if test "$indexversion" = "4"
30 own=3527df833c6c100d3d1d921a9a782d62a8be4b58
31 base=746f7ab2ed44fb839efdfbffcf399d0b113fb4cb
33 own=5e9b60117ece18da410ddecc8b8d43766a0e4204
34 base=4370042739b31cd17a5c5cd6043a77c9a00df113
42 test_cmp expect actual
45 test_expect_success 'add one file' '
46 create_non_racy_file one &&
47 git update-index --add one &&
48 git ls-files --stage >ls-files.actual &&
49 cat >ls-files.expect <<-EOF &&
50 100644 $EMPTY_BLOB 0 one
52 test_cmp ls-files.expect ls-files.actual &&
54 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
57 100644 $EMPTY_BLOB 0 one
61 test_cmp expect actual
64 test_expect_success 'disable split index' '
65 git update-index --no-split-index &&
66 git ls-files --stage >ls-files.actual &&
67 cat >ls-files.expect <<-EOF &&
68 100644 $EMPTY_BLOB 0 one
70 test_cmp ls-files.expect ls-files.actual &&
72 BASE=$(test-tool dump-split-index .git/index | sed -n "s/^own/base/p") &&
73 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
77 test_cmp expect actual
80 test_expect_success 'enable split index again, "one" now belongs to base index"' '
81 git update-index --split-index &&
82 git ls-files --stage >ls-files.actual &&
83 cat >ls-files.expect <<-EOF &&
84 100644 $EMPTY_BLOB 0 one
86 test_cmp ls-files.expect ls-files.actual &&
88 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
94 test_cmp expect actual
97 test_expect_success 'modify original file, base index untouched' '
98 echo modified | create_non_racy_file one &&
99 git update-index one &&
100 git ls-files --stage >ls-files.actual &&
101 cat >ls-files.expect <<-EOF &&
102 100644 2e0996000b7e9019eabcad29391bf0f5c7702f0b 0 one
104 test_cmp ls-files.expect ls-files.actual &&
106 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
107 q_to_tab >expect <<-EOF &&
109 100644 2e0996000b7e9019eabcad29391bf0f5c7702f0b 0Q
113 test_cmp expect actual
116 test_expect_success 'add another file, which stays index' '
117 create_non_racy_file two &&
118 git update-index --add two &&
119 git ls-files --stage >ls-files.actual &&
120 cat >ls-files.expect <<-EOF &&
121 100644 2e0996000b7e9019eabcad29391bf0f5c7702f0b 0 one
122 100644 $EMPTY_BLOB 0 two
124 test_cmp ls-files.expect ls-files.actual &&
126 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
127 q_to_tab >expect <<-EOF &&
129 100644 2e0996000b7e9019eabcad29391bf0f5c7702f0b 0Q
130 100644 $EMPTY_BLOB 0 two
134 test_cmp expect actual
137 test_expect_success 'remove file not in base index' '
138 git update-index --force-remove two &&
139 git ls-files --stage >ls-files.actual &&
140 cat >ls-files.expect <<-EOF &&
141 100644 2e0996000b7e9019eabcad29391bf0f5c7702f0b 0 one
143 test_cmp ls-files.expect ls-files.actual &&
145 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
146 q_to_tab >expect <<-EOF &&
148 100644 2e0996000b7e9019eabcad29391bf0f5c7702f0b 0Q
152 test_cmp expect actual
155 test_expect_success 'remove file in base index' '
156 git update-index --force-remove one &&
157 git ls-files --stage >ls-files.actual &&
158 test_must_be_empty ls-files.actual &&
160 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
161 cat >expect <<-EOF &&
166 test_cmp expect actual
169 test_expect_success 'add original file back' '
170 create_non_racy_file one &&
171 git update-index --add one &&
172 git ls-files --stage >ls-files.actual &&
173 cat >ls-files.expect <<-EOF &&
174 100644 $EMPTY_BLOB 0 one
176 test_cmp ls-files.expect ls-files.actual &&
178 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
179 cat >expect <<-EOF &&
181 100644 $EMPTY_BLOB 0 one
185 test_cmp expect actual
188 test_expect_success 'add new file' '
189 create_non_racy_file two &&
190 git update-index --add two &&
191 git ls-files --stage >actual &&
192 cat >expect <<-EOF &&
193 100644 $EMPTY_BLOB 0 one
194 100644 $EMPTY_BLOB 0 two
196 test_cmp expect actual
199 test_expect_success 'unify index, two files remain' '
200 git update-index --no-split-index &&
201 git ls-files --stage >ls-files.actual &&
202 cat >ls-files.expect <<-EOF &&
203 100644 $EMPTY_BLOB 0 one
204 100644 $EMPTY_BLOB 0 two
206 test_cmp ls-files.expect ls-files.actual &&
208 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
209 cat >expect <<-EOF &&
212 test_cmp expect actual
215 test_expect_success 'rev-parse --shared-index-path' '
216 test_create_repo split-index &&
219 git update-index --split-index &&
220 echo .git/sharedindex* >expect &&
221 git rev-parse --shared-index-path >actual &&
222 test_cmp expect actual &&
223 mkdir subdirectory &&
225 echo ../.git/sharedindex* >expect &&
226 git rev-parse --shared-index-path >actual &&
227 test_cmp expect actual
231 test_expect_success 'set core.splitIndex config variable to true' '
232 git config core.splitIndex true &&
233 create_non_racy_file three &&
234 git update-index --add three &&
235 git ls-files --stage >ls-files.actual &&
236 cat >ls-files.expect <<-EOF &&
237 100644 e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 0 one
238 100644 e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 0 three
239 100644 e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 0 two
241 test_cmp ls-files.expect ls-files.actual &&
242 BASE=$(test-tool dump-split-index .git/index | grep "^base") &&
243 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
244 cat >expect <<-EOF &&
249 test_cmp expect actual
252 test_expect_success 'set core.splitIndex config variable to false' '
253 git config core.splitIndex false &&
254 git update-index --force-remove three &&
255 git ls-files --stage >ls-files.actual &&
256 cat >ls-files.expect <<-EOF &&
257 100644 e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 0 one
258 100644 e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 0 two
260 test_cmp ls-files.expect ls-files.actual &&
261 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
262 cat >expect <<-EOF &&
265 test_cmp expect actual
268 test_expect_success 'set core.splitIndex config variable back to true' '
269 git config core.splitIndex true &&
270 create_non_racy_file three &&
271 git update-index --add three &&
272 BASE=$(test-tool dump-split-index .git/index | grep "^base") &&
273 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
274 cat >expect <<-EOF &&
279 test_cmp expect actual &&
280 create_non_racy_file four &&
281 git update-index --add four &&
282 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
283 cat >expect <<-EOF &&
285 100644 e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 0 four
289 test_cmp expect actual
292 test_expect_success 'check behavior with splitIndex.maxPercentChange unset' '
293 git config --unset splitIndex.maxPercentChange &&
294 create_non_racy_file five &&
295 git update-index --add five &&
296 BASE=$(test-tool dump-split-index .git/index | grep "^base") &&
297 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
298 cat >expect <<-EOF &&
303 test_cmp expect actual &&
304 create_non_racy_file six &&
305 git update-index --add six &&
306 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
307 cat >expect <<-EOF &&
309 100644 e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 0 six
313 test_cmp expect actual
316 test_expect_success 'check splitIndex.maxPercentChange set to 0' '
317 git config splitIndex.maxPercentChange 0 &&
318 create_non_racy_file seven &&
319 git update-index --add seven &&
320 BASE=$(test-tool dump-split-index .git/index | grep "^base") &&
321 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
322 cat >expect <<-EOF &&
327 test_cmp expect actual &&
328 create_non_racy_file eight &&
329 git update-index --add eight &&
330 BASE=$(test-tool dump-split-index .git/index | grep "^base") &&
331 test-tool dump-split-index .git/index | sed "/^own/d" >actual &&
332 cat >expect <<-EOF &&
337 test_cmp expect actual
340 test_expect_success 'shared index files expire after 2 weeks by default' '
341 create_non_racy_file ten &&
342 git update-index --add ten &&
343 test $(ls .git/sharedindex.* | wc -l) -gt 2 &&
344 just_under_2_weeks_ago=$((5-14*86400)) &&
345 test-tool chmtime =$just_under_2_weeks_ago .git/sharedindex.* &&
346 create_non_racy_file eleven &&
347 git update-index --add eleven &&
348 test $(ls .git/sharedindex.* | wc -l) -gt 2 &&
349 just_over_2_weeks_ago=$((-1-14*86400)) &&
350 test-tool chmtime =$just_over_2_weeks_ago .git/sharedindex.* &&
351 create_non_racy_file twelve &&
352 git update-index --add twelve &&
353 test $(ls .git/sharedindex.* | wc -l) -le 2
356 test_expect_success 'check splitIndex.sharedIndexExpire set to 16 days' '
357 git config splitIndex.sharedIndexExpire "16.days.ago" &&
358 test-tool chmtime =$just_over_2_weeks_ago .git/sharedindex.* &&
359 create_non_racy_file thirteen &&
360 git update-index --add thirteen &&
361 test $(ls .git/sharedindex.* | wc -l) -gt 2 &&
362 just_over_16_days_ago=$((-1-16*86400)) &&
363 test-tool chmtime =$just_over_16_days_ago .git/sharedindex.* &&
364 create_non_racy_file fourteen &&
365 git update-index --add fourteen &&
366 test $(ls .git/sharedindex.* | wc -l) -le 2
369 test_expect_success 'check splitIndex.sharedIndexExpire set to "never" and "now"' '
370 git config splitIndex.sharedIndexExpire never &&
371 just_10_years_ago=$((-365*10*86400)) &&
372 test-tool chmtime =$just_10_years_ago .git/sharedindex.* &&
373 create_non_racy_file fifteen &&
374 git update-index --add fifteen &&
375 test $(ls .git/sharedindex.* | wc -l) -gt 2 &&
376 git config splitIndex.sharedIndexExpire now &&
377 just_1_second_ago=-1 &&
378 test-tool chmtime =$just_1_second_ago .git/sharedindex.* &&
379 create_non_racy_file sixteen &&
380 git update-index --add sixteen &&
381 test $(ls .git/sharedindex.* | wc -l) -le 2
384 while read -r mode modebits
386 test_expect_success POSIXPERM "split index respects core.sharedrepository $mode" '
387 # Remove existing shared index files
388 git config core.splitIndex false &&
389 git update-index --force-remove one &&
390 rm -f .git/sharedindex.* &&
391 # Create one new shared index file
392 git config core.sharedrepository "$mode" &&
393 git config core.splitIndex true &&
394 create_non_racy_file one &&
395 git update-index --add one &&
396 echo "$modebits" >expect &&
397 test_modebits .git/index >actual &&
398 test_cmp expect actual &&
399 shared=$(ls .git/sharedindex.*) &&
402 # we have more than one???
405 test_modebits "$shared" >actual &&
406 test_cmp expect actual ;;
414 test_expect_success POSIXPERM,SANITY 'graceful handling when splitting index is not allowed' '
415 test_create_repo ro &&
418 test_commit initial &&
419 git update-index --split-index &&
420 test -f .git/sharedindex.*
422 cp ro/.git/index new-index &&
423 test_when_finished "chmod u+w ro/.git" &&
425 GIT_INDEX_FILE="$(pwd)/new-index" git -C ro update-index --split-index &&
427 rm ro/.git/sharedindex.* &&
428 GIT_INDEX_FILE=new-index git ls-files >actual &&
429 echo initial.t >expected &&
430 test_cmp expected actual
433 test_expect_success 'writing split index with null sha1 does not write cache tree' '
434 git config core.splitIndex true &&
435 git config splitIndex.maxPercentChange 0 &&
436 git commit -m "commit" &&
439 printf "160000 commit $ZERO_OID\\tbroken\\n"
441 echo "add broken entry" >msg &&
443 tree=$(git mktree <broken-tree) &&
445 commit=$(git commit-tree $tree -p HEAD <msg) &&
446 git update-ref HEAD "$commit" &&
447 GIT_ALLOW_NULL_SHA1=1 git reset --hard &&
448 test_might_fail test-tool dump-cache-tree >cache-tree.out &&
449 test_line_count = 0 cache-tree.out