os161/testscripts/runtest.py
2025-03-01 00:01:05 -05:00

203 lines
6.4 KiB
Python

#
# Usage:
# import runtest
# runtest.run(testcommands, outputfile,
# menuprompt=None, default "OS/161 kernel [? for menu]: "
# shellprompt=None, default "OS/161$ "
# conf=None, default is sys161 default behavior
# ram=None, default is per sys161 config
# cpus=None, default is per sys161 config
# doom=None, default is no doom counter
# progress=30, default is 30 seconds
# timeout=300, default is 300 seconds
# kernel=None) default is "kernel"
#
# Returns None on success or a (string) message if something apparently
# went wrong in the middle. (XXX: should it throw exceptions instead?)
#
# * The testcommands argument is a string containing a list of commands
# separated by semicolons. These can be either kernel menu commands
# or shell commands; the command 's' is recognized for switching from
# the menu to the shell and 'exit' for switching back to the menu.
# (This affects waiting for prompts - running the shell via 'p' or
# crashing out of the shell will confuse things.)
#
# The command 'q' from the menu is also recognized as causing a
# shutdown. This will be done automatically after everything else if
# not issued explicitly.
#
# The following commands are interpreted as macros:
# DOMOUNT expands to "mount sfs lhd1:; cd lhd1:"
# DOUNMOUNT expands to "cd /; unmount lhd1:"
# WAIT sleeps 3 seconds and just presses return
#
# * The outputfile argument should be a python file (e.g. sys.stdout)
# and receives a copy of the System/161 output.
#
# * The menuprompt and shellprompt arguments can be used to change the
# menu and shell prompt strings looked for. For the moment these can
# only be fixed strings, not regular expressions. (This is probably
# easy to improve, but I ran into some mysterious problems when I
# tried, so YMMV.) By default if you pass None prompt strings matching
# what OS/161 issues by default are used.
#
# * The conf argument can be used to supply an alternate sys161.conf
# file. If None is given (the default), sys161 will use its default
# config file.
#
# * The ram and cpus arguments can be used to override the RAM size
# and number-of-cpus settings in the sys161 config file. The number of
# cpus must be an integer, but any RAM size specification understood
# by sys161 can be used. Note: this feature requires System/161 2.0.5
# or higher.
#
# * The doom argument can be used to set the doom counter. If None is
# given (the default) the doom counter is not engaged.
#
# * The progress and timeout arguments can be used to set the timeouts
# for System/161 progress monitoring and pexpect-level global timeout,
# respectively. The defaults (somewhat arbitraily chosen) are 30 and
# 300 seconds. Passing progress=None disables progress monitoring; this
# is necessary for nontrivial tests that run within the kernel, as
# progress monitoring measures userland progress. Passing timeout=None
# probably either disables the global timeout or makes pexpect crash;
# I haven't tested it. I don't recommend trying: it is your defense
# against test runs hanging forever.
#
# Note that no-debugger unattended mode (sys161 -X) is always used.
# The purpose of this script is specifically to support unattended
# test runs...
#
# Depends on pexpect, which you may need to install specifically
# depending on your OS.
#
import time
import pexpect
#
# Macro commands
#
macros = {
"MOUNT" : ["mount sfs lhd1:", "cd lhd1:"],
"UNMOUNT" : ["cd /", "unmount lhd1:"],
# "WAIT" special-cased below
}
#
# Wait for a prompt; returns True if we got it, False if we need to
# bail.
#
def getprompt(proc, prompt):
which = proc.expect_exact([
prompt,
"panic: ", # panic message
"sys161: No progress in ", # sys161 deadman print
"sys161: Elapsed ", # sys161 shutdown print
pexpect.EOF,
pexpect.TIMEOUT
])
if which == 0:
# got the prompt
return None
if which == 1:
proc.expect_exact([pexpect.EOF, pexpect.TIMEOUT])
return "panic"
if which == 2:
proc.expect_exact([pexpect.EOF, pexpect.TIMEOUT])
return "progress timeout"
if which == 3:
proc.expect_exact([pexpect.EOF, pexpect.TIMEOUT])
return "unexpected shutdown"
if which == 4:
return "unexpected end of input"
if which == 5:
return "top-level timeout"
return "runtest: Internal error: pexpect returned out-of-range result"
# end getprompt
#
# main test function
#
def run(testcommands, outputfile,
menuprompt=None, shellprompt=None,
conf=None, ram=None, cpus=None,
doom=None,
progress=30, timeout=300,
kernel=None):
if menuprompt is None:
menuprompt = "OS/161 kernel [? for menu]: "
if shellprompt is None:
shellprompt = "OS/161$ "
if kernel is None:
kernel = "kernel"
args = ["-X"]
if conf is not None:
args.append("-c")
args.append(conf)
if cpus is not None:
args.append("-C")
args.append("31:cpus=%d" % cpus)
if doom is not None:
args.append("-D")
args.append("%d" % doom)
if progress is not None:
args.append("-Z")
args.append("%d" % progress)
if ram is not None:
args.append("-C")
args.append("31:ramsize=%s" % ram)
args.append(kernel)
proc = pexpect.spawn("sys161", args, timeout=timeout,
ignore_sighup=False)
proc.logfile_read = outputfile
commands = [s.strip() for s in testcommands.split(";")]
commands = [macros[c] if c in macros else [c] for c in commands]
# Apparently list flatten() is unpythonic...
commands = [c for sublist in commands for c in sublist]
prompts = { True: shellprompt, False: menuprompt }
inshell = False
quit = False
for cmd in commands:
msg = getprompt(proc, prompts[inshell])
if msg is not None:
return msg
if cmd == "WAIT":
time.sleep(3)
cmd = ""
proc.send("%s\r" % cmd)
if not inshell and cmd == "q":
quit = True
if not inshell and cmd == "s":
inshell = True
if inshell and cmd == "exit":
inshell = False
if not quit:
if inshell:
msg = getprompt(proc, prompts[inshell])
if msg is not None:
return msg
proc.send("exit\r")
inshell = False
msg = getprompt(proc, prompts[inshell])
if msg is not None:
return msg
proc.send("q\r")
quit = True
proc.expect_exact([pexpect.EOF, pexpect.TIMEOUT])
# Apparently if you call pexpect.wait() you must have
# explicitly read all the input, or it hangs; and the process
# can't be already dead, or it crashes. Therefore it appears
# to be entirely useless. I hope not calling it doesn't cause
# zombies to accumulate.
#proc.wait()
return None
# end run