Files
orchestrator/src/build_cache_pruner.py
claude-bot 664c2e945a feat(infra): auto-prune docker build cache on mva154 (ORCH-062)
Add src/build_cache_pruner.py — a background daemon thread modelled 1:1 on
src/disk_watchdog.py that periodically runs STRICTLY `docker builder prune -f
--filter until=<until>` (BuildKit GC) on the HOST over ssh. It is the "second
half" of the disk-watchdog (ORCH-063): the watchdog signals, the pruner cleans.
Removes the root cause of the 07.06.2026 incident (build cache ~11GB -> disk
100% -> whole self-hosting pipeline down) automatically, без оператора.

ADR-001 (Variant A): host-over-ssh, same channel as image_freshness/self_deploy
(no docker CLI in the image). Touches ONLY the build cache — no image/system
prune, no image/container removal, never restarts the docker daemon or the prod
container (self-hosting safety). No ssh target -> tick is a no-op.

- src/config.py: ORCH_BUILD_CACHE_PRUNE_* flags + defensive validators
  (interval/timeout >0, until ~ ^\d+[smhdw]?$, notify_min_gb >=0 -> safe default).
- src/main.py: start last (after disk_watchdog) / stop first in lifespan;
  additive read-only build_cache_prune block in GET /queue.
- never-raise on two levels (per-command + per-tick); kill-switch
  ORCH_BUILD_CACHE_PRUNE_ENABLED (false -> daemon does not start, 1:1 as before).
- STAGE_TRANSITIONS / QG_CHECKS / check_* / _parse_* / DB schema UNCHANGED;
  last-run/last-result is in-memory (no migration).
- tests/test_build_cache_pruner.py: TC-01..TC-12 (23 cases, docker fully mocked).
- .env.example + CHANGELOG.md updated; INFRA.md / architecture docs already
  carry the component (architecture stage).

Refs: ORCH-062

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-09 19:55:00 +03:00

352 lines
15 KiB
Python

