# Shell library to run an HTTP server for use in tests. # Ends the test early if httpd tests should not be run, # for example because the user has not enabled them. # # Usage: # # . ./test-lib.sh # . "$TEST_DIRECTORY"/lib-httpd.sh # start_httpd # # test_expect_success '...' ' # ... # ' # # test_expect_success ... # # test_done # # Can be configured using the following variables. # # GIT_TEST_HTTPD enable HTTPD tests # LIB_HTTPD_PATH web server path # LIB_HTTPD_MODULE_PATH web server modules path # LIB_HTTPD_PORT listening port # LIB_HTTPD_DAV enable DAV # LIB_HTTPD_SVN enable SVN at given location (e.g. "svn") # LIB_HTTPD_SSL enable SSL # LIB_HTTPD_PROXY enable proxy # # Copyright (c) 2008 Clemens Buchacher # if ! test_have_prereq LIBCURL then skip_all='skipping test, git built without http support' test_done fi if test -n "$NO_EXPAT" && test -n "$LIB_HTTPD_DAV" then skip_all='skipping test, git built without expat support' test_done fi if ! test_bool_env GIT_TEST_HTTPD true then skip_all="Network testing disabled (unset GIT_TEST_HTTPD to enable)" test_done fi if ! test_have_prereq NOT_ROOT; then test_skip_or_die GIT_TEST_HTTPD \ "Cannot run httpd tests as root" fi HTTPD_PARA="" for DEFAULT_HTTPD_PATH in '/usr/sbin/httpd' \ '/usr/sbin/apache2' \ "$(command -v httpd)" \ "$(command -v apache2)" do if test -n "$DEFAULT_HTTPD_PATH" && test -x "$DEFAULT_HTTPD_PATH" then break fi done if test -x "$DEFAULT_HTTPD_PATH" then DETECTED_HTTPD_ROOT="$("$DEFAULT_HTTPD_PATH" -V 2>/dev/null | sed -n 's/^ -D HTTPD_ROOT="\(.*\)"$/\1/p')" fi for DEFAULT_HTTPD_MODULE_PATH in '/usr/libexec/apache2' \ '/usr/lib/apache2/modules' \ '/usr/lib64/httpd/modules' \ '/usr/lib/httpd/modules' \ '/usr/libexec/httpd' \ '/usr/lib/apache2' \ "${DETECTED_HTTPD_ROOT:+${DETECTED_HTTPD_ROOT}/modules}" do if test -n "$DEFAULT_HTTPD_MODULE_PATH" && test -d "$DEFAULT_HTTPD_MODULE_PATH" then break fi done case $(uname) in Darwin) HTTPD_PARA="$HTTPD_PARA -DDarwin" ;; esac LIB_HTTPD_PATH=${LIB_HTTPD_PATH-"$DEFAULT_HTTPD_PATH"} test_set_port LIB_HTTPD_PORT TEST_PATH="$TEST_DIRECTORY"/lib-httpd HTTPD_ROOT_PATH="$PWD"/httpd HTTPD_DOCUMENT_ROOT_PATH=$HTTPD_ROOT_PATH/www # hack to suppress apache PassEnv warnings GIT_VALGRIND=$GIT_VALGRIND; export GIT_VALGRIND GIT_VALGRIND_OPTIONS=$GIT_VALGRIND_OPTIONS; export GIT_VALGRIND_OPTIONS GIT_TEST_SIDEBAND_ALL=$GIT_TEST_SIDEBAND_ALL; export GIT_TEST_SIDEBAND_ALL GIT_TRACE=$GIT_TRACE; export GIT_TRACE if ! test -x "$LIB_HTTPD_PATH" then test_skip_or_die GIT_TEST_HTTPD "no web server found at '$LIB_HTTPD_PATH'" fi HTTPD_VERSION=$($LIB_HTTPD_PATH -v | \ sed -n 's/^Server version: Apache\/\([0-9.]*\).*$/\1/p; q') HTTPD_VERSION_MAJOR=$(echo $HTTPD_VERSION | cut -d. -f1) HTTPD_VERSION_MINOR=$(echo $HTTPD_VERSION | cut -d. -f2) if test -n "$HTTPD_VERSION_MAJOR" then if test -z "$LIB_HTTPD_MODULE_PATH" then if ! test "$HTTPD_VERSION_MAJOR" -eq 2 || ! test "$HTTPD_VERSION_MINOR" -ge 4 then test_skip_or_die GIT_TEST_HTTPD \ "at least Apache version 2.4 is required" fi if ! test -d "$DEFAULT_HTTPD_MODULE_PATH" then test_skip_or_die GIT_TEST_HTTPD \ "Apache module directory not found" fi LIB_HTTPD_MODULE_PATH="$DEFAULT_HTTPD_MODULE_PATH" fi else test_skip_or_die GIT_TEST_HTTPD \ "Could not identify web server at '$LIB_HTTPD_PATH'" fi if test -n "$LIB_HTTPD_DAV" && test -f /etc/os-release then case "$(grep "^ID=" /etc/os-release | cut -d= -f2-)" in alpine) # The WebDAV module in Alpine Linux is broken at least up to # Alpine v3.16 as the default DBM driver is missing. # # https://gitlab.alpinelinux.org/alpine/aports/-/issues/13112 test_skip_or_die GIT_TEST_HTTPD \ "Apache WebDAV module does not have default DBM backend driver" ;; esac fi install_script () { write_script "$HTTPD_ROOT_PATH/$1" <"$TEST_PATH/$1" } prepare_httpd() { mkdir -p "$HTTPD_DOCUMENT_ROOT_PATH" cp "$TEST_PATH"/passwd "$HTTPD_ROOT_PATH" cp "$TEST_PATH"/proxy-passwd "$HTTPD_ROOT_PATH" install_script incomplete-length-upload-pack-v2-http.sh install_script incomplete-body-upload-pack-v2-http.sh install_script error-no-report.sh install_script broken-smart-http.sh install_script error-smart-http.sh install_script error.sh install_script apply-one-time-perl.sh install_script nph-custom-auth.sh ln -s "$LIB_HTTPD_MODULE_PATH" "$HTTPD_ROOT_PATH/modules" if test -n "$LIB_HTTPD_SSL" then HTTPD_PROTO=https RANDFILE_PATH="$HTTPD_ROOT_PATH"/.rnd openssl req \ -config "$TEST_PATH/ssl.cnf" \ -new -x509 -nodes \ -out "$HTTPD_ROOT_PATH/httpd.pem" \ -keyout "$HTTPD_ROOT_PATH/httpd.pem" GIT_SSL_NO_VERIFY=t export GIT_SSL_NO_VERIFY HTTPD_PARA="$HTTPD_PARA -DSSL" else HTTPD_PROTO=http fi HTTPD_DEST=127.0.0.1:$LIB_HTTPD_PORT HTTPD_URL=$HTTPD_PROTO://$HTTPD_DEST HTTPD_URL_USER=$HTTPD_PROTO://user%40host@$HTTPD_DEST HTTPD_URL_USER_PASS=$HTTPD_PROTO://user%40host:pass%40host@$HTTPD_DEST if test -n "$LIB_HTTPD_DAV" || test -n "$LIB_HTTPD_SVN" then HTTPD_PARA="$HTTPD_PARA -DDAV" if test -n "$LIB_HTTPD_SVN" then HTTPD_PARA="$HTTPD_PARA -DSVN" LIB_HTTPD_SVNPATH="$rawsvnrepo" svnrepo="http://127.0.0.1:$LIB_HTTPD_PORT/" svnrepo="$svnrepo$LIB_HTTPD_SVN" export LIB_HTTPD_SVN LIB_HTTPD_SVNPATH fi fi if test -n "$LIB_HTTPD_PROXY" then HTTPD_PARA="$HTTPD_PARA -DPROXY" fi } enable_http2 () { HTTPD_PARA="$HTTPD_PARA -DHTTP2" test_set_prereq HTTP2 } enable_cgipassauth () { # We are looking for 2.4.13 or more recent. Since we only support # 2.4 and up, no need to check for older major/minor. if test "$HTTPD_VERSION_MAJOR" = 2 && test "$HTTPD_VERSION_MINOR" = 4 && test "$(echo $HTTPD_VERSION | cut -d. -f3)" -lt 13 then echo >&4 "apache $HTTPD_VERSION too old for CGIPassAuth" return fi HTTPD_PARA="$HTTPD_PARA -DUSE_CGIPASSAUTH" test_set_prereq CGIPASSAUTH } start_httpd() { prepare_httpd >&3 2>&4 test_atexit stop_httpd "$LIB_HTTPD_PATH" -d "$HTTPD_ROOT_PATH" \ -f "$TEST_PATH/apache.conf" $HTTPD_PARA \ -c "Listen 127.0.0.1:$LIB_HTTPD_PORT" -k start \ >&3 2>&4 if test $? -ne 0 then cat "$HTTPD_ROOT_PATH"/error.log >&4 2>/dev/null test_skip_or_die GIT_TEST_HTTPD "web server setup failed" fi } stop_httpd() { "$LIB_HTTPD_PATH" -d "$HTTPD_ROOT_PATH" \ -f "$TEST_PATH/apache.conf" $HTTPD_PARA -k stop } test_http_push_nonff () { REMOTE_REPO=$1 LOCAL_REPO=$2 BRANCH=$3 EXPECT_CAS_RESULT=${4-failure} test_expect_success 'non-fast-forward push fails' ' cd "$REMOTE_REPO" && HEAD=$(git rev-parse --verify HEAD) && cd "$LOCAL_REPO" && git checkout $BRANCH && echo "changed" > path2 && git commit -a -m path2 --amend && test_must_fail git push -v origin >output 2>&1 && ( cd "$REMOTE_REPO" && echo "$HEAD" >expect && git rev-parse --verify HEAD >actual && test_cmp expect actual ) ' test_expect_success 'non-fast-forward push show ref status' ' grep "^ ! \[rejected\][ ]*$BRANCH -> $BRANCH (non-fast-forward)$" output ' test_expect_success 'non-fast-forward push shows help message' ' test_grep "Updates were rejected because" output ' test_expect_${EXPECT_CAS_RESULT} 'force with lease aka cas' ' HEAD=$( cd "$REMOTE_REPO" && git rev-parse --verify HEAD ) && test_when_finished '\'' (cd "$REMOTE_REPO" && git update-ref HEAD "$HEAD") '\'' && ( cd "$LOCAL_REPO" && git push -v --force-with-lease=$BRANCH:$HEAD origin ) && git rev-parse --verify "$BRANCH" >expect && ( cd "$REMOTE_REPO" && git rev-parse --verify HEAD ) >actual && test_cmp expect actual ' } setup_askpass_helper() { test_expect_success 'setup askpass helper' ' write_script "$TRASH_DIRECTORY/askpass" <<-\EOF && echo >>"$TRASH_DIRECTORY/askpass-query" "askpass: $*" && case "$*" in *Username*) what=user ;; *Password*) what=pass ;; esac && cat "$TRASH_DIRECTORY/askpass-$what" EOF GIT_ASKPASS="$TRASH_DIRECTORY/askpass" && export GIT_ASKPASS && export TRASH_DIRECTORY ' } set_askpass() { >"$TRASH_DIRECTORY/askpass-query" && echo "$1" >"$TRASH_DIRECTORY/askpass-user" && echo "$2" >"$TRASH_DIRECTORY/askpass-pass" } expect_askpass() { dest=$HTTPD_DEST${3+/$3} { case "$1" in none) ;; pass) echo "askpass: Password for '$HTTPD_PROTO://$2@$dest': " ;; both) echo "askpass: Username for '$HTTPD_PROTO://$dest': " echo "askpass: Password for '$HTTPD_PROTO://$2@$dest': " ;; *) false ;; esac } >"$TRASH_DIRECTORY/askpass-expect" && test_cmp "$TRASH_DIRECTORY/askpass-expect" \ "$TRASH_DIRECTORY/askpass-query" } strip_access_log() { sed -e " s/^.* \"// s/\"// s/ [1-9][0-9]*\$// s/^GET /GET / " "$HTTPD_ROOT_PATH"/access.log } # Requires one argument: the name of a file containing the expected stripped # access log entries. check_access_log() { sort "$1" >"$1".sorted && strip_access_log >access.log.stripped && sort access.log.stripped >access.log.sorted && if ! test_cmp "$1".sorted access.log.sorted then test_cmp "$1" access.log.stripped fi }