# Library of functions shared by all CI scripts if test true = "$GITHUB_ACTIONS" then begin_group () { need_to_end_group=t echo "::group::$1" >&2 set -x } end_group () { test -n "$need_to_end_group" || return 0 set +x need_to_end_group= echo '::endgroup::' >&2 } elif test true = "$GITLAB_CI" then begin_group () { need_to_end_group=t printf "\e[0Ksection_start:$(date +%s):$(echo "$1" | tr ' ' _)\r\e[0K$1\n" trap "end_group '$1'" EXIT set -x } end_group () { test -n "$need_to_end_group" || return 0 set +x need_to_end_group= printf "\e[0Ksection_end:$(date +%s):$(echo "$1" | tr ' ' _)\r\e[0K\n" trap - EXIT } else begin_group () { :; } end_group () { :; } set -x fi group () { group="$1" shift begin_group "$group" # work around `dash` not supporting `set -o pipefail` ( "$@" 2>&1 echo $? >exit.status ) | sed 's/^\(\([^ ]*\):\([0-9]*\):\([0-9]*:\) \)\(error\|warning\): /::\5 file=\2,line=\3::\1/' res=$(cat exit.status) rm exit.status end_group "$group" return $res } begin_group "CI setup" trap "end_group 'CI setup'" EXIT # Set 'exit on error' for all CI scripts to let the caller know that # something went wrong. # # We already enabled tracing executed commands earlier. This helps by showing # how # environment variables are set and and dependencies are installed. set -e skip_branch_tip_with_tag () { # Sometimes, a branch is pushed at the same time the tag that points # at the same commit as the tip of the branch is pushed, and building # both at the same time is a waste. # # When the build is triggered by a push to a tag, $CI_BRANCH will # have that tagname, e.g. v2.14.0. Let's see if $CI_BRANCH is # exactly at a tag, and if so, if it is different from $CI_BRANCH. # That way, we can tell if we are building the tip of a branch that # is tagged and we can skip the build because we won't be skipping a # build of a tag. if TAG=$(git describe --exact-match "$CI_BRANCH" 2>/dev/null) && test "$TAG" != "$CI_BRANCH" then echo "$(tput setaf 2)Tip of $CI_BRANCH is exactly at $TAG$(tput sgr0)" exit 0 fi } # Check whether we can use the path passed via the first argument as Git # repository. is_usable_git_repository () { # We require Git in our PATH, otherwise we cannot access repositories # at all. if ! command -v git >/dev/null then return 1 fi # And the target directory needs to be a proper Git repository. if ! git -C "$1" rev-parse 2>/dev/null then return 1 fi } # Save some info about the current commit's tree, so we can skip the build # job if we encounter the same tree again and can provide a useful info # message. save_good_tree () { if ! is_usable_git_repository . then return fi echo "$(git rev-parse $CI_COMMIT^{tree}) $CI_COMMIT $CI_JOB_NUMBER $CI_JOB_ID" >>"$good_trees_file" # limit the file size tail -1000 "$good_trees_file" >"$good_trees_file".tmp mv "$good_trees_file".tmp "$good_trees_file" } # Skip the build job if the same tree has already been built and tested # successfully before (e.g. because the branch got rebased, changing only # the commit messages). skip_good_tree () { if test true = "$GITHUB_ACTIONS" then return fi if ! is_usable_git_repository . then return fi if ! good_tree_info="$(grep "^$(git rev-parse $CI_COMMIT^{tree}) " "$good_trees_file")" then # Haven't seen this tree yet, or no cached good trees file yet. # Continue the build job. return fi echo "$good_tree_info" | { read tree prev_good_commit prev_good_job_number prev_good_job_id if test "$CI_JOB_ID" = "$prev_good_job_id" then cat <<-EOF $(tput setaf 2)Skipping build job for commit $CI_COMMIT.$(tput sgr0) This commit has already been built and tested successfully by this build job. To force a re-build delete the branch's cache and then hit 'Restart job'. EOF else cat <<-EOF $(tput setaf 2)Skipping build job for commit $CI_COMMIT.$(tput sgr0) This commit's tree has already been built and tested successfully in build job $prev_good_job_number for commit $prev_good_commit. The log of that build job is available at $SYSTEM_TASKDEFINITIONSURI$SYSTEM_TEAMPROJECT/_build/results?buildId=$prev_good_job_id To force a re-build delete the branch's cache and then hit 'Restart job'. EOF fi } exit 0 } check_unignored_build_artifacts () { if ! is_usable_git_repository . then return fi ! git ls-files --other --exclude-standard --error-unmatch \ -- ':/*' 2>/dev/null || { echo "$(tput setaf 1)error: found unignored build artifacts$(tput sgr0)" false } } handle_failed_tests () { return 1 } create_failed_test_artifacts () { mkdir -p t/failed-test-artifacts for test_exit in t/test-results/*.exit do test 0 != "$(cat "$test_exit")" || continue test_name="${test_exit%.exit}" test_name="${test_name##*/}" printf "\\e[33m\\e[1m=== Failed test: ${test_name} ===\\e[m\\n" echo "The full logs are in the 'print test failures' step below." echo "See also the 'failed-tests-*' artifacts attached to this run." cat "t/test-results/$test_name.markup" trash_dir="t/trash directory.$test_name" cp "t/test-results/$test_name.out" t/failed-test-artifacts/ tar czf t/failed-test-artifacts/"$test_name".trash.tar.gz "$trash_dir" done } # GitHub Action doesn't set TERM, which is required by tput export TERM=${TERM:-dumb} # Clear MAKEFLAGS that may come from the outside world. export MAKEFLAGS= if test -n "$SYSTEM_COLLECTIONURI" || test -n "$SYSTEM_TASKDEFINITIONSURI" then CI_TYPE=azure-pipelines # We are running in Azure Pipelines CI_BRANCH="$BUILD_SOURCEBRANCH" CI_COMMIT="$BUILD_SOURCEVERSION" CI_JOB_ID="$BUILD_BUILDID" CI_JOB_NUMBER="$BUILD_BUILDNUMBER" CI_OS_NAME="$(echo "$AGENT_OS" | tr A-Z a-z)" test darwin != "$CI_OS_NAME" || CI_OS_NAME=osx CI_REPO_SLUG="$(expr "$BUILD_REPOSITORY_URI" : '.*/\([^/]*/[^/]*\)$')" CC="${CC:-gcc}" # use a subdirectory of the cache dir (because the file share is shared # among *all* phases) cache_dir="$HOME/test-cache/$SYSTEM_PHASENAME" GIT_TEST_OPTS="--write-junit-xml" JOBS=10 elif test true = "$GITHUB_ACTIONS" then CI_TYPE=github-actions CI_BRANCH="$GITHUB_REF" CI_COMMIT="$GITHUB_SHA" CI_OS_NAME="$(echo "$RUNNER_OS" | tr A-Z a-z)" test macos != "$CI_OS_NAME" || CI_OS_NAME=osx CI_REPO_SLUG="$GITHUB_REPOSITORY" CI_JOB_ID="$GITHUB_RUN_ID" CC="${CC_PACKAGE:-${CC:-gcc}}" DONT_SKIP_TAGS=t handle_failed_tests () { echo "FAILED_TEST_ARTIFACTS=t/failed-test-artifacts" >>$GITHUB_ENV create_failed_test_artifacts return 1 } cache_dir="$HOME/none" GIT_TEST_OPTS="--github-workflow-markup" JOBS=10 elif test true = "$GITLAB_CI" then CI_TYPE=gitlab-ci CI_BRANCH="$CI_COMMIT_REF_NAME" CI_COMMIT="$CI_COMMIT_SHA" case "$CI_JOB_IMAGE" in macos-*) # GitLab CI has Python installed via multiple package managers, # most notably via asdf and Homebrew. Ensure that our builds # pick up the Homebrew one by prepending it to our PATH as the # asdf one breaks tests. export PATH="$(brew --prefix)/bin:$PATH" CI_OS_NAME=osx ;; alpine:*|fedora:*|ubuntu:*) CI_OS_NAME=linux;; *) echo "Could not identify OS image" >&2 env >&2 exit 1 ;; esac CI_REPO_SLUG="$CI_PROJECT_PATH" CI_JOB_ID="$CI_JOB_ID" CC="${CC_PACKAGE:-${CC:-gcc}}" DONT_SKIP_TAGS=t handle_failed_tests () { create_failed_test_artifacts return 1 } cache_dir="$HOME/none" distro=$(echo "$CI_JOB_IMAGE" | tr : -) JOBS=$(nproc) else echo "Could not identify CI type" >&2 env >&2 exit 1 fi MAKEFLAGS="$MAKEFLAGS --jobs=$JOBS" GIT_PROVE_OPTS="--timer --jobs $JOBS" GIT_TEST_OPTS="$GIT_TEST_OPTS --verbose-log -x" case "$CI_OS_NAME" in windows|windows_nt) GIT_TEST_OPTS="$GIT_TEST_OPTS --no-chain-lint --no-bin-wrappers" ;; esac export GIT_TEST_OPTS export GIT_PROVE_OPTS good_trees_file="$cache_dir/good-trees" mkdir -p "$cache_dir" test -n "${DONT_SKIP_TAGS-}" || skip_branch_tip_with_tag skip_good_tree if test -z "$jobname" then jobname="$CI_OS_NAME-$CC" fi export DEVELOPER=1 export DEFAULT_TEST_TARGET=prove export GIT_TEST_CLONE_2GB=true export SKIP_DASHED_BUILT_INS=YesPlease case "$distro" in ubuntu-*) if test "$jobname" = "linux-gcc-default" then break fi PYTHON_PACKAGE=python2 if test "$jobname" = linux-gcc then PYTHON_PACKAGE=python3 fi MAKEFLAGS="$MAKEFLAGS PYTHON_PATH=/usr/bin/$PYTHON_PACKAGE" export GIT_TEST_HTTPD=true # The Linux build installs the defined dependency versions below. # The OS X build installs much more recent versions, whichever # were recorded in the Homebrew database upon creating the OS X # image. # Keep that in mind when you encounter a broken OS X build! export LINUX_GIT_LFS_VERSION="1.5.2" ;; macos-*) MAKEFLAGS="$MAKEFLAGS PYTHON_PATH=$(which python3)" if [ "$jobname" != osx-gcc ] then MAKEFLAGS="$MAKEFLAGS APPLE_COMMON_CRYPTO_SHA1=Yes" fi ;; esac CUSTOM_PATH="${CUSTOM_PATH:-$HOME/path}" export PATH="$CUSTOM_PATH:$PATH" case "$jobname" in linux32) CC=gcc ;; linux-musl) CC=gcc MAKEFLAGS="$MAKEFLAGS PYTHON_PATH=/usr/bin/python3 USE_LIBPCRE2=Yes" MAKEFLAGS="$MAKEFLAGS NO_REGEX=Yes ICONV_OMITS_BOM=Yes" MAKEFLAGS="$MAKEFLAGS GIT_TEST_UTF8_LOCALE=C.UTF-8" ;; linux-leaks|linux-reftable-leaks) export SANITIZE=leak export GIT_TEST_PASSING_SANITIZE_LEAK=true export GIT_TEST_SANITIZE_LEAK_LOG=true ;; linux-asan-ubsan) export SANITIZE=address,undefined export NO_SVN_TESTS=LetsSaveSomeTime MAKEFLAGS="$MAKEFLAGS NO_PYTHON=YepBecauseP4FlakesTooOften" ;; esac MAKEFLAGS="$MAKEFLAGS CC=${CC:-cc}" end_group "CI setup" set -x