"""ORCH-062: build-cache-pruner — periodic ``docker builder prune`` on the host.
On 07.06.2026 the mva154 host disk silently grew to 100% and took down the WHOLE
self-hosting pipeline of every project. The dominant consumer was the **docker
build cache** (~11 GB accumulated by frequent rebuilds: ``docker compose up
--build`` on prod deploy, the ``--profile staging`` rebuild, the build-once retag
behind ``check_staging_image_fresh``). ORCH-063 added the disk-watchdog, which
only **signals** (Telegram alert at >=85%) and explicitly deferred the cleanup to
this task. **This module is that cleanup: the watchdog signals — the pruner
cleans.**
It is a background daemon thread modelled **1:1 on** ``src/disk_watchdog.py``
(``threading.Thread(daemon=True)`` + ``threading.Event`` for a clean stop, the
``start()`` / ``stop(timeout)`` / ``status()`` contract, a ``/queue`` snapshot,
per-tick never-raise and a kill-switch ``ORCH_BUILD_CACHE_PRUNE_ENABLED``). Each
tick runs **strictly** ``docker builder prune -f --filter until=<until>`` (BuildKit
GC) on the **host over ssh** — the prod container ships no docker CLI, only
``openssh-client`` (``Dockerfile:11``), so docker operations run over ssh on the
host, the same channel ``image_freshness``/``self_deploy`` already use.
Invariants (TRZ §5/§6 / ADR-001 D2/D6):
* The command touches **only** the BuildKit build cache. There is NO
``docker image prune``, NO ``docker system prune``, no image/container removal
of running services and no container stop/restart. The prod ``orchestrator``
container is NEVER restarted (self-hosting blast radius). ``-a/--all`` is only
ever added **paired with** the ``until`` age filter — never a bare
"nuke everything".
* ``STAGE_TRANSITIONS`` / ``QG_CHECKS`` / ``check_*`` / ``_parse_*`` /
``src/stage_engine.py`` / the DB schema are UNCHANGED — the pruner is an
operational daemon, not a Quality Gate (like ``reconciler`` / ``job_reaper`` /
``disk_watchdog``). No new migration (last-run / last-result is in-memory,
best-effort, may reset on restart — safe: at worst one extra safe prune).
* never-raise on two levels: per-command (non-zero rc / timeout / ``OSError`` /
no ssh target / output-parse error -> logged and swallowed, the tick lives)
and per-tick (outer ``try/except`` in ``_run``, like ``disk_watchdog._run``).
The background loop and the pipeline never fall over.
* No ssh target configured (``deploy_ssh_host`` empty) -> the tick is a no-op
(logged, reflected in ``status().last_error``). This scopes the feature to the
self-hosting prod (where ssh is configured) and makes the default safe in any
environment without host access — parallel to how ``self_deploy`` /
``image_freshness`` degrade without a target.
* Kill-switch ``build_cache_prune_enabled=False`` -> the daemon does not start
(``main.lifespan`` guard + ``start()`` guard) and ``/queue`` returns
``{"enabled": false, ...}`` — behaviour 1:1 as before the task.
See docs/work-items/ORCH-062/06-adr/ADR-001-build-cache-pruner.md and the
cross-cutting docs/architecture/adr/adr-0025-build-cache-pruner.md.
"""
import logging
import re
import shlex
import subprocess
import threading
import time
from .config import settings
from .notifications import send_telegram
logger = logging.getLogger("orchestrator.build_cache_pruner")
_BYTES_PER_GB = 1024 ** 3
# Multipliers for the "Total reclaimed space: <n><unit>" line emitted by
# `docker builder prune`. Decimal units are base-1000 (docker's HumanSize),
# the *i* binary units base-1024. Best-effort — only used for observability /
# the optional notify threshold, never for a decision.
_SIZE_UNITS = {
"B": 1,
"KB": 1000, "MB": 1000 ** 2, "GB": 1000 ** 3, "TB": 1000 ** 4,
"KIB": 1024, "MIB": 1024 ** 2, "GIB": 1024 ** 3, "TIB": 1024 ** 4,
}
_RECLAIMED_RE = re.compile(
r"Total reclaimed space:\s*([\d.]+)\s*([KMGT]?i?B)", re.IGNORECASE
)
def decide_prune(prev_run_ts: float | None, now: float, interval_s: float) -> bool:
"""Pure decision (anti-frequency, NFR-4): should this tick prune?
Returns ``True`` when no prune has run yet (``prev_run_ts is None``) or at
least ``interval_s`` seconds have elapsed since the last attempt; ``False``
otherwise. Testable without a thread or a real timer (TC-01/TC-02). A
non-positive / unusable ``interval_s`` falls open to ``True`` (prune) — the
config validator already guards the value, this is belt-and-braces.
"""
if prev_run_ts is None:
return True
try:
return (now - prev_run_ts) >= interval_s
except TypeError: # pragma: no cover - defensive, inputs are numbers
return True
def _ssh_target() -> str | None:
"""ssh ``user@host`` for the host prune, or ``None`` when no host is
configured (tests / non-self contexts). Mirrors ``image_freshness._ssh_target``.
"""
host = (settings.deploy_ssh_host or "").strip()
if not host:
return None
user = (settings.deploy_ssh_user or "").strip()
return f"{user}@{host}" if user else host
def build_prune_command(
ssh_target: str, until: str, prune_all: bool = False
) -> list[str]:
"""Build the ssh command that runs ``docker builder prune`` on the host.
The remote is **strictly** ``docker builder prune -f`` (BuildKit GC), with the
age filter ``--filter until=<until>`` appended whenever ``until`` is set so the
warm recent cache is kept (BR-2/AC-2), and ``-a`` added **only** when
``prune_all`` is set — always paired with the age filter (D2). It NEVER emits
``docker image prune`` / ``docker system prune`` / any image/container removal
(BR-3/AC-3). The ``until`` value is ``shlex.quote``-d for the remote shell.
"""
remote = "docker builder prune -f"
if prune_all:
remote += " -a"
if until:
remote += " --filter until=" + shlex.quote(until)
return ["ssh", "-o", "StrictHostKeyChecking=no", ssh_target, remote]
def parse_reclaimed(output: str) -> int | None:
"""Best-effort parse of ``Total reclaimed space: <n><unit>`` -> bytes.
Returns the reclaimed size in bytes, or ``None`` when the line is absent /
unparseable (FR-4: observability is best-effort, never a decision). Never
raises.
"""
try:
m = _RECLAIMED_RE.search(output or "")
if not m:
return None
value = float(m.group(1))
unit = m.group(2).upper()
mult = _SIZE_UNITS.get(unit)
if mult is None:
return None
return int(value * mult)
except Exception as e: # noqa: BLE001 - parsing is best-effort
logger.warning("build-cache-pruner: cannot parse reclaimed space: %s", e)
return None
class BuildCachePruner:
"""Background daemon running ``docker builder prune`` on the host on a period.
Modelled on ``DiskWatchdog``: a ``threading.Thread(daemon=True)`` + a
``threading.Event`` for a clean stop. The only in-memory state is the
best-effort ``last_run_ts`` / ``_last_reclaimed`` / ``_last_error`` — all reset
on restart, which is safe (at worst one extra safe prune; D6).
``now_provider`` is injectable so the anti-frequency decision is testable
deterministically without a real timer.
"""
def __init__(self, interval_s: float | None = None, now_provider=None):
self.interval_s = (
interval_s
if interval_s is not None
else settings.build_cache_prune_interval_s
)
self._now = now_provider or time.time
self._stop = threading.Event()
self._thread: threading.Thread | None = None
# Best-effort in-memory state (no DB row, no migration).
self.last_run_ts: float | None = None
self._last_reclaimed: int | None = None
self._last_reclaimed_human: str | None = None
self._last_error: str | None = None
# -- config helpers ----------------------------------------------------
@property
def _until(self) -> str:
return settings.build_cache_prune_until
@property
def _all(self) -> bool:
return settings.build_cache_prune_all
@property
def _timeout_s(self) -> int:
return settings.build_cache_prune_timeout_s
@property
def _notify_min_gb(self) -> float:
return settings.build_cache_prune_notify_min_gb
# -- tick --------------------------------------------------------------
def tick(self) -> None:
"""One pass: prune if the anti-frequency window has elapsed (never-raise).
Runs the pure ``decide_prune`` against the injected clock; on a PRUNE
decision it performs the host prune (``_prune``), which is itself
never-raise. A SKIP decision leaves all state untouched.
"""
now = self._now()
if not decide_prune(self.last_run_ts, now, self.interval_s):
return
self._prune(now)
def _prune(self, now: float) -> None:
"""Run ``docker builder prune`` on the host over ssh. Never raises (AC-4).
Records the attempt time (``last_run_ts``) up front so the anti-frequency
window advances even when the command fails or there is no ssh target.
Every failure mode — no target, timeout, non-zero rc, ``OSError`` — is
logged, stored in ``_last_error`` and swallowed; the loop stays alive.
"""
self.last_run_ts = now
target = _ssh_target()
if not target:
self._last_error = "no ssh host configured (deploy_ssh_host empty)"
logger.info("build-cache-pruner: %s — tick is a no-op", self._last_error)
return
cmd = build_prune_command(target, self._until, self._all)
try:
r = subprocess.run(
cmd, capture_output=True, text=True, timeout=self._timeout_s
)
except subprocess.TimeoutExpired:
self._last_error = f"timeout after {self._timeout_s}s"
logger.warning("build-cache-pruner: prune %s", self._last_error)
return
except (subprocess.SubprocessError, OSError) as e:
self._last_error = f"ssh/subprocess error: {e}"
logger.warning("build-cache-pruner: %s", self._last_error)
return
if r.returncode != 0:
self._last_error = (
f"rc={r.returncode}: {(r.stderr or '').strip()[:200]}"
)
logger.warning("build-cache-pruner: prune %s", self._last_error)
return
# Success: parse the best-effort reclaimed size and clear the error.
self._last_error = None
reclaimed = parse_reclaimed(r.stdout or "")
self._last_reclaimed = reclaimed
self._last_reclaimed_human = self._format_reclaimed(reclaimed)
logger.info(
"build-cache-pruner: pruned host build cache (until=%s, all=%s), "
"reclaimed=%s",
self._until, self._all, self._last_reclaimed_human or "unknown",
)
self._maybe_notify(reclaimed)
@staticmethod
def _format_reclaimed(reclaimed: int | None) -> str | None:
"""Human GB label for a reclaimed byte count (best-effort, never raises)."""
if reclaimed is None:
return None
try:
return f"{reclaimed / _BYTES_PER_GB:.2f} GB"
except Exception: # noqa: BLE001 - observability only
return None
def _maybe_notify(self, reclaimed: int | None) -> None:
"""Telegram when reclaimed >= ``notify_min_gb`` (>0 to enable). Never raises."""
try:
min_gb = self._notify_min_gb
if not min_gb or min_gb <= 0 or reclaimed is None:
return
gb = reclaimed / _BYTES_PER_GB
if gb < min_gb:
return
self._send(
f"\U0001f9f9 build-cache-pruner: освобождено {gb:.2f} ГБ "
f"docker build cache на хосте (until={self._until})."
)
except Exception as e: # noqa: BLE001 - notify is best-effort
logger.warning("build-cache-pruner: notify decision failed: %s", e)
def _send(self, text: str) -> None:
"""Send a Telegram message (notifying). Never raises (best-effort)."""
try:
send_telegram(text)
except Exception as e: # noqa: BLE001 - delivery is best-effort
logger.warning("build-cache-pruner: telegram send failed: %s", e)
# -- loop / lifecycle --------------------------------------------------
def _tick(self) -> None:
try:
self.tick()
except Exception as e: # noqa: BLE001 - inner never-raise
logger.error("build-cache-pruner: tick error: %s", e)
def _run(self) -> None:
logger.info(
"BuildCachePruner started (interval=%ss, until=%s, all=%s, "
"timeout=%ss, enabled=%s)",
self.interval_s, self._until, self._all, self._timeout_s,
settings.build_cache_prune_enabled,
)
while not self._stop.is_set():
try:
self._tick()
except Exception as e: # noqa: BLE001 - outer never-raise
logger.error("BuildCachePruner loop error: %s", e)
self._stop.wait(self.interval_s)
logger.info("BuildCachePruner stopped")
def start(self) -> None:
"""Start the daemon thread (idempotent: a live thread is a no-op).
Honours the kill-switch: ``build_cache_prune_enabled=False`` -> no-op (the
daemon never starts; ``main.lifespan`` also guards, AC-5/TC-07).
"""
if not settings.build_cache_prune_enabled:
return
if self._thread and self._thread.is_alive():
return
self._stop.clear()
self._thread = threading.Thread(
target=self._run, name="build-cache-pruner", daemon=True
)
self._thread.start()
def stop(self, timeout: float = 5.0) -> None:
self._stop.set()
if self._thread:
self._thread.join(timeout=timeout)
def status(self) -> dict:
"""Build-cache-pruner snapshot for /queue observability (FR-4/AC-7).
Never raises — returns a minimal ``{"enabled": ...}`` on any error.
"""
try:
return {
"enabled": settings.build_cache_prune_enabled,
"interval_s": self.interval_s,
"until": self._until,
"all": self._all,
"last_run_ts": self.last_run_ts,
"last_reclaimed_bytes": self._last_reclaimed,
"last_reclaimed": self._last_reclaimed_human,
"last_error": self._last_error,
}
except Exception as e: # noqa: BLE001 - observability must never raise
logger.warning("build-cache-pruner: status() failed: %s", e)
return {"enabled": settings.build_cache_prune_enabled}
# Module-level singleton used by the FastAPI lifespan.
build_cache_pruner = BuildCachePruner()