"""SmolmachinesBottleBackend — the smolmachines implementation of BottleBackend (PRD 0023). Per PRD 0050 the per-provider provisioning steps (prompt, skills, the declarative provision-plan apply, supervise MCP registration) live on the `AgentProvider` plugin under `bot_bottle/contrib/`. The smolmachines backend only owns the steps that are about backend infrastructure: CA install (no-op for now), workspace, git copy-in.""" from __future__ import annotations from contextlib import contextmanager from pathlib import Path from typing import Generator, Sequence from ...agent_provider import AgentProvisionPlan from ...egress import EgressPlan from ...env import ResolvedEnv from ...git_gate import GitGatePlan from ...supervise import SupervisePlan from .. import ActiveAgent, BottleBackend, BottleSpec from . import cleanup as _cleanup from . import enumerate as _enumerate from . import launch as _launch from . import resolve_plan as _resolve_plan from . import smolvm as _smolvm from .bottle import SmolmachinesBottle from .bottle_cleanup_plan import SmolmachinesBottleCleanupPlan from .bottle_plan import SmolmachinesBottlePlan class SmolmachinesBottleBackend( BottleBackend["SmolmachinesBottlePlan", "SmolmachinesBottleCleanupPlan"] ): """smolmachines backend. Selected by `BOT_BOTTLE_BACKEND=smolmachines`.""" name = "smolmachines" @classmethod def is_available(cls) -> bool: """`smolvm` on PATH. The backend additionally needs macOS for libkrun + TSI, but `enumerate_active` / `cleanup` are host-shell ops that gracefully no-op on Linux too — the runtime check happens at `prepare`.""" return _smolvm.is_available() def _preflight(self) -> None: _resolve_plan.preflight() def _build_guest_env(self, resolved_env: ResolvedEnv) -> dict[str, str]: return _resolve_plan.build_guest_env(resolved_env) def _resolve_plan( self, spec: BottleSpec, *, slug: str, resolved_env: ResolvedEnv, agent_provision_plan: AgentProvisionPlan, egress_plan: EgressPlan, git_gate_plan: GitGatePlan, supervise_plan: SupervisePlan | None, stage_dir: Path, ) -> SmolmachinesBottlePlan: return _resolve_plan.resolve_plan( spec, slug=slug, resolved_env=resolved_env, agent_provision_plan=agent_provision_plan, egress_plan=egress_plan, supervise_plan=supervise_plan, git_gate_plan=git_gate_plan, stage_dir=stage_dir, ) @contextmanager def launch( self, plan: SmolmachinesBottlePlan ) -> Generator[SmolmachinesBottle, None, None]: with _launch.launch(plan, provision=self.provision) as bottle: yield bottle def supervise_mcp_url(self, plan: SmolmachinesBottlePlan) -> str: """The smolmachines guest reaches the supervise sidecar via a host-published random port the launch step pinned earlier (`http://:/`). `agent_supervise_url` on the plan is "" when the bottle has no sidecar.""" return plan.agent_supervise_url def prepare_cleanup(self) -> SmolmachinesBottleCleanupPlan: return _cleanup.prepare_cleanup() def cleanup(self, plan: SmolmachinesBottleCleanupPlan) -> None: _cleanup.cleanup(plan) def enumerate_active(self) -> Sequence[ActiveAgent]: return _enumerate.enumerate_active()