2024-04-19 09:17:58 +02:00
#!/usr/bin/python3
# SPDX-License-Identifier: LGPL-2.1-or-later
''' Test wrapper command for driving integration tests.
'''
import argparse
2024-05-07 12:24:51 +02:00
import json
2024-04-19 09:17:58 +02:00
import os
import shlex
import subprocess
import sys
import textwrap
from pathlib import Path
EMERGENCY_EXIT_DROPIN = """ \
[ Unit ]
Wants = emergency - exit . service
"""
EMERGENCY_EXIT_SERVICE = """ \
[ Unit ]
DefaultDependencies = no
Conflicts = shutdown . target
Conflicts = rescue . service
Before = shutdown . target
Before = rescue . service
FailureAction = exit
[ Service ]
ExecStart = false
"""
def main ( ) :
parser = argparse . ArgumentParser ( description = __doc__ )
2024-05-27 11:15:02 +02:00
parser . add_argument ( ' --mkosi ' , required = True )
2024-04-19 09:17:58 +02:00
parser . add_argument ( ' --meson-source-dir ' , required = True , type = Path )
parser . add_argument ( ' --meson-build-dir ' , required = True , type = Path )
2024-05-11 19:40:03 +02:00
parser . add_argument ( ' --name ' , required = True )
2024-05-12 10:50:47 +02:00
parser . add_argument ( ' --unit ' , required = True )
2024-04-30 19:06:00 +02:00
parser . add_argument ( ' --storage ' , required = True )
2024-05-05 18:14:44 +02:00
parser . add_argument ( ' --firmware ' , required = True )
2024-05-07 15:20:44 +02:00
parser . add_argument ( ' --slow ' , action = argparse . BooleanOptionalAction )
2024-05-28 15:54:35 +02:00
parser . add_argument ( ' --vm ' , action = argparse . BooleanOptionalAction )
2024-05-15 10:09:53 +02:00
parser . add_argument ( ' --exit-code ' , required = True , type = int )
2024-04-19 09:17:58 +02:00
parser . add_argument ( ' mkosi_args ' , nargs = " * " )
args = parser . parse_args ( )
2024-05-07 15:20:44 +02:00
if not bool ( int ( os . getenv ( " SYSTEMD_INTEGRATION_TESTS " , " 0 " ) ) ) :
2024-05-11 19:40:03 +02:00
print ( f " SYSTEMD_INTEGRATION_TESTS=1 not found in environment, skipping { args . name } " , file = sys . stderr )
2024-05-07 15:20:44 +02:00
exit ( 77 )
if args . slow and not bool ( int ( os . getenv ( " SYSTEMD_SLOW_TESTS " , " 0 " ) ) ) :
2024-05-11 19:40:03 +02:00
print ( f " SYSTEMD_SLOW_TESTS=1 not found in environment, skipping { args . name } " , file = sys . stderr )
2024-05-07 15:20:44 +02:00
exit ( 77 )
2024-05-11 19:40:03 +02:00
name = args . name + ( f " - { i } " if ( i := os . getenv ( " MESON_TEST_ITERATION " ) ) else " " )
2024-04-19 09:17:58 +02:00
dropin = textwrap . dedent (
""" \
[ Unit ]
2024-05-05 11:38:09 +02:00
SuccessAction = exit
SuccessActionExitStatus = 123
2024-04-19 09:17:58 +02:00
[ Service ]
StandardOutput = journal + console
"""
)
2024-05-03 10:27:58 +02:00
if os . getenv ( " TEST_MATCH_SUBTEST " ) :
dropin + = textwrap . dedent (
f """
[ Service ]
Environment = TEST_MATCH_SUBTEST = { os . environ [ " TEST_MATCH_SUBTEST " ] }
"""
)
if os . getenv ( " TEST_MATCH_TESTCASE " ) :
dropin + = textwrap . dedent (
f """
[ Service ]
Environment = TEST_MATCH_TESTCASE = { os . environ [ " TEST_MATCH_TESTCASE " ] }
"""
)
2024-04-19 09:17:58 +02:00
if not sys . stderr . isatty ( ) :
dropin + = textwrap . dedent (
"""
[ Unit ]
FailureAction = exit
"""
)
2024-05-03 10:57:50 +02:00
journal_file = ( args . meson_build_dir / ( f " test/journal/ { name } .journal " ) ) . absolute ( )
2024-04-19 09:17:58 +02:00
journal_file . unlink ( missing_ok = True )
else :
journal_file = None
cmd = [
2024-05-27 11:15:02 +02:00
args . mkosi ,
2024-04-19 09:17:58 +02:00
' --directory ' , os . fspath ( args . meson_source_dir ) ,
' --output-dir ' , os . fspath ( args . meson_build_dir / ' mkosi.output ' ) ,
' --extra-search-path ' , os . fspath ( args . meson_build_dir ) ,
2024-05-03 10:57:50 +02:00
' --machine ' , name ,
2024-04-19 09:17:58 +02:00
' --ephemeral ' ,
* ( [ ' --forward-journal ' , journal_file ] if journal_file else [ ] ) ,
* (
[
' --credential ' ,
f " systemd.extra-unit.emergency-exit.service= { shlex . quote ( EMERGENCY_EXIT_SERVICE ) } " ,
' --credential ' ,
f " systemd.unit-dropin.emergency.target= { shlex . quote ( EMERGENCY_EXIT_DROPIN ) } " ,
]
if not sys . stderr . isatty ( )
else [ ]
) ,
' --credential ' ,
2024-05-12 10:50:47 +02:00
f " systemd.unit-dropin. { args . unit } = { shlex . quote ( dropin ) } " ,
2024-04-25 10:00:15 +01:00
' --runtime-network=none ' ,
2024-04-28 20:46:14 +02:00
' --runtime-scratch=no ' ,
2024-05-12 15:16:37 +02:00
* args . mkosi_args ,
2024-04-19 09:17:58 +02:00
' --append ' ,
2024-05-05 18:14:44 +02:00
' --qemu-firmware ' , args . firmware ,
2024-06-16 19:15:24 +01:00
' --qemu-kvm ' , " auto " if not bool ( int ( os . getenv ( " TEST_NO_KVM " , " 0 " ) ) ) else " no " ,
2024-04-19 09:17:58 +02:00
' --kernel-command-line-extra ' ,
' ' . join ( [
' systemd.hostname=H ' ,
2024-05-11 19:40:03 +02:00
f " SYSTEMD_UNIT_PATH=/usr/lib/systemd/tests/testdata/ { args . name } .units:/usr/lib/systemd/tests/testdata/units: " ,
2024-05-12 10:50:47 +02:00
f " systemd.unit= { args . unit } " ,
2024-04-28 22:51:41 +02:00
' systemd.mask=systemd-networkd-wait-online.service ' ,
2024-04-28 19:28:37 +02:00
* (
[
" systemd.mask=serial-getty@.service " ,
" systemd.show_status=no " ,
" systemd.crash_shell=0 " ,
2024-04-29 10:47:25 +02:00
" systemd.crash_action=poweroff " ,
2024-04-28 19:28:37 +02:00
]
if not sys . stderr . isatty ( )
else [ ]
) ,
2024-04-19 09:17:58 +02:00
] ) ,
2024-05-02 08:52:50 +02:00
' --credential ' , f " journal.storage= { ' persistent ' if sys . stderr . isatty ( ) else args . storage } " ,
2024-05-28 15:54:35 +02:00
' qemu ' if args . vm or os . getuid ( ) != 0 else ' boot ' ,
2024-04-19 09:17:58 +02:00
]
2024-04-23 14:13:22 +01:00
result = subprocess . run ( cmd )
2024-05-07 11:50:11 +02:00
2024-05-15 10:09:53 +02:00
if result . returncode in ( args . exit_code , 77 ) :
2024-05-07 11:50:11 +02:00
# Do not keep journal files for tests that don't fail.
if journal_file :
journal_file . unlink ( missing_ok = True )
2024-05-15 10:09:53 +02:00
exit ( 0 if result . returncode == args . exit_code else 77 )
2024-04-19 09:17:58 +02:00
if journal_file :
2024-05-07 12:24:51 +02:00
ops = [ ]
if os . getenv ( " GITHUB_ACTIONS " ) :
id = os . environ [ " GITHUB_RUN_ID " ]
iteration = os . environ [ " GITHUB_RUN_ATTEMPT " ]
j = json . loads (
subprocess . run (
[
2024-05-27 11:15:02 +02:00
args . mkosi ,
2024-05-07 12:24:51 +02:00
" --directory " , os . fspath ( args . meson_source_dir ) ,
" --json " ,
" summary " ,
] ,
stdout = subprocess . PIPE ,
text = True ,
) . stdout
)
images = { image [ " Image " ] : image for image in j [ " Images " ] }
distribution = images [ " system " ] [ " Distribution " ]
release = images [ " system " ] [ " Release " ]
artifact = f " ci-mkosi- { id } - { iteration } - { distribution } - { release } -failed-test-journals "
ops + = [ f " gh run download { id } --name { artifact } -D ci/ { artifact } " ]
journal_file = Path ( f " ci/ { artifact } /test/journal/ { name } .journal " )
2024-05-12 10:50:47 +02:00
ops + = [ f " journalctl --file { journal_file } --no-hostname -o short-monotonic -u { args . unit } -p info " ]
2024-05-07 12:24:51 +02:00
2024-05-07 11:50:11 +02:00
print ( " Test failed, relevant logs can be viewed with: \n \n "
2024-05-07 12:24:51 +02:00
f " { ( ' && ' . join ( ops ) ) } \n " , file = sys . stderr )
2024-05-07 11:50:11 +02:00
# 0 also means we failed so translate that to a non-zero exit code to mark the test as failed.
exit ( result . returncode or 1 )
2024-04-19 09:17:58 +02:00
if __name__ == ' __main__ ' :
main ( )