"""Shared pytest fixtures for recipe CI (plan §4.3). A run is parameterized by env: RECIPE, REF (PR head sha), PR, SRC (head repo). The harness computes a unique app domain per run so concurrent runs never collide, and GUARANTEES teardown (undeploy + volume + secret removal) via a finalizer, even on failure. """ from __future__ import annotations import hashlib import os import sys import time import pytest sys.path.insert(0, os.path.join(os.path.dirname(__file__), "..", "runner")) from harness import lifecycle # noqa: E402 def _short(s: str, n: int = 8) -> str: return "".join(c for c in s if c.isalnum())[:n] or "local" @pytest.fixture(scope="session") def recipe() -> str: return os.environ.get("RECIPE", "custom-html") @pytest.fixture(scope="session") def app_domain(recipe) -> str: # Docker swarm config/secret names = __ must be <= 64 chars, and # stackname is the sanitized domain. ".ci.commoninternet.net" alone is 22 chars, so the # subdomain label must stay short. Use -<6hex(recipe|pr|ref)> — unique per run, # collision-safe across recipes (full recipe in the hash), readable context lives in the # Drone build params + PR comment. (Deviation from plan §4.0 long name; see DECISIONS.md.) pr = os.environ.get("PR", "0") ref = os.environ.get("REF", "local" + str(int(time.time()))) tag = _short(recipe, 4).lower() h = hashlib.sha1(f"{recipe}|{pr}|{ref}".encode()).hexdigest()[:6] return f"{tag}-{h}.ci.commoninternet.net" @pytest.fixture def deployed(recipe, app_domain, request): """Function-scoped: deploy the current/$REF version healthy, guaranteed teardown after. Used by stages that start from current (install/backup).""" version = os.environ.get("VERSION") or None lifecycle.janitor() request.addfinalizer(lambda: lifecycle.teardown_app(app_domain)) lifecycle.deploy_app(recipe, app_domain, version=version) lifecycle.wait_healthy(app_domain) return app_domain @pytest.fixture(scope="session") def deployed_app(recipe, app_domain): """Install stage: deploy the recipe and wait until healthy; tear down at session end.""" version = os.environ.get("VERSION") or None lifecycle.janitor() # sweep orphans from crashed runs first try: lifecycle.deploy_app(recipe, app_domain, version=version, secrets=True) lifecycle.wait_healthy(app_domain) yield app_domain finally: lifecycle.teardown_app(app_domain)