| #!/bin/sh | 
 | # | 
 | # Copyright (c) 2005 Johannes Schindelin | 
 | # | 
 |  | 
 | test_description='Testing multi_ack pack fetching' | 
 |  | 
 | GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME=main | 
 | export GIT_TEST_DEFAULT_INITIAL_BRANCH_NAME | 
 |  | 
 | . ./test-lib.sh | 
 |  | 
 | # Test fetch-pack/upload-pack pair. | 
 |  | 
 | # Some convenience functions | 
 |  | 
 | add () { | 
 | 	name=$1 && | 
 | 	text="$@" && | 
 | 	branch=$(echo $name | sed -e 's/^\(.\).*$/\1/') && | 
 | 	parents="" && | 
 |  | 
 | 	shift && | 
 | 	while test $1; do | 
 | 		parents="$parents -p $1" && | 
 | 		shift | 
 | 	done && | 
 |  | 
 | 	echo "$text" > test.txt && | 
 | 	git update-index --add test.txt && | 
 | 	tree=$(git write-tree) && | 
 | 	# make sure timestamps are in correct order | 
 | 	test_tick && | 
 | 	commit=$(echo "$text" | git commit-tree $tree $parents) && | 
 | 	eval "$name=$commit; export $name" && | 
 | 	git update-ref "refs/heads/$branch" "$commit" && | 
 | 	eval ${branch}TIP=$commit | 
 | } | 
 |  | 
 | pull_to_client () { | 
 | 	number=$1 && | 
 | 	heads=$2 && | 
 | 	count=$3 && | 
 | 	test_expect_success "$number pull" ' | 
 | 		( | 
 | 			cd client && | 
 | 			git fetch-pack -k -v .. $heads && | 
 |  | 
 | 			case "$heads" in | 
 | 			    *A*) | 
 | 				    git update-ref refs/heads/A "$ATIP";; | 
 | 			esac && | 
 | 			case "$heads" in *B*) | 
 | 			    git update-ref refs/heads/B "$BTIP";; | 
 | 			esac && | 
 |  | 
 | 			git symbolic-ref HEAD refs/heads/$( | 
 | 				echo $heads | | 
 | 				sed -e "s/^\(.\).*$/\1/" | 
 | 			) && | 
 |  | 
 | 			git fsck --full && | 
 |  | 
 | 			mv .git/objects/pack/pack-* . && | 
 | 			p=$(ls -1 pack-*.pack) && | 
 | 			git unpack-objects <$p && | 
 | 			git fsck --full && | 
 |  | 
 | 			idx=$(echo pack-*.idx) && | 
 | 			pack_count=$(git show-index <$idx | wc -l) && | 
 | 			test $pack_count = $count && | 
 | 			rm -f pack-* | 
 | 		) | 
 | 	' | 
 | } | 
 |  | 
 | # Here begins the actual testing | 
 |  | 
 | # A1 - ... - A20 - A21 | 
 | #    \ | 
 | #      B1  -   B2 - .. - B70 | 
 |  | 
 | # client pulls A20, B1. Then tracks only B. Then pulls A. | 
 |  | 
 | test_expect_success 'setup' ' | 
 | 	mkdir client && | 
 | 	( | 
 | 		cd client && | 
 | 		git init && | 
 | 		git config transfer.unpacklimit 0 | 
 | 	) && | 
 | 	add A1 && | 
 | 	prev=1 && | 
 | 	cur=2 && | 
 | 	while [ $cur -le 10 ]; do | 
 | 		add A$cur $(eval echo \$A$prev) && | 
 | 		prev=$cur && | 
 | 		cur=$(($cur+1)) || return 1 | 
 | 	done && | 
 | 	add B1 $A1 && | 
 | 	git update-ref refs/heads/A "$ATIP" && | 
 | 	git update-ref refs/heads/B "$BTIP" && | 
 | 	git symbolic-ref HEAD refs/heads/B | 
 | ' | 
 |  | 
 | pull_to_client 1st "refs/heads/B refs/heads/A" $((11*3)) | 
 |  | 
 | test_expect_success 'post 1st pull setup' ' | 
 | 	add A11 $A10 && | 
 | 	prev=1 && | 
 | 	cur=2 && | 
 | 	while [ $cur -le 65 ]; do | 
 | 		add B$cur $(eval echo \$B$prev) && | 
 | 		prev=$cur && | 
 | 		cur=$(($cur+1)) || return 1 | 
 | 	done | 
 | ' | 
 |  | 
 | pull_to_client 2nd "refs/heads/B" $((64*3)) | 
 |  | 
 | pull_to_client 3rd "refs/heads/A" $((1*3)) | 
 |  | 
 | test_expect_success 'single branch clone' ' | 
 | 	git clone --single-branch "file://$(pwd)/." singlebranch | 
 | ' | 
 |  | 
 | test_expect_success 'single branch object count' ' | 
 | 	GIT_DIR=singlebranch/.git git count-objects -v | | 
 | 		grep "^in-pack:" > count.singlebranch && | 
 | 	echo "in-pack: 198" >expected && | 
 | 	test_cmp expected count.singlebranch | 
 | ' | 
 |  | 
 | test_expect_success 'single given branch clone' ' | 
 | 	GIT_TRACE2_EVENT="$(pwd)/branch-a/trace2_event" \ | 
 | 		git clone --single-branch --branch A "file://$(pwd)/." branch-a && | 
 | 	test_must_fail git --git-dir=branch-a/.git rev-parse origin/B && | 
 | 	grep \"fetch-info\".*\"haves\":0 branch-a/trace2_event && | 
 | 	grep \"fetch-info\".*\"wants\":1 branch-a/trace2_event | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow depth 1' ' | 
 | 	GIT_TRACE2_EVENT="$(pwd)/shallow0/trace2_event" \ | 
 | 		git clone --no-single-branch --depth 1 "file://$(pwd)/." shallow0 && | 
 | 	test "$(git --git-dir=shallow0/.git rev-list --count HEAD)" = 1 && | 
 | 	grep \"fetch-info\".*\"depth\":1 shallow0/trace2_event | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow depth 1 with fsck' ' | 
 | 	git config --global fetch.fsckobjects true && | 
 | 	git clone --no-single-branch --depth 1 "file://$(pwd)/." shallow0fsck && | 
 | 	test "$(git --git-dir=shallow0fsck/.git rev-list --count HEAD)" = 1 && | 
 | 	git config --global --unset fetch.fsckobjects | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow' ' | 
 | 	git clone --no-single-branch --depth 2 "file://$(pwd)/." shallow | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow depth count' ' | 
 | 	test "$(git --git-dir=shallow/.git rev-list --count HEAD)" = 2 | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow object count' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git count-objects -v | 
 | 	) > count.shallow && | 
 | 	grep "^in-pack: 12" count.shallow | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow object count (part 2)' ' | 
 | 	sed -e "/^in-pack:/d" -e "/^packs:/d" -e "/^size-pack:/d" \ | 
 | 	    -e "/: 0$/d" count.shallow > count_output && | 
 | 	test_must_be_empty count_output | 
 | ' | 
 |  | 
 | test_expect_success 'fsck in shallow repo' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git fsck --full | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'simple fetch in shallow repo' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git fetch | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'no changes expected' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git count-objects -v | 
 | 	) > count.shallow.2 && | 
 | 	cmp count.shallow count.shallow.2 | 
 | ' | 
 |  | 
 | test_expect_success 'fetch same depth in shallow repo' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git fetch --depth=2 | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'no changes expected' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git count-objects -v | 
 | 	) > count.shallow.3 && | 
 | 	cmp count.shallow count.shallow.3 | 
 | ' | 
 |  | 
 | test_expect_success 'add two more' ' | 
 | 	add B66 $B65 && | 
 | 	add B67 $B66 | 
 | ' | 
 |  | 
 | test_expect_success 'pull in shallow repo' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git pull .. B | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow object count' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git count-objects -v | 
 | 	) > count.shallow && | 
 | 	grep "^count: 6" count.shallow | 
 | ' | 
 |  | 
 | test_expect_success 'add two more (part 2)' ' | 
 | 	add B68 $B67 && | 
 | 	add B69 $B68 | 
 | ' | 
 |  | 
 | test_expect_success 'deepening pull in shallow repo' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		GIT_TRACE2_EVENT="$(pwd)/trace2_event" \ | 
 | 			git pull --depth 4 .. B && | 
 | 		grep \"fetch-info\".*\"depth\":4 trace2_event && | 
 | 		grep \"fetch-info\".*\"shallows\":2 trace2_event | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow object count' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git count-objects -v | 
 | 	) > count.shallow && | 
 | 	grep "^count: 12" count.shallow | 
 | ' | 
 |  | 
 | test_expect_success 'deepening fetch in shallow repo' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git fetch --depth 4 .. A:A | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow object count' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git count-objects -v | 
 | 	) > count.shallow && | 
 | 	grep "^count: 18" count.shallow | 
 | ' | 
 |  | 
 | test_expect_success 'pull in shallow repo with missing merge base' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git fetch --depth 4 .. A && | 
 | 		test_must_fail git merge --allow-unrelated-histories FETCH_HEAD | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'additional simple shallow deepenings' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git fetch --depth=8 && | 
 | 		git fetch --depth=10 && | 
 | 		git fetch --depth=11 | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow depth count' ' | 
 | 	test "$(git --git-dir=shallow/.git rev-list --count HEAD)" = 11 | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow object count' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		git prune && | 
 | 		git count-objects -v | 
 | 	) > count.shallow && | 
 | 	grep "^count: 54" count.shallow | 
 | ' | 
 |  | 
 | test_expect_success 'fetch --no-shallow on full repo' ' | 
 | 	test_must_fail git fetch --noshallow | 
 | ' | 
 |  | 
 | test_expect_success 'fetch --depth --no-shallow' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		test_must_fail git fetch --depth=1 --noshallow | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'turn shallow to complete repository' ' | 
 | 	( | 
 | 		cd shallow && | 
 | 		GIT_TRACE2_EVENT="$(pwd)/trace2_event" \ | 
 | 			git fetch --unshallow && | 
 | 		! test -f .git/shallow && | 
 | 		git fsck --full && | 
 | 		grep \"fetch-info\".*\"shallows\":2 trace2_event && | 
 | 		grep \"fetch-info\".*\"depth\":2147483647 trace2_event | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow without --no-single-branch' ' | 
 | 	git clone --depth 1 "file://$(pwd)/." shallow2 | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow object count' ' | 
 | 	( | 
 | 		cd shallow2 && | 
 | 		git count-objects -v | 
 | 	) > count.shallow2 && | 
 | 	grep "^in-pack: 3" count.shallow2 | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow with --branch' ' | 
 | 	git clone --depth 1 --branch A "file://$(pwd)/." shallow3 | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow object count' ' | 
 | 	echo "in-pack: 3" > count3.expected && | 
 | 	GIT_DIR=shallow3/.git git count-objects -v | | 
 | 		grep "^in-pack" > count3.actual && | 
 | 	test_cmp count3.expected count3.actual | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow with detached HEAD' ' | 
 | 	git checkout HEAD^ && | 
 | 	git clone --depth 1 "file://$(pwd)/." shallow5 && | 
 | 	git checkout - && | 
 | 	GIT_DIR=shallow5/.git git rev-parse HEAD >actual && | 
 | 	git rev-parse HEAD^ >expected && | 
 | 	test_cmp expected actual | 
 | ' | 
 |  | 
 | test_expect_success 'shallow clone pulling tags' ' | 
 | 	git tag -a -m A TAGA1 A && | 
 | 	git tag -a -m B TAGB1 B && | 
 | 	git tag TAGA2 A && | 
 | 	git tag TAGB2 B && | 
 | 	git clone --depth 1 "file://$(pwd)/." shallow6 && | 
 |  | 
 | 	cat >taglist.expected <<\EOF && | 
 | TAGB1 | 
 | TAGB2 | 
 | EOF | 
 | 	GIT_DIR=shallow6/.git git tag -l >taglist.actual && | 
 | 	test_cmp taglist.expected taglist.actual && | 
 |  | 
 | 	echo "in-pack: 4" > count6.expected && | 
 | 	GIT_DIR=shallow6/.git git count-objects -v | | 
 | 		grep "^in-pack" > count6.actual && | 
 | 	test_cmp count6.expected count6.actual | 
 | ' | 
 |  | 
 | test_expect_success 'shallow cloning single tag' ' | 
 | 	git clone --depth 1 --branch=TAGB1 "file://$(pwd)/." shallow7 && | 
 | 	cat >taglist.expected <<\EOF && | 
 | TAGB1 | 
 | TAGB2 | 
 | EOF | 
 | 	GIT_DIR=shallow7/.git git tag -l >taglist.actual && | 
 | 	test_cmp taglist.expected taglist.actual && | 
 |  | 
 | 	echo "in-pack: 4" > count7.expected && | 
 | 	GIT_DIR=shallow7/.git git count-objects -v | | 
 | 		grep "^in-pack" > count7.actual && | 
 | 	test_cmp count7.expected count7.actual | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow with packed refs' ' | 
 | 	git pack-refs --all && | 
 | 	git clone --depth 1 --branch A "file://$(pwd)/." shallow8 && | 
 | 	echo "in-pack: 4" > count8.expected && | 
 | 	GIT_DIR=shallow8/.git git count-objects -v | | 
 | 		grep "^in-pack" > count8.actual && | 
 | 	test_cmp count8.expected count8.actual | 
 | ' | 
 |  | 
 | test_expect_success 'in_vain not triggered before first ACK' ' | 
 | 	rm -rf myserver myclient && | 
 | 	git init myserver && | 
 | 	test_commit -C myserver foo && | 
 | 	git clone "file://$(pwd)/myserver" myclient && | 
 |  | 
 | 	# MAX_IN_VAIN is 256. Because of batching, the client will send 496 | 
 | 	# (16+32+64+128+256) commits, not 256, before giving up. So create 496 | 
 | 	# irrelevant commits. | 
 | 	test_commit_bulk -C myclient 496 && | 
 |  | 
 | 	# The new commit that the client wants to fetch. | 
 | 	test_commit -C myserver bar && | 
 |  | 
 | 	git -C myclient fetch --progress origin 2>log && | 
 | 	test_grep "remote: Total 3 " log | 
 | ' | 
 |  | 
 | test_expect_success 'in_vain reset upon ACK' ' | 
 | 	test_when_finished rm -f log trace2 && | 
 | 	rm -rf myserver myclient && | 
 | 	git init myserver && | 
 |  | 
 | 	# Linked list of commits on main. The first is common; the rest are | 
 | 	# not. | 
 | 	test_commit -C myserver first_main_commit && | 
 | 	git clone "file://$(pwd)/myserver" myclient && | 
 | 	test_commit_bulk -C myclient 255 && | 
 |  | 
 | 	# Another linked list of commits on anotherbranch with no connection to | 
 | 	# main. The first is common; the rest are not. | 
 | 	git -C myserver checkout --orphan anotherbranch && | 
 | 	test_commit -C myserver first_anotherbranch_commit && | 
 | 	git -C myclient fetch origin anotherbranch:refs/heads/anotherbranch && | 
 | 	git -C myclient checkout anotherbranch && | 
 | 	test_commit_bulk -C myclient 255 && | 
 |  | 
 | 	# The new commit that the client wants to fetch. | 
 | 	git -C myserver checkout main && | 
 | 	test_commit -C myserver to_fetch && | 
 |  | 
 | 	# The client will send (as "have"s) all 256 commits in anotherbranch | 
 | 	# first. The 256th commit is common between the client and the server, | 
 | 	# and should reset in_vain. This allows negotiation to continue until | 
 | 	# the client reports that first_anotherbranch_commit is common. | 
 | 	GIT_TRACE2_EVENT="$(pwd)/trace2" git -C myclient fetch --progress origin main 2>log && | 
 | 	grep \"key\":\"total_rounds\",\"value\":\"6\" trace2 && | 
 | 	test_grep "Total 3 " log | 
 | ' | 
 |  | 
 | test_expect_success 'fetch in shallow repo unreachable shallow objects' ' | 
 | 	( | 
 | 		git clone --bare --branch B --single-branch "file://$(pwd)/." no-reflog && | 
 | 		git clone --depth 1 "file://$(pwd)/no-reflog" shallow9 && | 
 | 		cd no-reflog && | 
 | 		git tag -d TAGB1 TAGB2 && | 
 | 		git update-ref refs/heads/B B~~ && | 
 | 		git gc --prune=now && | 
 | 		cd ../shallow9 && | 
 | 		git fetch origin && | 
 | 		git fsck --no-dangling | 
 | 	) | 
 | ' | 
 | test_expect_success 'fetch creating new shallow root' ' | 
 | 	( | 
 | 		git clone "file://$(pwd)/." shallow10 && | 
 | 		git commit --allow-empty -m empty && | 
 | 		cd shallow10 && | 
 | 		git fetch --depth=1 --progress 2>actual && | 
 | 		# This should fetch only the empty commit, no tree or | 
 | 		# blob objects | 
 | 		test_grep "remote: Total 1" actual | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'setup tests for the --stdin parameter' ' | 
 | 	for head in C D E F | 
 | 	do | 
 | 		add $head || return 1 | 
 | 	done && | 
 | 	for head in A B C D E F | 
 | 	do | 
 | 		git tag $head $head || return 1 | 
 | 	done && | 
 | 	cat >input <<-\EOF && | 
 | 	refs/heads/C | 
 | 	refs/heads/A | 
 | 	refs/heads/D | 
 | 	refs/tags/C | 
 | 	refs/heads/B | 
 | 	refs/tags/A | 
 | 	refs/heads/E | 
 | 	refs/tags/B | 
 | 	refs/tags/E | 
 | 	refs/tags/D | 
 | 	EOF | 
 | 	sort <input >expect && | 
 | 	( | 
 | 		echo refs/heads/E && | 
 | 		echo refs/tags/E && | 
 | 		cat input | 
 | 	) >input.dup | 
 | ' | 
 |  | 
 | test_expect_success 'setup fetch refs from cmdline v[12]' ' | 
 | 	cp -r client client0 && | 
 | 	cp -r client client1 && | 
 | 	cp -r client client2 | 
 | ' | 
 |  | 
 | for version in '' 0 1 2 | 
 | do | 
 | 	test_expect_success "protocol.version=$version fetch refs from cmdline" " | 
 | 		( | 
 | 			cd client$version && | 
 | 			GIT_TEST_PROTOCOL_VERSION=$version git fetch-pack --no-progress .. \$(cat ../input) | 
 | 		) >output && | 
 | 		cut -d ' ' -f 2 <output | sort >actual && | 
 | 		test_cmp expect actual | 
 | 	" | 
 | done | 
 |  | 
 | test_expect_success 'fetch refs from stdin' ' | 
 | 	( | 
 | 		cd client && | 
 | 		git fetch-pack --stdin --no-progress .. <../input | 
 | 	) >output && | 
 | 	cut -d " " -f 2 <output | sort >actual && | 
 | 	test_cmp expect actual | 
 | ' | 
 |  | 
 | test_expect_success 'fetch mixed refs from cmdline and stdin' ' | 
 | 	( | 
 | 		cd client && | 
 | 		tail -n +5 ../input | | 
 | 		git fetch-pack --stdin --no-progress .. $(head -n 4 ../input) | 
 | 	) >output && | 
 | 	cut -d " " -f 2 <output | sort >actual && | 
 | 	test_cmp expect actual | 
 | ' | 
 |  | 
 | test_expect_success 'test duplicate refs from stdin' ' | 
 | 	( | 
 | 	cd client && | 
 | 	git fetch-pack --stdin --no-progress .. <../input.dup | 
 | 	) >output && | 
 | 	cut -d " " -f 2 <output | sort >actual && | 
 | 	test_cmp expect actual | 
 | ' | 
 |  | 
 | test_expect_success 'set up tests of missing reference' ' | 
 | 	cat >expect-error <<-\EOF | 
 | 	error: no such remote ref refs/heads/xyzzy | 
 | 	EOF | 
 | ' | 
 |  | 
 | test_expect_success 'test lonely missing ref' ' | 
 | 	( | 
 | 		cd client && | 
 | 		test_must_fail git fetch-pack --no-progress .. refs/heads/xyzzy 2>../error-m | 
 | 	) && | 
 | 	test_cmp expect-error error-m | 
 | ' | 
 |  | 
 | test_expect_success 'test missing ref after existing' ' | 
 | 	( | 
 | 		cd client && | 
 | 		test_must_fail git fetch-pack --no-progress .. refs/heads/A refs/heads/xyzzy 2>../error-em | 
 | 	) && | 
 | 	test_cmp expect-error error-em | 
 | ' | 
 |  | 
 | test_expect_success 'test missing ref before existing' ' | 
 | 	( | 
 | 		cd client && | 
 | 		test_must_fail git fetch-pack --no-progress .. refs/heads/xyzzy refs/heads/A 2>../error-me | 
 | 	) && | 
 | 	test_cmp expect-error error-me | 
 | ' | 
 |  | 
 | test_expect_success 'test --all, --depth, and explicit head' ' | 
 | 	( | 
 | 		cd client && | 
 | 		git fetch-pack --no-progress --all --depth=1 .. refs/heads/A | 
 | 	) >out-adh 2>error-adh | 
 | ' | 
 |  | 
 | test_expect_success 'test --all, --depth, and explicit tag' ' | 
 | 	git tag OLDTAG refs/heads/B~5 && | 
 | 	( | 
 | 		cd client && | 
 | 		git fetch-pack --no-progress --all --depth=1 .. refs/tags/OLDTAG | 
 | 	) >out-adt 2>error-adt | 
 | ' | 
 |  | 
 | test_expect_success 'test --all with tag to non-tip' ' | 
 | 	git commit --allow-empty -m non-tip && | 
 | 	git commit --allow-empty -m tip && | 
 | 	git tag -m "annotated" non-tip HEAD^ && | 
 | 	( | 
 | 		cd client && | 
 | 		git fetch-pack --all .. | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'test --all wrt tag to non-commits' ' | 
 | 	# create tag-to-{blob,tree,commit,tag}, making sure all tagged objects | 
 | 	# are reachable only via created tag references. | 
 | 	blob=$(echo "hello blob" | git hash-object -t blob -w --stdin) && | 
 | 	git tag -a -m "tag -> blob" tag-to-blob $blob && | 
 |  | 
 | 	tree=$(printf "100644 blob $blob\tfile" | git mktree) && | 
 | 	git tag -a -m "tag -> tree" tag-to-tree $tree && | 
 |  | 
 | 	tree2=$(printf "100644 blob $blob\tfile2" | git mktree) && | 
 | 	commit=$(git commit-tree -m "hello commit" $tree) && | 
 | 	git tag -a -m "tag -> commit" tag-to-commit $commit && | 
 |  | 
 | 	blob2=$(echo "hello blob2" | git hash-object -t blob -w --stdin) && | 
 | 	tag=$(git mktag <<-EOF | 
 | 		object $blob2 | 
 | 		type blob | 
 | 		tag tag-to-blob2 | 
 | 		tagger author A U Thor <author@example.com> 0 +0000 | 
 |  | 
 | 		hello tag | 
 | 	EOF | 
 | 	) && | 
 | 	git tag -a -m "tag -> tag" tag-to-tag $tag && | 
 |  | 
 | 	# `fetch-pack --all` should succeed fetching all those objects. | 
 | 	mkdir fetchall && | 
 | 	( | 
 | 		cd fetchall && | 
 | 		git init && | 
 | 		git fetch-pack --all .. && | 
 | 		git cat-file blob $blob >/dev/null && | 
 | 		git cat-file tree $tree >/dev/null && | 
 | 		git cat-file commit $commit >/dev/null && | 
 | 		git cat-file tag $tag >/dev/null | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'shallow fetch with tags does not break the repository' ' | 
 | 	mkdir repo1 && | 
 | 	( | 
 | 		cd repo1 && | 
 | 		git init && | 
 | 		test_commit 1 && | 
 | 		test_commit 2 && | 
 | 		test_commit 3 && | 
 | 		mkdir repo2 && | 
 | 		cd repo2 && | 
 | 		git init && | 
 | 		git fetch --depth=2 ../.git main:branch && | 
 | 		git fsck | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'fetch-pack can fetch a raw sha1' ' | 
 | 	git init hidden && | 
 | 	( | 
 | 		cd hidden && | 
 | 		test_commit 1 && | 
 | 		test_commit 2 && | 
 | 		git update-ref refs/hidden/one HEAD^ && | 
 | 		git config transfer.hiderefs refs/hidden && | 
 | 		git config uploadpack.allowtipsha1inwant true | 
 | 	) && | 
 | 	git fetch-pack hidden $(git -C hidden rev-parse refs/hidden/one) | 
 | ' | 
 |  | 
 | test_expect_success 'fetch-pack can fetch a raw sha1 that is advertised as a ref' ' | 
 | 	rm -rf server client && | 
 | 	git init server && | 
 | 	test_commit -C server 1 && | 
 |  | 
 | 	git init client && | 
 | 	git -C client fetch-pack ../server \ | 
 | 		$(git -C server rev-parse refs/heads/main) | 
 | ' | 
 |  | 
 | test_expect_success 'fetch-pack can fetch a raw sha1 overlapping a named ref' ' | 
 | 	rm -rf server client && | 
 | 	git init server && | 
 | 	test_commit -C server 1 && | 
 | 	test_commit -C server 2 && | 
 |  | 
 | 	git init client && | 
 | 	git -C client fetch-pack ../server \ | 
 | 		$(git -C server rev-parse refs/tags/1) refs/tags/1 | 
 | ' | 
 |  | 
 | test_expect_success 'fetch-pack cannot fetch a raw sha1 that is not advertised as a ref' ' | 
 | 	rm -rf server && | 
 |  | 
 | 	git init server && | 
 | 	test_commit -C server 5 && | 
 | 	git -C server tag -d 5 && | 
 | 	test_commit -C server 6 && | 
 |  | 
 | 	git init client && | 
 | 	# Some protocol versions (e.g. 2) support fetching | 
 | 	# unadvertised objects, so restrict this test to v0. | 
 | 	test_must_fail env GIT_TEST_PROTOCOL_VERSION=0 git -C client fetch-pack ../server \ | 
 | 		$(git -C server rev-parse refs/heads/main^) 2>err && | 
 | 	test_grep "Server does not allow request for unadvertised object" err | 
 | ' | 
 |  | 
 | check_prot_path () { | 
 | 	cat >expected <<-EOF && | 
 | 	Diag: url=$1 | 
 | 	Diag: protocol=$2 | 
 | 	Diag: path=$3 | 
 | 	EOF | 
 | 	git fetch-pack --diag-url "$1" | grep -v hostandport= >actual && | 
 | 	test_cmp expected actual | 
 | } | 
 |  | 
 | check_prot_host_port_path () { | 
 | 	case "$2" in | 
 | 		*ssh*) | 
 | 		pp=ssh | 
 | 		uah=userandhost | 
 | 		ehost=$(echo $3 | tr -d "[]") | 
 | 		diagport="Diag: port=$4" | 
 | 		;; | 
 | 		*) | 
 | 		pp=$p | 
 | 		uah=hostandport | 
 | 		ehost=$(echo $3$4 | sed -e "s/22$/:22/" -e "s/NONE//") | 
 | 		diagport="" | 
 | 		;; | 
 | 	esac | 
 | 	cat >exp <<-EOF && | 
 | 	Diag: url=$1 | 
 | 	Diag: protocol=$pp | 
 | 	Diag: $uah=$ehost | 
 | 	$diagport | 
 | 	Diag: path=$5 | 
 | 	EOF | 
 | 	grep -v "^$" exp >expected | 
 | 	git fetch-pack --diag-url "$1" >actual && | 
 | 	test_cmp expected actual | 
 | } | 
 |  | 
 | for r in repo re:po re/po | 
 | do | 
 | 	# git or ssh with scheme | 
 | 	for p in "ssh+git" "git+ssh" git ssh | 
 | 	do | 
 | 		for h in host user@host user@[::1] user@::1 | 
 | 		do | 
 | 			for c in "" : | 
 | 			do | 
 | 				test_expect_success "fetch-pack --diag-url $p://$h$c/$r" ' | 
 | 					check_prot_host_port_path $p://$h/$r $p "$h" NONE "/$r" | 
 | 				' | 
 | 				# "/~" -> "~" conversion | 
 | 				test_expect_success "fetch-pack --diag-url $p://$h$c/~$r" ' | 
 | 					check_prot_host_port_path $p://$h/~$r $p "$h" NONE "~$r" | 
 | 				' | 
 | 			done | 
 | 		done | 
 | 		for h in host User@host User@[::1] | 
 | 		do | 
 | 			test_expect_success "fetch-pack --diag-url $p://$h:22/$r" ' | 
 | 				check_prot_host_port_path $p://$h:22/$r $p "$h" 22 "/$r" | 
 | 			' | 
 | 		done | 
 | 	done | 
 | 	# file with scheme | 
 | 	for p in file | 
 | 	do | 
 | 		test_expect_success !WINDOWS "fetch-pack --diag-url $p://$h/$r" ' | 
 | 			check_prot_path $p://$h/$r $p "/$r" | 
 | 		' | 
 | 		test_expect_success MINGW "fetch-pack --diag-url $p://$h/$r" ' | 
 | 			check_prot_path $p://$h/$r $p "//$h/$r" | 
 | 		' | 
 | 		test_expect_success MINGW "fetch-pack --diag-url $p:///$r" ' | 
 | 			check_prot_path $p:///$r $p "/$r" | 
 | 		' | 
 | 		# No "/~" -> "~" conversion for file | 
 | 		test_expect_success !WINDOWS "fetch-pack --diag-url $p://$h/~$r" ' | 
 | 			check_prot_path $p://$h/~$r $p "/~$r" | 
 | 		' | 
 | 		test_expect_success MINGW "fetch-pack --diag-url $p://$h/~$r" ' | 
 | 			check_prot_path $p://$h/~$r $p "//$h/~$r" | 
 | 		' | 
 | 	done | 
 | 	# file without scheme | 
 | 	for h in nohost nohost:12 [::1] [::1]:23 [ [:aa | 
 | 	do | 
 | 		test_expect_success "fetch-pack --diag-url ./$h:$r" ' | 
 | 			check_prot_path ./$h:$r $p "./$h:$r" | 
 | 		' | 
 | 		# No "/~" -> "~" conversion for file | 
 | 		test_expect_success "fetch-pack --diag-url ./$p:$h/~$r" ' | 
 | 		check_prot_path ./$p:$h/~$r $p "./$p:$h/~$r" | 
 | 		' | 
 | 	done | 
 | 	#ssh without scheme | 
 | 	p=ssh | 
 | 	for h in host [::1] | 
 | 	do | 
 | 		expectation="success" | 
 | 		if test_have_prereq CYGWIN && test "$h" = "[::1]" | 
 | 		then | 
 | 			expectation="failure" | 
 | 		fi | 
 |  | 
 | 		test_expect_$expectation "fetch-pack --diag-url $h:$r" ' | 
 | 			check_prot_host_port_path $h:$r $p "$h" NONE "$r" | 
 | 		' | 
 | 		# Do "/~" -> "~" conversion | 
 | 		test_expect_$expectation "fetch-pack --diag-url $h:/~$r" ' | 
 | 			check_prot_host_port_path $h:/~$r $p "$h" NONE "~$r" | 
 | 		' | 
 | 	done | 
 | done | 
 |  | 
 | test_expect_success MINGW 'fetch-pack --diag-url file://c:/repo' ' | 
 | 	check_prot_path file://c:/repo file c:/repo | 
 | ' | 
 | test_expect_success MINGW 'fetch-pack --diag-url c:repo' ' | 
 | 	check_prot_path c:repo file c:repo | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow since ...' ' | 
 | 	test_create_repo shallow-since && | 
 | 	( | 
 | 	cd shallow-since && | 
 | 	GIT_COMMITTER_DATE="100000000 +0700" git commit --allow-empty -m one && | 
 | 	GIT_COMMITTER_DATE="200000000 +0700" git commit --allow-empty -m two && | 
 | 	GIT_COMMITTER_DATE="300000000 +0700" git commit --allow-empty -m three && | 
 | 	git clone --shallow-since "300000000 +0700" "file://$(pwd)/." ../shallow11 && | 
 | 	git -C ../shallow11 log --pretty=tformat:%s HEAD >actual && | 
 | 	echo three >expected && | 
 | 	test_cmp expected actual | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'fetch shallow since ...' ' | 
 | 	GIT_TRACE2_EVENT=$(pwd)/shallow11/trace2_event \ | 
 | 		git -C shallow11 fetch --shallow-since "200000000 +0700" origin && | 
 | 	git -C shallow11 log --pretty=tformat:%s origin/main >actual && | 
 | 	cat >expected <<-\EOF && | 
 | 	three | 
 | 	two | 
 | 	EOF | 
 | 	test_cmp expected actual && | 
 | 	grep \"fetch-info\".*\"deepen-since\":true shallow11/trace2_event | 
 | ' | 
 |  | 
 | test_expect_success 'clone shallow since selects no commits' ' | 
 | 	test_create_repo shallow-since-the-future && | 
 | 	( | 
 | 	cd shallow-since-the-future && | 
 | 	GIT_COMMITTER_DATE="100000000 +0700" git commit --allow-empty -m one && | 
 | 	GIT_COMMITTER_DATE="200000000 +0700" git commit --allow-empty -m two && | 
 | 	GIT_COMMITTER_DATE="300000000 +0700" git commit --allow-empty -m three && | 
 | 	test_must_fail git clone --shallow-since "900000000 +0700" "file://$(pwd)/." ../shallow111 | 
 | 	) | 
 | ' | 
 |  | 
 | # A few subtle things about the request in this test: | 
 | # | 
 | #  - the server must have commit-graphs present and enabled | 
 | # | 
 | #  - the history is such that our want/have share a common ancestor ("base" | 
 | #    here) | 
 | # | 
 | #  - we send only a single have, which is fewer than a normal client would | 
 | #    send. This ensures that we don't parse "base" up front with | 
 | #    parse_object(), but rather traverse to it as a parent while deciding if we | 
 | #    can stop the "have" negotiation, and call parse_commit(). The former | 
 | #    sees the actual object data and so always loads the three oid, whereas the | 
 | #    latter will try to load it lazily. | 
 | # | 
 | #  - we must use protocol v2, because it handles the "have" negotiation before | 
 | #    processing the shallow directives | 
 | # | 
 | test_expect_success 'shallow since with commit graph and already-seen commit' ' | 
 | 	test_create_repo shallow-since-graph && | 
 | 	( | 
 | 	cd shallow-since-graph && | 
 | 	test_commit base && | 
 | 	test_commit main && | 
 | 	git checkout -b other HEAD^ && | 
 | 	test_commit other && | 
 | 	git commit-graph write --reachable && | 
 | 	git config core.commitGraph true && | 
 |  | 
 | 	GIT_PROTOCOL=version=2 git upload-pack . <<-EOF >/dev/null | 
 | 	0012command=fetch | 
 | 	$(echo "object-format=$(test_oid algo)" | packetize) | 
 | 	00010013deepen-since 1 | 
 | 	$(echo "want $(git rev-parse other)" | packetize) | 
 | 	$(echo "have $(git rev-parse main)" | packetize) | 
 | 	0000 | 
 | 	EOF | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'shallow clone exclude tag two' ' | 
 | 	test_create_repo shallow-exclude && | 
 | 	( | 
 | 	cd shallow-exclude && | 
 | 	test_commit one && | 
 | 	test_commit two && | 
 | 	test_commit three && | 
 | 	git clone --shallow-exclude two "file://$(pwd)/." ../shallow12 && | 
 | 	git -C ../shallow12 log --pretty=tformat:%s HEAD >actual && | 
 | 	echo three >expected && | 
 | 	test_cmp expected actual | 
 | 	) | 
 | ' | 
 |  | 
 | test_expect_success 'fetch exclude tag one' ' | 
 | 	git -C shallow12 fetch --shallow-exclude one origin && | 
 | 	git -C shallow12 log --pretty=tformat:%s origin/main >actual && | 
 | 	test_write_lines three two >expected && | 
 | 	test_cmp expected actual | 
 | ' | 
 |  | 
 | test_expect_success 'fetch exclude tag one as revision' ' | 
 | 	test_when_finished rm -f rev err && | 
 | 	git -C shallow-exclude rev-parse one >rev && | 
 | 	test_must_fail git -C shallow12 fetch --shallow-exclude $(cat rev) origin 2>err && | 
 | 	grep "deepen-not is not a ref:" err | 
 | ' | 
 |  | 
 | test_expect_success 'fetching deepen' ' | 
 | 	test_create_repo shallow-deepen && | 
 | 	( | 
 | 	cd shallow-deepen && | 
 | 	test_commit one && | 
 | 	test_commit two && | 
 | 	test_commit three && | 
 | 	git clone --depth 1 "file://$(pwd)/." deepen && | 
 | 	test_commit four && | 
 | 	git -C deepen log --pretty=tformat:%s main >actual && | 
 | 	echo three >expected && | 
 | 	test_cmp expected actual && | 
 | 	git -C deepen fetch --deepen=1 && | 
 | 	git -C deepen log --pretty=tformat:%s origin/main >actual && | 
 | 	cat >expected <<-\EOF && | 
 | 	four | 
 | 	three | 
 | 	two | 
 | 	EOF | 
 | 	test_cmp expected actual | 
 | 	) | 
 | ' | 
 |  | 
 | test_negotiation_algorithm_default () { | 
 | 	test_when_finished rm -rf clientv0 clientv2 && | 
 | 	rm -rf server client && | 
 | 	git init server && | 
 | 	test_commit -C server both_have_1 && | 
 | 	git -C server tag -d both_have_1 && | 
 | 	test_commit -C server both_have_2 && | 
 |  | 
 | 	git clone server client && | 
 | 	test_commit -C server server_has && | 
 | 	test_commit -C client client_has && | 
 |  | 
 | 	# In both protocol v0 and v2, ensure that the parent of both_have_2 is | 
 | 	# not sent as a "have" line. The client should know that the server has | 
 | 	# both_have_2, so it only needs to inform the server that it has | 
 | 	# both_have_2, and the server can infer the rest. | 
 |  | 
 | 	rm -f trace && | 
 | 	cp -r client clientv0 && | 
 | 	GIT_TRACE_PACKET="$(pwd)/trace" git -C clientv0 \ | 
 | 		"$@" fetch origin server_has both_have_2 && | 
 | 	grep "have $(git -C client rev-parse client_has)" trace && | 
 | 	grep "have $(git -C client rev-parse both_have_2)" trace && | 
 | 	! grep "have $(git -C client rev-parse both_have_2^)" trace && | 
 |  | 
 | 	rm -f trace && | 
 | 	cp -r client clientv2 && | 
 | 	GIT_TRACE_PACKET="$(pwd)/trace" git -C clientv2 -c protocol.version=2 \ | 
 | 		"$@" fetch origin server_has both_have_2 && | 
 | 	grep "have $(git -C client rev-parse client_has)" trace && | 
 | 	grep "have $(git -C client rev-parse both_have_2)" trace && | 
 | 	! grep "have $(git -C client rev-parse both_have_2^)" trace | 
 | } | 
 |  | 
 | test_expect_success 'use ref advertisement to prune "have" lines sent' ' | 
 | 	test_negotiation_algorithm_default | 
 | ' | 
 |  | 
 | test_expect_success 'same as last but with config overrides' ' | 
 | 	test_negotiation_algorithm_default \ | 
 | 		-c feature.experimental=true \ | 
 | 		-c fetch.negotiationAlgorithm=consecutive | 
 | ' | 
 |  | 
 | test_expect_success 'ensure bogus fetch.negotiationAlgorithm yields error' ' | 
 | 	test_when_finished rm -rf clientv0 && | 
 | 	cp -r client clientv0 && | 
 | 	test_must_fail git -C clientv0 --fetch.negotiationAlgorithm=bogus \ | 
 | 		       fetch origin server_has both_have_2 | 
 | ' | 
 |  | 
 | test_expect_success 'fetch-pack with fsckObjects and keep-file does not segfault' ' | 
 | 	rm -rf server client && | 
 | 	test_create_repo server && | 
 | 	test_commit -C server one && | 
 |  | 
 | 	test_create_repo client && | 
 | 	git -c fetch.fsckObjects=true \ | 
 | 	    -C client fetch-pack -k -k ../server HEAD | 
 | ' | 
 |  | 
 | test_expect_success 'filtering by size' ' | 
 | 	rm -rf server client && | 
 | 	test_create_repo server && | 
 | 	test_commit -C server one && | 
 | 	test_config -C server uploadpack.allowfilter 1 && | 
 |  | 
 | 	test_create_repo client && | 
 | 	GIT_TRACE2_EVENT=$(pwd)/client/trace2_event \ | 
 | 		git -C client fetch-pack --filter=blob:limit=0 ../server HEAD && | 
 |  | 
 | 	# Ensure that object is not inadvertently fetched | 
 | 	commit=$(git -C server rev-parse HEAD) && | 
 | 	blob=$(git hash-object server/one.t) && | 
 | 	git -C client rev-list --objects --missing=allow-any "$commit" >oids && | 
 | 	! grep "$blob" oids && | 
 |  | 
 | 	grep \"fetch-info\".*\"filter\":\"blob:limit\" client/trace2_event | 
 | ' | 
 |  | 
 | test_expect_success 'filtering by size has no effect if support for it is not advertised' ' | 
 | 	rm -rf server client && | 
 | 	test_create_repo server && | 
 | 	test_commit -C server one && | 
 |  | 
 | 	test_create_repo client && | 
 | 	git -C client fetch-pack --filter=blob:limit=0 ../server HEAD 2> err && | 
 |  | 
 | 	# Ensure that object is fetched | 
 | 	commit=$(git -C server rev-parse HEAD) && | 
 | 	blob=$(git hash-object server/one.t) && | 
 | 	git -C client rev-list --objects --missing=allow-any "$commit" >oids && | 
 | 	grep "$blob" oids && | 
 |  | 
 | 	test_grep "filtering not recognized by server" err | 
 | ' | 
 |  | 
 | fetch_filter_blob_limit_zero () { | 
 | 	SERVER="$1" | 
 | 	URL="$2" | 
 |  | 
 | 	rm -rf "$SERVER" client && | 
 | 	test_create_repo "$SERVER" && | 
 | 	test_commit -C "$SERVER" one && | 
 | 	test_config -C "$SERVER" uploadpack.allowfilter 1 && | 
 |  | 
 | 	git clone "$URL" client && | 
 |  | 
 | 	test_commit -C "$SERVER" two && | 
 |  | 
 | 	git -C client fetch --filter=blob:limit=0 origin HEAD:somewhere && | 
 |  | 
 | 	# Ensure that commit is fetched, but blob is not | 
 | 	commit=$(git -C "$SERVER" rev-parse two) && | 
 | 	blob=$(git hash-object "$SERVER/two.t") && | 
 | 	git -C client rev-list --objects --missing=allow-any "$commit" >oids && | 
 | 	grep "$commit" oids && | 
 | 	! grep "$blob" oids | 
 | } | 
 |  | 
 | test_expect_success 'fetch with --filter=blob:limit=0' ' | 
 | 	fetch_filter_blob_limit_zero server server | 
 | ' | 
 |  | 
 | . "$TEST_DIRECTORY"/lib-httpd.sh | 
 | start_httpd | 
 |  | 
 | test_expect_success 'fetch with --filter=blob:limit=0 and HTTP' ' | 
 | 	fetch_filter_blob_limit_zero "$HTTPD_DOCUMENT_ROOT_PATH/server" "$HTTPD_URL/smart/server" | 
 | ' | 
 |  | 
 | # DO NOT add non-httpd-specific tests here, because the last part of this | 
 | # test script is only executed when httpd is available and enabled. | 
 |  | 
 | test_done |