2020-04-24 23:36:52 +00:00
|
|
|
|
#! /somewhere/python3
|
|
|
|
|
from contextlib import contextmanager, _GeneratorContextManager
|
|
|
|
|
from queue import Queue, Empty
|
2021-04-25 03:57:28 +00:00
|
|
|
|
from typing import Tuple, Any, Callable, Dict, Iterator, Optional, List, Iterable
|
2020-04-24 23:36:52 +00:00
|
|
|
|
from xml.sax.saxutils import XMLGenerator
|
2021-05-20 23:08:51 +00:00
|
|
|
|
from colorama import Style
|
2020-07-18 16:06:22 +00:00
|
|
|
|
import queue
|
|
|
|
|
import io
|
2020-04-24 23:36:52 +00:00
|
|
|
|
import _thread
|
2020-07-18 16:06:22 +00:00
|
|
|
|
import argparse
|
2020-04-24 23:36:52 +00:00
|
|
|
|
import atexit
|
|
|
|
|
import base64
|
|
|
|
|
import codecs
|
|
|
|
|
import os
|
|
|
|
|
import pathlib
|
|
|
|
|
import ptpython.repl
|
|
|
|
|
import pty
|
|
|
|
|
import re
|
|
|
|
|
import shlex
|
|
|
|
|
import shutil
|
|
|
|
|
import socket
|
|
|
|
|
import subprocess
|
|
|
|
|
import sys
|
2021-06-04 09:07:49 +00:00
|
|
|
|
import telnetlib
|
2020-04-24 23:36:52 +00:00
|
|
|
|
import tempfile
|
|
|
|
|
import time
|
|
|
|
|
import unicodedata
|
|
|
|
|
|
|
|
|
|
CHAR_TO_KEY = {
|
|
|
|
|
"A": "shift-a",
|
|
|
|
|
"N": "shift-n",
|
|
|
|
|
"-": "0x0C",
|
|
|
|
|
"_": "shift-0x0C",
|
|
|
|
|
"B": "shift-b",
|
|
|
|
|
"O": "shift-o",
|
|
|
|
|
"=": "0x0D",
|
|
|
|
|
"+": "shift-0x0D",
|
|
|
|
|
"C": "shift-c",
|
|
|
|
|
"P": "shift-p",
|
|
|
|
|
"[": "0x1A",
|
|
|
|
|
"{": "shift-0x1A",
|
|
|
|
|
"D": "shift-d",
|
|
|
|
|
"Q": "shift-q",
|
|
|
|
|
"]": "0x1B",
|
|
|
|
|
"}": "shift-0x1B",
|
|
|
|
|
"E": "shift-e",
|
|
|
|
|
"R": "shift-r",
|
|
|
|
|
";": "0x27",
|
|
|
|
|
":": "shift-0x27",
|
|
|
|
|
"F": "shift-f",
|
|
|
|
|
"S": "shift-s",
|
|
|
|
|
"'": "0x28",
|
|
|
|
|
'"': "shift-0x28",
|
|
|
|
|
"G": "shift-g",
|
|
|
|
|
"T": "shift-t",
|
|
|
|
|
"`": "0x29",
|
|
|
|
|
"~": "shift-0x29",
|
|
|
|
|
"H": "shift-h",
|
|
|
|
|
"U": "shift-u",
|
|
|
|
|
"\\": "0x2B",
|
|
|
|
|
"|": "shift-0x2B",
|
|
|
|
|
"I": "shift-i",
|
|
|
|
|
"V": "shift-v",
|
|
|
|
|
",": "0x33",
|
|
|
|
|
"<": "shift-0x33",
|
|
|
|
|
"J": "shift-j",
|
|
|
|
|
"W": "shift-w",
|
|
|
|
|
".": "0x34",
|
|
|
|
|
">": "shift-0x34",
|
|
|
|
|
"K": "shift-k",
|
|
|
|
|
"X": "shift-x",
|
|
|
|
|
"/": "0x35",
|
|
|
|
|
"?": "shift-0x35",
|
|
|
|
|
"L": "shift-l",
|
|
|
|
|
"Y": "shift-y",
|
|
|
|
|
" ": "spc",
|
|
|
|
|
"M": "shift-m",
|
|
|
|
|
"Z": "shift-z",
|
|
|
|
|
"\n": "ret",
|
|
|
|
|
"!": "shift-0x02",
|
|
|
|
|
"@": "shift-0x03",
|
|
|
|
|
"#": "shift-0x04",
|
|
|
|
|
"$": "shift-0x05",
|
|
|
|
|
"%": "shift-0x06",
|
|
|
|
|
"^": "shift-0x07",
|
|
|
|
|
"&": "shift-0x08",
|
|
|
|
|
"*": "shift-0x09",
|
|
|
|
|
"(": "shift-0x0A",
|
|
|
|
|
")": "shift-0x0B",
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-22 07:53:02 +00:00
|
|
|
|
global log, machines, test_script
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def eprint(*args: object, **kwargs: Any) -> None:
|
|
|
|
|
print(*args, file=sys.stderr, **kwargs)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def make_command(args: list) -> str:
|
|
|
|
|
return " ".join(map(shlex.quote, (map(str, args))))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def create_vlan(vlan_nr: str) -> Tuple[str, str, "subprocess.Popen[bytes]", Any]:
|
|
|
|
|
log.log("starting VDE switch for network {}".format(vlan_nr))
|
|
|
|
|
vde_socket = tempfile.mkdtemp(
|
|
|
|
|
prefix="nixos-test-vde-", suffix="-vde{}.ctl".format(vlan_nr)
|
|
|
|
|
)
|
|
|
|
|
pty_master, pty_slave = pty.openpty()
|
|
|
|
|
vde_process = subprocess.Popen(
|
|
|
|
|
["vde_switch", "-s", vde_socket, "--dirmode", "0700"],
|
|
|
|
|
stdin=pty_slave,
|
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
|
stderr=subprocess.PIPE,
|
|
|
|
|
shell=False,
|
|
|
|
|
)
|
|
|
|
|
fd = os.fdopen(pty_master, "w")
|
|
|
|
|
fd.write("version\n")
|
|
|
|
|
# TODO: perl version checks if this can be read from
|
|
|
|
|
# an if not, dies. we could hang here forever. Fix it.
|
|
|
|
|
assert vde_process.stdout is not None
|
|
|
|
|
vde_process.stdout.readline()
|
|
|
|
|
if not os.path.exists(os.path.join(vde_socket, "ctl")):
|
|
|
|
|
raise Exception("cannot start vde_switch")
|
|
|
|
|
|
|
|
|
|
return (vlan_nr, vde_socket, vde_process, fd)
|
|
|
|
|
|
|
|
|
|
|
2021-06-01 10:57:12 +00:00
|
|
|
|
def retry(fn: Callable, timeout: int = 900) -> None:
|
2020-04-24 23:36:52 +00:00
|
|
|
|
"""Call the given function repeatedly, with 1 second intervals,
|
|
|
|
|
until it returns True or a timeout is reached.
|
|
|
|
|
"""
|
|
|
|
|
|
2021-06-01 10:57:12 +00:00
|
|
|
|
for _ in range(timeout):
|
2020-04-24 23:36:52 +00:00
|
|
|
|
if fn(False):
|
|
|
|
|
return
|
|
|
|
|
time.sleep(1)
|
|
|
|
|
|
|
|
|
|
if not fn(True):
|
2021-06-01 10:57:12 +00:00
|
|
|
|
raise Exception(f"action timed out after {timeout} seconds")
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Logger:
|
|
|
|
|
def __init__(self) -> None:
|
|
|
|
|
self.logfile = os.environ.get("LOGFILE", "/dev/null")
|
|
|
|
|
self.logfile_handle = codecs.open(self.logfile, "wb")
|
|
|
|
|
self.xml = XMLGenerator(self.logfile_handle, encoding="utf-8")
|
2020-05-15 21:57:56 +00:00
|
|
|
|
self.queue: "Queue[Dict[str, str]]" = Queue()
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
self.xml.startDocument()
|
|
|
|
|
self.xml.startElement("logfile", attrs={})
|
|
|
|
|
|
2021-05-20 23:08:51 +00:00
|
|
|
|
self._print_serial_logs = True
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
def close(self) -> None:
|
|
|
|
|
self.xml.endElement("logfile")
|
|
|
|
|
self.xml.endDocument()
|
|
|
|
|
self.logfile_handle.close()
|
|
|
|
|
|
|
|
|
|
def sanitise(self, message: str) -> str:
|
|
|
|
|
return "".join(ch for ch in message if unicodedata.category(ch)[0] != "C")
|
|
|
|
|
|
|
|
|
|
def maybe_prefix(self, message: str, attributes: Dict[str, str]) -> str:
|
|
|
|
|
if "machine" in attributes:
|
|
|
|
|
return "{}: {}".format(attributes["machine"], message)
|
|
|
|
|
return message
|
|
|
|
|
|
|
|
|
|
def log_line(self, message: str, attributes: Dict[str, str]) -> None:
|
|
|
|
|
self.xml.startElement("line", attributes)
|
|
|
|
|
self.xml.characters(message)
|
|
|
|
|
self.xml.endElement("line")
|
|
|
|
|
|
|
|
|
|
def log(self, message: str, attributes: Dict[str, str] = {}) -> None:
|
|
|
|
|
eprint(self.maybe_prefix(message, attributes))
|
|
|
|
|
self.drain_log_queue()
|
|
|
|
|
self.log_line(message, attributes)
|
|
|
|
|
|
2021-05-20 23:08:51 +00:00
|
|
|
|
def log_serial(self, message: str, machine: str) -> None:
|
|
|
|
|
self.enqueue({"msg": message, "machine": machine, "type": "serial"})
|
|
|
|
|
if self._print_serial_logs:
|
|
|
|
|
eprint(Style.DIM + "{} # {}".format(machine, message) + Style.RESET_ALL)
|
|
|
|
|
|
|
|
|
|
def enqueue(self, item: Dict[str, str]) -> None:
|
|
|
|
|
self.queue.put(item)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
def drain_log_queue(self) -> None:
|
|
|
|
|
try:
|
|
|
|
|
while True:
|
|
|
|
|
item = self.queue.get_nowait()
|
2021-05-20 23:08:51 +00:00
|
|
|
|
msg = self.sanitise(item["msg"])
|
|
|
|
|
del item["msg"]
|
|
|
|
|
self.log_line(msg, item)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
except Empty:
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
@contextmanager
|
|
|
|
|
def nested(self, message: str, attributes: Dict[str, str] = {}) -> Iterator[None]:
|
|
|
|
|
eprint(self.maybe_prefix(message, attributes))
|
|
|
|
|
|
|
|
|
|
self.xml.startElement("nest", attrs={})
|
|
|
|
|
self.xml.startElement("head", attributes)
|
|
|
|
|
self.xml.characters(message)
|
|
|
|
|
self.xml.endElement("head")
|
|
|
|
|
|
|
|
|
|
tic = time.time()
|
|
|
|
|
self.drain_log_queue()
|
|
|
|
|
yield
|
|
|
|
|
self.drain_log_queue()
|
|
|
|
|
toc = time.time()
|
|
|
|
|
self.log("({:.2f} seconds)".format(toc - tic))
|
|
|
|
|
|
|
|
|
|
self.xml.endElement("nest")
|
|
|
|
|
|
|
|
|
|
|
2021-04-25 03:57:28 +00:00
|
|
|
|
def _perform_ocr_on_screenshot(
|
|
|
|
|
screenshot_path: str, model_ids: Iterable[int]
|
|
|
|
|
) -> List[str]:
|
|
|
|
|
if shutil.which("tesseract") is None:
|
|
|
|
|
raise Exception("OCR requested but enableOCR is false")
|
|
|
|
|
|
|
|
|
|
magick_args = (
|
|
|
|
|
"-filter Catrom -density 72 -resample 300 "
|
|
|
|
|
+ "-contrast -normalize -despeckle -type grayscale "
|
|
|
|
|
+ "-sharpen 1 -posterize 3 -negate -gamma 100 "
|
|
|
|
|
+ "-blur 1x65535"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
tess_args = f"-c debug_file=/dev/null --psm 11"
|
|
|
|
|
|
|
|
|
|
cmd = f"convert {magick_args} {screenshot_path} tiff:{screenshot_path}.tiff"
|
|
|
|
|
ret = subprocess.run(cmd, shell=True, capture_output=True)
|
|
|
|
|
if ret.returncode != 0:
|
|
|
|
|
raise Exception(f"TIFF conversion failed with exit code {ret.returncode}")
|
|
|
|
|
|
|
|
|
|
model_results = []
|
|
|
|
|
for model_id in model_ids:
|
|
|
|
|
cmd = f"tesseract {screenshot_path}.tiff - {tess_args} --oem {model_id}"
|
|
|
|
|
ret = subprocess.run(cmd, shell=True, capture_output=True)
|
|
|
|
|
if ret.returncode != 0:
|
|
|
|
|
raise Exception(f"OCR failed with exit code {ret.returncode}")
|
|
|
|
|
model_results.append(ret.stdout.decode("utf-8"))
|
|
|
|
|
|
|
|
|
|
return model_results
|
|
|
|
|
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
class Machine:
|
2021-08-22 07:53:02 +00:00
|
|
|
|
def __repr__(self) -> str:
|
|
|
|
|
return f"<Machine '{self.name}'>"
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
def __init__(self, args: Dict[str, Any]) -> None:
|
|
|
|
|
if "name" in args:
|
|
|
|
|
self.name = args["name"]
|
|
|
|
|
else:
|
|
|
|
|
self.name = "machine"
|
|
|
|
|
cmd = args.get("startCommand", None)
|
|
|
|
|
if cmd:
|
|
|
|
|
match = re.search("run-(.+)-vm$", cmd)
|
|
|
|
|
if match:
|
|
|
|
|
self.name = match.group(1)
|
2020-09-25 04:45:31 +00:00
|
|
|
|
self.logger = args["log"]
|
2020-04-24 23:36:52 +00:00
|
|
|
|
self.script = args.get("startCommand", self.create_startcommand(args))
|
|
|
|
|
|
|
|
|
|
tmp_dir = os.environ.get("TMPDIR", tempfile.gettempdir())
|
|
|
|
|
|
|
|
|
|
def create_dir(name: str) -> str:
|
|
|
|
|
path = os.path.join(tmp_dir, name)
|
|
|
|
|
os.makedirs(path, mode=0o700, exist_ok=True)
|
|
|
|
|
return path
|
|
|
|
|
|
2020-09-25 04:45:31 +00:00
|
|
|
|
self.state_dir = os.path.join(tmp_dir, f"vm-state-{self.name}")
|
|
|
|
|
if not args.get("keepVmState", False):
|
|
|
|
|
self.cleanup_statedir()
|
|
|
|
|
os.makedirs(self.state_dir, mode=0o700, exist_ok=True)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
self.shared_dir = create_dir("shared-xchg")
|
|
|
|
|
|
|
|
|
|
self.booted = False
|
|
|
|
|
self.connected = False
|
|
|
|
|
self.pid: Optional[int] = None
|
|
|
|
|
self.socket = None
|
|
|
|
|
self.monitor: Optional[socket.socket] = None
|
|
|
|
|
self.allow_reboot = args.get("allowReboot", False)
|
|
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
|
def create_startcommand(args: Dict[str, str]) -> str:
|
|
|
|
|
net_backend = "-netdev user,id=net0"
|
|
|
|
|
net_frontend = "-device virtio-net-pci,netdev=net0"
|
|
|
|
|
|
|
|
|
|
if "netBackendArgs" in args:
|
|
|
|
|
net_backend += "," + args["netBackendArgs"]
|
|
|
|
|
|
|
|
|
|
if "netFrontendArgs" in args:
|
|
|
|
|
net_frontend += "," + args["netFrontendArgs"]
|
|
|
|
|
|
|
|
|
|
start_command = (
|
2021-07-14 22:03:04 +00:00
|
|
|
|
args.get("qemuBinary", "qemu-kvm")
|
|
|
|
|
+ " -m 384 "
|
|
|
|
|
+ net_backend
|
|
|
|
|
+ " "
|
|
|
|
|
+ net_frontend
|
|
|
|
|
+ " $QEMU_OPTS "
|
2020-04-24 23:36:52 +00:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
if "hda" in args:
|
|
|
|
|
hda_path = os.path.abspath(args["hda"])
|
|
|
|
|
if args.get("hdaInterface", "") == "scsi":
|
|
|
|
|
start_command += (
|
|
|
|
|
"-drive id=hda,file="
|
|
|
|
|
+ hda_path
|
|
|
|
|
+ ",werror=report,if=none "
|
|
|
|
|
+ "-device scsi-hd,drive=hda "
|
|
|
|
|
)
|
|
|
|
|
else:
|
|
|
|
|
start_command += (
|
|
|
|
|
"-drive file="
|
|
|
|
|
+ hda_path
|
|
|
|
|
+ ",if="
|
|
|
|
|
+ args["hdaInterface"]
|
|
|
|
|
+ ",werror=report "
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
if "cdrom" in args:
|
|
|
|
|
start_command += "-cdrom " + args["cdrom"] + " "
|
|
|
|
|
|
|
|
|
|
if "usb" in args:
|
2021-05-28 09:39:13 +00:00
|
|
|
|
# https://github.com/qemu/qemu/blob/master/docs/usb2.txt
|
2020-04-24 23:36:52 +00:00
|
|
|
|
start_command += (
|
2021-05-28 09:39:13 +00:00
|
|
|
|
"-device usb-ehci -drive "
|
2020-04-24 23:36:52 +00:00
|
|
|
|
+ "id=usbdisk,file="
|
|
|
|
|
+ args["usb"]
|
|
|
|
|
+ ",if=none,readonly "
|
|
|
|
|
+ "-device usb-storage,drive=usbdisk "
|
|
|
|
|
)
|
|
|
|
|
if "bios" in args:
|
|
|
|
|
start_command += "-bios " + args["bios"] + " "
|
|
|
|
|
|
|
|
|
|
start_command += args.get("qemuFlags", "")
|
|
|
|
|
|
|
|
|
|
return start_command
|
|
|
|
|
|
|
|
|
|
def is_up(self) -> bool:
|
|
|
|
|
return self.booted and self.connected
|
|
|
|
|
|
|
|
|
|
def log(self, msg: str) -> None:
|
|
|
|
|
self.logger.log(msg, {"machine": self.name})
|
|
|
|
|
|
2021-05-20 23:08:51 +00:00
|
|
|
|
def log_serial(self, msg: str) -> None:
|
|
|
|
|
self.logger.log_serial(msg, self.name)
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
def nested(self, msg: str, attrs: Dict[str, str] = {}) -> _GeneratorContextManager:
|
|
|
|
|
my_attrs = {"machine": self.name}
|
|
|
|
|
my_attrs.update(attrs)
|
|
|
|
|
return self.logger.nested(msg, my_attrs)
|
|
|
|
|
|
|
|
|
|
def wait_for_monitor_prompt(self) -> str:
|
|
|
|
|
assert self.monitor is not None
|
|
|
|
|
answer = ""
|
|
|
|
|
while True:
|
|
|
|
|
undecoded_answer = self.monitor.recv(1024)
|
|
|
|
|
if not undecoded_answer:
|
|
|
|
|
break
|
|
|
|
|
answer += undecoded_answer.decode()
|
|
|
|
|
if answer.endswith("(qemu) "):
|
|
|
|
|
break
|
|
|
|
|
return answer
|
|
|
|
|
|
|
|
|
|
def send_monitor_command(self, command: str) -> str:
|
|
|
|
|
message = ("{}\n".format(command)).encode()
|
|
|
|
|
self.log("sending monitor command: {}".format(command))
|
|
|
|
|
assert self.monitor is not None
|
|
|
|
|
self.monitor.send(message)
|
|
|
|
|
return self.wait_for_monitor_prompt()
|
|
|
|
|
|
|
|
|
|
def wait_for_unit(self, unit: str, user: Optional[str] = None) -> None:
|
|
|
|
|
"""Wait for a systemd unit to get into "active" state.
|
|
|
|
|
Throws exceptions on "failed" and "inactive" states as well as
|
|
|
|
|
after timing out.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def check_active(_: Any) -> bool:
|
|
|
|
|
info = self.get_unit_info(unit, user)
|
|
|
|
|
state = info["ActiveState"]
|
|
|
|
|
if state == "failed":
|
|
|
|
|
raise Exception('unit "{}" reached state "{}"'.format(unit, state))
|
|
|
|
|
|
|
|
|
|
if state == "inactive":
|
|
|
|
|
status, jobs = self.systemctl("list-jobs --full 2>&1", user)
|
|
|
|
|
if "No jobs" in jobs:
|
|
|
|
|
info = self.get_unit_info(unit, user)
|
|
|
|
|
if info["ActiveState"] == state:
|
|
|
|
|
raise Exception(
|
|
|
|
|
(
|
|
|
|
|
'unit "{}" is inactive and there ' "are no pending jobs"
|
|
|
|
|
).format(unit)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
return state == "active"
|
|
|
|
|
|
|
|
|
|
retry(check_active)
|
|
|
|
|
|
|
|
|
|
def get_unit_info(self, unit: str, user: Optional[str] = None) -> Dict[str, str]:
|
|
|
|
|
status, lines = self.systemctl('--no-pager show "{}"'.format(unit), user)
|
|
|
|
|
if status != 0:
|
|
|
|
|
raise Exception(
|
|
|
|
|
'retrieving systemctl info for unit "{}" {} failed with exit code {}'.format(
|
|
|
|
|
unit, "" if user is None else 'under user "{}"'.format(user), status
|
|
|
|
|
)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
line_pattern = re.compile(r"^([^=]+)=(.*)$")
|
|
|
|
|
|
|
|
|
|
def tuple_from_line(line: str) -> Tuple[str, str]:
|
|
|
|
|
match = line_pattern.match(line)
|
|
|
|
|
assert match is not None
|
|
|
|
|
return match[1], match[2]
|
|
|
|
|
|
|
|
|
|
return dict(
|
|
|
|
|
tuple_from_line(line)
|
|
|
|
|
for line in lines.split("\n")
|
|
|
|
|
if line_pattern.match(line)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def systemctl(self, q: str, user: Optional[str] = None) -> Tuple[int, str]:
|
|
|
|
|
if user is not None:
|
|
|
|
|
q = q.replace("'", "\\'")
|
|
|
|
|
return self.execute(
|
|
|
|
|
(
|
2020-05-15 21:57:56 +00:00
|
|
|
|
"su -l {} --shell /bin/sh -c "
|
2020-04-24 23:36:52 +00:00
|
|
|
|
"$'XDG_RUNTIME_DIR=/run/user/`id -u` "
|
|
|
|
|
"systemctl --user {}'"
|
|
|
|
|
).format(user, q)
|
|
|
|
|
)
|
|
|
|
|
return self.execute("systemctl {}".format(q))
|
|
|
|
|
|
|
|
|
|
def require_unit_state(self, unit: str, require_state: str = "active") -> None:
|
|
|
|
|
with self.nested(
|
|
|
|
|
"checking if unit ‘{}’ has reached state '{}'".format(unit, require_state)
|
|
|
|
|
):
|
|
|
|
|
info = self.get_unit_info(unit)
|
|
|
|
|
state = info["ActiveState"]
|
|
|
|
|
if state != require_state:
|
|
|
|
|
raise Exception(
|
|
|
|
|
"Expected unit ‘{}’ to to be in state ".format(unit)
|
|
|
|
|
+ "'{}' but it is in state ‘{}’".format(require_state, state)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def execute(self, command: str) -> Tuple[int, str]:
|
|
|
|
|
self.connect()
|
|
|
|
|
|
2021-06-28 23:13:55 +00:00
|
|
|
|
out_command = "( set -euo pipefail; {} ); echo '|!=EOF' $?\n".format(command)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
self.shell.send(out_command.encode())
|
|
|
|
|
|
|
|
|
|
output = ""
|
2020-05-15 21:57:56 +00:00
|
|
|
|
status_code_pattern = re.compile(r"(.*)\|\!=EOF\s+(\d+)")
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
while True:
|
|
|
|
|
chunk = self.shell.recv(4096).decode(errors="ignore")
|
|
|
|
|
match = status_code_pattern.match(chunk)
|
|
|
|
|
if match:
|
|
|
|
|
output += match[1]
|
|
|
|
|
status_code = int(match[2])
|
|
|
|
|
return (status_code, output)
|
|
|
|
|
output += chunk
|
|
|
|
|
|
2021-06-04 09:07:49 +00:00
|
|
|
|
def shell_interact(self) -> None:
|
|
|
|
|
"""Allows you to interact with the guest shell
|
|
|
|
|
|
|
|
|
|
Should only be used during test development, not in the production test."""
|
|
|
|
|
self.connect()
|
|
|
|
|
self.log("Terminal is ready (there is no prompt):")
|
2021-06-28 23:13:55 +00:00
|
|
|
|
subprocess.run(
|
|
|
|
|
["socat", "READLINE", f"FD:{self.shell.fileno()}"],
|
|
|
|
|
pass_fds=[self.shell.fileno()],
|
|
|
|
|
)
|
2021-06-04 09:07:49 +00:00
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
def succeed(self, *commands: str) -> str:
|
|
|
|
|
"""Execute each command and check that it succeeds."""
|
|
|
|
|
output = ""
|
|
|
|
|
for command in commands:
|
|
|
|
|
with self.nested("must succeed: {}".format(command)):
|
|
|
|
|
(status, out) = self.execute(command)
|
|
|
|
|
if status != 0:
|
|
|
|
|
self.log("output: {}".format(out))
|
|
|
|
|
raise Exception(
|
|
|
|
|
"command `{}` failed (exit code {})".format(command, status)
|
|
|
|
|
)
|
|
|
|
|
output += out
|
|
|
|
|
return output
|
|
|
|
|
|
2020-09-25 04:45:31 +00:00
|
|
|
|
def fail(self, *commands: str) -> str:
|
2020-04-24 23:36:52 +00:00
|
|
|
|
"""Execute each command and check that it fails."""
|
2020-09-25 04:45:31 +00:00
|
|
|
|
output = ""
|
2020-04-24 23:36:52 +00:00
|
|
|
|
for command in commands:
|
|
|
|
|
with self.nested("must fail: {}".format(command)):
|
2020-09-25 04:45:31 +00:00
|
|
|
|
(status, out) = self.execute(command)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
if status == 0:
|
|
|
|
|
raise Exception(
|
|
|
|
|
"command `{}` unexpectedly succeeded".format(command)
|
|
|
|
|
)
|
2020-09-25 04:45:31 +00:00
|
|
|
|
output += out
|
|
|
|
|
return output
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
2021-07-21 07:28:18 +00:00
|
|
|
|
def wait_until_succeeds(self, command: str, timeout: int = 900) -> str:
|
2020-04-24 23:36:52 +00:00
|
|
|
|
"""Wait until a command returns success and return its output.
|
|
|
|
|
Throws an exception on timeout.
|
|
|
|
|
"""
|
|
|
|
|
output = ""
|
|
|
|
|
|
|
|
|
|
def check_success(_: Any) -> bool:
|
|
|
|
|
nonlocal output
|
|
|
|
|
status, output = self.execute(command)
|
|
|
|
|
return status == 0
|
|
|
|
|
|
|
|
|
|
with self.nested("waiting for success: {}".format(command)):
|
2021-07-21 07:28:18 +00:00
|
|
|
|
retry(check_success, timeout)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
return output
|
|
|
|
|
|
|
|
|
|
def wait_until_fails(self, command: str) -> str:
|
|
|
|
|
"""Wait until a command returns failure.
|
|
|
|
|
Throws an exception on timeout.
|
|
|
|
|
"""
|
|
|
|
|
output = ""
|
|
|
|
|
|
|
|
|
|
def check_failure(_: Any) -> bool:
|
|
|
|
|
nonlocal output
|
|
|
|
|
status, output = self.execute(command)
|
|
|
|
|
return status != 0
|
|
|
|
|
|
|
|
|
|
with self.nested("waiting for failure: {}".format(command)):
|
|
|
|
|
retry(check_failure)
|
|
|
|
|
return output
|
|
|
|
|
|
|
|
|
|
def wait_for_shutdown(self) -> None:
|
|
|
|
|
if not self.booted:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
with self.nested("waiting for the VM to power off"):
|
|
|
|
|
sys.stdout.flush()
|
|
|
|
|
self.process.wait()
|
|
|
|
|
|
|
|
|
|
self.pid = None
|
|
|
|
|
self.booted = False
|
|
|
|
|
self.connected = False
|
|
|
|
|
|
|
|
|
|
def get_tty_text(self, tty: str) -> str:
|
|
|
|
|
status, output = self.execute(
|
|
|
|
|
"fold -w$(stty -F /dev/tty{0} size | "
|
|
|
|
|
"awk '{{print $2}}') /dev/vcs{0}".format(tty)
|
|
|
|
|
)
|
|
|
|
|
return output
|
|
|
|
|
|
|
|
|
|
def wait_until_tty_matches(self, tty: str, regexp: str) -> None:
|
|
|
|
|
"""Wait until the visible output on the chosen TTY matches regular
|
|
|
|
|
expression. Throws an exception on timeout.
|
|
|
|
|
"""
|
|
|
|
|
matcher = re.compile(regexp)
|
|
|
|
|
|
|
|
|
|
def tty_matches(last: bool) -> bool:
|
|
|
|
|
text = self.get_tty_text(tty)
|
|
|
|
|
if last:
|
|
|
|
|
self.log(
|
|
|
|
|
f"Last chance to match /{regexp}/ on TTY{tty}, "
|
|
|
|
|
f"which currently contains: {text}"
|
|
|
|
|
)
|
|
|
|
|
return len(matcher.findall(text)) > 0
|
|
|
|
|
|
|
|
|
|
with self.nested("waiting for {} to appear on tty {}".format(regexp, tty)):
|
|
|
|
|
retry(tty_matches)
|
|
|
|
|
|
|
|
|
|
def send_chars(self, chars: List[str]) -> None:
|
|
|
|
|
with self.nested("sending keys ‘{}‘".format(chars)):
|
|
|
|
|
for char in chars:
|
|
|
|
|
self.send_key(char)
|
|
|
|
|
|
|
|
|
|
def wait_for_file(self, filename: str) -> None:
|
|
|
|
|
"""Waits until the file exists in machine's file system."""
|
|
|
|
|
|
|
|
|
|
def check_file(_: Any) -> bool:
|
|
|
|
|
status, _ = self.execute("test -e {}".format(filename))
|
|
|
|
|
return status == 0
|
|
|
|
|
|
|
|
|
|
with self.nested("waiting for file ‘{}‘".format(filename)):
|
|
|
|
|
retry(check_file)
|
|
|
|
|
|
|
|
|
|
def wait_for_open_port(self, port: int) -> None:
|
|
|
|
|
def port_is_open(_: Any) -> bool:
|
|
|
|
|
status, _ = self.execute("nc -z localhost {}".format(port))
|
|
|
|
|
return status == 0
|
|
|
|
|
|
|
|
|
|
with self.nested("waiting for TCP port {}".format(port)):
|
|
|
|
|
retry(port_is_open)
|
|
|
|
|
|
|
|
|
|
def wait_for_closed_port(self, port: int) -> None:
|
|
|
|
|
def port_is_closed(_: Any) -> bool:
|
|
|
|
|
status, _ = self.execute("nc -z localhost {}".format(port))
|
|
|
|
|
return status != 0
|
|
|
|
|
|
|
|
|
|
retry(port_is_closed)
|
|
|
|
|
|
|
|
|
|
def start_job(self, jobname: str, user: Optional[str] = None) -> Tuple[int, str]:
|
|
|
|
|
return self.systemctl("start {}".format(jobname), user)
|
|
|
|
|
|
|
|
|
|
def stop_job(self, jobname: str, user: Optional[str] = None) -> Tuple[int, str]:
|
|
|
|
|
return self.systemctl("stop {}".format(jobname), user)
|
|
|
|
|
|
|
|
|
|
def wait_for_job(self, jobname: str) -> None:
|
|
|
|
|
self.wait_for_unit(jobname)
|
|
|
|
|
|
|
|
|
|
def connect(self) -> None:
|
|
|
|
|
if self.connected:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
with self.nested("waiting for the VM to finish booting"):
|
|
|
|
|
self.start()
|
|
|
|
|
|
|
|
|
|
tic = time.time()
|
|
|
|
|
self.shell.recv(1024)
|
|
|
|
|
# TODO: Timeout
|
|
|
|
|
toc = time.time()
|
|
|
|
|
|
|
|
|
|
self.log("connected to guest root shell")
|
|
|
|
|
self.log("(connecting took {:.2f} seconds)".format(toc - tic))
|
|
|
|
|
self.connected = True
|
|
|
|
|
|
|
|
|
|
def screenshot(self, filename: str) -> None:
|
|
|
|
|
out_dir = os.environ.get("out", os.getcwd())
|
|
|
|
|
word_pattern = re.compile(r"^\w+$")
|
|
|
|
|
if word_pattern.match(filename):
|
|
|
|
|
filename = os.path.join(out_dir, "{}.png".format(filename))
|
|
|
|
|
tmp = "{}.ppm".format(filename)
|
|
|
|
|
|
|
|
|
|
with self.nested(
|
|
|
|
|
"making screenshot {}".format(filename),
|
|
|
|
|
{"image": os.path.basename(filename)},
|
|
|
|
|
):
|
|
|
|
|
self.send_monitor_command("screendump {}".format(tmp))
|
|
|
|
|
ret = subprocess.run("pnmtopng {} > {}".format(tmp, filename), shell=True)
|
|
|
|
|
os.unlink(tmp)
|
|
|
|
|
if ret.returncode != 0:
|
|
|
|
|
raise Exception("Cannot convert screenshot")
|
|
|
|
|
|
|
|
|
|
def copy_from_host_via_shell(self, source: str, target: str) -> None:
|
|
|
|
|
"""Copy a file from the host into the guest by piping it over the
|
|
|
|
|
shell into the destination file. Works without host-guest shared folder.
|
|
|
|
|
Prefer copy_from_host for whenever possible.
|
|
|
|
|
"""
|
|
|
|
|
with open(source, "rb") as fh:
|
|
|
|
|
content_b64 = base64.b64encode(fh.read()).decode()
|
|
|
|
|
self.succeed(
|
|
|
|
|
f"mkdir -p $(dirname {target})",
|
|
|
|
|
f"echo -n {content_b64} | base64 -d > {target}",
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def copy_from_host(self, source: str, target: str) -> None:
|
|
|
|
|
"""Copy a file from the host into the guest via the `shared_dir` shared
|
|
|
|
|
among all the VMs (using a temporary directory).
|
|
|
|
|
"""
|
|
|
|
|
host_src = pathlib.Path(source)
|
|
|
|
|
vm_target = pathlib.Path(target)
|
|
|
|
|
with tempfile.TemporaryDirectory(dir=self.shared_dir) as shared_td:
|
|
|
|
|
shared_temp = pathlib.Path(shared_td)
|
|
|
|
|
host_intermediate = shared_temp / host_src.name
|
|
|
|
|
vm_shared_temp = pathlib.Path("/tmp/shared") / shared_temp.name
|
|
|
|
|
vm_intermediate = vm_shared_temp / host_src.name
|
|
|
|
|
|
|
|
|
|
self.succeed(make_command(["mkdir", "-p", vm_shared_temp]))
|
|
|
|
|
if host_src.is_dir():
|
|
|
|
|
shutil.copytree(host_src, host_intermediate)
|
|
|
|
|
else:
|
|
|
|
|
shutil.copy(host_src, host_intermediate)
|
|
|
|
|
self.succeed(make_command(["mkdir", "-p", vm_target.parent]))
|
|
|
|
|
self.succeed(make_command(["cp", "-r", vm_intermediate, vm_target]))
|
|
|
|
|
|
|
|
|
|
def copy_from_vm(self, source: str, target_dir: str = "") -> None:
|
|
|
|
|
"""Copy a file from the VM (specified by an in-VM source path) to a path
|
|
|
|
|
relative to `$out`. The file is copied via the `shared_dir` shared among
|
|
|
|
|
all the VMs (using a temporary directory).
|
|
|
|
|
"""
|
|
|
|
|
# Compute the source, target, and intermediate shared file names
|
|
|
|
|
out_dir = pathlib.Path(os.environ.get("out", os.getcwd()))
|
|
|
|
|
vm_src = pathlib.Path(source)
|
|
|
|
|
with tempfile.TemporaryDirectory(dir=self.shared_dir) as shared_td:
|
|
|
|
|
shared_temp = pathlib.Path(shared_td)
|
|
|
|
|
vm_shared_temp = pathlib.Path("/tmp/shared") / shared_temp.name
|
|
|
|
|
vm_intermediate = vm_shared_temp / vm_src.name
|
|
|
|
|
intermediate = shared_temp / vm_src.name
|
|
|
|
|
# Copy the file to the shared directory inside VM
|
|
|
|
|
self.succeed(make_command(["mkdir", "-p", vm_shared_temp]))
|
|
|
|
|
self.succeed(make_command(["cp", "-r", vm_src, vm_intermediate]))
|
|
|
|
|
abs_target = out_dir / target_dir / vm_src.name
|
|
|
|
|
abs_target.parent.mkdir(exist_ok=True, parents=True)
|
|
|
|
|
# Copy the file from the shared directory outside VM
|
|
|
|
|
if intermediate.is_dir():
|
|
|
|
|
shutil.copytree(intermediate, abs_target)
|
|
|
|
|
else:
|
|
|
|
|
shutil.copy(intermediate, abs_target)
|
|
|
|
|
|
|
|
|
|
def dump_tty_contents(self, tty: str) -> None:
|
2020-11-21 19:51:51 +00:00
|
|
|
|
"""Debugging: Dump the contents of the TTY<n>"""
|
2020-04-24 23:36:52 +00:00
|
|
|
|
self.execute("fold -w 80 /dev/vcs{} | systemd-cat".format(tty))
|
|
|
|
|
|
2021-04-25 03:57:28 +00:00
|
|
|
|
def _get_screen_text_variants(self, model_ids: Iterable[int]) -> List[str]:
|
|
|
|
|
with tempfile.TemporaryDirectory() as tmpdir:
|
|
|
|
|
screenshot_path = os.path.join(tmpdir, "ppm")
|
|
|
|
|
self.send_monitor_command(f"screendump {screenshot_path}")
|
|
|
|
|
return _perform_ocr_on_screenshot(screenshot_path, model_ids)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
2021-04-25 03:57:28 +00:00
|
|
|
|
def get_screen_text_variants(self) -> List[str]:
|
|
|
|
|
return self._get_screen_text_variants([0, 1, 2])
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
2021-04-25 03:57:28 +00:00
|
|
|
|
def get_screen_text(self) -> str:
|
|
|
|
|
return self._get_screen_text_variants([2])[0]
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
def wait_for_text(self, regex: str) -> None:
|
|
|
|
|
def screen_matches(last: bool) -> bool:
|
2021-04-25 03:57:28 +00:00
|
|
|
|
variants = self.get_screen_text_variants()
|
|
|
|
|
for text in variants:
|
|
|
|
|
if re.search(regex, text) is not None:
|
|
|
|
|
return True
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
2021-04-25 03:57:28 +00:00
|
|
|
|
if last:
|
|
|
|
|
self.log("Last OCR attempt failed. Text was: {}".format(variants))
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
2021-04-25 03:57:28 +00:00
|
|
|
|
return False
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
with self.nested("waiting for {} to appear on screen".format(regex)):
|
|
|
|
|
retry(screen_matches)
|
|
|
|
|
|
2020-07-18 16:06:22 +00:00
|
|
|
|
def wait_for_console_text(self, regex: str) -> None:
|
|
|
|
|
self.log("waiting for {} to appear on console".format(regex))
|
|
|
|
|
# Buffer the console output, this is needed
|
|
|
|
|
# to match multiline regexes.
|
|
|
|
|
console = io.StringIO()
|
|
|
|
|
while True:
|
|
|
|
|
try:
|
|
|
|
|
console.write(self.last_lines.get())
|
|
|
|
|
except queue.Empty:
|
|
|
|
|
self.sleep(1)
|
|
|
|
|
continue
|
|
|
|
|
console.seek(0)
|
|
|
|
|
matches = re.search(regex, console.read())
|
|
|
|
|
if matches is not None:
|
|
|
|
|
return
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
def send_key(self, key: str) -> None:
|
|
|
|
|
key = CHAR_TO_KEY.get(key, key)
|
|
|
|
|
self.send_monitor_command("sendkey {}".format(key))
|
|
|
|
|
|
|
|
|
|
def start(self) -> None:
|
|
|
|
|
if self.booted:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
self.log("starting vm")
|
|
|
|
|
|
|
|
|
|
def create_socket(path: str) -> socket.socket:
|
|
|
|
|
if os.path.exists(path):
|
|
|
|
|
os.unlink(path)
|
|
|
|
|
s = socket.socket(family=socket.AF_UNIX, type=socket.SOCK_STREAM)
|
|
|
|
|
s.bind(path)
|
|
|
|
|
s.listen(1)
|
|
|
|
|
return s
|
|
|
|
|
|
|
|
|
|
monitor_path = os.path.join(self.state_dir, "monitor")
|
|
|
|
|
self.monitor_socket = create_socket(monitor_path)
|
|
|
|
|
|
|
|
|
|
shell_path = os.path.join(self.state_dir, "shell")
|
|
|
|
|
self.shell_socket = create_socket(shell_path)
|
|
|
|
|
|
2021-05-04 21:07:42 +00:00
|
|
|
|
display_available = any(x in os.environ for x in ["DISPLAY", "WAYLAND_DISPLAY"])
|
2020-04-24 23:36:52 +00:00
|
|
|
|
qemu_options = (
|
|
|
|
|
" ".join(
|
|
|
|
|
[
|
|
|
|
|
"" if self.allow_reboot else "-no-reboot",
|
|
|
|
|
"-monitor unix:{}".format(monitor_path),
|
|
|
|
|
"-chardev socket,id=shell,path={}".format(shell_path),
|
|
|
|
|
"-device virtio-serial",
|
|
|
|
|
"-device virtconsole,chardev=shell",
|
|
|
|
|
"-device virtio-rng-pci",
|
2021-05-04 21:07:42 +00:00
|
|
|
|
"-serial stdio" if display_available else "-nographic",
|
2020-04-24 23:36:52 +00:00
|
|
|
|
]
|
|
|
|
|
)
|
|
|
|
|
+ " "
|
|
|
|
|
+ os.environ.get("QEMU_OPTS", "")
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
environment = dict(os.environ)
|
|
|
|
|
environment.update(
|
|
|
|
|
{
|
|
|
|
|
"TMPDIR": self.state_dir,
|
|
|
|
|
"SHARED_DIR": self.shared_dir,
|
|
|
|
|
"USE_TMPDIR": "1",
|
|
|
|
|
"QEMU_OPTS": qemu_options,
|
|
|
|
|
}
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
self.process = subprocess.Popen(
|
|
|
|
|
self.script,
|
|
|
|
|
stdin=subprocess.DEVNULL,
|
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
|
stderr=subprocess.STDOUT,
|
|
|
|
|
shell=True,
|
|
|
|
|
cwd=self.state_dir,
|
|
|
|
|
env=environment,
|
|
|
|
|
)
|
|
|
|
|
self.monitor, _ = self.monitor_socket.accept()
|
|
|
|
|
self.shell, _ = self.shell_socket.accept()
|
|
|
|
|
|
2020-07-18 16:06:22 +00:00
|
|
|
|
# Store last serial console lines for use
|
|
|
|
|
# of wait_for_console_text
|
|
|
|
|
self.last_lines: Queue = Queue()
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
def process_serial_output() -> None:
|
|
|
|
|
assert self.process.stdout is not None
|
|
|
|
|
for _line in self.process.stdout:
|
|
|
|
|
# Ignore undecodable bytes that may occur in boot menus
|
|
|
|
|
line = _line.decode(errors="ignore").replace("\r", "").rstrip()
|
2020-07-18 16:06:22 +00:00
|
|
|
|
self.last_lines.put(line)
|
2021-05-20 23:08:51 +00:00
|
|
|
|
self.log_serial(line)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
_thread.start_new_thread(process_serial_output, ())
|
|
|
|
|
|
|
|
|
|
self.wait_for_monitor_prompt()
|
|
|
|
|
|
|
|
|
|
self.pid = self.process.pid
|
|
|
|
|
self.booted = True
|
|
|
|
|
|
|
|
|
|
self.log("QEMU running (pid {})".format(self.pid))
|
|
|
|
|
|
2020-07-18 16:06:22 +00:00
|
|
|
|
def cleanup_statedir(self) -> None:
|
2020-09-25 04:45:31 +00:00
|
|
|
|
if os.path.isdir(self.state_dir):
|
2020-07-18 16:06:22 +00:00
|
|
|
|
shutil.rmtree(self.state_dir)
|
2020-09-25 04:45:31 +00:00
|
|
|
|
self.logger.log(f"deleting VM state directory {self.state_dir}")
|
|
|
|
|
self.logger.log("if you want to keep the VM state, pass --keep-vm-state")
|
2020-07-18 16:06:22 +00:00
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
def shutdown(self) -> None:
|
|
|
|
|
if not self.booted:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
self.shell.send("poweroff\n".encode())
|
|
|
|
|
self.wait_for_shutdown()
|
|
|
|
|
|
|
|
|
|
def crash(self) -> None:
|
|
|
|
|
if not self.booted:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
self.log("forced crash")
|
|
|
|
|
self.send_monitor_command("quit")
|
|
|
|
|
self.wait_for_shutdown()
|
|
|
|
|
|
|
|
|
|
def wait_for_x(self) -> None:
|
|
|
|
|
"""Wait until it is possible to connect to the X server. Note that
|
|
|
|
|
testing the existence of /tmp/.X11-unix/X0 is insufficient.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def check_x(_: Any) -> bool:
|
|
|
|
|
cmd = (
|
|
|
|
|
"journalctl -b SYSLOG_IDENTIFIER=systemd | "
|
|
|
|
|
+ 'grep "Reached target Current graphical"'
|
|
|
|
|
)
|
|
|
|
|
status, _ = self.execute(cmd)
|
|
|
|
|
if status != 0:
|
|
|
|
|
return False
|
|
|
|
|
status, _ = self.execute("[ -e /tmp/.X11-unix/X0 ]")
|
|
|
|
|
return status == 0
|
|
|
|
|
|
|
|
|
|
with self.nested("waiting for the X11 server"):
|
|
|
|
|
retry(check_x)
|
|
|
|
|
|
|
|
|
|
def get_window_names(self) -> List[str]:
|
|
|
|
|
return self.succeed(
|
|
|
|
|
r"xwininfo -root -tree | sed 's/.*0x[0-9a-f]* \"\([^\"]*\)\".*/\1/; t; d'"
|
|
|
|
|
).splitlines()
|
|
|
|
|
|
|
|
|
|
def wait_for_window(self, regexp: str) -> None:
|
|
|
|
|
pattern = re.compile(regexp)
|
|
|
|
|
|
|
|
|
|
def window_is_visible(last_try: bool) -> bool:
|
|
|
|
|
names = self.get_window_names()
|
|
|
|
|
if last_try:
|
|
|
|
|
self.log(
|
|
|
|
|
"Last chance to match {} on the window list,".format(regexp)
|
|
|
|
|
+ " which currently contains: "
|
|
|
|
|
+ ", ".join(names)
|
|
|
|
|
)
|
|
|
|
|
return any(pattern.search(name) for name in names)
|
|
|
|
|
|
|
|
|
|
with self.nested("Waiting for a window to appear"):
|
|
|
|
|
retry(window_is_visible)
|
|
|
|
|
|
|
|
|
|
def sleep(self, secs: int) -> None:
|
2020-09-25 04:45:31 +00:00
|
|
|
|
# We want to sleep in *guest* time, not *host* time.
|
|
|
|
|
self.succeed(f"sleep {secs}")
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
def forward_port(self, host_port: int = 8080, guest_port: int = 80) -> None:
|
|
|
|
|
"""Forward a TCP port on the host to a TCP port on the guest.
|
|
|
|
|
Useful during interactive testing.
|
|
|
|
|
"""
|
|
|
|
|
self.send_monitor_command(
|
|
|
|
|
"hostfwd_add tcp::{}-:{}".format(host_port, guest_port)
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def block(self) -> None:
|
|
|
|
|
"""Make the machine unreachable by shutting down eth1 (the multicast
|
|
|
|
|
interface used to talk to the other VMs). We keep eth0 up so that
|
|
|
|
|
the test driver can continue to talk to the machine.
|
|
|
|
|
"""
|
|
|
|
|
self.send_monitor_command("set_link virtio-net-pci.1 off")
|
|
|
|
|
|
|
|
|
|
def unblock(self) -> None:
|
2020-11-21 19:51:51 +00:00
|
|
|
|
"""Make the machine reachable."""
|
2020-04-24 23:36:52 +00:00
|
|
|
|
self.send_monitor_command("set_link virtio-net-pci.1 on")
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def create_machine(args: Dict[str, Any]) -> Machine:
|
|
|
|
|
args["log"] = log
|
|
|
|
|
return Machine(args)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def start_all() -> None:
|
|
|
|
|
with log.nested("starting all VMs"):
|
|
|
|
|
for machine in machines:
|
|
|
|
|
machine.start()
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def join_all() -> None:
|
|
|
|
|
with log.nested("waiting for all VMs to finish"):
|
|
|
|
|
for machine in machines:
|
|
|
|
|
machine.wait_for_shutdown()
|
|
|
|
|
|
|
|
|
|
|
2021-08-08 23:34:03 +00:00
|
|
|
|
def run_tests(interactive: bool = False) -> None:
|
|
|
|
|
if interactive:
|
2021-08-22 07:53:02 +00:00
|
|
|
|
ptpython.repl.embed(test_symbols(), {})
|
2020-04-24 23:36:52 +00:00
|
|
|
|
else:
|
2021-08-08 23:34:03 +00:00
|
|
|
|
test_script()
|
|
|
|
|
# TODO: Collect coverage data
|
|
|
|
|
for machine in machines:
|
|
|
|
|
if machine.is_up():
|
|
|
|
|
machine.execute("sync")
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
|
2021-05-20 23:08:51 +00:00
|
|
|
|
def serial_stdout_on() -> None:
|
|
|
|
|
log._print_serial_logs = True
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def serial_stdout_off() -> None:
|
|
|
|
|
log._print_serial_logs = False
|
|
|
|
|
|
|
|
|
|
|
2021-08-08 23:34:03 +00:00
|
|
|
|
class EnvDefault(argparse.Action):
|
|
|
|
|
"""An argpars Action that takes values from the specified
|
|
|
|
|
environment variable as the flags default value.
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
def __init__(self, envvar, required=False, default=None, nargs=None, **kwargs): # type: ignore
|
|
|
|
|
if not default and envvar:
|
|
|
|
|
if envvar in os.environ:
|
|
|
|
|
if nargs is not None and (nargs.isdigit() or nargs in ["*", "+"]):
|
|
|
|
|
default = os.environ[envvar].split()
|
|
|
|
|
else:
|
|
|
|
|
default = os.environ[envvar]
|
|
|
|
|
kwargs["help"] = (
|
|
|
|
|
kwargs["help"] + f" (default from environment: {default})"
|
|
|
|
|
)
|
|
|
|
|
if required and default:
|
|
|
|
|
required = False
|
|
|
|
|
super(EnvDefault, self).__init__(
|
|
|
|
|
default=default, required=required, nargs=nargs, **kwargs
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def __call__(self, parser, namespace, values, option_string=None): # type: ignore
|
|
|
|
|
setattr(namespace, self.dest, values)
|
|
|
|
|
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
@contextmanager
|
|
|
|
|
def subtest(name: str) -> Iterator[None]:
|
|
|
|
|
with log.nested(name):
|
|
|
|
|
try:
|
|
|
|
|
yield
|
|
|
|
|
return True
|
|
|
|
|
except Exception as e:
|
2020-05-03 17:38:23 +00:00
|
|
|
|
log.log(f'Test "{name}" failed with error: "{e}"')
|
|
|
|
|
raise e
|
2020-04-24 23:36:52 +00:00
|
|
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
2021-08-22 07:53:02 +00:00
|
|
|
|
def _test_symbols() -> Dict[str, Any]:
|
|
|
|
|
general_symbols = dict(
|
|
|
|
|
start_all=start_all,
|
|
|
|
|
test_script=globals().get("test_script"), # same
|
|
|
|
|
machines=globals().get("machines"), # without being initialized
|
|
|
|
|
log=globals().get("log"), # extracting those symbol keys
|
|
|
|
|
os=os,
|
|
|
|
|
create_machine=create_machine,
|
|
|
|
|
subtest=subtest,
|
|
|
|
|
run_tests=run_tests,
|
|
|
|
|
join_all=join_all,
|
|
|
|
|
retry=retry,
|
|
|
|
|
serial_stdout_off=serial_stdout_off,
|
|
|
|
|
serial_stdout_on=serial_stdout_on,
|
|
|
|
|
Machine=Machine, # for typing
|
|
|
|
|
)
|
|
|
|
|
return general_symbols
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def test_symbols() -> Dict[str, Any]:
|
|
|
|
|
|
|
|
|
|
general_symbols = _test_symbols()
|
|
|
|
|
|
|
|
|
|
machine_symbols = {m.name: machines[idx] for idx, m in enumerate(machines)}
|
|
|
|
|
print(
|
|
|
|
|
"additionally exposed symbols:\n "
|
|
|
|
|
+ ", ".join(map(lambda m: m.name, machines))
|
|
|
|
|
+ ",\n "
|
|
|
|
|
+ ", ".join(list(general_symbols.keys()))
|
|
|
|
|
)
|
|
|
|
|
return {**general_symbols, **machine_symbols}
|
|
|
|
|
|
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
if __name__ == "__main__":
|
2021-06-28 23:13:55 +00:00
|
|
|
|
arg_parser = argparse.ArgumentParser(prog="nixos-test-driver")
|
2020-07-18 16:06:22 +00:00
|
|
|
|
arg_parser.add_argument(
|
|
|
|
|
"-K",
|
|
|
|
|
"--keep-vm-state",
|
|
|
|
|
help="re-use a VM state coming from a previous run",
|
|
|
|
|
action="store_true",
|
|
|
|
|
)
|
2021-08-08 23:34:03 +00:00
|
|
|
|
arg_parser.add_argument(
|
|
|
|
|
"-I",
|
|
|
|
|
"--interactive",
|
|
|
|
|
help="drop into a python repl and run the tests interactively",
|
|
|
|
|
action="store_true",
|
|
|
|
|
)
|
|
|
|
|
arg_parser.add_argument(
|
|
|
|
|
"--start-scripts",
|
|
|
|
|
metavar="START-SCRIPT",
|
|
|
|
|
action=EnvDefault,
|
|
|
|
|
envvar="startScripts",
|
|
|
|
|
nargs="*",
|
|
|
|
|
help="start scripts for participating virtual machines",
|
|
|
|
|
)
|
|
|
|
|
arg_parser.add_argument(
|
|
|
|
|
"--vlans",
|
|
|
|
|
metavar="VLAN",
|
|
|
|
|
action=EnvDefault,
|
|
|
|
|
envvar="vlans",
|
|
|
|
|
nargs="*",
|
|
|
|
|
help="vlans to span by the driver",
|
|
|
|
|
)
|
|
|
|
|
arg_parser.add_argument(
|
|
|
|
|
"testscript",
|
|
|
|
|
action=EnvDefault,
|
|
|
|
|
envvar="testScript",
|
|
|
|
|
help="the test script to run",
|
|
|
|
|
type=pathlib.Path,
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
args = arg_parser.parse_args()
|
2021-08-18 13:19:15 +00:00
|
|
|
|
testscript = pathlib.Path(args.testscript).read_text()
|
2021-08-08 23:34:03 +00:00
|
|
|
|
|
2021-08-22 07:53:02 +00:00
|
|
|
|
global log, machines, test_script
|
2020-07-18 16:06:22 +00:00
|
|
|
|
|
2020-04-24 23:36:52 +00:00
|
|
|
|
log = Logger()
|
|
|
|
|
|
2021-08-08 23:34:03 +00:00
|
|
|
|
vde_sockets = [create_vlan(v) for v in args.vlans]
|
2020-04-24 23:36:52 +00:00
|
|
|
|
for nr, vde_socket, _, _ in vde_sockets:
|
|
|
|
|
os.environ["QEMU_VDE_SOCKET_{}".format(nr)] = vde_socket
|
|
|
|
|
|
2020-09-25 04:45:31 +00:00
|
|
|
|
machines = [
|
2021-08-08 23:34:03 +00:00
|
|
|
|
create_machine({"startCommand": s, "keepVmState": args.keep_vm_state})
|
|
|
|
|
for s in args.start_scripts
|
2020-09-25 04:45:31 +00:00
|
|
|
|
]
|
2020-04-24 23:36:52 +00:00
|
|
|
|
machine_eval = [
|
|
|
|
|
"{0} = machines[{1}]".format(m.name, idx) for idx, m in enumerate(machines)
|
|
|
|
|
]
|
|
|
|
|
exec("\n".join(machine_eval))
|
|
|
|
|
|
|
|
|
|
@atexit.register
|
|
|
|
|
def clean_up() -> None:
|
|
|
|
|
with log.nested("cleaning up"):
|
|
|
|
|
for machine in machines:
|
|
|
|
|
if machine.pid is None:
|
|
|
|
|
continue
|
|
|
|
|
log.log("killing {} (pid {})".format(machine.name, machine.pid))
|
|
|
|
|
machine.process.kill()
|
|
|
|
|
for _, _, process, _ in vde_sockets:
|
|
|
|
|
process.terminate()
|
|
|
|
|
log.close()
|
|
|
|
|
|
2021-08-22 07:53:02 +00:00
|
|
|
|
def test_script() -> None:
|
|
|
|
|
with log.nested("running the VM test script"):
|
|
|
|
|
symbols = test_symbols() # call eagerly
|
|
|
|
|
exec(testscript, symbols, None)
|
|
|
|
|
|
2021-08-18 13:19:15 +00:00
|
|
|
|
interactive = args.interactive or (not bool(testscript))
|
2020-04-24 23:36:52 +00:00
|
|
|
|
tic = time.time()
|
2021-08-18 13:19:15 +00:00
|
|
|
|
run_tests(interactive)
|
2020-04-24 23:36:52 +00:00
|
|
|
|
toc = time.time()
|
|
|
|
|
print("test script finished in {:.2f}s".format(toc - tic))
|