]> git.ipfire.org Git - thirdparty/git.git/blob - t/t5500-fetch-pack.sh
fetch-pack: in protocol v2, in_vain only after ACK
[thirdparty/git.git] / t / t5500-fetch-pack.sh
1 #!/bin/sh
2 #
3 # Copyright (c) 2005 Johannes Schindelin
4 #
5
6 test_description='Testing multi_ack pack fetching'
7
8 . ./test-lib.sh
9
10 # Test fetch-pack/upload-pack pair.
11
12 # Some convenience functions
13
14 add () {
15 name=$1 &&
16 text="$@" &&
17 branch=$(echo $name | sed -e 's/^\(.\).*$/\1/') &&
18 parents="" &&
19
20 shift &&
21 while test $1; do
22 parents="$parents -p $1" &&
23 shift
24 done &&
25
26 echo "$text" > test.txt &&
27 git update-index --add test.txt &&
28 tree=$(git write-tree) &&
29 # make sure timestamps are in correct order
30 test_tick &&
31 commit=$(echo "$text" | git commit-tree $tree $parents) &&
32 eval "$name=$commit; export $name" &&
33 git update-ref "refs/heads/$branch" "$commit" &&
34 eval ${branch}TIP=$commit
35 }
36
37 pull_to_client () {
38 number=$1 &&
39 heads=$2 &&
40 count=$3 &&
41 test_expect_success "$number pull" '
42 (
43 cd client &&
44 git fetch-pack -k -v .. $heads &&
45
46 case "$heads" in
47 *A*)
48 git update-ref refs/heads/A "$ATIP";;
49 esac &&
50 case "$heads" in *B*)
51 git update-ref refs/heads/B "$BTIP";;
52 esac &&
53
54 git symbolic-ref HEAD refs/heads/$(
55 echo $heads |
56 sed -e "s/^\(.\).*$/\1/"
57 ) &&
58
59 git fsck --full &&
60
61 mv .git/objects/pack/pack-* . &&
62 p=$(ls -1 pack-*.pack) &&
63 git unpack-objects <$p &&
64 git fsck --full &&
65
66 idx=$(echo pack-*.idx) &&
67 pack_count=$(git show-index <$idx | wc -l) &&
68 test $pack_count = $count &&
69 rm -f pack-*
70 )
71 '
72 }
73
74 # Here begins the actual testing
75
76 # A1 - ... - A20 - A21
77 # \
78 # B1 - B2 - .. - B70
79
80 # client pulls A20, B1. Then tracks only B. Then pulls A.
81
82 test_expect_success 'setup' '
83 mkdir client &&
84 (
85 cd client &&
86 git init &&
87 git config transfer.unpacklimit 0
88 ) &&
89 add A1 &&
90 prev=1 &&
91 cur=2 &&
92 while [ $cur -le 10 ]; do
93 add A$cur $(eval echo \$A$prev) &&
94 prev=$cur &&
95 cur=$(($cur+1))
96 done &&
97 add B1 $A1 &&
98 git update-ref refs/heads/A "$ATIP" &&
99 git update-ref refs/heads/B "$BTIP" &&
100 git symbolic-ref HEAD refs/heads/B
101 '
102
103 pull_to_client 1st "refs/heads/B refs/heads/A" $((11*3))
104
105 test_expect_success 'post 1st pull setup' '
106 add A11 $A10 &&
107 prev=1 &&
108 cur=2 &&
109 while [ $cur -le 65 ]; do
110 add B$cur $(eval echo \$B$prev) &&
111 prev=$cur &&
112 cur=$(($cur+1))
113 done
114 '
115
116 pull_to_client 2nd "refs/heads/B" $((64*3))
117
118 pull_to_client 3rd "refs/heads/A" $((1*3))
119
120 test_expect_success 'single branch clone' '
121 git clone --single-branch "file://$(pwd)/." singlebranch
122 '
123
124 test_expect_success 'single branch object count' '
125 GIT_DIR=singlebranch/.git git count-objects -v |
126 grep "^in-pack:" > count.singlebranch &&
127 echo "in-pack: 198" >expected &&
128 test_cmp expected count.singlebranch
129 '
130
131 test_expect_success 'single given branch clone' '
132 git clone --single-branch --branch A "file://$(pwd)/." branch-a &&
133 test_must_fail git --git-dir=branch-a/.git rev-parse origin/B
134 '
135
136 test_expect_success 'clone shallow depth 1' '
137 git clone --no-single-branch --depth 1 "file://$(pwd)/." shallow0 &&
138 test "$(git --git-dir=shallow0/.git rev-list --count HEAD)" = 1
139 '
140
141 test_expect_success 'clone shallow depth 1 with fsck' '
142 git config --global fetch.fsckobjects true &&
143 git clone --no-single-branch --depth 1 "file://$(pwd)/." shallow0fsck &&
144 test "$(git --git-dir=shallow0fsck/.git rev-list --count HEAD)" = 1 &&
145 git config --global --unset fetch.fsckobjects
146 '
147
148 test_expect_success 'clone shallow' '
149 git clone --no-single-branch --depth 2 "file://$(pwd)/." shallow
150 '
151
152 test_expect_success 'clone shallow depth count' '
153 test "$(git --git-dir=shallow/.git rev-list --count HEAD)" = 2
154 '
155
156 test_expect_success 'clone shallow object count' '
157 (
158 cd shallow &&
159 git count-objects -v
160 ) > count.shallow &&
161 grep "^in-pack: 12" count.shallow
162 '
163
164 test_expect_success 'clone shallow object count (part 2)' '
165 sed -e "/^in-pack:/d" -e "/^packs:/d" -e "/^size-pack:/d" \
166 -e "/: 0$/d" count.shallow > count_output &&
167 test_must_be_empty count_output
168 '
169
170 test_expect_success 'fsck in shallow repo' '
171 (
172 cd shallow &&
173 git fsck --full
174 )
175 '
176
177 test_expect_success 'simple fetch in shallow repo' '
178 (
179 cd shallow &&
180 git fetch
181 )
182 '
183
184 test_expect_success 'no changes expected' '
185 (
186 cd shallow &&
187 git count-objects -v
188 ) > count.shallow.2 &&
189 cmp count.shallow count.shallow.2
190 '
191
192 test_expect_success 'fetch same depth in shallow repo' '
193 (
194 cd shallow &&
195 git fetch --depth=2
196 )
197 '
198
199 test_expect_success 'no changes expected' '
200 (
201 cd shallow &&
202 git count-objects -v
203 ) > count.shallow.3 &&
204 cmp count.shallow count.shallow.3
205 '
206
207 test_expect_success 'add two more' '
208 add B66 $B65 &&
209 add B67 $B66
210 '
211
212 test_expect_success 'pull in shallow repo' '
213 (
214 cd shallow &&
215 git pull .. B
216 )
217 '
218
219 test_expect_success 'clone shallow object count' '
220 (
221 cd shallow &&
222 git count-objects -v
223 ) > count.shallow &&
224 grep "^count: 6" count.shallow
225 '
226
227 test_expect_success 'add two more (part 2)' '
228 add B68 $B67 &&
229 add B69 $B68
230 '
231
232 test_expect_success 'deepening pull in shallow repo' '
233 (
234 cd shallow &&
235 git pull --depth 4 .. B
236 )
237 '
238
239 test_expect_success 'clone shallow object count' '
240 (
241 cd shallow &&
242 git count-objects -v
243 ) > count.shallow &&
244 grep "^count: 12" count.shallow
245 '
246
247 test_expect_success 'deepening fetch in shallow repo' '
248 (
249 cd shallow &&
250 git fetch --depth 4 .. A:A
251 )
252 '
253
254 test_expect_success 'clone shallow object count' '
255 (
256 cd shallow &&
257 git count-objects -v
258 ) > count.shallow &&
259 grep "^count: 18" count.shallow
260 '
261
262 test_expect_success 'pull in shallow repo with missing merge base' '
263 (
264 cd shallow &&
265 git fetch --depth 4 .. A &&
266 test_must_fail git merge --allow-unrelated-histories FETCH_HEAD
267 )
268 '
269
270 test_expect_success 'additional simple shallow deepenings' '
271 (
272 cd shallow &&
273 git fetch --depth=8 &&
274 git fetch --depth=10 &&
275 git fetch --depth=11
276 )
277 '
278
279 test_expect_success 'clone shallow depth count' '
280 test "$(git --git-dir=shallow/.git rev-list --count HEAD)" = 11
281 '
282
283 test_expect_success 'clone shallow object count' '
284 (
285 cd shallow &&
286 git prune &&
287 git count-objects -v
288 ) > count.shallow &&
289 grep "^count: 54" count.shallow
290 '
291
292 test_expect_success 'fetch --no-shallow on full repo' '
293 test_must_fail git fetch --noshallow
294 '
295
296 test_expect_success 'fetch --depth --no-shallow' '
297 (
298 cd shallow &&
299 test_must_fail git fetch --depth=1 --noshallow
300 )
301 '
302
303 test_expect_success 'turn shallow to complete repository' '
304 (
305 cd shallow &&
306 git fetch --unshallow &&
307 ! test -f .git/shallow &&
308 git fsck --full
309 )
310 '
311
312 test_expect_success 'clone shallow without --no-single-branch' '
313 git clone --depth 1 "file://$(pwd)/." shallow2
314 '
315
316 test_expect_success 'clone shallow object count' '
317 (
318 cd shallow2 &&
319 git count-objects -v
320 ) > count.shallow2 &&
321 grep "^in-pack: 3" count.shallow2
322 '
323
324 test_expect_success 'clone shallow with --branch' '
325 git clone --depth 1 --branch A "file://$(pwd)/." shallow3
326 '
327
328 test_expect_success 'clone shallow object count' '
329 echo "in-pack: 3" > count3.expected &&
330 GIT_DIR=shallow3/.git git count-objects -v |
331 grep "^in-pack" > count3.actual &&
332 test_cmp count3.expected count3.actual
333 '
334
335 test_expect_success 'clone shallow with detached HEAD' '
336 git checkout HEAD^ &&
337 git clone --depth 1 "file://$(pwd)/." shallow5 &&
338 git checkout - &&
339 GIT_DIR=shallow5/.git git rev-parse HEAD >actual &&
340 git rev-parse HEAD^ >expected &&
341 test_cmp expected actual
342 '
343
344 test_expect_success 'shallow clone pulling tags' '
345 git tag -a -m A TAGA1 A &&
346 git tag -a -m B TAGB1 B &&
347 git tag TAGA2 A &&
348 git tag TAGB2 B &&
349 git clone --depth 1 "file://$(pwd)/." shallow6 &&
350
351 cat >taglist.expected <<\EOF &&
352 TAGB1
353 TAGB2
354 EOF
355 GIT_DIR=shallow6/.git git tag -l >taglist.actual &&
356 test_cmp taglist.expected taglist.actual &&
357
358 echo "in-pack: 4" > count6.expected &&
359 GIT_DIR=shallow6/.git git count-objects -v |
360 grep "^in-pack" > count6.actual &&
361 test_cmp count6.expected count6.actual
362 '
363
364 test_expect_success 'shallow cloning single tag' '
365 git clone --depth 1 --branch=TAGB1 "file://$(pwd)/." shallow7 &&
366 cat >taglist.expected <<\EOF &&
367 TAGB1
368 TAGB2
369 EOF
370 GIT_DIR=shallow7/.git git tag -l >taglist.actual &&
371 test_cmp taglist.expected taglist.actual &&
372
373 echo "in-pack: 4" > count7.expected &&
374 GIT_DIR=shallow7/.git git count-objects -v |
375 grep "^in-pack" > count7.actual &&
376 test_cmp count7.expected count7.actual
377 '
378
379 test_expect_success 'clone shallow with packed refs' '
380 git pack-refs --all &&
381 git clone --depth 1 --branch A "file://$(pwd)/." shallow8 &&
382 echo "in-pack: 4" > count8.expected &&
383 GIT_DIR=shallow8/.git git count-objects -v |
384 grep "^in-pack" > count8.actual &&
385 test_cmp count8.expected count8.actual
386 '
387
388 test_expect_success 'in_vain not triggered before first ACK' '
389 rm -rf myserver myclient trace &&
390 git init myserver &&
391 test_commit -C myserver foo &&
392 git clone "file://$(pwd)/myserver" myclient &&
393
394 # MAX_IN_VAIN is 256. Because of batching, the client will send 496
395 # (16+32+64+128+256) commits, not 256, before giving up. So create 496
396 # irrelevant commits.
397 test_commit_bulk -C myclient 496 &&
398
399 # The new commit that the client wants to fetch.
400 test_commit -C myserver bar &&
401
402 GIT_TRACE_PACKET="$(pwd)/trace" git -C myclient fetch --progress origin &&
403 test_i18ngrep "Total 3 " trace
404 '
405
406 test_expect_success 'fetch in shallow repo unreachable shallow objects' '
407 (
408 git clone --bare --branch B --single-branch "file://$(pwd)/." no-reflog &&
409 git clone --depth 1 "file://$(pwd)/no-reflog" shallow9 &&
410 cd no-reflog &&
411 git tag -d TAGB1 TAGB2 &&
412 git update-ref refs/heads/B B~~ &&
413 git gc --prune=now &&
414 cd ../shallow9 &&
415 git fetch origin &&
416 git fsck --no-dangling
417 )
418 '
419 test_expect_success 'fetch creating new shallow root' '
420 (
421 git clone "file://$(pwd)/." shallow10 &&
422 git commit --allow-empty -m empty &&
423 cd shallow10 &&
424 git fetch --depth=1 --progress 2>actual &&
425 # This should fetch only the empty commit, no tree or
426 # blob objects
427 test_i18ngrep "remote: Total 1" actual
428 )
429 '
430
431 test_expect_success 'setup tests for the --stdin parameter' '
432 for head in C D E F
433 do
434 add $head
435 done &&
436 for head in A B C D E F
437 do
438 git tag $head $head
439 done &&
440 cat >input <<-\EOF &&
441 refs/heads/C
442 refs/heads/A
443 refs/heads/D
444 refs/tags/C
445 refs/heads/B
446 refs/tags/A
447 refs/heads/E
448 refs/tags/B
449 refs/tags/E
450 refs/tags/D
451 EOF
452 sort <input >expect &&
453 (
454 echo refs/heads/E &&
455 echo refs/tags/E &&
456 cat input
457 ) >input.dup
458 '
459
460 test_expect_success 'setup fetch refs from cmdline v[12]' '
461 cp -r client client1 &&
462 cp -r client client2
463 '
464
465 for version in '' 1 2
466 do
467 test_expect_success "protocol.version=$version fetch refs from cmdline" "
468 (
469 cd client$version &&
470 GIT_TEST_PROTOCOL_VERSION=$version git fetch-pack --no-progress .. \$(cat ../input)
471 ) >output &&
472 cut -d ' ' -f 2 <output | sort >actual &&
473 test_cmp expect actual
474 "
475 done
476
477 test_expect_success 'fetch refs from stdin' '
478 (
479 cd client &&
480 git fetch-pack --stdin --no-progress .. <../input
481 ) >output &&
482 cut -d " " -f 2 <output | sort >actual &&
483 test_cmp expect actual
484 '
485
486 test_expect_success 'fetch mixed refs from cmdline and stdin' '
487 (
488 cd client &&
489 tail -n +5 ../input |
490 git fetch-pack --stdin --no-progress .. $(head -n 4 ../input)
491 ) >output &&
492 cut -d " " -f 2 <output | sort >actual &&
493 test_cmp expect actual
494 '
495
496 test_expect_success 'test duplicate refs from stdin' '
497 (
498 cd client &&
499 git fetch-pack --stdin --no-progress .. <../input.dup
500 ) >output &&
501 cut -d " " -f 2 <output | sort >actual &&
502 test_cmp expect actual
503 '
504
505 test_expect_success 'set up tests of missing reference' '
506 cat >expect-error <<-\EOF
507 error: no such remote ref refs/heads/xyzzy
508 EOF
509 '
510
511 test_expect_success 'test lonely missing ref' '
512 (
513 cd client &&
514 test_must_fail git fetch-pack --no-progress .. refs/heads/xyzzy 2>../error-m
515 ) &&
516 test_i18ncmp expect-error error-m
517 '
518
519 test_expect_success 'test missing ref after existing' '
520 (
521 cd client &&
522 test_must_fail git fetch-pack --no-progress .. refs/heads/A refs/heads/xyzzy 2>../error-em
523 ) &&
524 test_i18ncmp expect-error error-em
525 '
526
527 test_expect_success 'test missing ref before existing' '
528 (
529 cd client &&
530 test_must_fail git fetch-pack --no-progress .. refs/heads/xyzzy refs/heads/A 2>../error-me
531 ) &&
532 test_i18ncmp expect-error error-me
533 '
534
535 test_expect_success 'test --all, --depth, and explicit head' '
536 (
537 cd client &&
538 git fetch-pack --no-progress --all --depth=1 .. refs/heads/A
539 ) >out-adh 2>error-adh
540 '
541
542 test_expect_success 'test --all, --depth, and explicit tag' '
543 git tag OLDTAG refs/heads/B~5 &&
544 (
545 cd client &&
546 git fetch-pack --no-progress --all --depth=1 .. refs/tags/OLDTAG
547 ) >out-adt 2>error-adt
548 '
549
550 test_expect_success 'test --all with tag to non-tip' '
551 git commit --allow-empty -m non-tip &&
552 git commit --allow-empty -m tip &&
553 git tag -m "annotated" non-tip HEAD^ &&
554 (
555 cd client &&
556 git fetch-pack --all ..
557 )
558 '
559
560 test_expect_success 'test --all wrt tag to non-commits' '
561 # create tag-to-{blob,tree,commit,tag}, making sure all tagged objects
562 # are reachable only via created tag references.
563 blob=$(echo "hello blob" | git hash-object -t blob -w --stdin) &&
564 git tag -a -m "tag -> blob" tag-to-blob $blob &&
565
566 tree=$(printf "100644 blob $blob\tfile" | git mktree) &&
567 git tag -a -m "tag -> tree" tag-to-tree $tree &&
568
569 tree2=$(printf "100644 blob $blob\tfile2" | git mktree) &&
570 commit=$(git commit-tree -m "hello commit" $tree) &&
571 git tag -a -m "tag -> commit" tag-to-commit $commit &&
572
573 blob2=$(echo "hello blob2" | git hash-object -t blob -w --stdin) &&
574 tag=$(git mktag <<-EOF
575 object $blob2
576 type blob
577 tag tag-to-blob2
578 tagger author A U Thor <author@example.com> 0 +0000
579
580 hello tag
581 EOF
582 ) &&
583 git tag -a -m "tag -> tag" tag-to-tag $tag &&
584
585 # `fetch-pack --all` should succeed fetching all those objects.
586 mkdir fetchall &&
587 (
588 cd fetchall &&
589 git init &&
590 git fetch-pack --all .. &&
591 git cat-file blob $blob >/dev/null &&
592 git cat-file tree $tree >/dev/null &&
593 git cat-file commit $commit >/dev/null &&
594 git cat-file tag $tag >/dev/null
595 )
596 '
597
598 test_expect_success 'shallow fetch with tags does not break the repository' '
599 mkdir repo1 &&
600 (
601 cd repo1 &&
602 git init &&
603 test_commit 1 &&
604 test_commit 2 &&
605 test_commit 3 &&
606 mkdir repo2 &&
607 cd repo2 &&
608 git init &&
609 git fetch --depth=2 ../.git master:branch &&
610 git fsck
611 )
612 '
613
614 test_expect_success 'fetch-pack can fetch a raw sha1' '
615 git init hidden &&
616 (
617 cd hidden &&
618 test_commit 1 &&
619 test_commit 2 &&
620 git update-ref refs/hidden/one HEAD^ &&
621 git config transfer.hiderefs refs/hidden &&
622 git config uploadpack.allowtipsha1inwant true
623 ) &&
624 git fetch-pack hidden $(git -C hidden rev-parse refs/hidden/one)
625 '
626
627 test_expect_success 'fetch-pack can fetch a raw sha1 that is advertised as a ref' '
628 rm -rf server client &&
629 git init server &&
630 test_commit -C server 1 &&
631
632 git init client &&
633 git -C client fetch-pack ../server \
634 $(git -C server rev-parse refs/heads/master)
635 '
636
637 test_expect_success 'fetch-pack can fetch a raw sha1 overlapping a named ref' '
638 rm -rf server client &&
639 git init server &&
640 test_commit -C server 1 &&
641 test_commit -C server 2 &&
642
643 git init client &&
644 git -C client fetch-pack ../server \
645 $(git -C server rev-parse refs/tags/1) refs/tags/1
646 '
647
648 test_expect_success 'fetch-pack cannot fetch a raw sha1 that is not advertised as a ref' '
649 rm -rf server &&
650
651 git init server &&
652 test_commit -C server 5 &&
653 git -C server tag -d 5 &&
654 test_commit -C server 6 &&
655
656 git init client &&
657 # Some protocol versions (e.g. 2) support fetching
658 # unadvertised objects, so restrict this test to v0.
659 test_must_fail env GIT_TEST_PROTOCOL_VERSION= git -C client fetch-pack ../server \
660 $(git -C server rev-parse refs/heads/master^) 2>err &&
661 test_i18ngrep "Server does not allow request for unadvertised object" err
662 '
663
664 check_prot_path () {
665 cat >expected <<-EOF &&
666 Diag: url=$1
667 Diag: protocol=$2
668 Diag: path=$3
669 EOF
670 git fetch-pack --diag-url "$1" | grep -v hostandport= >actual &&
671 test_cmp expected actual
672 }
673
674 check_prot_host_port_path () {
675 case "$2" in
676 *ssh*)
677 pp=ssh
678 uah=userandhost
679 ehost=$(echo $3 | tr -d "[]")
680 diagport="Diag: port=$4"
681 ;;
682 *)
683 pp=$p
684 uah=hostandport
685 ehost=$(echo $3$4 | sed -e "s/22$/:22/" -e "s/NONE//")
686 diagport=""
687 ;;
688 esac
689 cat >exp <<-EOF &&
690 Diag: url=$1
691 Diag: protocol=$pp
692 Diag: $uah=$ehost
693 $diagport
694 Diag: path=$5
695 EOF
696 grep -v "^$" exp >expected
697 git fetch-pack --diag-url "$1" >actual &&
698 test_cmp expected actual
699 }
700
701 for r in repo re:po re/po
702 do
703 # git or ssh with scheme
704 for p in "ssh+git" "git+ssh" git ssh
705 do
706 for h in host user@host user@[::1] user@::1
707 do
708 for c in "" :
709 do
710 test_expect_success "fetch-pack --diag-url $p://$h$c/$r" '
711 check_prot_host_port_path $p://$h/$r $p "$h" NONE "/$r"
712 '
713 # "/~" -> "~" conversion
714 test_expect_success "fetch-pack --diag-url $p://$h$c/~$r" '
715 check_prot_host_port_path $p://$h/~$r $p "$h" NONE "~$r"
716 '
717 done
718 done
719 for h in host User@host User@[::1]
720 do
721 test_expect_success "fetch-pack --diag-url $p://$h:22/$r" '
722 check_prot_host_port_path $p://$h:22/$r $p "$h" 22 "/$r"
723 '
724 done
725 done
726 # file with scheme
727 for p in file
728 do
729 test_expect_success !MINGW "fetch-pack --diag-url $p://$h/$r" '
730 check_prot_path $p://$h/$r $p "/$r"
731 '
732 test_expect_success MINGW "fetch-pack --diag-url $p://$h/$r" '
733 check_prot_path $p://$h/$r $p "//$h/$r"
734 '
735 test_expect_success MINGW "fetch-pack --diag-url $p:///$r" '
736 check_prot_path $p:///$r $p "/$r"
737 '
738 # No "/~" -> "~" conversion for file
739 test_expect_success !MINGW "fetch-pack --diag-url $p://$h/~$r" '
740 check_prot_path $p://$h/~$r $p "/~$r"
741 '
742 test_expect_success MINGW "fetch-pack --diag-url $p://$h/~$r" '
743 check_prot_path $p://$h/~$r $p "//$h/~$r"
744 '
745 done
746 # file without scheme
747 for h in nohost nohost:12 [::1] [::1]:23 [ [:aa
748 do
749 test_expect_success "fetch-pack --diag-url ./$h:$r" '
750 check_prot_path ./$h:$r $p "./$h:$r"
751 '
752 # No "/~" -> "~" conversion for file
753 test_expect_success "fetch-pack --diag-url ./$p:$h/~$r" '
754 check_prot_path ./$p:$h/~$r $p "./$p:$h/~$r"
755 '
756 done
757 #ssh without scheme
758 p=ssh
759 for h in host [::1]
760 do
761 test_expect_success "fetch-pack --diag-url $h:$r" '
762 check_prot_host_port_path $h:$r $p "$h" NONE "$r"
763 '
764 # Do "/~" -> "~" conversion
765 test_expect_success "fetch-pack --diag-url $h:/~$r" '
766 check_prot_host_port_path $h:/~$r $p "$h" NONE "~$r"
767 '
768 done
769 done
770
771 test_expect_success MINGW 'fetch-pack --diag-url file://c:/repo' '
772 check_prot_path file://c:/repo file c:/repo
773 '
774 test_expect_success MINGW 'fetch-pack --diag-url c:repo' '
775 check_prot_path c:repo file c:repo
776 '
777
778 test_expect_success 'clone shallow since ...' '
779 test_create_repo shallow-since &&
780 (
781 cd shallow-since &&
782 GIT_COMMITTER_DATE="100000000 +0700" git commit --allow-empty -m one &&
783 GIT_COMMITTER_DATE="200000000 +0700" git commit --allow-empty -m two &&
784 GIT_COMMITTER_DATE="300000000 +0700" git commit --allow-empty -m three &&
785 git clone --shallow-since "300000000 +0700" "file://$(pwd)/." ../shallow11 &&
786 git -C ../shallow11 log --pretty=tformat:%s HEAD >actual &&
787 echo three >expected &&
788 test_cmp expected actual
789 )
790 '
791
792 test_expect_success 'fetch shallow since ...' '
793 git -C shallow11 fetch --shallow-since "200000000 +0700" origin &&
794 git -C shallow11 log --pretty=tformat:%s origin/master >actual &&
795 cat >expected <<-\EOF &&
796 three
797 two
798 EOF
799 test_cmp expected actual
800 '
801
802 test_expect_success 'clone shallow since selects no commits' '
803 test_create_repo shallow-since-the-future &&
804 (
805 cd shallow-since-the-future &&
806 GIT_COMMITTER_DATE="100000000 +0700" git commit --allow-empty -m one &&
807 GIT_COMMITTER_DATE="200000000 +0700" git commit --allow-empty -m two &&
808 GIT_COMMITTER_DATE="300000000 +0700" git commit --allow-empty -m three &&
809 test_must_fail git clone --shallow-since "900000000 +0700" "file://$(pwd)/." ../shallow111
810 )
811 '
812
813 # A few subtle things about the request in this test:
814 #
815 # - the server must have commit-graphs present and enabled
816 #
817 # - the history is such that our want/have share a common ancestor ("base"
818 # here)
819 #
820 # - we send only a single have, which is fewer than a normal client would
821 # send. This ensures that we don't parse "base" up front with
822 # parse_object(), but rather traverse to it as a parent while deciding if we
823 # can stop the "have" negotiation, and call parse_commit(). The former
824 # sees the actual object data and so always loads the three oid, whereas the
825 # latter will try to load it lazily.
826 #
827 # - we must use protocol v2, because it handles the "have" negotiation before
828 # processing the shallow directives
829 #
830 test_expect_success 'shallow since with commit graph and already-seen commit' '
831 test_create_repo shallow-since-graph &&
832 (
833 cd shallow-since-graph &&
834 test_commit base &&
835 test_commit master &&
836 git checkout -b other HEAD^ &&
837 test_commit other &&
838 git commit-graph write --reachable &&
839 git config core.commitGraph true &&
840
841 GIT_PROTOCOL=version=2 git upload-pack . <<-EOF >/dev/null
842 0012command=fetch
843 00010013deepen-since 1
844 0032want $(git rev-parse other)
845 0032have $(git rev-parse master)
846 0000
847 EOF
848 )
849 '
850
851 test_expect_success 'shallow clone exclude tag two' '
852 test_create_repo shallow-exclude &&
853 (
854 cd shallow-exclude &&
855 test_commit one &&
856 test_commit two &&
857 test_commit three &&
858 git clone --shallow-exclude two "file://$(pwd)/." ../shallow12 &&
859 git -C ../shallow12 log --pretty=tformat:%s HEAD >actual &&
860 echo three >expected &&
861 test_cmp expected actual
862 )
863 '
864
865 test_expect_success 'fetch exclude tag one' '
866 git -C shallow12 fetch --shallow-exclude one origin &&
867 git -C shallow12 log --pretty=tformat:%s origin/master >actual &&
868 test_write_lines three two >expected &&
869 test_cmp expected actual
870 '
871
872 test_expect_success 'fetching deepen' '
873 test_create_repo shallow-deepen &&
874 (
875 cd shallow-deepen &&
876 test_commit one &&
877 test_commit two &&
878 test_commit three &&
879 git clone --depth 1 "file://$(pwd)/." deepen &&
880 test_commit four &&
881 git -C deepen log --pretty=tformat:%s master >actual &&
882 echo three >expected &&
883 test_cmp expected actual &&
884 git -C deepen fetch --deepen=1 &&
885 git -C deepen log --pretty=tformat:%s origin/master >actual &&
886 cat >expected <<-\EOF &&
887 four
888 three
889 two
890 EOF
891 test_cmp expected actual
892 )
893 '
894
895 test_expect_success 'use ref advertisement to prune "have" lines sent' '
896 rm -rf server client &&
897 git init server &&
898 test_commit -C server both_have_1 &&
899 git -C server tag -d both_have_1 &&
900 test_commit -C server both_have_2 &&
901
902 git clone server client &&
903 test_commit -C server server_has &&
904 test_commit -C client client_has &&
905
906 # In both protocol v0 and v2, ensure that the parent of both_have_2 is
907 # not sent as a "have" line. The client should know that the server has
908 # both_have_2, so it only needs to inform the server that it has
909 # both_have_2, and the server can infer the rest.
910
911 rm -f trace &&
912 cp -r client clientv0 &&
913 GIT_TRACE_PACKET="$(pwd)/trace" git -C clientv0 \
914 fetch origin server_has both_have_2 &&
915 grep "have $(git -C client rev-parse client_has)" trace &&
916 grep "have $(git -C client rev-parse both_have_2)" trace &&
917 ! grep "have $(git -C client rev-parse both_have_2^)" trace &&
918
919 rm -f trace &&
920 cp -r client clientv2 &&
921 GIT_TRACE_PACKET="$(pwd)/trace" git -C clientv2 -c protocol.version=2 \
922 fetch origin server_has both_have_2 &&
923 grep "have $(git -C client rev-parse client_has)" trace &&
924 grep "have $(git -C client rev-parse both_have_2)" trace &&
925 ! grep "have $(git -C client rev-parse both_have_2^)" trace
926 '
927
928 test_expect_success 'filtering by size' '
929 rm -rf server client &&
930 test_create_repo server &&
931 test_commit -C server one &&
932 test_config -C server uploadpack.allowfilter 1 &&
933
934 test_create_repo client &&
935 git -C client fetch-pack --filter=blob:limit=0 ../server HEAD &&
936
937 # Ensure that object is not inadvertently fetched
938 test_must_fail git -C client cat-file -e $(git hash-object server/one.t)
939 '
940
941 test_expect_success 'filtering by size has no effect if support for it is not advertised' '
942 rm -rf server client &&
943 test_create_repo server &&
944 test_commit -C server one &&
945
946 test_create_repo client &&
947 git -C client fetch-pack --filter=blob:limit=0 ../server HEAD 2> err &&
948
949 # Ensure that object is fetched
950 git -C client cat-file -e $(git hash-object server/one.t) &&
951
952 test_i18ngrep "filtering not recognized by server" err
953 '
954
955 fetch_filter_blob_limit_zero () {
956 SERVER="$1"
957 URL="$2"
958
959 rm -rf "$SERVER" client &&
960 test_create_repo "$SERVER" &&
961 test_commit -C "$SERVER" one &&
962 test_config -C "$SERVER" uploadpack.allowfilter 1 &&
963
964 git clone "$URL" client &&
965 test_config -C client extensions.partialclone origin &&
966
967 test_commit -C "$SERVER" two &&
968
969 git -C client fetch --filter=blob:limit=0 origin HEAD:somewhere &&
970
971 # Ensure that commit is fetched, but blob is not
972 test_config -C client extensions.partialclone "arbitrary string" &&
973 git -C client cat-file -e $(git -C "$SERVER" rev-parse two) &&
974 test_must_fail git -C client cat-file -e $(git hash-object "$SERVER/two.t")
975 }
976
977 test_expect_success 'fetch with --filter=blob:limit=0' '
978 fetch_filter_blob_limit_zero server server
979 '
980
981 . "$TEST_DIRECTORY"/lib-httpd.sh
982 start_httpd
983
984 test_expect_success 'fetch with --filter=blob:limit=0 and HTTP' '
985 fetch_filter_blob_limit_zero "$HTTPD_DOCUMENT_ROOT_PATH/server" "$HTTPD_URL/smart/server"
986 '
987
988 # DO NOT add non-httpd-specific tests here, because the last part of this
989 # test script is only executed when httpd is available and enabled.
990
991 test_done