3 test_description
='test fetching bundles with --bundle-uri'
6 .
"$TEST_DIRECTORY"/lib-bundle.sh
8 test_expect_success
'fail to clone from non-existent file' '
9 test_when_finished rm -rf test &&
10 git clone --bundle-uri="$(pwd)/does-not-exist" . test 2>err &&
11 grep "failed to download bundle from URI" err
14 test_expect_success
'fail to clone from non-bundle file' '
15 test_when_finished rm -rf test &&
17 git clone --bundle-uri="$(pwd)/bogus" . test 2>err &&
18 grep "is not a bundle" err
21 test_expect_success
'create bundle' '
22 git init clone-from &&
25 git checkout -b topic &&
28 git bundle create A.bundle topic &&
31 git bundle create B.bundle topic &&
33 # Create a bundle with reference pointing to non-existent object.
34 commit_a=$(git rev-parse A) &&
35 commit_b=$(git rev-parse B) &&
36 sed -e "/^$/q" -e "s/$commit_a /$commit_b /" \
37 <A.bundle >bad-header.bundle &&
38 convert_bundle_to_pack \
39 <A.bundle >>bad-header.bundle &&
41 tree_b=$(git rev-parse B^{tree}) &&
48 commit: this is a commit with bad emails
51 bad_commit=$(git hash-object --literally -t commit -w --stdin <data) &&
52 git branch bad $bad_commit &&
53 git bundle create bad-object.bundle bad &&
54 git update-ref -d refs/heads/bad
58 test_expect_success
'clone with path bundle' '
59 git clone --bundle-uri="clone-from/B.bundle" \
60 clone-from clone-path &&
61 git -C clone-path rev-parse refs/bundles/topic >actual &&
62 git -C clone-from rev-parse topic >expect &&
63 test_cmp expect actual
66 test_expect_success
'clone with bundle that has bad header' '
67 # Write bundle ref fails, but clone can still proceed.
68 git clone --bundle-uri="clone-from/bad-header.bundle" \
69 clone-from clone-bad-header 2>err &&
70 commit_b=$(git -C clone-from rev-parse B) &&
71 test_grep "trying to write ref '\''refs/bundles/topic'\'' with nonexistent object $commit_b" err &&
72 git -C clone-bad-header for-each-ref --format="%(refname)" >refs &&
73 test_grep ! "refs/bundles/" refs
76 test_expect_success
'clone with bundle that has bad object' '
77 # Unbundle succeeds if no fsckObjects configured.
78 git clone --bundle-uri="clone-from/bad-object.bundle" \
79 clone-from clone-bad-object-no-fsck &&
80 git -C clone-bad-object-no-fsck for-each-ref --format="%(refname)" >refs &&
81 grep "refs/bundles/" refs >actual &&
82 test_write_lines refs/bundles/bad >expect &&
83 test_cmp expect actual &&
85 # Unbundle fails with fsckObjects set true, but clone can still proceed.
86 git -c fetch.fsckObjects=true clone --bundle-uri="clone-from/bad-object.bundle" \
87 clone-from clone-bad-object-fsck 2>err &&
88 test_grep "missingEmail" err &&
89 git -C clone-bad-object-fsck for-each-ref --format="%(refname)" >refs &&
90 test_grep ! "refs/bundles/" refs
93 test_expect_success
'clone with path bundle and non-default hash' '
94 test_when_finished "rm -rf clone-path-non-default-hash" &&
95 GIT_DEFAULT_HASH=sha256 git clone --bundle-uri="clone-from/B.bundle" \
96 clone-from clone-path-non-default-hash &&
97 git -C clone-path-non-default-hash rev-parse refs/bundles/topic >actual &&
98 git -C clone-from rev-parse topic >expect &&
99 test_cmp expect actual
102 test_expect_success
'clone with file:// bundle' '
103 git clone --bundle-uri="file://$(pwd)/clone-from/B.bundle" \
104 clone-from clone-file &&
105 git -C clone-file rev-parse refs/bundles/topic >actual &&
106 git -C clone-from rev-parse topic >expect &&
107 test_cmp expect actual
110 # To get interesting tests for bundle lists, we need to construct a
111 # somewhat-interesting commit history.
113 # ---------------- bundle-4
117 # ----|---|------- bundle-3
121 # ----|---|------- bundle-2
125 # ----|---|------- bundle-1
130 test_expect_success
'construct incremental bundle list' '
133 git checkout -b base &&
135 git checkout -b left &&
137 git checkout -b right base &&
139 git checkout -b merge left &&
140 git merge right -m "4" &&
142 git bundle create bundle-1.bundle base &&
143 git bundle create bundle-2.bundle base..left &&
144 git bundle create bundle-3.bundle base..right &&
145 git bundle create bundle-4.bundle merge --not left right
149 test_expect_success
'clone bundle list (file, no heuristic)' '
150 cat >bundle-list <<-EOF &&
156 uri = file://$(pwd)/clone-from/bundle-1.bundle
159 uri = file://$(pwd)/clone-from/bundle-2.bundle
162 uri = file://$(pwd)/clone-from/bundle-3.bundle
165 uri = file://$(pwd)/clone-from/bundle-4.bundle
168 git clone --bundle-uri="file://$(pwd)/bundle-list" \
169 clone-from clone-list-file 2>err &&
170 ! grep "Repository lacks these prerequisite commits" err &&
172 git -C clone-from for-each-ref --format="%(objectname)" >oids &&
173 git -C clone-list-file cat-file --batch-check <oids &&
175 git -C clone-list-file for-each-ref --format="%(refname)" >refs &&
176 grep "refs/bundles/" refs >actual &&
177 cat >expect <<-\EOF &&
183 test_cmp expect actual
186 test_expect_success
'clone bundle list (file, all mode, some failures)' '
187 cat >bundle-list <<-EOF &&
192 # Does not exist. Should be skipped.
194 uri = file://$(pwd)/clone-from/bundle-0.bundle
197 uri = file://$(pwd)/clone-from/bundle-1.bundle
200 uri = file://$(pwd)/clone-from/bundle-2.bundle
202 # No bundle-3 means bundle-4 will not apply.
205 uri = file://$(pwd)/clone-from/bundle-4.bundle
207 # Does not exist. Should be skipped.
209 uri = file://$(pwd)/clone-from/bundle-5.bundle
213 git clone --bundle-uri="file://$(pwd)/bundle-list" \
214 clone-from clone-all-some 2>err &&
215 ! grep "Repository lacks these prerequisite commits" err &&
216 ! grep "fatal" err &&
217 grep "warning: failed to download bundle from URI" err &&
219 git -C clone-from for-each-ref --format="%(objectname)" >oids &&
220 git -C clone-all-some cat-file --batch-check <oids &&
222 git -C clone-all-some for-each-ref --format="%(refname)" >refs &&
223 grep "refs/bundles/" refs >actual &&
224 cat >expect <<-\EOF &&
228 test_cmp expect actual
231 test_expect_success
'clone bundle list (file, all mode, all failures)' '
232 cat >bundle-list <<-EOF &&
237 # Does not exist. Should be skipped.
239 uri = file://$(pwd)/clone-from/bundle-0.bundle
241 # Does not exist. Should be skipped.
243 uri = file://$(pwd)/clone-from/bundle-5.bundle
246 git clone --bundle-uri="file://$(pwd)/bundle-list" \
247 clone-from clone-all-fail 2>err &&
248 ! grep "Repository lacks these prerequisite commits" err &&
249 ! grep "fatal" err &&
250 grep "warning: failed to download bundle from URI" err &&
252 git -C clone-from for-each-ref --format="%(objectname)" >oids &&
253 git -C clone-all-fail cat-file --batch-check <oids &&
255 git -C clone-all-fail for-each-ref --format="%(refname)" >refs &&
256 ! grep "refs/bundles/" refs
259 test_expect_success
'clone bundle list (file, any mode)' '
260 cat >bundle-list <<-EOF &&
265 # Does not exist. Should be skipped.
267 uri = file://$(pwd)/clone-from/bundle-0.bundle
270 uri = file://$(pwd)/clone-from/bundle-1.bundle
272 # Does not exist. Should be skipped.
274 uri = file://$(pwd)/clone-from/bundle-5.bundle
277 git clone --bundle-uri="file://$(pwd)/bundle-list" \
278 clone-from clone-any-file 2>err &&
279 ! grep "Repository lacks these prerequisite commits" err &&
281 git -C clone-from for-each-ref --format="%(objectname)" >oids &&
282 git -C clone-any-file cat-file --batch-check <oids &&
284 git -C clone-any-file for-each-ref --format="%(refname)" >refs &&
285 grep "refs/bundles/" refs >actual &&
286 cat >expect <<-\EOF &&
289 test_cmp expect actual
292 test_expect_success
'clone bundle list (file, any mode, all failures)' '
293 cat >bundle-list <<-EOF &&
298 # Does not exist. Should be skipped.
300 uri = $HTTPD_URL/bundle-0.bundle
302 # Does not exist. Should be skipped.
304 uri = $HTTPD_URL/bundle-5.bundle
307 git clone --bundle-uri="file://$(pwd)/bundle-list" \
308 clone-from clone-any-fail 2>err &&
309 ! grep "fatal" err &&
310 grep "warning: failed to download bundle from URI" err &&
312 git -C clone-from for-each-ref --format="%(objectname)" >oids &&
313 git -C clone-any-fail cat-file --batch-check <oids &&
315 git -C clone-any-fail for-each-ref --format="%(refname)" >refs &&
316 ! grep "refs/bundles/" refs
319 test_expect_success
'negotiation: bundle with part of wanted commits' '
320 test_when_finished "rm -f trace*.txt" &&
321 GIT_TRACE_PACKET="$(pwd)/trace-packet.txt" \
322 git clone --no-local --bundle-uri="clone-from/A.bundle" \
323 clone-from nego-bundle-part &&
324 git -C nego-bundle-part for-each-ref --format="%(refname)" >refs &&
325 grep "refs/bundles/" refs >actual &&
326 test_write_lines refs/bundles/topic >expect &&
327 test_cmp expect actual &&
328 # Ensure that refs/bundles/topic are sent as "have".
329 tip=$(git -C clone-from rev-parse A) &&
330 test_grep "clone> have $tip" trace-packet.txt
333 test_expect_success
'negotiation: bundle with all wanted commits' '
334 test_when_finished "rm -f trace*.txt" &&
335 GIT_TRACE_PACKET="$(pwd)/trace-packet.txt" \
336 git clone --no-local --single-branch --branch=topic --no-tags \
337 --bundle-uri="clone-from/B.bundle" \
338 clone-from nego-bundle-all &&
339 git -C nego-bundle-all for-each-ref --format="%(refname)" >refs &&
340 grep "refs/bundles/" refs >actual &&
341 test_write_lines refs/bundles/topic >expect &&
342 test_cmp expect actual &&
343 # We already have all needed commits so no "want" needed.
344 test_grep ! "clone> want " trace-packet.txt
347 test_expect_success
'negotiation: bundle list (no heuristic)' '
348 test_when_finished "rm -f trace*.txt" &&
349 cat >bundle-list <<-EOF &&
355 uri = file://$(pwd)/clone-from/bundle-1.bundle
358 uri = file://$(pwd)/clone-from/bundle-2.bundle
361 GIT_TRACE_PACKET="$(pwd)/trace-packet.txt" \
362 git clone --no-local --bundle-uri="file://$(pwd)/bundle-list" \
363 clone-from nego-bundle-list-no-heuristic &&
365 git -C nego-bundle-list-no-heuristic for-each-ref --format="%(refname)" >refs &&
366 grep "refs/bundles/" refs >actual &&
367 cat >expect <<-\EOF &&
371 test_cmp expect actual &&
372 tip=$(git -C nego-bundle-list-no-heuristic rev-parse refs/bundles/left) &&
373 test_grep "clone> have $tip" trace-packet.txt
376 test_expect_success
'negotiation: bundle list (creationToken)' '
377 test_when_finished "rm -f trace*.txt" &&
378 cat >bundle-list <<-EOF &&
382 heuristic = creationToken
385 uri = file://$(pwd)/clone-from/bundle-1.bundle
389 uri = file://$(pwd)/clone-from/bundle-2.bundle
393 GIT_TRACE_PACKET="$(pwd)/trace-packet.txt" \
394 git clone --no-local --bundle-uri="file://$(pwd)/bundle-list" \
395 clone-from nego-bundle-list-heuristic &&
397 git -C nego-bundle-list-heuristic for-each-ref --format="%(refname)" >refs &&
398 grep "refs/bundles/" refs >actual &&
399 cat >expect <<-\EOF &&
403 test_cmp expect actual &&
404 tip=$(git -C nego-bundle-list-heuristic rev-parse refs/bundles/left) &&
405 test_grep "clone> have $tip" trace-packet.txt
408 test_expect_success
'negotiation: bundle list with all wanted commits' '
409 test_when_finished "rm -f trace*.txt" &&
410 cat >bundle-list <<-EOF &&
414 heuristic = creationToken
417 uri = file://$(pwd)/clone-from/bundle-1.bundle
421 uri = file://$(pwd)/clone-from/bundle-2.bundle
425 GIT_TRACE_PACKET="$(pwd)/trace-packet.txt" \
426 git clone --no-local --single-branch --branch=left --no-tags \
427 --bundle-uri="file://$(pwd)/bundle-list" \
428 clone-from nego-bundle-list-all &&
430 git -C nego-bundle-list-all for-each-ref --format="%(refname)" >refs &&
431 grep "refs/bundles/" refs >actual &&
432 cat >expect <<-\EOF &&
436 test_cmp expect actual &&
437 # We already have all needed commits so no "want" needed.
438 test_grep ! "clone> want " trace-packet.txt
441 #########################################################################
442 # HTTP tests begin here
444 .
"$TEST_DIRECTORY"/lib-httpd.sh
447 test_expect_success
'fail to fetch from non-existent HTTP URL' '
448 test_when_finished rm -rf test &&
449 git clone --bundle-uri="$HTTPD_URL/does-not-exist" . test 2>err &&
450 grep "failed to download bundle from URI" err
453 test_expect_success
'fail to fetch from non-bundle HTTP URL' '
454 test_when_finished rm -rf test &&
455 echo bogus >"$HTTPD_DOCUMENT_ROOT_PATH/bogus" &&
456 git clone --bundle-uri="$HTTPD_URL/bogus" . test 2>err &&
457 grep "is not a bundle" err
460 test_expect_success
'clone HTTP bundle' '
461 cp clone-from/B.bundle "$HTTPD_DOCUMENT_ROOT_PATH/B.bundle" &&
463 git clone --no-local --mirror clone-from \
464 "$HTTPD_DOCUMENT_ROOT_PATH/fetch.git" &&
466 git clone --bundle-uri="$HTTPD_URL/B.bundle" \
467 "$HTTPD_URL/smart/fetch.git" clone-http &&
468 git -C clone-http rev-parse refs/bundles/topic >actual &&
469 git -C clone-from rev-parse topic >expect &&
470 test_cmp expect actual &&
472 test_config -C clone-http log.excludedecoration refs/bundle/
475 test_expect_success
'clone HTTP bundle with non-default hash' '
476 test_when_finished "rm -rf clone-http-non-default-hash" &&
477 GIT_DEFAULT_HASH=sha256 git clone --bundle-uri="$HTTPD_URL/B.bundle" \
478 "$HTTPD_URL/smart/fetch.git" clone-http-non-default-hash &&
479 git -C clone-http-non-default-hash rev-parse refs/bundles/topic >actual &&
480 git -C clone-from rev-parse topic >expect &&
481 test_cmp expect actual
484 test_expect_success
'clone bundle list (HTTP, no heuristic)' '
485 test_when_finished rm -f trace*.txt &&
487 cp clone-from/bundle-*.bundle "$HTTPD_DOCUMENT_ROOT_PATH/" &&
488 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
494 uri = $HTTPD_URL/bundle-1.bundle
497 uri = $HTTPD_URL/bundle-2.bundle
500 uri = $HTTPD_URL/bundle-3.bundle
503 uri = $HTTPD_URL/bundle-4.bundle
506 GIT_TRACE2_EVENT="$(pwd)/trace-clone.txt" \
507 git clone --bundle-uri="$HTTPD_URL/bundle-list" \
508 clone-from clone-list-http 2>err &&
509 ! grep "Repository lacks these prerequisite commits" err &&
511 git -C clone-from for-each-ref --format="%(objectname)" >oids &&
512 git -C clone-list-http cat-file --batch-check <oids &&
514 cat >expect <<-EOF &&
515 $HTTPD_URL/bundle-1.bundle
516 $HTTPD_URL/bundle-2.bundle
517 $HTTPD_URL/bundle-3.bundle
518 $HTTPD_URL/bundle-4.bundle
519 $HTTPD_URL/bundle-list
522 # Sort the list, since the order is not well-defined
523 # without a heuristic.
524 test_remote_https_urls <trace-clone.txt | sort >actual &&
525 test_cmp expect actual
528 test_expect_success
'clone bundle list (HTTP, any mode)' '
529 cp clone-from/bundle-*.bundle "$HTTPD_DOCUMENT_ROOT_PATH/" &&
530 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
535 # Does not exist. Should be skipped.
537 uri = $HTTPD_URL/bundle-0.bundle
540 uri = $HTTPD_URL/bundle-1.bundle
542 # Does not exist. Should be skipped.
544 uri = $HTTPD_URL/bundle-5.bundle
547 git clone --bundle-uri="$HTTPD_URL/bundle-list" \
548 clone-from clone-any-http 2>err &&
549 ! grep "fatal" err &&
550 grep "warning: failed to download bundle from URI" err &&
552 git -C clone-from for-each-ref --format="%(objectname)" >oids &&
553 git -C clone-any-http cat-file --batch-check <oids &&
555 git -C clone-list-file for-each-ref --format="%(refname)" >refs &&
556 grep "refs/bundles/" refs >actual &&
557 cat >expect <<-\EOF &&
563 test_cmp expect actual
566 test_expect_success
'clone bundle list (http, creationToken)' '
567 test_when_finished rm -f trace*.txt &&
569 cp clone-from/bundle-*.bundle "$HTTPD_DOCUMENT_ROOT_PATH/" &&
570 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
574 heuristic = creationToken
577 uri = bundle-1.bundle
581 uri = bundle-2.bundle
585 uri = bundle-3.bundle
589 uri = bundle-4.bundle
593 GIT_TRACE2_EVENT="$(pwd)/trace-clone.txt" git \
594 clone --bundle-uri="$HTTPD_URL/bundle-list" \
595 "$HTTPD_URL/smart/fetch.git" clone-list-http-2 &&
597 git -C clone-from for-each-ref --format="%(objectname)" >oids &&
598 git -C clone-list-http-2 cat-file --batch-check <oids &&
600 cat >expect <<-EOF &&
601 $HTTPD_URL/bundle-list
602 $HTTPD_URL/bundle-4.bundle
603 $HTTPD_URL/bundle-3.bundle
604 $HTTPD_URL/bundle-2.bundle
605 $HTTPD_URL/bundle-1.bundle
608 test_remote_https_urls <trace-clone.txt >actual &&
609 test_cmp expect actual
612 test_expect_success
'clone incomplete bundle list (http, creationToken)' '
613 test_when_finished rm -f trace*.txt &&
615 cp clone-from/bundle-*.bundle "$HTTPD_DOCUMENT_ROOT_PATH/" &&
616 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
620 heuristic = creationToken
623 uri = bundle-1.bundle
627 GIT_TRACE2_EVENT=$(pwd)/trace-clone.txt \
628 git clone --bundle-uri="$HTTPD_URL/bundle-list" \
629 --single-branch --branch=base --no-tags \
630 "$HTTPD_URL/smart/fetch.git" clone-token-http &&
632 test_cmp_config -C clone-token-http "$HTTPD_URL/bundle-list" fetch.bundleuri &&
633 test_cmp_config -C clone-token-http 1 fetch.bundlecreationtoken &&
635 cat >expect <<-EOF &&
636 $HTTPD_URL/bundle-list
637 $HTTPD_URL/bundle-1.bundle
640 test_remote_https_urls <trace-clone.txt >actual &&
641 test_cmp expect actual &&
643 # We now have only one bundle ref.
644 git -C clone-token-http for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
645 cat >expect <<-\EOF &&
648 test_cmp expect refs &&
650 # Add remaining bundles, exercising the "deepening" strategy
651 # for downloading via the creationToken heurisitc.
652 cat >>"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
654 uri = bundle-2.bundle
658 uri = bundle-3.bundle
662 uri = bundle-4.bundle
666 GIT_TRACE2_EVENT="$(pwd)/trace1.txt" \
667 git -C clone-token-http fetch origin --no-tags \
668 refs/heads/merge:refs/heads/merge &&
669 test_cmp_config -C clone-token-http 4 fetch.bundlecreationtoken &&
671 cat >expect <<-EOF &&
672 $HTTPD_URL/bundle-list
673 $HTTPD_URL/bundle-4.bundle
674 $HTTPD_URL/bundle-3.bundle
675 $HTTPD_URL/bundle-2.bundle
678 test_remote_https_urls <trace1.txt >actual &&
679 test_cmp expect actual &&
681 # We now have all bundle refs.
682 git -C clone-token-http for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
684 cat >expect <<-\EOF &&
693 test_expect_success
'http clone with bundle.heuristic creates fetch.bundleURI' '
694 test_when_finished rm -rf fetch-http-4 trace*.txt &&
696 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
700 heuristic = creationToken
703 uri = bundle-1.bundle
707 GIT_TRACE2_EVENT="$(pwd)/trace-clone.txt" \
708 git clone --single-branch --branch=base \
709 --bundle-uri="$HTTPD_URL/bundle-list" \
710 "$HTTPD_URL/smart/fetch.git" fetch-http-4 &&
712 test_cmp_config -C fetch-http-4 "$HTTPD_URL/bundle-list" fetch.bundleuri &&
713 test_cmp_config -C fetch-http-4 1 fetch.bundlecreationtoken &&
715 cat >expect <<-EOF &&
716 $HTTPD_URL/bundle-list
717 $HTTPD_URL/bundle-1.bundle
720 test_remote_https_urls <trace-clone.txt >actual &&
721 test_cmp expect actual &&
723 # only received base ref from bundle-1
724 git -C fetch-http-4 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
725 cat >expect <<-\EOF &&
728 test_cmp expect refs &&
730 cat >>"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
732 uri = bundle-2.bundle
736 # Fetch the objects for bundle-2 _and_ bundle-3.
737 GIT_TRACE2_EVENT="$(pwd)/trace1.txt" \
738 git -C fetch-http-4 fetch origin --no-tags \
739 refs/heads/left:refs/heads/left \
740 refs/heads/right:refs/heads/right &&
741 test_cmp_config -C fetch-http-4 2 fetch.bundlecreationtoken &&
743 cat >expect <<-EOF &&
744 $HTTPD_URL/bundle-list
745 $HTTPD_URL/bundle-2.bundle
748 test_remote_https_urls <trace1.txt >actual &&
749 test_cmp expect actual &&
751 # received left from bundle-2
752 git -C fetch-http-4 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
753 cat >expect <<-\EOF &&
757 test_cmp expect refs &&
760 GIT_TRACE2_EVENT="$(pwd)/trace1b.txt" \
761 git -C fetch-http-4 fetch origin --no-tags \
762 refs/heads/left:refs/heads/left \
763 refs/heads/right:refs/heads/right &&
765 cat >expect <<-EOF &&
766 $HTTPD_URL/bundle-list
768 test_remote_https_urls <trace1b.txt >actual &&
769 test_cmp expect actual &&
771 cat >>"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
773 uri = bundle-3.bundle
777 uri = bundle-4.bundle
781 # This fetch should skip bundle-3.bundle, since its objects are
782 # already local (we have the requisite commits for bundle-4.bundle).
783 GIT_TRACE2_EVENT="$(pwd)/trace2.txt" \
784 git -C fetch-http-4 fetch origin --no-tags \
785 refs/heads/merge:refs/heads/merge &&
786 test_cmp_config -C fetch-http-4 4 fetch.bundlecreationtoken &&
788 cat >expect <<-EOF &&
789 $HTTPD_URL/bundle-list
790 $HTTPD_URL/bundle-4.bundle
793 test_remote_https_urls <trace2.txt >actual &&
794 test_cmp expect actual &&
796 # received merge ref from bundle-4, but right is missing
797 # because we did not download bundle-3.
798 git -C fetch-http-4 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
800 cat >expect <<-\EOF &&
805 test_cmp expect refs &&
808 GIT_TRACE2_EVENT="$(pwd)/trace2b.txt" \
809 git -C fetch-http-4 fetch origin &&
811 cat >expect <<-EOF &&
812 $HTTPD_URL/bundle-list
814 test_remote_https_urls <trace2b.txt >actual &&
815 test_cmp expect actual
818 test_expect_success
'creationToken heuristic with failed downloads (clone)' '
819 test_when_finished rm -rf download-* trace*.txt &&
821 # Case 1: base bundle does not exist, nothing can unbundle
822 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
826 heuristic = creationToken
833 uri = bundle-2.bundle
837 uri = bundle-3.bundle
841 uri = bundle-4.bundle
845 GIT_TRACE2_EVENT="$(pwd)/trace-clone-1.txt" \
846 git clone --single-branch --branch=base \
847 --bundle-uri="$HTTPD_URL/bundle-list" \
848 "$HTTPD_URL/smart/fetch.git" download-1 &&
850 # Bundle failure does not set these configs.
851 test_must_fail git -C download-1 config fetch.bundleuri &&
852 test_must_fail git -C download-1 config fetch.bundlecreationtoken &&
854 cat >expect <<-EOF &&
855 $HTTPD_URL/bundle-list
856 $HTTPD_URL/bundle-4.bundle
857 $HTTPD_URL/bundle-3.bundle
858 $HTTPD_URL/bundle-2.bundle
859 $HTTPD_URL/fake.bundle
861 test_remote_https_urls <trace-clone-1.txt >actual &&
862 test_cmp expect actual &&
864 # All bundles failed to unbundle
865 git -C download-1 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
866 test_must_be_empty refs &&
868 # Case 2: middle bundle does not exist, only two bundles can unbundle
869 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
873 heuristic = creationToken
876 uri = bundle-1.bundle
884 uri = bundle-3.bundle
888 uri = bundle-4.bundle
892 GIT_TRACE2_EVENT="$(pwd)/trace-clone-2.txt" \
893 git clone --single-branch --branch=base \
894 --bundle-uri="$HTTPD_URL/bundle-list" \
895 "$HTTPD_URL/smart/fetch.git" download-2 &&
897 # Bundle failure does not set these configs.
898 test_must_fail git -C download-2 config fetch.bundleuri &&
899 test_must_fail git -C download-2 config fetch.bundlecreationtoken &&
901 cat >expect <<-EOF &&
902 $HTTPD_URL/bundle-list
903 $HTTPD_URL/bundle-4.bundle
904 $HTTPD_URL/bundle-3.bundle
905 $HTTPD_URL/fake.bundle
906 $HTTPD_URL/bundle-1.bundle
908 test_remote_https_urls <trace-clone-2.txt >actual &&
909 test_cmp expect actual &&
911 # bundle-1 and bundle-3 could unbundle, but bundle-4 could not
912 git -C download-2 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
913 cat >expect <<-EOF &&
917 test_cmp expect refs &&
919 # Case 3: top bundle does not exist, rest unbundle fine.
920 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
924 heuristic = creationToken
927 uri = bundle-1.bundle
931 uri = bundle-2.bundle
935 uri = bundle-3.bundle
943 GIT_TRACE2_EVENT="$(pwd)/trace-clone-3.txt" \
944 git clone --single-branch --branch=base \
945 --bundle-uri="$HTTPD_URL/bundle-list" \
946 "$HTTPD_URL/smart/fetch.git" download-3 &&
948 # As long as we have contiguous successful downloads,
949 # we _do_ set these configs.
950 test_cmp_config -C download-3 "$HTTPD_URL/bundle-list" fetch.bundleuri &&
951 test_cmp_config -C download-3 3 fetch.bundlecreationtoken &&
953 cat >expect <<-EOF &&
954 $HTTPD_URL/bundle-list
955 $HTTPD_URL/fake.bundle
956 $HTTPD_URL/bundle-3.bundle
957 $HTTPD_URL/bundle-2.bundle
958 $HTTPD_URL/bundle-1.bundle
960 test_remote_https_urls <trace-clone-3.txt >actual &&
961 test_cmp expect actual &&
963 # fake.bundle did not unbundle, but the others did.
964 git -C download-3 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
965 cat >expect <<-EOF &&
973 # Expand the bundle list to include other interesting shapes, specifically
974 # interesting for use when fetching from a previous state.
976 # ---------------- bundle-7
979 # ---/--|--\------ bundle-6
981 # --|---|---|----- bundle-4
984 # --|-|---|/------ bundle-3 (the client will be caught up to this point.)
986 # ---\|---|------- bundle-2
988 # ----|---|------- bundle-1
993 test_expect_success
'expand incremental bundle list' '
996 git checkout -b lefter left &&
998 git checkout -b righter right &&
1000 git checkout -b top lefter &&
1001 git merge -m "7" merge righter &&
1003 git bundle create bundle-6.bundle lefter righter --not left right &&
1004 git bundle create bundle-7.bundle top --not lefter merge righter &&
1006 cp bundle-*.bundle "$HTTPD_DOCUMENT_ROOT_PATH/"
1008 git -C "$HTTPD_DOCUMENT_ROOT_PATH/fetch.git" fetch origin +refs/heads/*:refs/heads/*
1011 test_expect_success
'creationToken heuristic with failed downloads (fetch)' '
1012 test_when_finished rm -rf download-* trace*.txt &&
1014 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
1018 heuristic = creationToken
1021 uri = bundle-1.bundle
1025 uri = bundle-2.bundle
1029 uri = bundle-3.bundle
1033 git clone --single-branch --branch=left \
1034 --bundle-uri="$HTTPD_URL/bundle-list" \
1035 "$HTTPD_URL/smart/fetch.git" fetch-base &&
1036 test_cmp_config -C fetch-base "$HTTPD_URL/bundle-list" fetch.bundleURI &&
1037 test_cmp_config -C fetch-base 3 fetch.bundleCreationToken &&
1039 # Case 1: all bundles exist: successful unbundling of all bundles
1040 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
1044 heuristic = creationToken
1047 uri = bundle-1.bundle
1051 uri = bundle-2.bundle
1055 uri = bundle-3.bundle
1059 uri = bundle-4.bundle
1063 uri = bundle-6.bundle
1067 uri = bundle-7.bundle
1071 cp -r fetch-base fetch-1 &&
1072 GIT_TRACE2_EVENT="$(pwd)/trace-fetch-1.txt" \
1073 git -C fetch-1 fetch origin &&
1074 test_cmp_config -C fetch-1 7 fetch.bundlecreationtoken &&
1076 cat >expect <<-EOF &&
1077 $HTTPD_URL/bundle-list
1078 $HTTPD_URL/bundle-7.bundle
1079 $HTTPD_URL/bundle-6.bundle
1080 $HTTPD_URL/bundle-4.bundle
1082 test_remote_https_urls <trace-fetch-1.txt >actual &&
1083 test_cmp expect actual &&
1085 # Check which bundles have unbundled by refs
1086 git -C fetch-1 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
1087 cat >expect <<-EOF &&
1093 refs/bundles/righter
1096 test_cmp expect refs &&
1098 # Case 2: middle bundle does not exist, only bundle-4 can unbundle
1099 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
1103 heuristic = creationToken
1106 uri = bundle-1.bundle
1110 uri = bundle-2.bundle
1114 uri = bundle-3.bundle
1118 uri = bundle-4.bundle
1126 uri = bundle-7.bundle
1130 cp -r fetch-base fetch-2 &&
1131 GIT_TRACE2_EVENT="$(pwd)/trace-fetch-2.txt" \
1132 git -C fetch-2 fetch origin &&
1134 # Since bundle-7 fails to unbundle, do not update creation token.
1135 test_cmp_config -C fetch-2 3 fetch.bundlecreationtoken &&
1137 cat >expect <<-EOF &&
1138 $HTTPD_URL/bundle-list
1139 $HTTPD_URL/bundle-7.bundle
1140 $HTTPD_URL/fake.bundle
1141 $HTTPD_URL/bundle-4.bundle
1143 test_remote_https_urls <trace-fetch-2.txt >actual &&
1144 test_cmp expect actual &&
1146 # Check which bundles have unbundled by refs
1147 git -C fetch-2 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
1148 cat >expect <<-EOF &&
1154 test_cmp expect refs &&
1156 # Case 3: top bundle does not exist, rest unbundle fine.
1157 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
1161 heuristic = creationToken
1164 uri = bundle-1.bundle
1168 uri = bundle-2.bundle
1172 uri = bundle-3.bundle
1176 uri = bundle-4.bundle
1180 uri = bundle-6.bundle
1188 cp -r fetch-base fetch-3 &&
1189 GIT_TRACE2_EVENT="$(pwd)/trace-fetch-3.txt" \
1190 git -C fetch-3 fetch origin &&
1192 # As long as we have contiguous successful downloads,
1193 # we _do_ set the maximum creation token.
1194 test_cmp_config -C fetch-3 6 fetch.bundlecreationtoken &&
1196 # NOTE: the fetch skips bundle-4 since bundle-6 successfully
1197 # unbundles itself and bundle-7 failed to download.
1198 cat >expect <<-EOF &&
1199 $HTTPD_URL/bundle-list
1200 $HTTPD_URL/fake.bundle
1201 $HTTPD_URL/bundle-6.bundle
1203 test_remote_https_urls <trace-fetch-3.txt >actual &&
1204 test_cmp expect actual &&
1206 # Check which bundles have unbundled by refs
1207 git -C fetch-3 for-each-ref --format="%(refname)" "refs/bundles/*" >refs &&
1208 cat >expect <<-EOF &&
1213 refs/bundles/righter
1215 test_cmp expect refs
1218 test_expect_success
'bundles are downloaded once during fetch --all' '
1219 test_when_finished rm -rf download-* trace*.txt fetch-mult &&
1221 cat >"$HTTPD_DOCUMENT_ROOT_PATH/bundle-list" <<-EOF &&
1225 heuristic = creationToken
1228 uri = bundle-1.bundle
1232 uri = bundle-2.bundle
1236 uri = bundle-3.bundle
1240 git clone --single-branch --branch=left \
1241 --bundle-uri="$HTTPD_URL/bundle-list" \
1242 "$HTTPD_URL/smart/fetch.git" fetch-mult &&
1243 git -C fetch-mult remote add dup1 "$HTTPD_URL/smart/fetch.git" &&
1244 git -C fetch-mult remote add dup2 "$HTTPD_URL/smart/fetch.git" &&
1246 GIT_TRACE2_EVENT="$(pwd)/trace-mult.txt" \
1247 git -C fetch-mult fetch --all &&
1248 grep "\"child_start\".*\"git-remote-https\",\"$HTTPD_URL/bundle-list\"" \
1249 trace-mult.txt >bundle-fetches &&
1250 test_line_count = 1 bundle-fetches
1252 # Do not add tests here unless they use the HTTP server, as they will
1253 # not run unless the HTTP dependencies exist.