2021-12-09 17:26:46 +02:00
|
|
|
|
set -eu -o pipefail
|
2006-03-01 14:15:33 +02:00
|
|
|
|
|
2021-12-09 17:16:18 +02:00
|
|
|
|
if [[ -z "${COMMON_VARS_AND_FUNCTIONS_SH_SOURCED-}" ]]; then
|
2021-07-26 07:54:55 +03:00
|
|
|
|
|
2021-12-09 17:16:18 +02:00
|
|
|
|
COMMON_VARS_AND_FUNCTIONS_SH_SOURCED=1
|
2021-07-26 07:54:55 +03:00
|
|
|
|
|
2023-11-08 07:30:55 +02:00
|
|
|
|
set +x
|
2023-02-09 23:14:53 +02:00
|
|
|
|
|
2023-10-05 19:12:18 +03:00
|
|
|
|
export TEST_ROOT=$(realpath ${TMPDIR:-/tmp}/nix-test)/${TEST_NAME:-default/tests\/functional//}
|
2006-07-21 15:46:54 +03:00
|
|
|
|
export NIX_STORE_DIR
|
2007-08-14 16:43:51 +03:00
|
|
|
|
if ! NIX_STORE_DIR=$(readlink -f $TEST_ROOT/store 2> /dev/null); then
|
2006-07-21 15:46:54 +03:00
|
|
|
|
# Maybe the build directory is symlinked.
|
|
|
|
|
export NIX_IGNORE_SYMLINK_STORE=1
|
|
|
|
|
NIX_STORE_DIR=$TEST_ROOT/store
|
|
|
|
|
fi
|
2006-03-01 14:15:33 +02:00
|
|
|
|
export NIX_LOCALSTATE_DIR=$TEST_ROOT/var
|
|
|
|
|
export NIX_LOG_DIR=$TEST_ROOT/var/log/nix
|
|
|
|
|
export NIX_STATE_DIR=$TEST_ROOT/var/nix
|
|
|
|
|
export NIX_CONF_DIR=$TEST_ROOT/etc
|
2020-11-09 17:04:18 +02:00
|
|
|
|
export NIX_DAEMON_SOCKET_PATH=$TEST_ROOT/dSocket
|
2020-03-30 16:31:14 +03:00
|
|
|
|
unset NIX_USER_CONF_FILES
|
2013-11-25 19:47:03 +02:00
|
|
|
|
export _NIX_TEST_SHARED=$TEST_ROOT/shared
|
2017-06-19 15:24:48 +03:00
|
|
|
|
if [[ -n $NIX_STORE ]]; then
|
|
|
|
|
export _NIX_TEST_NO_SANDBOX=1
|
|
|
|
|
fi
|
|
|
|
|
export _NIX_IN_TEST=$TEST_ROOT/shared
|
2019-07-30 12:29:03 +03:00
|
|
|
|
export _NIX_TEST_NO_LSOF=1
|
2021-12-09 17:16:18 +02:00
|
|
|
|
export NIX_REMOTE=${NIX_REMOTE_-}
|
2016-07-21 15:25:06 +03:00
|
|
|
|
unset NIX_PATH
|
2016-05-30 21:22:30 +03:00
|
|
|
|
export TEST_HOME=$TEST_ROOT/test-home
|
|
|
|
|
export HOME=$TEST_HOME
|
2021-11-17 22:35:21 +02:00
|
|
|
|
unset XDG_STATE_HOME
|
|
|
|
|
unset XDG_DATA_HOME
|
2020-03-30 16:31:14 +03:00
|
|
|
|
unset XDG_CONFIG_HOME
|
|
|
|
|
unset XDG_CONFIG_DIRS
|
2017-11-24 04:50:01 +02:00
|
|
|
|
unset XDG_CACHE_HOME
|
2016-05-30 21:22:30 +03:00
|
|
|
|
mkdir -p $TEST_HOME
|
2006-03-01 14:15:33 +02:00
|
|
|
|
|
2012-03-19 02:20:02 +02:00
|
|
|
|
export PATH=@bindir@:$PATH
|
2021-03-16 14:43:08 +02:00
|
|
|
|
if [[ -n "${NIX_CLIENT_PACKAGE:-}" ]]; then
|
|
|
|
|
export PATH="$NIX_CLIENT_PACKAGE/bin":$PATH
|
|
|
|
|
fi
|
2021-11-05 12:11:33 +02:00
|
|
|
|
DAEMON_PATH="$PATH"
|
2021-03-16 14:43:08 +02:00
|
|
|
|
if [[ -n "${NIX_DAEMON_PACKAGE:-}" ]]; then
|
2021-11-19 00:22:33 +02:00
|
|
|
|
DAEMON_PATH="${NIX_DAEMON_PACKAGE}/bin:$DAEMON_PATH"
|
2021-03-16 14:43:08 +02:00
|
|
|
|
fi
|
2016-05-31 14:07:24 +03:00
|
|
|
|
coreutils=@coreutils@
|
2023-11-16 16:12:31 +02:00
|
|
|
|
lsof=@lsof@
|
2011-10-11 00:11:08 +03:00
|
|
|
|
|
2006-03-01 14:15:33 +02:00
|
|
|
|
export dot=@dot@
|
2013-11-25 19:47:03 +02:00
|
|
|
|
export SHELL="@bash@"
|
2014-09-23 16:18:44 +03:00
|
|
|
|
export PAGER=cat
|
2020-06-25 19:26:34 +03:00
|
|
|
|
export busybox="@sandbox_shell@"
|
2006-03-01 14:15:33 +02:00
|
|
|
|
|
2013-11-25 19:47:03 +02:00
|
|
|
|
export version=@PACKAGE_VERSION@
|
2006-03-01 17:46:22 +02:00
|
|
|
|
export system=@system@
|
2006-03-01 14:51:18 +02:00
|
|
|
|
|
2022-06-22 23:41:14 +03:00
|
|
|
|
export BUILD_SHARED_LIBS=@BUILD_SHARED_LIBS@
|
|
|
|
|
|
2021-07-26 07:54:55 +03:00
|
|
|
|
export IMPURE_VAR1=foo
|
|
|
|
|
export IMPURE_VAR2=bar
|
|
|
|
|
|
2014-02-17 13:22:50 +02:00
|
|
|
|
cacheDir=$TEST_ROOT/binary-cache
|
|
|
|
|
|
2006-03-01 16:26:03 +02:00
|
|
|
|
readLink() {
|
|
|
|
|
ls -l "$1" | sed 's/.*->\ //'
|
|
|
|
|
}
|
2006-09-21 21:54:08 +03:00
|
|
|
|
|
2009-03-18 18:35:35 +02:00
|
|
|
|
clearProfiles() {
|
2021-11-17 22:35:21 +02:00
|
|
|
|
profiles="$HOME"/.local/state/nix/profiles
|
|
|
|
|
rm -rf "$profiles"
|
2009-03-18 18:35:35 +02:00
|
|
|
|
}
|
|
|
|
|
|
2006-09-21 21:54:08 +03:00
|
|
|
|
clearStore() {
|
|
|
|
|
echo "clearing store..."
|
|
|
|
|
chmod -R +w "$NIX_STORE_DIR"
|
|
|
|
|
rm -rf "$NIX_STORE_DIR"
|
|
|
|
|
mkdir "$NIX_STORE_DIR"
|
2016-07-27 18:14:41 +03:00
|
|
|
|
rm -rf "$NIX_STATE_DIR"
|
|
|
|
|
mkdir "$NIX_STATE_DIR"
|
2009-03-18 18:35:35 +02:00
|
|
|
|
clearProfiles
|
2006-09-21 21:54:08 +03:00
|
|
|
|
}
|
2007-08-13 16:15:02 +03:00
|
|
|
|
|
2014-02-17 13:22:50 +02:00
|
|
|
|
clearCache() {
|
|
|
|
|
rm -rf "$cacheDir"
|
|
|
|
|
}
|
|
|
|
|
|
2016-05-30 21:22:30 +03:00
|
|
|
|
clearCacheCache() {
|
|
|
|
|
rm -f $TEST_HOME/.cache/nix/binary-cache*
|
|
|
|
|
}
|
|
|
|
|
|
2011-07-20 14:50:13 +03:00
|
|
|
|
startDaemon() {
|
2021-07-26 07:54:55 +03:00
|
|
|
|
# Don’t start the daemon twice, as this would just make it loop indefinitely
|
2021-12-09 17:16:18 +02:00
|
|
|
|
if [[ "${_NIX_TEST_DAEMON_PID-}" != '' ]]; then
|
|
|
|
|
return
|
2021-07-26 07:54:55 +03:00
|
|
|
|
fi
|
2022-03-02 23:22:55 +02:00
|
|
|
|
# Start the daemon, wait for the socket to appear.
|
2021-07-27 12:12:32 +03:00
|
|
|
|
rm -f $NIX_DAEMON_SOCKET_PATH
|
2023-12-18 20:36:18 +02:00
|
|
|
|
PATH=$DAEMON_PATH nix --extra-experimental-features 'nix-command' daemon &
|
2021-12-09 17:16:18 +02:00
|
|
|
|
_NIX_TEST_DAEMON_PID=$!
|
|
|
|
|
export _NIX_TEST_DAEMON_PID
|
2022-02-24 15:57:27 +02:00
|
|
|
|
for ((i = 0; i < 300; i++)); do
|
2022-02-28 15:41:09 +02:00
|
|
|
|
if [[ -S $NIX_DAEMON_SOCKET_PATH ]]; then
|
|
|
|
|
DAEMON_STARTED=1
|
|
|
|
|
break;
|
|
|
|
|
fi
|
2022-02-24 15:57:27 +02:00
|
|
|
|
sleep 0.1
|
2011-10-11 14:14:30 +03:00
|
|
|
|
done
|
2022-02-28 15:41:09 +02:00
|
|
|
|
if [[ -z ${DAEMON_STARTED+x} ]]; then
|
|
|
|
|
fail "Didn’t manage to start the daemon"
|
|
|
|
|
fi
|
2021-07-26 07:54:55 +03:00
|
|
|
|
trap "killDaemon" EXIT
|
2021-12-09 17:16:18 +02:00
|
|
|
|
# Save for if daemon is killed
|
|
|
|
|
NIX_REMOTE_OLD=$NIX_REMOTE
|
2011-07-20 14:50:13 +03:00
|
|
|
|
export NIX_REMOTE=daemon
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
killDaemon() {
|
2021-12-09 17:16:18 +02:00
|
|
|
|
# Don’t fail trying to stop a non-existant daemon twice
|
|
|
|
|
if [[ "${_NIX_TEST_DAEMON_PID-}" == '' ]]; then
|
|
|
|
|
return
|
|
|
|
|
fi
|
|
|
|
|
kill $_NIX_TEST_DAEMON_PID
|
2022-02-24 15:57:27 +02:00
|
|
|
|
for i in {0..100}; do
|
2021-12-09 17:16:18 +02:00
|
|
|
|
kill -0 $_NIX_TEST_DAEMON_PID 2> /dev/null || break
|
2022-02-24 15:57:27 +02:00
|
|
|
|
sleep 0.1
|
2021-07-26 07:54:55 +03:00
|
|
|
|
done
|
2021-12-09 17:16:18 +02:00
|
|
|
|
kill -9 $_NIX_TEST_DAEMON_PID 2> /dev/null || true
|
|
|
|
|
wait $_NIX_TEST_DAEMON_PID || true
|
|
|
|
|
rm -f $NIX_DAEMON_SOCKET_PATH
|
|
|
|
|
# Indicate daemon is stopped
|
|
|
|
|
unset _NIX_TEST_DAEMON_PID
|
|
|
|
|
# Restore old nix remote
|
|
|
|
|
NIX_REMOTE=$NIX_REMOTE_OLD
|
2011-07-20 14:50:13 +03:00
|
|
|
|
trap "" EXIT
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-26 07:54:55 +03:00
|
|
|
|
restartDaemon() {
|
2021-12-09 17:16:18 +02:00
|
|
|
|
[[ -z "${_NIX_TEST_DAEMON_PID:-}" ]] && return 0
|
2021-07-26 07:54:55 +03:00
|
|
|
|
|
2022-07-14 16:06:11 +03:00
|
|
|
|
killDaemon
|
|
|
|
|
startDaemon
|
2021-07-26 07:54:55 +03:00
|
|
|
|
}
|
|
|
|
|
|
2023-07-13 22:06:34 +03:00
|
|
|
|
if [[ $(uname) == Linux ]] && [[ -L /proc/self/ns/user ]] && unshare --user true; then
|
2018-11-07 18:08:28 +02:00
|
|
|
|
_canUseSandbox=1
|
|
|
|
|
fi
|
2018-01-13 15:18:35 +02:00
|
|
|
|
|
2021-07-26 07:54:55 +03:00
|
|
|
|
isDaemonNewer () {
|
|
|
|
|
[[ -n "${NIX_DAEMON_PACKAGE:-}" ]] || return 0
|
|
|
|
|
local requiredVersion="$1"
|
2023-12-18 20:36:18 +02:00
|
|
|
|
local daemonVersion=$($NIX_DAEMON_PACKAGE/bin/nix daemon --version | cut -d' ' -f3)
|
2021-10-14 16:42:54 +03:00
|
|
|
|
[[ $(nix eval --expr "builtins.compareVersions ''$daemonVersion'' ''$requiredVersion''") -ge 0 ]]
|
2021-07-26 07:54:55 +03:00
|
|
|
|
}
|
|
|
|
|
|
2023-03-16 22:00:20 +02:00
|
|
|
|
skipTest () {
|
|
|
|
|
echo "$1, skipping this test..." >&2
|
|
|
|
|
exit 99
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-26 07:54:55 +03:00
|
|
|
|
requireDaemonNewerThan () {
|
2023-03-16 22:00:20 +02:00
|
|
|
|
isDaemonNewer "$1" || skipTest "Daemon is too old"
|
2021-07-26 07:54:55 +03:00
|
|
|
|
}
|
|
|
|
|
|
2018-11-07 18:08:28 +02:00
|
|
|
|
canUseSandbox() {
|
2023-03-16 22:00:20 +02:00
|
|
|
|
[[ ${_canUseSandbox-} ]]
|
|
|
|
|
}
|
2018-01-13 15:18:35 +02:00
|
|
|
|
|
2023-03-16 22:00:20 +02:00
|
|
|
|
requireSandboxSupport () {
|
|
|
|
|
canUseSandbox || skipTest "Sandboxing not supported"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
requireGit() {
|
|
|
|
|
[[ $(type -p git) ]] || skipTest "Git not installed"
|
2018-01-13 15:18:35 +02:00
|
|
|
|
}
|
|
|
|
|
|
2009-03-17 19:38:32 +02:00
|
|
|
|
fail() {
|
2023-03-16 22:00:20 +02:00
|
|
|
|
echo "$1" >&2
|
2009-03-17 19:38:32 +02:00
|
|
|
|
exit 1
|
|
|
|
|
}
|
2012-07-27 21:33:01 +03:00
|
|
|
|
|
2021-12-09 17:26:46 +02:00
|
|
|
|
# Run a command failing if it didn't exit with the expected exit code.
|
|
|
|
|
#
|
|
|
|
|
# Has two advantages over the built-in `!`:
|
|
|
|
|
#
|
|
|
|
|
# 1. `!` conflates all non-0 codes. `expect` allows testing for an exact
|
|
|
|
|
# code.
|
|
|
|
|
#
|
|
|
|
|
# 2. `!` unexpectedly negates `set -e`, and cannot be used on individual
|
|
|
|
|
# pipeline stages with `set -o pipefail`. It only works on the entire
|
|
|
|
|
# pipeline, which is useless if we want, say, `nix ...` invocation to
|
|
|
|
|
# *fail*, but a grep on the error message it outputs to *succeed*.
|
2017-11-15 13:23:31 +02:00
|
|
|
|
expect() {
|
|
|
|
|
local expected res
|
|
|
|
|
expected="$1"
|
|
|
|
|
shift
|
2021-12-09 17:26:46 +02:00
|
|
|
|
"$@" && res=0 || res="$?"
|
2022-05-02 16:12:50 +03:00
|
|
|
|
if [[ $res -ne $expected ]]; then
|
2023-07-07 16:08:25 +03:00
|
|
|
|
echo "Expected exit code '$expected' but got '$res' from command ${*@Q}" >&2
|
2021-12-09 17:26:46 +02:00
|
|
|
|
return 1
|
|
|
|
|
fi
|
|
|
|
|
return 0
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Better than just doing `expect ... >&2` because the "Expected..."
|
|
|
|
|
# message below will *not* be redirected.
|
|
|
|
|
expectStderr() {
|
|
|
|
|
local expected res
|
|
|
|
|
expected="$1"
|
|
|
|
|
shift
|
|
|
|
|
"$@" 2>&1 && res=0 || res="$?"
|
|
|
|
|
if [[ $res -ne $expected ]]; then
|
2023-07-07 16:08:25 +03:00
|
|
|
|
echo "Expected exit code '$expected' but got '$res' from command ${*@Q}" >&2
|
2022-05-02 16:12:50 +03:00
|
|
|
|
return 1
|
|
|
|
|
fi
|
|
|
|
|
return 0
|
2017-11-15 13:23:31 +02:00
|
|
|
|
}
|
|
|
|
|
|
2024-02-29 00:35:10 +02:00
|
|
|
|
# Run a command and check whether the stderr matches stdin.
|
|
|
|
|
# Show a diff when output does not match.
|
|
|
|
|
# Usage:
|
|
|
|
|
#
|
|
|
|
|
# assertStderr nix profile remove nothing << EOF
|
|
|
|
|
# error: This error is expected
|
|
|
|
|
# EOF
|
|
|
|
|
assertStderr() {
|
|
|
|
|
diff -u /dev/stdin <($@ 2>/dev/null 2>&1)
|
|
|
|
|
}
|
|
|
|
|
|
2021-07-26 07:54:55 +03:00
|
|
|
|
needLocalStore() {
|
|
|
|
|
if [[ "$NIX_REMOTE" == "daemon" ]]; then
|
2023-03-16 22:00:20 +02:00
|
|
|
|
skipTest "Can’t run through the daemon ($1)"
|
2021-07-26 07:54:55 +03:00
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Just to make it easy to find which tests should be fixed
|
2022-03-02 22:48:25 +02:00
|
|
|
|
buggyNeedLocalStore() {
|
2021-12-09 17:26:46 +02:00
|
|
|
|
needLocalStore "$1"
|
2021-07-26 07:54:55 +03:00
|
|
|
|
}
|
|
|
|
|
|
2022-03-02 22:48:25 +02:00
|
|
|
|
enableFeatures() {
|
|
|
|
|
local features="$1"
|
|
|
|
|
sed -i 's/experimental-features .*/& '"$features"'/' "$NIX_CONF_DIR"/nix.conf
|
|
|
|
|
}
|
|
|
|
|
|
2012-07-27 21:33:01 +03:00
|
|
|
|
set -x
|
2021-07-26 07:54:55 +03:00
|
|
|
|
|
2022-07-14 16:06:11 +03:00
|
|
|
|
onError() {
|
|
|
|
|
set +x
|
|
|
|
|
echo "$0: test failed at:" >&2
|
2022-07-27 17:41:26 +03:00
|
|
|
|
for ((i = 1; i < ${#BASH_SOURCE[@]}; i++)); do
|
2022-07-14 16:06:11 +03:00
|
|
|
|
if [[ -z ${BASH_SOURCE[i]} ]]; then break; fi
|
|
|
|
|
echo " ${FUNCNAME[i]} in ${BASH_SOURCE[i]}:${BASH_LINENO[i-1]}" >&2
|
|
|
|
|
done
|
|
|
|
|
}
|
|
|
|
|
|
2021-12-09 17:26:46 +02:00
|
|
|
|
# `grep -v` doesn't work well for exit codes. We want `!(exist line l. l
|
|
|
|
|
# matches)`. It gives us `exist line l. !(l matches)`.
|
|
|
|
|
#
|
|
|
|
|
# `!` normally doesn't work well with `set -e`, but when we wrap in a
|
|
|
|
|
# function it *does*.
|
|
|
|
|
grepInverse() {
|
|
|
|
|
! grep "$@"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# A shorthand, `> /dev/null` is a bit noisy.
|
|
|
|
|
#
|
|
|
|
|
# `grep -q` would seem to do this, no function necessary, but it is a
|
|
|
|
|
# bad fit with pipes and `set -o pipefail`: `-q` will exit after the
|
|
|
|
|
# first match, and then subsequent writes will result in broken pipes.
|
|
|
|
|
#
|
|
|
|
|
# Note that reproducing the above is a bit tricky as it depends on
|
|
|
|
|
# non-deterministic properties such as the timing between the match and
|
|
|
|
|
# the closing of the pipe, the buffering of the pipe, and the speed of
|
|
|
|
|
# the producer into the pipe. But rest assured we've seen it happen in
|
|
|
|
|
# CI reliably.
|
|
|
|
|
grepQuiet() {
|
|
|
|
|
grep "$@" > /dev/null
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# The previous two, combined
|
|
|
|
|
grepQuietInverse() {
|
|
|
|
|
! grep "$@" > /dev/null
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-14 16:06:11 +03:00
|
|
|
|
trap onError ERR
|
|
|
|
|
|
2021-12-09 17:16:18 +02:00
|
|
|
|
fi # COMMON_VARS_AND_FUNCTIONS_SH_SOURCED
|