All of our git-protocol tests rely on invoking the client
and having it make a request of a server. That gives a nice
real-world test of how the two behave together, but it
doesn't leave any room for testing how a server might react
to _other_ clients.
Let's add a few test helper functions which can be used to
manually conduct a git-protocol conversation with a remote
git-daemon:
  1. To connect to a remote git-daemon, we need something
     like "netcat". But not everybody will have netcat. And
     even if they do, the behavior with respect to
     half-duplex shutdowns is not portable (openbsd netcat
     has "-N", with others you must rely on "-q 1", which is
     racy).
     Here we provide a "fake_nc" that is capable of doing
     a client-side netcat, with sane half-duplex semantics.
     It relies on perl's IO::Socket::INET. That's been in
     the base distribution since 5.6.0, so it's probably
     available everywhere. But just to be on the safe side,
     we'll add a prereq.
  2. To help tests speak and read pktline, this patch adds
     packetize() and depacketize() functions.
I've put fake_nc() into lib-git-daemon.sh, since that's
really the only server where we'd need to use a network
socket.  Whereas the pktline helpers may be of more general
use, so I've added them to test-lib-functions.sh. Programs
like upload-pack speak pktline, but can talk directly over
stdio without a network socket.
Signed-off-by: Jeff King <peff@peff.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
		
	
		
			
				
	
	
		
			124 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
			
		
		
	
	
			124 lines
		
	
	
		
			2.7 KiB
		
	
	
	
		
			Bash
		
	
	
	
	
	
# Shell library to run git-daemon in tests.  Ends the test early if
 | 
						|
# GIT_TEST_GIT_DAEMON is not set.
 | 
						|
#
 | 
						|
# Usage:
 | 
						|
#
 | 
						|
#	. ./test-lib.sh
 | 
						|
#	. "$TEST_DIRECTORY"/lib-git-daemon.sh
 | 
						|
#	start_git_daemon
 | 
						|
#
 | 
						|
#	test_expect_success '...' '
 | 
						|
#		...
 | 
						|
#	'
 | 
						|
#
 | 
						|
#	test_expect_success ...
 | 
						|
#
 | 
						|
#	stop_git_daemon
 | 
						|
#	test_done
 | 
						|
 | 
						|
test_tristate GIT_TEST_GIT_DAEMON
 | 
						|
if test "$GIT_TEST_GIT_DAEMON" = false
 | 
						|
then
 | 
						|
	skip_all="git-daemon testing disabled (unset GIT_TEST_GIT_DAEMON to enable)"
 | 
						|
	test_done
 | 
						|
fi
 | 
						|
 | 
						|
if test_have_prereq !PIPE
 | 
						|
then
 | 
						|
	test_skip_or_die $GIT_TEST_GIT_DAEMON "file system does not support FIFOs"
 | 
						|
fi
 | 
						|
 | 
						|
LIB_GIT_DAEMON_PORT=${LIB_GIT_DAEMON_PORT-${this_test#t}}
 | 
						|
 | 
						|
GIT_DAEMON_PID=
 | 
						|
GIT_DAEMON_DOCUMENT_ROOT_PATH="$PWD"/repo
 | 
						|
GIT_DAEMON_HOST_PORT=127.0.0.1:$LIB_GIT_DAEMON_PORT
 | 
						|
GIT_DAEMON_URL=git://$GIT_DAEMON_HOST_PORT
 | 
						|
 | 
						|
start_git_daemon() {
 | 
						|
	if test -n "$GIT_DAEMON_PID"
 | 
						|
	then
 | 
						|
		error "start_git_daemon already called"
 | 
						|
	fi
 | 
						|
 | 
						|
	mkdir -p "$GIT_DAEMON_DOCUMENT_ROOT_PATH"
 | 
						|
 | 
						|
	trap 'code=$?; stop_git_daemon; (exit $code); die' EXIT
 | 
						|
 | 
						|
	say >&3 "Starting git daemon ..."
 | 
						|
	mkfifo git_daemon_output
 | 
						|
	${LIB_GIT_DAEMON_COMMAND:-git daemon} \
 | 
						|
		--listen=127.0.0.1 --port="$LIB_GIT_DAEMON_PORT" \
 | 
						|
		--reuseaddr --verbose \
 | 
						|
		--base-path="$GIT_DAEMON_DOCUMENT_ROOT_PATH" \
 | 
						|
		"$@" "$GIT_DAEMON_DOCUMENT_ROOT_PATH" \
 | 
						|
		>&3 2>git_daemon_output &
 | 
						|
	GIT_DAEMON_PID=$!
 | 
						|
	>daemon.log
 | 
						|
	{
 | 
						|
		read -r line <&7
 | 
						|
		printf "%s\n" "$line"
 | 
						|
		printf >&4 "%s\n" "$line"
 | 
						|
		(
 | 
						|
			while read -r line <&7
 | 
						|
			do
 | 
						|
				printf "%s\n" "$line"
 | 
						|
				printf >&4 "%s\n" "$line"
 | 
						|
			done
 | 
						|
		) &
 | 
						|
	} 7<git_daemon_output >>"$TRASH_DIRECTORY/daemon.log" &&
 | 
						|
 | 
						|
	# Check expected output
 | 
						|
	if test x"$(expr "$line" : "\[[0-9]*\] \(.*\)")" != x"Ready to rumble"
 | 
						|
	then
 | 
						|
		kill "$GIT_DAEMON_PID"
 | 
						|
		wait "$GIT_DAEMON_PID"
 | 
						|
		trap 'die' EXIT
 | 
						|
		test_skip_or_die $GIT_TEST_GIT_DAEMON \
 | 
						|
			"git daemon failed to start"
 | 
						|
	fi
 | 
						|
}
 | 
						|
 | 
						|
stop_git_daemon() {
 | 
						|
	if test -z "$GIT_DAEMON_PID"
 | 
						|
	then
 | 
						|
		return
 | 
						|
	fi
 | 
						|
 | 
						|
	trap 'die' EXIT
 | 
						|
 | 
						|
	# kill git-daemon child of git
 | 
						|
	say >&3 "Stopping git daemon ..."
 | 
						|
	kill "$GIT_DAEMON_PID"
 | 
						|
	wait "$GIT_DAEMON_PID" >&3 2>&4
 | 
						|
	ret=$?
 | 
						|
	if test_match_signal 15 $?
 | 
						|
	then
 | 
						|
		error "git daemon exited with status: $ret"
 | 
						|
	fi
 | 
						|
	GIT_DAEMON_PID=
 | 
						|
	rm -f git_daemon_output
 | 
						|
}
 | 
						|
 | 
						|
# A stripped-down version of a netcat client, that connects to a "host:port"
 | 
						|
# given in $1, sends its stdin followed by EOF, then dumps the response (until
 | 
						|
# EOF) to stdout.
 | 
						|
fake_nc() {
 | 
						|
	if ! test_declared_prereq FAKENC
 | 
						|
	then
 | 
						|
		echo >&4 "fake_nc: need to declare FAKENC prerequisite"
 | 
						|
		return 127
 | 
						|
	fi
 | 
						|
	perl -Mstrict -MIO::Socket::INET -e '
 | 
						|
		my $s = IO::Socket::INET->new(shift)
 | 
						|
			or die "unable to open socket: $!";
 | 
						|
		print $s <STDIN>;
 | 
						|
		$s->shutdown(1);
 | 
						|
		print <$s>;
 | 
						|
	' "$@"
 | 
						|
}
 | 
						|
 | 
						|
test_lazy_prereq FAKENC '
 | 
						|
	perl -MIO::Socket::INET -e "exit 0"
 | 
						|
'
 |