#!/usr/bin/env bash set -e if [ -n "$SKIP_TESTS" ]; then if [ -z "$SKIP_OFFLINE_TESTS" ]; then SKIP_OFFLINE_TESTS=1; fi if [ -z "$SKIP_ONLINE_TESTS" ]; then SKIP_ONLINE_TESTS=1; fi if [ -z "$SKIP_GITDAEMON_TESTS" ]; then SKIP_GITDAEMON_TESTS=1; fi if [ -z "$SKIP_PROXY_TESTS" ]; then SKIP_PROXY_TESTS=1; fi if [ -z "$SKIP_NTLM_TESTS" ]; then SKIP_NTLM_TESTS=1; fi if [ -z "$SKIP_NEGOTIATE_TESTS" ]; then SKIP_NEGOTIATE_TESTS=1; fi if [ -z "$SKIP_SSH_TESTS" ]; then SKIP_SSH_TESTS=1; fi if [ -z "$SKIP_FUZZERS" ]; then SKIP_FUZZERS=1; fi fi # Windows doesn't run the NTLM tests properly (yet) if [[ "$(uname -s)" == MINGW* ]]; then SKIP_NTLM_TESTS=1 fi # older versions of git don't support push options if [ -z "$SKIP_PUSHOPTIONS_TESTS" ]; then export GITTEST_PUSH_OPTIONS=true fi SOURCE_DIR=${SOURCE_DIR:-$( cd "$( dirname "${BASH_SOURCE[0]}" )" && dirname $( pwd ) )} BUILD_DIR=$(pwd) BUILD_PATH=${BUILD_PATH:=$PATH} CTEST=$(which ctest) TMPDIR=${TMPDIR:-/tmp} USER=${USER:-$(whoami)} GITTEST_SSH_KEYTYPE=${GITTEST_SSH_KEYTYPE:="ecdsa"} HOME=`mktemp -d ${TMPDIR}/home.XXXXXXXX` export CLAR_HOMEDIR=${HOME} SUCCESS=1 CONTINUE_ON_FAILURE=0 should_run() { eval "skip=\${SKIP_${1}}" [ -z "$skip" \ -o "$skip" == "no" -o "$skip" == "NO" \ -o "$skip" == "n" -o "$skip" == "N" \ -o "$skip" == "false" -o "$skip" == "FALSE" \ -o "$skip" == "f" -o "$skip" == "F" \ -o "$skip" == "0" ] } cleanup() { echo "Cleaning up..." if [ ! -z "$GIT_STANDARD_PID" ]; then echo "Stopping git daemon (standard)..." kill $GIT_STANDARD_PID fi if [ ! -z "$GIT_NAMESPACE_PID" ]; then echo "Stopping git daemon (namespace)..." kill $GIT_NAMESPACE_PID fi if [ ! -z "$GIT_SHA256_PID" ]; then echo "Stopping git daemon (sha256)..." kill $GIT_SHA256_PID fi if [ ! -z "$PROXY_BASIC_PID" ]; then echo "Stopping proxy (Basic)..." kill $PROXY_BASIC_PID fi if [ ! -z "$PROXY_NTLM_PID" ]; then echo "Stopping proxy (NTLM)..." kill $PROXY_NTLM_PID fi if [ ! -z "$HTTP_PID" ]; then echo "Stopping HTTP server..." kill $HTTP_PID fi if [ ! -z "$SSHD_DIR" -a -f "${SSHD_DIR}/pid" ]; then echo "Stopping SSH server..." kill $(cat "${SSHD_DIR}/pid") fi echo "Done." } run_test() { if [[ "$GITTEST_FLAKY_RETRY" > 0 ]]; then ATTEMPTS_REMAIN=$GITTEST_FLAKY_RETRY else ATTEMPTS_REMAIN=1 fi FAILED=0 while [[ "$ATTEMPTS_REMAIN" > 0 ]]; do if [ "$FAILED" -eq 1 ]; then echo "" echo "Re-running flaky ${1} tests..." echo "" sleep 2 fi RETURN_CODE=0 ( export PATH="${BUILD_PATH}" export CLAR_SUMMARY="${BUILD_DIR}/results_${1}.xml" "${CTEST}" -V -R "^${1}$" ) || RETURN_CODE=$? && true if [ "$RETURN_CODE" -eq 0 ]; then FAILED=0 break fi echo "Test exited with code: $RETURN_CODE" ATTEMPTS_REMAIN="$(($ATTEMPTS_REMAIN-1))" FAILED=1 done if [ "$FAILED" -ne 0 ]; then if [ "$CONTINUE_ON_FAILURE" -ne 1 ]; then exit 1 fi SUCCESS=0 fi } indent() { sed "s/^/ /"; } cygfullpath() { result=$(echo "${1}" | tr \; \\n | while read -r element; do if [ "${last}" != "" ]; then echo -n ":"; fi echo -n $(cygpath "${element}") last="${element}" done) if [ "${result}" = "" ]; then exit 1; fi echo "${result}" } if [[ "$(uname -s)" == MINGW* ]]; then BUILD_PATH=$(cygfullpath "$BUILD_PATH") fi # Configure the test environment; run them early so that we're certain # that they're started by the time we need them. echo "CTest version:" env PATH="${BUILD_PATH}" "${CTEST}" --version | head -1 2>&1 | indent echo "" echo "##############################################################################" echo "## Configuring test environment" echo "##############################################################################" echo "" if should_run "GITDAEMON_TESTS"; then echo "Starting git daemon (standard)..." GIT_STANDARD_DIR=`mktemp -d ${TMPDIR}/git_standard.XXXXXXXX` cp -R "${SOURCE_DIR}/tests/resources/pushoptions.git" "${GIT_STANDARD_DIR}/test.git" git daemon --listen=localhost --export-all --enable=receive-pack --base-path="${GIT_STANDARD_DIR}" "${GIT_STANDARD_DIR}" 2>/dev/null & GIT_STANDARD_PID=$! echo "Starting git daemon (namespace)..." GIT_NAMESPACE_DIR=`mktemp -d ${TMPDIR}/git_namespace.XXXXXXXX` cp -R "${SOURCE_DIR}/tests/resources/namespace.git" "${GIT_NAMESPACE_DIR}/namespace.git" GIT_NAMESPACE="name1" git daemon --listen=localhost --port=9419 --export-all --enable=receive-pack --base-path="${GIT_NAMESPACE_DIR}" "${GIT_NAMESPACE_DIR}" & GIT_NAMESPACE_PID=$! echo "Starting git daemon (sha256)..." GIT_SHA256_DIR=`mktemp -d ${TMPDIR}/git_sha256.XXXXXXXX` cp -R "${SOURCE_DIR}/tests/resources/testrepo_256.git" "${GIT_SHA256_DIR}/testrepo_256.git" git daemon --listen=localhost --port=9420 --export-all --enable=receive-pack --base-path="${GIT_SHA256_DIR}" "${GIT_SHA256_DIR}" & GIT_SHA256_PID=$! fi if should_run "PROXY_TESTS"; then curl --location --silent --show-error https://github.com/ethomson/poxyproxy/releases/download/v0.7.0/poxyproxy-0.7.0.jar >poxyproxy.jar echo "Starting HTTP proxy (Basic)..." java -jar poxyproxy.jar --address 127.0.0.1 --port 8080 --credentials foo:bar --auth-type basic --quiet & PROXY_BASIC_PID=$! echo "Starting HTTP proxy (NTLM)..." java -jar poxyproxy.jar --address 127.0.0.1 --port 8090 --credentials foo:bar --auth-type ntlm --quiet & PROXY_NTLM_PID=$! fi if should_run "NTLM_TESTS" || should_run "ONLINE_TESTS"; then curl --location --silent --show-error https://github.com/ethomson/poxygit/releases/download/v0.6.0/poxygit-0.6.0.jar >poxygit.jar echo "Starting HTTP server..." HTTP_DIR=`mktemp -d ${TMPDIR}/http.XXXXXXXX` cp -R "${SOURCE_DIR}/tests/resources/pushoptions.git" "${HTTP_DIR}/test.git" java -jar poxygit.jar --address 127.0.0.1 --port 9000 --credentials foo:baz --quiet "${HTTP_DIR}" & HTTP_PID=$! fi if should_run "SSH_TESTS"; then echo "Starting SSH server..." SSHD_DIR=`mktemp -d ${TMPDIR}/sshd.XXXXXXXX` cp -R "${SOURCE_DIR}/tests/resources/pushoptions.git" "${SSHD_DIR}/test.git" cat >"${SSHD_DIR}/sshd_config" <<-EOF Port 2222 ListenAddress 0.0.0.0 Protocol 2 HostKey ${SSHD_DIR}/id_${GITTEST_SSH_KEYTYPE} PidFile ${SSHD_DIR}/pid AuthorizedKeysFile ${HOME}/.ssh/authorized_keys LogLevel DEBUG RSAAuthentication yes PasswordAuthentication yes PubkeyAuthentication yes ChallengeResponseAuthentication no StrictModes no HostCertificate ${SSHD_DIR}/id_${GITTEST_SSH_KEYTYPE}.pub HostKey ${SSHD_DIR}/id_${GITTEST_SSH_KEYTYPE} # Required here as sshd will simply close connection otherwise UsePAM no EOF ssh-keygen -t "${GITTEST_SSH_KEYTYPE}" -f "${SSHD_DIR}/id_${GITTEST_SSH_KEYTYPE}" -N "" -q /usr/sbin/sshd -f "${SSHD_DIR}/sshd_config" -E "${SSHD_DIR}/log" # Set up keys mkdir "${HOME}/.ssh" ssh-keygen -t "${GITTEST_SSH_KEYTYPE}" -f "${HOME}/.ssh/id_${GITTEST_SSH_KEYTYPE}" -N "" -q cat "${HOME}/.ssh/id_${GITTEST_SSH_KEYTYPE}.pub" >>"${HOME}/.ssh/authorized_keys" while read algorithm key comment; do echo "[localhost]:2222 $algorithm $key" >>"${HOME}/.ssh/known_hosts" done <"${SSHD_DIR}/id_${GITTEST_SSH_KEYTYPE}.pub" # Append the github.com keys for the tests that don't override checks. # We ask for ssh-rsa to test that the selection based off of known_hosts # is working. ssh-keyscan -t ssh-rsa github.com >>"${HOME}/.ssh/known_hosts" # Get the fingerprint for localhost and remove the colons so we can # parse it as a hex number. Older versions have a different output # format. if [[ $(ssh -V 2>&1) == OpenSSH_6* ]]; then SSH_FINGERPRINT=$(ssh-keygen -F '[localhost]:2222' -f "${HOME}/.ssh/known_hosts" -l | tail -n 1 | cut -d ' ' -f 2 | tr -d ':') else SSH_FINGERPRINT=$(ssh-keygen -E md5 -F '[localhost]:2222' -f "${HOME}/.ssh/known_hosts" -l | tail -n 1 | cut -d ' ' -f 3 | cut -d : -f2- | tr -d :) fi fi # Run the tests that do not require network connectivity. if should_run "OFFLINE_TESTS"; then echo "" echo "##############################################################################" echo "## Running core tests" echo "##############################################################################" echo "" echo "Running libgit2 integration (offline) tests" echo "" run_test offline echo "" echo "Running utility tests" echo "" run_test util fi if [ -n "$RUN_INVASIVE_TESTS" ]; then echo "" echo "Running invasive tests" echo "" export GITTEST_INVASIVE_FS_SIZE=1 export GITTEST_INVASIVE_MEMORY=1 export GITTEST_INVASIVE_SPEED=1 run_test invasive unset GITTEST_INVASIVE_FS_SIZE unset GITTEST_INVASIVE_MEMORY unset GITTEST_INVASIVE_SPEED fi # the various network tests can fail due to network connectivity problems; # allow them to retry up to 5 times export GITTEST_FLAKY_RETRY=5 if should_run "ONLINE_TESTS"; then # Run the online tests. The "online" test suite only includes the # default online tests that do not require additional configuration. # The "proxy" and "ssh" test suites require further setup. echo "" echo "##############################################################################" echo "## Running networking (online) tests" echo "##############################################################################" export GITTEST_REMOTE_REDIRECT_INITIAL="http://localhost:9000/initial-redirect/libgit2/TestGitRepository" export GITTEST_REMOTE_REDIRECT_SUBSEQUENT="http://localhost:9000/subsequent-redirect/libgit2/TestGitRepository" export GITTEST_REMOTE_SPEED_SLOW="http://localhost:9000/speed-9600/test.git" export GITTEST_REMOTE_SPEED_TIMESOUT="http://localhost:9000/speed-0.5/test.git" run_test online unset GITTEST_REMOTE_REDIRECT_INITIAL unset GITTEST_REMOTE_REDIRECT_SUBSEQUENT unset GITTEST_REMOTE_SPEED_SLOW unset GITTEST_REMOTE_SPEED_TIMESOUT # Run the online tests that immutably change global state separately # to avoid polluting the test environment. echo "" echo "Running custom certificate (online_customcert) tests" echo "" run_test online_customcert fi if should_run "GITDAEMON_TESTS"; then echo "" echo "Running gitdaemon (standard) tests" echo "" export GITTEST_REMOTE_URL="git://localhost/test.git" run_test gitdaemon unset GITTEST_REMOTE_URL echo "" echo "Running gitdaemon (namespace) tests" echo "" export GITTEST_REMOTE_URL="git://localhost:9419/namespace.git" export GITTEST_REMOTE_BRANCH="four" run_test gitdaemon_namespace unset GITTEST_REMOTE_URL unset GITTEST_REMOTE_BRANCH echo "" echo "Running gitdaemon (sha256) tests" echo "" export GITTEST_REMOTE_URL="git://localhost:9420/testrepo_256.git" run_test gitdaemon_sha256 unset GITTEST_REMOTE_URL fi if should_run "PROXY_TESTS"; then echo "" echo "Running proxy tests (Basic authentication)" echo "" export GITTEST_REMOTE_PROXY_HOST="localhost:8080" export GITTEST_REMOTE_PROXY_USER="foo" export GITTEST_REMOTE_PROXY_PASS="bar" run_test proxy unset GITTEST_REMOTE_PROXY_HOST unset GITTEST_REMOTE_PROXY_USER unset GITTEST_REMOTE_PROXY_PASS echo "" echo "Running proxy tests (NTLM authentication)" echo "" export GITTEST_REMOTE_PROXY_HOST="localhost:8090" export GITTEST_REMOTE_PROXY_USER="foo" export GITTEST_REMOTE_PROXY_PASS="bar" run_test proxy unset GITTEST_REMOTE_PROXY_HOST unset GITTEST_REMOTE_PROXY_USER unset GITTEST_REMOTE_PROXY_PASS fi if should_run "NTLM_TESTS"; then echo "" echo "Running NTLM tests (IIS emulation)" echo "" export GITTEST_REMOTE_URL="http://localhost:9000/ntlm/test.git" export GITTEST_REMOTE_USER="foo" export GITTEST_REMOTE_PASS="baz" run_test auth_clone_and_push unset GITTEST_REMOTE_URL unset GITTEST_REMOTE_USER unset GITTEST_REMOTE_PASS echo "" echo "Running NTLM tests (Apache emulation)" echo "" export GITTEST_REMOTE_URL="http://localhost:9000/broken-ntlm/test.git" export GITTEST_REMOTE_USER="foo" export GITTEST_REMOTE_PASS="baz" run_test auth_clone_and_push unset GITTEST_REMOTE_URL unset GITTEST_REMOTE_USER unset GITTEST_REMOTE_PASS fi if should_run "NEGOTIATE_TESTS" && -n "$GITTEST_NEGOTIATE_PASSWORD" ; then echo "" echo "Running SPNEGO tests" echo "" if [ "$(uname -s)" = "Darwin" ]; then KINIT_FLAGS="--password-file=STDIN" fi echo $GITTEST_NEGOTIATE_PASSWORD | kinit $KINIT_FLAGS test@LIBGIT2.ORG klist -5f export GITTEST_REMOTE_URL="https://test.libgit2.org/kerberos/empty.git" export GITTEST_REMOTE_DEFAULT="true" run_test auth_clone unset GITTEST_REMOTE_URL unset GITTEST_REMOTE_DEFAULT echo "" echo "Running SPNEGO tests (expect/continue)" echo "" export GITTEST_REMOTE_URL="https://test.libgit2.org/kerberos/empty.git" export GITTEST_REMOTE_DEFAULT="true" export GITTEST_REMOTE_EXPECTCONTINUE="true" run_test auth_clone unset GITTEST_REMOTE_URL unset GITTEST_REMOTE_DEFAULT unset GITTEST_REMOTE_EXPECTCONTINUE kdestroy -A fi if should_run "SSH_TESTS"; then export GITTEST_REMOTE_USER=$USER export GITTEST_REMOTE_SSH_KEY="${HOME}/.ssh/id_${GITTEST_SSH_KEYTYPE}" export GITTEST_REMOTE_SSH_PUBKEY="${HOME}/.ssh/id_${GITTEST_SSH_KEYTYPE}.pub" export GITTEST_REMOTE_SSH_PASSPHRASE="" export GITTEST_REMOTE_SSH_FINGERPRINT="${SSH_FINGERPRINT}" export GITTEST_SSH_CMD="ssh -i ${HOME}/.ssh/id_${GITTEST_SSH_KEYTYPE} -o UserKnownHostsFile=${HOME}/.ssh/known_hosts" echo "" echo "Running ssh tests" echo "" export GITTEST_REMOTE_URL="ssh://localhost:2222/$SSHD_DIR/test.git" run_test ssh unset GITTEST_REMOTE_URL echo "" echo "Running ssh tests (scp-style paths)" echo "" export GITTEST_REMOTE_URL="[localhost:2222]:$SSHD_DIR/test.git" run_test ssh unset GITTEST_REMOTE_URL unset GITTEST_SSH_CMD unset GITTEST_REMOTE_USER unset GITTEST_REMOTE_SSH_KEY unset GITTEST_REMOTE_SSH_PUBKEY unset GITTEST_REMOTE_SSH_PASSPHRASE unset GITTEST_REMOTE_SSH_FINGERPRINT fi unset GITTEST_FLAKY_RETRY if should_run "FUZZERS"; then echo "" echo "##############################################################################" echo "## Running fuzzers" echo "##############################################################################" env PATH="${BUILD_PATH}" "${CTEST}" -V -R 'fuzzer' fi cleanup if [ "$SUCCESS" -ne 1 ]; then echo "Some tests failed." exit 1 fi echo "Success." exit 0