mirror of
https://github.com/openSUSE/osc.git
synced 2025-01-25 22:36:13 +01:00
behave: Speed running tests up by preparing containers in advance
This commit is contained in:
parent
c1f0cfa1b7
commit
a66d40fe3a
2
.github/workflows/tests.yaml
vendored
2
.github/workflows/tests.yaml
vendored
@ -108,4 +108,4 @@ jobs:
|
|||||||
- name: "Run tests"
|
- name: "Run tests"
|
||||||
run: |
|
run: |
|
||||||
cd behave
|
cd behave
|
||||||
behave -Dosc=../osc-wrapper.py
|
behave -Dosc=../osc-wrapper.py -Dmax_podman_containers=2
|
||||||
|
@ -22,7 +22,7 @@ def after_scenario(context, scenario):
|
|||||||
# start a new container after a destructive test
|
# start a new container after a destructive test
|
||||||
# we must use an existing podman instance defined in `before_all` due to context attribute life-cycle:
|
# we must use an existing podman instance defined in `before_all` due to context attribute life-cycle:
|
||||||
# https://behave.readthedocs.io/en/stable/context_attributes.html
|
# https://behave.readthedocs.io/en/stable/context_attributes.html
|
||||||
context.podman.restart()
|
context.podman.new_container()
|
||||||
context.osc.clear()
|
context.osc.clear()
|
||||||
common.check_exit_code(context)
|
common.check_exit_code(context)
|
||||||
|
|
||||||
@ -47,7 +47,12 @@ def before_all(context):
|
|||||||
# absolute path to .../behave/fixtures
|
# absolute path to .../behave/fixtures
|
||||||
context.fixtures = os.path.join(os.path.dirname(__file__), "..", "fixtures")
|
context.fixtures = os.path.join(os.path.dirname(__file__), "..", "fixtures")
|
||||||
|
|
||||||
context.podman = podman.Podman(context)
|
podman_max_containers = context.config.userdata.get("podman_max_containers", None)
|
||||||
|
if podman_max_containers:
|
||||||
|
podman_max_containers = int(podman_max_containers)
|
||||||
|
context.podman = podman.ThreadedPodman(context, container_name_prefix="osc-behave-", max_containers=podman_max_containers)
|
||||||
|
else:
|
||||||
|
context.podman = podman.Podman(context, container_name="osc-behave")
|
||||||
context.osc = osc.Osc(context)
|
context.osc = osc.Osc(context)
|
||||||
|
|
||||||
|
|
||||||
|
@ -35,7 +35,7 @@ class Osc:
|
|||||||
with open(self.oscrc, "w") as f:
|
with open(self.oscrc, "w") as f:
|
||||||
f.write("[general]\n")
|
f.write("[general]\n")
|
||||||
f.write("\n")
|
f.write("\n")
|
||||||
f.write(f"[https://localhost:{self.context.podman.port}]\n")
|
f.write(f"[https://localhost:{self.context.podman.container.port}]\n")
|
||||||
f.write("user=Admin\n")
|
f.write("user=Admin\n")
|
||||||
f.write("pass=opensuse\n")
|
f.write("pass=opensuse\n")
|
||||||
f.write("credentials_mgr_class=osc.credentials.PlaintextConfigFileCredentialsManager\n")
|
f.write("credentials_mgr_class=osc.credentials.PlaintextConfigFileCredentialsManager\n")
|
||||||
@ -48,7 +48,7 @@ class Osc:
|
|||||||
osc_cmd = self.context.config.userdata.get("osc", "osc")
|
osc_cmd = self.context.config.userdata.get("osc", "osc")
|
||||||
cmd = [osc_cmd]
|
cmd = [osc_cmd]
|
||||||
cmd += ["--config", self.oscrc]
|
cmd += ["--config", self.oscrc]
|
||||||
cmd += ["-A", f"https://localhost:{self.context.podman.port}"]
|
cmd += ["-A", f"https://localhost:{self.context.podman.container.port}"]
|
||||||
return cmd
|
return cmd
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,13 +1,126 @@
|
|||||||
|
import queue
|
||||||
import subprocess
|
import subprocess
|
||||||
|
import threading
|
||||||
|
|
||||||
from steps.common import debug
|
from steps.common import debug
|
||||||
|
|
||||||
|
|
||||||
class Podman:
|
class Podman:
|
||||||
def __init__(self, context):
|
def __init__(self, context, container_name):
|
||||||
self.context = context
|
self.context = context
|
||||||
debug(context, "Podman.__init__()")
|
self.container_name = container_name
|
||||||
|
self.container = None
|
||||||
|
debug(self.context, "Podman.__init__()")
|
||||||
|
|
||||||
|
self.new_container()
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
debug(self.context, "Podman.__del__()")
|
||||||
|
try:
|
||||||
|
self.kill()
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def kill(self):
|
||||||
|
debug(self.context, "Podman.kill()")
|
||||||
|
if not self.container:
|
||||||
|
return
|
||||||
|
self.container.kill()
|
||||||
|
self.container = None
|
||||||
|
|
||||||
|
def new_container(self):
|
||||||
|
debug(self.context, "Podman.new_container()")
|
||||||
|
# no need to stop the running container
|
||||||
|
# becuse the new container replaces an old container with the identical name
|
||||||
|
self.container = Container(self.context, name=self.container_name)
|
||||||
|
|
||||||
|
|
||||||
|
class ThreadedPodman:
|
||||||
|
def __init__(self, context, container_name_prefix, max_containers=1):
|
||||||
|
self.context = context
|
||||||
|
self.container = None
|
||||||
|
debug(self.context, "ThreadedPodman.__init__()")
|
||||||
|
|
||||||
|
self.max_containers = max_containers
|
||||||
|
self.container_name_prefix = container_name_prefix
|
||||||
|
self.container_name_num = 0
|
||||||
|
|
||||||
|
# produce new containers
|
||||||
|
self.container_producer_queue = queue.Queue(maxsize=self.max_containers)
|
||||||
|
self.container_producer_queue_is_stopping = threading.Event()
|
||||||
|
self.container_producer_queue_is_stopped = threading.Event()
|
||||||
|
self.container_producer_thread = threading.Thread(target=self.container_producer, daemon=True)
|
||||||
|
self.container_producer_thread.start()
|
||||||
|
|
||||||
|
# consume (kill) used containers
|
||||||
|
self.container_consumer_queue = queue.Queue()
|
||||||
|
self.container_consumer_thread = threading.Thread(target=self.container_consumer, daemon=True)
|
||||||
|
self.container_consumer_thread.start()
|
||||||
|
|
||||||
|
self.new_container()
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
debug(self.context, "ThreadedPodman.__del__()")
|
||||||
|
try:
|
||||||
|
self.kill()
|
||||||
|
except Exception:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def kill(self):
|
||||||
|
debug(self.context, "ThreadedPodman.kill()")
|
||||||
|
self.container_producer_queue_is_stopping.set()
|
||||||
|
|
||||||
|
container = getattr(self, "container", None)
|
||||||
|
if container:
|
||||||
|
self.container_consumer_queue.put(container)
|
||||||
|
self.container = None
|
||||||
|
|
||||||
|
while not self.container_producer_queue_is_stopped.is_set():
|
||||||
|
try:
|
||||||
|
container = self.container_producer_queue.get(block=True, timeout=1)
|
||||||
|
self.container_consumer_queue.put(container)
|
||||||
|
except queue.Empty:
|
||||||
|
continue
|
||||||
|
|
||||||
|
# 'None' is a signal to finish processing the queue
|
||||||
|
self.container_consumer_queue.put(None)
|
||||||
|
|
||||||
|
self.container_producer_thread.join()
|
||||||
|
self.container_consumer_thread.join()
|
||||||
|
|
||||||
|
def container_producer(self):
|
||||||
|
while not self.container_producer_queue_is_stopping.is_set():
|
||||||
|
if self.container_name_prefix:
|
||||||
|
self.container_name_num += 1
|
||||||
|
container_name = f"{self.container_name_prefix}{self.container_name_num}"
|
||||||
|
else:
|
||||||
|
container_name = None
|
||||||
|
container = Container(self.context, name=container_name)
|
||||||
|
self.container_producer_queue.put(container, block=True)
|
||||||
|
self.container_producer_queue_is_stopped.set()
|
||||||
|
|
||||||
|
def container_consumer(self):
|
||||||
|
while True:
|
||||||
|
container = self.container_consumer_queue.get(block=True)
|
||||||
|
if container is None:
|
||||||
|
break
|
||||||
|
container.kill()
|
||||||
|
|
||||||
|
def new_container(self):
|
||||||
|
debug(self.context, "ThreadedPodman.new_container()")
|
||||||
|
if getattr(self, "container", None):
|
||||||
|
self.container_consumer_queue.put(self.container)
|
||||||
|
self.container = self.container_producer_queue.get(block=True)
|
||||||
|
debug(self.context, f"> {self.container}")
|
||||||
|
|
||||||
|
|
||||||
|
class Container:
|
||||||
|
def __init__(self, context, name=None):
|
||||||
|
self.context = context
|
||||||
|
debug(self.context, "Container.__init__()")
|
||||||
|
self.container_name = name
|
||||||
self.container_id = None
|
self.container_id = None
|
||||||
|
self.port = None
|
||||||
self.start()
|
self.start()
|
||||||
|
|
||||||
def __del__(self):
|
def __del__(self):
|
||||||
@ -16,6 +129,11 @@ class Podman:
|
|||||||
except Exception:
|
except Exception:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
result = super().__repr__()
|
||||||
|
result += f"(port:{self.port}, id:{self.container_id}, name:{self.container_name})"
|
||||||
|
return result
|
||||||
|
|
||||||
def _run(self, args, check=True):
|
def _run(self, args, check=True):
|
||||||
cmd = ["podman"] + args
|
cmd = ["podman"] + args
|
||||||
debug(self.context, "Running command:", cmd)
|
debug(self.context, "Running command:", cmd)
|
||||||
@ -32,12 +150,18 @@ class Podman:
|
|||||||
return proc
|
return proc
|
||||||
|
|
||||||
def start(self):
|
def start(self):
|
||||||
debug(self.context, "Podman.start()")
|
debug(self.context, "Container.start()")
|
||||||
args = [
|
args = [
|
||||||
"run",
|
"run",
|
||||||
"--name", "obs-server-behave",
|
|
||||||
"--hostname", "obs-server-behave",
|
"--hostname", "obs-server-behave",
|
||||||
|
]
|
||||||
|
if self.container_name:
|
||||||
|
args += [
|
||||||
|
"--name", self.container_name,
|
||||||
"--replace",
|
"--replace",
|
||||||
|
"--stop-signal", "SIGKILL",
|
||||||
|
]
|
||||||
|
args += [
|
||||||
"--rm",
|
"--rm",
|
||||||
"--detach",
|
"--detach",
|
||||||
"--interactive",
|
"--interactive",
|
||||||
@ -54,13 +178,13 @@ class Podman:
|
|||||||
def kill(self):
|
def kill(self):
|
||||||
if not self.container_id:
|
if not self.container_id:
|
||||||
return
|
return
|
||||||
debug(self.context, "Podman.kill()")
|
debug(self.context, "Container.kill()")
|
||||||
args = ["kill", self.container_id]
|
args = ["kill", self.container_id]
|
||||||
self._run(args)
|
self._run(args)
|
||||||
self.container_id = None
|
self.container_id = None
|
||||||
|
|
||||||
def restart(self):
|
def restart(self):
|
||||||
debug(self.context, "Podman.restart()")
|
debug(self.context, "Container.restart()")
|
||||||
self.kill()
|
self.kill()
|
||||||
self.start()
|
self.start()
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user