| # 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 <drizzd@aon.at> |
| # |
| |
| 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 |
| } |