2023-05-10 21:12:01 +02:00
# SPDX-License-Identifier: LGPL-2.1-or-later
# shellcheck shell=bash
if [ [ " ${ BASH_SOURCE [0] } " -ef " $0 " ] ] ; then
echo >& 2 "This file should not be executed directly"
exit 1
fi
2023-05-22 13:24:12 +02:00
declare -i _CHILD_PID = 0
_PASSED_TESTS = ( )
2023-05-10 21:12:01 +02:00
# Like trap, but passes the signal name as the first argument
2023-05-22 13:24:12 +02:00
_trap_with_sig( ) {
2023-05-10 21:12:01 +02:00
local fun = " ${ 1 : ? } "
local sig
shift
for sig in " $@ " ; do
# shellcheck disable=SC2064
trap " $fun $sig " " $sig "
done
}
# Propagate the caught signal to the current child process
2023-05-22 13:24:12 +02:00
_handle_signal( ) {
2023-05-10 21:12:01 +02:00
local sig = " ${ 1 : ? } "
2023-05-22 13:24:12 +02:00
if [ [ $_CHILD_PID -gt 0 ] ] ; then
echo " Propagating signal $sig to child process $_CHILD_PID "
kill -s " $sig " " $_CHILD_PID "
2023-05-10 21:12:01 +02:00
fi
}
2023-05-22 13:24:12 +02:00
# In order to make the _handle_signal() stuff above work, we have to execute
2023-05-10 21:12:01 +02:00
# each script asynchronously, since bash won't execute traps until the currently
# executed command finishes. This, however, introduces another issue regarding
# how bash's wait works. Quoting:
#
# When bash is waiting for an asynchronous command via the wait builtin,
# the reception of a signal for which a trap has been set will cause the wait
# builtin to return immediately with an exit status greater than 128,
# immediately after which the trap is executed.
#
# In other words - every time we propagate a signal, wait returns with
# 128+signal, so we have to wait again - repeat until the process dies.
2023-05-22 13:24:12 +02:00
_wait_harder( ) {
2023-05-10 21:12:01 +02:00
local pid = " ${ 1 : ? } "
while kill -0 " $pid " & >/dev/null; do
wait " $pid " || :
done
wait " $pid "
}
2023-05-22 13:24:12 +02:00
_show_summary( ) { (
set +x
2023-06-14 20:14:25 +02:00
if [ [ ${# _PASSED_TESTS [@] } -eq 0 ] ] ; then
2023-05-22 13:24:12 +02:00
echo >& 2 "No tests were executed, this is most likely an error"
exit 1
fi
printf "PASSED TESTS: %3d:\n" " ${# _PASSED_TESTS [@] } "
echo "------------------"
for t in " ${ _PASSED_TESTS [@] } " ; do
echo " $t "
done
) }
2023-05-10 21:12:01 +02:00
# Like run_subtests, but propagate specified signals to the subtest script
run_subtests_with_signals( ) {
local subtests = ( " ${ 0 %.sh } " .*.sh)
local subtest
if [ [ " ${# subtests [@] } " -eq 0 ] ] ; then
echo >& 2 " No subtests found for file $0 "
exit 1
fi
if [ [ " $# " -eq 0 ] ] ; then
echo >& 2 "No signals to propagate were specified"
exit 1
fi
2023-05-22 13:24:12 +02:00
_trap_with_sig _handle_signal " $@ "
2023-05-10 21:12:01 +02:00
for subtest in " ${ subtests [@] } " ; do
2024-02-19 20:37:31 +01:00
if [ [ -n " ${ TEST_MATCH_SUBTEST :- } " ] ] && ! [ [ " $subtest " = ~ $TEST_MATCH_SUBTEST ] ] ; then
echo " Skipping $subtest (not matching ' $TEST_MATCH_SUBTEST ') "
continue
fi
2024-12-10 11:56:04 +00:00
for skip in ${ TEST_SKIP_SUBTESTS :- } ; do
if [ [ " $subtest " = ~ $skip ] ] ; then
echo " Skipping $subtest (matching ' $skip ') "
continue 2
fi
done
2023-05-10 21:12:01 +02:00
: " --- $subtest BEGIN --- "
2023-07-04 10:42:47 +02:00
SECONDS = 0
2023-05-10 21:12:01 +02:00
" ./ $subtest " &
2023-05-22 13:24:12 +02:00
_CHILD_PID = $!
2023-06-24 20:43:26 +02:00
if ! _wait_harder " $_CHILD_PID " ; then
echo " Subtest $subtest failed "
return 1
fi
_PASSED_TESTS += ( " $subtest " )
2023-07-04 10:42:47 +02:00
: " --- $subtest END ( ${ SECONDS } s) --- "
2023-05-10 21:12:01 +02:00
done
2023-05-22 13:24:12 +02:00
_show_summary
2023-05-10 21:12:01 +02:00
}
2024-05-11 19:17:13 +02:00
# Run all subtests (i.e. files named as $TESTNAME.<subtest_name>.sh)
2023-05-10 21:12:01 +02:00
run_subtests( ) {
local subtests = ( " ${ 0 %.sh } " .*.sh)
local subtest
if [ [ " ${# subtests [@] } " -eq 0 ] ] ; then
echo >& 2 " No subtests found for file $0 "
exit 1
fi
for subtest in " ${ subtests [@] } " ; do
2023-06-05 10:47:21 +02:00
if [ [ -n " ${ TEST_MATCH_SUBTEST :- } " ] ] && ! [ [ " $subtest " = ~ $TEST_MATCH_SUBTEST ] ] ; then
echo " Skipping $subtest (not matching ' $TEST_MATCH_SUBTEST ') "
continue
fi
2024-12-10 11:56:04 +00:00
for skip in ${ TEST_SKIP_SUBTESTS :- } ; do
if [ [ " $subtest " = ~ $skip ] ] ; then
echo " Skipping $subtest (matching ' $skip ') "
continue 2
fi
done
2023-05-10 21:12:01 +02:00
: " --- $subtest BEGIN --- "
2023-07-04 10:42:47 +02:00
SECONDS = 0
2023-06-24 20:43:26 +02:00
if ! " ./ $subtest " ; then
echo " Subtest $subtest failed "
return 1
fi
_PASSED_TESTS += ( " $subtest " )
2023-07-04 10:42:47 +02:00
: " --- $subtest END ( ${ SECONDS } s) --- "
2023-05-10 21:12:01 +02:00
done
2023-05-22 13:24:12 +02:00
_show_summary
2023-05-10 21:12:01 +02:00
}
2023-05-22 12:39:25 +02:00
# Run all test cases (i.e. functions prefixed with testcase_ in the current namespace)
run_testcases( ) {
local testcase testcases
# Create a list of all functions prefixed with testcase_
mapfile -t testcases < <( declare -F | awk '$3 ~ /^testcase_/ {print $3;}' )
if [ [ " ${# testcases [@] } " -eq 0 ] ] ; then
echo >& 2 "No test cases found, this is most likely an error"
exit 1
fi
for testcase in " ${ testcases [@] } " ; do
2023-06-05 10:47:21 +02:00
if [ [ -n " ${ TEST_MATCH_TESTCASE :- } " ] ] && ! [ [ " $testcase " = ~ $TEST_MATCH_TESTCASE ] ] ; then
echo " Skipping $testcase (not matching ' $TEST_MATCH_TESTCASE ') "
continue
fi
2024-12-10 11:56:04 +00:00
for skip in ${ TEST_SKIP_TESTCASES :- } ; do
if [ [ " $testcase " = ~ $skip ] ] ; then
echo " Skipping $testcase (matching ' $skip ') "
continue 2
fi
done
2023-05-22 12:39:25 +02:00
: " +++ $testcase BEGIN +++ "
# Note: the subshell here is used purposefully, otherwise we might
# unexpectedly inherit a RETURN trap handler from the called
# function and call it for the second time once we return,
# causing a "double-free"
( " $testcase " )
: " +++ $testcase END +++ "
done
}