88 lines
3 KiB
Python
88 lines
3 KiB
Python
|
#!/usr/bin/env python3
|
||
|
|
||
|
"""
|
||
|
This script is a helper for this project's Meson buildsystem to run Lix's
|
||
|
functional tests. It is an analogue to mk/run-test.sh in the autoconf+Make
|
||
|
buildsystem.
|
||
|
|
||
|
These tests are run in the installCheckPhase in Lix's derivation, and as such
|
||
|
expect to be run after the project has already been "installed" to some extent.
|
||
|
Look at meson/setup-functional-tests.py for more details.
|
||
|
"""
|
||
|
|
||
|
import argparse
|
||
|
from pathlib import Path
|
||
|
import os
|
||
|
import shutil
|
||
|
import subprocess
|
||
|
import sys
|
||
|
|
||
|
name = 'run-test.py'
|
||
|
|
||
|
if 'MESON_BUILD_ROOT' not in os.environ:
|
||
|
raise ValueError(f'{name}: this script must be run from the Meson build system')
|
||
|
|
||
|
def main():
|
||
|
|
||
|
tests_dir = Path(os.path.join(os.environ['MESON_BUILD_ROOT'], 'tests/functional'))
|
||
|
|
||
|
parser = argparse.ArgumentParser(name)
|
||
|
parser.add_argument('target', help='the script path relative to tests/functional to run')
|
||
|
args = parser.parse_args()
|
||
|
|
||
|
target = Path(args.target)
|
||
|
# The test suite considers the test's name to be the path to the test relative to
|
||
|
# `tests/functional`, but without the file extension.
|
||
|
# e.g. for `tests/functional/flakes/develop.sh`, the test name is `flakes/develop`
|
||
|
test_name = target.with_suffix('').as_posix()
|
||
|
if not target.is_absolute():
|
||
|
target = tests_dir.joinpath(target).resolve()
|
||
|
|
||
|
assert target.exists(), f'{name}: test {target} does not exist; did you run `meson install`?'
|
||
|
|
||
|
bash = os.environ.get('BASH', shutil.which('bash'))
|
||
|
if bash is None:
|
||
|
raise ValueError(f'{name}: bash executable not found and BASH environment variable not set')
|
||
|
|
||
|
test_environment = os.environ | {
|
||
|
'TEST_NAME': test_name,
|
||
|
# mk/run-test.sh did this, but I don't know if it has any effect since it seems
|
||
|
# like the tests that interact with remote stores set it themselves?
|
||
|
'NIX_REMOTE': '',
|
||
|
}
|
||
|
|
||
|
# Initialize testing.
|
||
|
init_result = subprocess.run([bash, '-e', 'init.sh'], cwd=tests_dir, env=test_environment)
|
||
|
if init_result.returncode != 0:
|
||
|
print(f'{name}: internal error initializing {args.target}', file=sys.stderr)
|
||
|
print('[ERROR]')
|
||
|
# Meson interprets exit code 99 as indicating an *error* in the testing process.
|
||
|
return 99
|
||
|
|
||
|
# Run the test itself.
|
||
|
test_result = subprocess.run([bash, '-e', target.name], cwd=target.parent, env=test_environment)
|
||
|
|
||
|
if test_result.returncode == 0:
|
||
|
print('[PASS]')
|
||
|
elif test_result.returncode == 99:
|
||
|
print('[SKIP]')
|
||
|
# Meson interprets exit code 77 as indicating a skipped test.
|
||
|
return 77
|
||
|
else:
|
||
|
print('[FAIL]')
|
||
|
|
||
|
return test_result.returncode
|
||
|
|
||
|
try:
|
||
|
sys.exit(main())
|
||
|
except AssertionError as e:
|
||
|
# This should mean that this test was run not-from-Meson, probably without
|
||
|
# having run `meson install` first, which is not an bug in this script.
|
||
|
print(e, file=sys.stderr)
|
||
|
sys.exit(99)
|
||
|
except Exception as e:
|
||
|
print(f'{name}: INTERNAL ERROR running test ({sys.argv}): {e}', file=sys.stderr)
|
||
|
print(f'this is a bug in {name}')
|
||
|
sys.exit(99)
|
||
|
|