2024-07-08 20:07:06 +00:00
# shellcheck shell=bash
2024-05-06 13:42:49 +00:00
2021-12-09 15:26:46 +00:00
set -eu -o pipefail
2006-03-01 12:15:33 +00:00
2024-07-08 20:07:06 +00:00
if [ [ -z " ${ COMMON_FUNCTIONS_SH_SOURCED - } " ] ] ; then
2021-07-26 04:54:55 +00:00
2024-07-08 20:07:06 +00:00
COMMON_FUNCTIONS_SH_SOURCED = 1
2021-07-26 04:54:55 +00:00
2024-06-16 10:13:07 +00:00
isTestOnNixOS( ) {
[ [ " ${ isTestOnNixOS :- } " = = 1 ] ]
}
die( ) {
2024-06-24 16:11:10 +00:00
echo " unexpected fatal error: $* " >& 2
2024-06-16 10:13:07 +00:00
exit 1
}
2006-03-01 14:26:03 +00:00
readLink( ) {
2024-07-08 20:07:06 +00:00
# TODO fix this
# shellcheck disable=SC2012
2006-03-01 14:26:03 +00:00
ls -l " $1 " | sed 's/.*->\ //'
}
2006-09-21 18:54:08 +00:00
2009-03-18 16:35:35 +00:00
clearProfiles( ) {
2024-07-08 20:07:06 +00:00
profiles = " $HOME /.local/state/nix/profiles "
2021-11-17 20:35:21 +00:00
rm -rf " $profiles "
2009-03-18 16:35:35 +00:00
}
2024-06-16 15:56:50 +00:00
# Clear the store, but do not fail if we're in an environment where we can't.
# This allows the test to run in a NixOS test environment, where we use the system store.
2024-10-10 16:04:33 +00:00
# See doc/manual/source/contributing/testing.md / Running functional tests on NixOS.
2024-06-16 15:56:50 +00:00
clearStoreIfPossible( ) {
if isTestOnNixOS; then
echo "clearStoreIfPossible: Not clearing store, because we're on NixOS. Moving on."
else
doClearStore
fi
}
2006-09-21 18:54:08 +00:00
clearStore( ) {
2024-06-16 10:13:07 +00:00
if isTestOnNixOS; then
2024-06-16 15:56:50 +00:00
die "clearStore: not supported when testing on NixOS. If not essential, call clearStoreIfPossible. If really needed, add conditionals; e.g. if ! isTestOnNixOS; then ..."
2024-06-16 10:13:07 +00:00
fi
2024-06-16 15:56:50 +00:00
doClearStore
}
2024-06-16 10:13:07 +00:00
2024-06-16 15:56:50 +00:00
doClearStore( ) {
2006-09-21 18:54:08 +00:00
echo "clearing store..."
chmod -R +w " $NIX_STORE_DIR "
rm -rf " $NIX_STORE_DIR "
mkdir " $NIX_STORE_DIR "
2016-07-27 15:14:41 +00:00
rm -rf " $NIX_STATE_DIR "
mkdir " $NIX_STATE_DIR "
2009-03-18 16:35:35 +00:00
clearProfiles
2006-09-21 18:54:08 +00:00
}
2007-08-13 13:15:02 +00:00
2014-02-17 11:22:50 +00:00
clearCache( ) {
2024-07-08 20:07:06 +00:00
rm -rf " ${ cacheDir ? } "
2014-02-17 11:22:50 +00:00
}
2016-05-30 18:22:30 +00:00
clearCacheCache( ) {
2024-07-08 20:07:06 +00:00
rm -f " $TEST_HOME /.cache/nix/binary-cache " *
2016-05-30 18:22:30 +00:00
}
2011-07-20 11:50:13 +00:00
startDaemon( ) {
2024-06-16 10:13:07 +00:00
if isTestOnNixOS; then
die "startDaemon: not supported when testing on NixOS. Is it really needed? If so add conditionals; e.g. if ! isTestOnNixOS; then ..."
fi
2021-07-26 04:54:55 +00:00
# Don’ t start the daemon twice, as this would just make it loop indefinitely
2021-12-09 15:16:18 +00:00
if [ [ " ${ _NIX_TEST_DAEMON_PID - } " != '' ] ] ; then
return
2021-07-26 04:54:55 +00:00
fi
2022-03-02 21:22:55 +00:00
# Start the daemon, wait for the socket to appear.
2024-07-08 20:07:06 +00:00
rm -f " $NIX_DAEMON_SOCKET_PATH "
2023-12-18 18:36:18 +00:00
PATH = $DAEMON_PATH nix --extra-experimental-features 'nix-command' daemon &
2021-12-09 15:16:18 +00:00
_NIX_TEST_DAEMON_PID = $!
export _NIX_TEST_DAEMON_PID
2022-02-24 13:57:27 +00:00
for ( ( i = 0; i < 300; i++) ) ; do
2022-02-28 13:41:09 +00:00
if [ [ -S $NIX_DAEMON_SOCKET_PATH ] ] ; then
DAEMON_STARTED = 1
break;
fi
2022-02-24 13:57:27 +00:00
sleep 0.1
2011-10-11 11:14:30 +00:00
done
2022-02-28 13:41:09 +00:00
if [ [ -z ${ DAEMON_STARTED +x } ] ] ; then
fail "Didn’ t manage to start the daemon"
fi
2021-07-26 04:54:55 +00:00
trap "killDaemon" EXIT
2021-12-09 15:16:18 +00:00
# Save for if daemon is killed
NIX_REMOTE_OLD = $NIX_REMOTE
2011-07-20 11:50:13 +00:00
export NIX_REMOTE = daemon
}
killDaemon( ) {
2024-06-16 10:13:07 +00:00
if isTestOnNixOS; then
die "killDaemon: not supported when testing on NixOS. Is it really needed? If so add conditionals; e.g. if ! isTestOnNixOS; then ..."
fi
2021-12-09 15:16:18 +00:00
# Don’ t fail trying to stop a non-existant daemon twice
if [ [ " ${ _NIX_TEST_DAEMON_PID - } " = = '' ] ] ; then
return
fi
2024-07-08 20:07:06 +00:00
kill " $_NIX_TEST_DAEMON_PID "
2022-02-24 13:57:27 +00:00
for i in { 0..100} ; do
2024-07-08 20:07:06 +00:00
kill -0 " $_NIX_TEST_DAEMON_PID " 2> /dev/null || break
2022-02-24 13:57:27 +00:00
sleep 0.1
2021-07-26 04:54:55 +00:00
done
2024-07-08 20:07:06 +00:00
kill -9 " $_NIX_TEST_DAEMON_PID " 2> /dev/null || true
wait " $_NIX_TEST_DAEMON_PID " || true
rm -f " $NIX_DAEMON_SOCKET_PATH "
2021-12-09 15:16:18 +00:00
# Indicate daemon is stopped
unset _NIX_TEST_DAEMON_PID
# Restore old nix remote
NIX_REMOTE = $NIX_REMOTE_OLD
2011-07-20 11:50:13 +00:00
trap "" EXIT
}
2021-07-26 04:54:55 +00:00
restartDaemon( ) {
2024-06-16 10:13:07 +00:00
if isTestOnNixOS; then
die "restartDaemon: not supported when testing on NixOS. Is it really needed? If so add conditionals; e.g. if ! isTestOnNixOS; then ..."
fi
2021-12-09 15:16:18 +00:00
[ [ -z " ${ _NIX_TEST_DAEMON_PID :- } " ] ] && return 0
2021-07-26 04:54:55 +00:00
2022-07-14 13:06:11 +00:00
killDaemon
startDaemon
2021-07-26 04:54:55 +00:00
}
isDaemonNewer ( ) {
[ [ -n " ${ NIX_DAEMON_PACKAGE :- } " ] ] || return 0
local requiredVersion = " $1 "
2024-07-08 20:07:06 +00:00
local daemonVersion
daemonVersion = $( " $NIX_DAEMON_PACKAGE /bin/nix " daemon --version | cut -d' ' -f3)
2021-10-14 13:42:54 +00:00
[ [ $( nix eval --expr " builtins.compareVersions '' $daemonVersion '' '' $requiredVersion '' " ) -ge 0 ] ]
2021-07-26 04:54:55 +00:00
}
2023-03-16 20:00:20 +00:00
skipTest ( ) {
echo " $1 , skipping this test... " >& 2
2024-07-25 02:36:43 +00:00
exit 77
2023-03-16 20:00:20 +00:00
}
2024-06-16 10:13:07 +00:00
TODO_NixOS( ) {
if isTestOnNixOS; then
skipTest "This test has not been adapted for NixOS yet"
fi
}
2021-07-26 04:54:55 +00:00
requireDaemonNewerThan ( ) {
2023-03-16 20:00:20 +00:00
isDaemonNewer " $1 " || skipTest "Daemon is too old"
2021-07-26 04:54:55 +00:00
}
2018-11-07 16:08:28 +00:00
canUseSandbox( ) {
2023-03-16 20:00:20 +00:00
[ [ ${ _canUseSandbox - } ] ]
}
2018-01-13 13:18:35 +00:00
2023-03-16 20:00:20 +00:00
requireSandboxSupport ( ) {
canUseSandbox || skipTest "Sandboxing not supported"
}
requireGit( ) {
[ [ $( type -p git) ] ] || skipTest "Git not installed"
2018-01-13 13:18:35 +00:00
}
2009-03-17 17:38:32 +00:00
fail( ) {
2024-06-24 16:11:58 +00:00
echo " test failed: $* " >& 2
2009-03-17 17:38:32 +00:00
exit 1
}
2012-07-27 18:33:01 +00:00
2021-12-09 15:26:46 +00: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 11:23:31 +00:00
expect( ) {
local expected res
expected = " $1 "
shift
2021-12-09 15:26:46 +00:00
" $@ " && res = 0 || res = " $? "
2024-07-15 23:28:28 +00:00
# also match "negative" codes, which wrap around to >127
2024-07-08 20:07:06 +00:00
if [ [ $res -ne $expected && $res -ne $(( 256 + expected)) ] ] ; then
2023-07-07 13:08:25 +00:00
echo " Expected exit code ' $expected ' but got ' $res ' from command ${ *@Q } " >& 2
2021-12-09 15:26:46 +00: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 = " $? "
2024-07-15 23:28:28 +00:00
# also match "negative" codes, which wrap around to >127
2024-07-08 20:07:06 +00:00
if [ [ $res -ne $expected && $res -ne $(( 256 + expected)) ] ] ; then
2023-07-07 13:08:25 +00:00
echo " Expected exit code ' $expected ' but got ' $res ' from command ${ *@Q } " >& 2
2022-05-02 13:12:50 +00:00
return 1
fi
return 0
2017-11-15 11:23:31 +00:00
}
2024-02-28 22:35:10 +00: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( ) {
2024-07-08 20:07:06 +00:00
diff -u /dev/stdin <( " $@ " 2>/dev/null 2>& 1)
2024-02-28 22:35:10 +00:00
}
2021-07-26 04:54:55 +00:00
needLocalStore( ) {
if [ [ " $NIX_REMOTE " = = "daemon" ] ] ; then
2023-03-16 20:00:20 +00:00
skipTest " Can’ t run through the daemon ( $1 ) "
2021-07-26 04:54:55 +00:00
fi
}
# Just to make it easy to find which tests should be fixed
2022-03-02 20:48:25 +00:00
buggyNeedLocalStore( ) {
2021-12-09 15:26:46 +00:00
needLocalStore " $1 "
2021-07-26 04:54:55 +00:00
}
2022-03-02 20:48:25 +00:00
enableFeatures( ) {
local features = " $1 "
2024-07-08 20:07:06 +00:00
sed -i 's/experimental-features .*/& ' " $features " '/' " ${ test_nix_conf ? } "
2022-03-02 20:48:25 +00:00
}
2022-07-14 13:06:11 +00:00
onError( ) {
set +x
echo " $0 : test failed at: " >& 2
2022-07-27 14:41:26 +00:00
for ( ( i = 1; i < ${# BASH_SOURCE [@] } ; i++) ) ; do
2022-07-14 13:06:11 +00:00
if [ [ -z ${ BASH_SOURCE [i] } ] ] ; then break; fi
echo " ${ FUNCNAME [i] } in ${ BASH_SOURCE [i] } : ${ BASH_LINENO [i-1] } " >& 2
done
}
2024-07-15 23:40:14 +00:00
# Prints an error message prefix referring to the last call into this file.
# Ignores `expect` and `expectStderr` calls.
# Set a special exit code when test suite functions are misused, so that
# functions like expectStderr won't mistake them for expected Nix CLI errors.
# Suggestion: -101 (negative to indicate very abnormal, and beyond the normal
# range of signals)
# Example (showns as string): 'repl.sh:123: in call to grepQuiet: '
# This function is inefficient, so it should only be used in error messages.
callerPrefix( ) {
2024-07-23 08:24:18 +00:00
# Find the closest caller that's not from this file
# using the bash `caller` builtin.
2024-07-15 23:40:14 +00:00
local i file line fn savedFn
# Use `caller`
for i in $( seq 0 100) ; do
2024-07-08 20:07:06 +00:00
caller " $i " > /dev/null || {
2024-07-15 23:40:14 +00:00
if [ [ -n " ${ file :- } " ] ] ; then
echo " $file : $line : ${ savedFn +in call to $savedFn : } "
fi
break
}
2024-07-08 20:07:06 +00:00
line = " $( caller " $i " | cut -d' ' -f1) "
fn = " $( caller " $i " | cut -d' ' -f2) "
file = " $( caller " $i " | cut -d' ' -f3) "
2024-07-15 23:40:14 +00:00
if [ [ $file != " ${ BASH_SOURCE [0] } " ] ] ; then
echo " $file : $line : ${ savedFn +in call to $savedFn : } "
return
fi
case " $fn " in
# Ignore higher order functions that don't report any misuse of themselves
# This way a misuse of a foo in `expectStderr 1 foo` will be reported as
# calling foo, not expectStderr.
expect| expectStderr| callerPrefix)
; ;
*)
savedFn = " $fn "
; ;
esac
done
}
2024-07-15 23:41:22 +00:00
checkGrepArgs( ) {
local arg
for arg in " $@ " ; do
if [ [ " $arg " != " ${ arg // $'\n' /_ } " ] ] ; then
echo " $( callerPrefix) newline not allowed in arguments; grep would try each line individually as if connected by an OR operator " >& 2
2024-07-08 20:07:06 +00:00
return 155 # = -101 mod 256
2024-07-15 23:41:22 +00:00
fi
done
}
2021-12-09 15:26:46 +00: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*.
2024-07-15 23:54:12 +00:00
#
# `command grep` lets us avoid re-checking the args by going directly to the
# executable.
2021-12-09 15:26:46 +00:00
grepInverse( ) {
2024-07-15 23:41:22 +00:00
checkGrepArgs " $@ " && \
2024-07-15 23:54:12 +00:00
! command grep " $@ "
2021-12-09 15:26:46 +00:00
}
# 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.
2024-07-15 23:54:12 +00:00
#
# `command grep` lets us avoid re-checking the args by going directly to the
# executable.
2021-12-09 15:26:46 +00:00
grepQuiet( ) {
2024-07-15 23:41:22 +00:00
checkGrepArgs " $@ " && \
2024-07-15 23:54:12 +00:00
command grep " $@ " > /dev/null
2021-12-09 15:26:46 +00:00
}
# The previous two, combined
grepQuietInverse( ) {
2024-07-15 23:41:22 +00:00
checkGrepArgs " $@ " && \
2024-07-15 23:54:12 +00:00
! command grep " $@ " > /dev/null
}
# Wrap grep to remove its newline footgun; see checkGrepArgs.
# Note that we keep the checkGrepArgs calls in the other helpers, because some
# of them are negated and that would defeat this check.
grep( ) {
checkGrepArgs " $@ " && \
command grep " $@ "
2021-12-09 15:26:46 +00:00
}
2024-03-24 14:04:06 +00:00
# Return the number of arguments
count( ) {
echo $#
}
2022-07-14 13:06:11 +00:00
trap onError ERR
2024-07-08 20:07:06 +00:00
fi # COMMON_FUNCTIONS_SH_SOURCED