import os import re import shutil import tempfile import time import behave from steps.common import debug from steps.common import run_in_context class Osc: def __init__(self, context): if not hasattr(context, "podman"): raise RuntimeError("context doesn't have 'podman' object set") self.context = context debug(self.context, "Osc.__init__()") self.temp = None self.clear() def __del__(self): try: shutil.rmtree(self.temp) except Exception: pass def clear(self): debug(self.context, "Osc.clear()") if self.temp: shutil.rmtree(self.temp) self.temp = tempfile.mkdtemp(prefix="osc_behave_") self.oscrc = os.path.join(self.temp, "oscrc") self.write_oscrc() def write_oscrc(self, username=None, password=None): with open(self.oscrc, "w") as f: f.write("[general]\n") f.write("\n") f.write(f"[https://localhost:{self.context.podman.container.port}]\n") f.write(f"user={username or 'Admin'}\n") f.write(f"pass={password or 'opensuse'}\n") f.write("credentials_mgr_class=osc.credentials.PlaintextConfigFileCredentialsManager\n") f.write("sslcertck=0\n") if not any((username, password)): f.write("http_headers =\n") # avoid the initial 401 response by setting auth to Admin:opensuse directly # write the header only when the default user/pass are used f.write(" authorization: Basic QWRtaW46b3BlbnN1c2U=\n") def get_cmd(self): osc_cmd = self.context.config.userdata.get("osc", "osc") cmd = [osc_cmd] cmd += ["--config", self.oscrc] cmd += ["-A", f"https://localhost:{self.context.podman.container.port}"] return cmd @behave.step("I execute osc with args \"{args}\"") def step_impl(context, args): args = args.format(context=context) cmd = context.osc.get_cmd() + [args] cmd = " ".join(cmd) run_in_context(context, cmd, can_fail=True) # remove InsecureRequestWarning that is irrelevant to the tests context.cmd_stderr = re.sub(r"^.*InsecureRequestWarning.*\n warnings.warn\(\n", "", context.cmd_stderr) @behave.step("I configure osc user \"{username}\" with password \"{password}\"") def step_impl(context, username, password): context.osc.write_oscrc(username=username, password=password) @behave.step('I wait for osc results for "{project}" "{package}"') def step_impl(context, project, package): args = f"results {project} {package} --csv --format='%(code)s,%(dirty)s'" cmd = context.osc.get_cmd() + [args] cmd = " ".join(cmd) while True: # wait for a moment before checking the status even for the first time # for some reason, packages appear to be "broken" for a while after they get commited time.sleep(5) run_in_context(context, cmd, can_fail=True) results = [] for line in context.cmd_stdout.splitlines(): code, dirty = line.split(",") dirty = dirty.lower() == "true" results.append((code, dirty)) if all((code == "succeeded" and not dirty for code, dirty in results)): # all builds have succeeded and all dirty flags are false break if any((code in ("unresolvable", "failed", "broken", "blocked", "locked", "excluded") and not dirty for code, dirty in results)): # failed build with dirty flag false raise AssertionError("Package build failed:\n" + context.cmd_stdout)