294a6ed023
Manifest now holds exactly one agent and one effective bottle (with git_user overlay already applied). The old multi-agent/bottle collection is renamed ManifestIndex. BottleSpec.manifest starts as ManifestIndex from the CLI and becomes Manifest after _validate() calls load_for_agent(); all provisioning code downstream reads spec.manifest.agent / spec.manifest.bottle instead of indexing by name.
152 lines
5.2 KiB
Python
152 lines
5.2 KiB
Python
"""Unit: shared backend prepare wiring.
|
|
|
|
These tests keep the base `BottleBackend.prepare` template honest:
|
|
backend-specific preflight/env hooks must be wired through, and launch
|
|
metadata must record the backend that actually prepared the plan.
|
|
"""
|
|
|
|
from __future__ import annotations
|
|
|
|
import tempfile
|
|
import unittest
|
|
from pathlib import Path
|
|
from unittest.mock import patch
|
|
|
|
from bot_bottle import bottle_state
|
|
from bot_bottle import supervise
|
|
from bot_bottle.backend import BottleSpec
|
|
from bot_bottle.backend.docker import DockerBottleBackend
|
|
from bot_bottle.backend.resolve_common import mint_slug
|
|
from bot_bottle.backend.smolmachines import SmolmachinesBottleBackend
|
|
from bot_bottle.manifest import ManifestIndex
|
|
|
|
|
|
def _manifest() -> ManifestIndex:
|
|
return ManifestIndex.from_json_obj({
|
|
"bottles": {
|
|
"dev": {
|
|
"env": {
|
|
"LITERAL_ENV": "literal-value",
|
|
"FORWARDED_ENV": "${HOST_SECRET_ENV}",
|
|
},
|
|
},
|
|
},
|
|
"agents": {
|
|
"demo": {
|
|
"bottle": "dev",
|
|
"skills": [],
|
|
"prompt": "hello",
|
|
},
|
|
},
|
|
})
|
|
|
|
|
|
def _spec(tmp: Path, *, identity: str) -> BottleSpec:
|
|
return BottleSpec(
|
|
manifest=_manifest(),
|
|
agent_name="demo",
|
|
copy_cwd=False,
|
|
user_cwd=str(tmp),
|
|
identity=identity,
|
|
)
|
|
|
|
|
|
class _FakeStateMixin:
|
|
def setUp(self) -> None:
|
|
self.tmp = tempfile.TemporaryDirectory(prefix="backend-prepare.")
|
|
self.root = Path(self.tmp.name) / ".bot-bottle"
|
|
self.original_root = supervise.bot_bottle_root
|
|
supervise.bot_bottle_root = lambda: self.root # type: ignore[assignment]
|
|
|
|
def tearDown(self) -> None:
|
|
supervise.bot_bottle_root = self.original_root # type: ignore[assignment]
|
|
self.tmp.cleanup()
|
|
|
|
|
|
class TestDockerPrepare(_FakeStateMixin, unittest.TestCase):
|
|
def test_records_backend_and_preserves_env_split(self) -> None:
|
|
backend = DockerBottleBackend()
|
|
spec = _spec(Path(self.tmp.name), identity="demo-docker")
|
|
|
|
with (
|
|
patch.dict("os.environ", {"HOST_SECRET_ENV": "secret-value"}),
|
|
patch(
|
|
"bot_bottle.backend.docker.resolve_plan.docker_mod.require_docker",
|
|
) as require_docker,
|
|
patch(
|
|
"bot_bottle.backend.docker.resolve_plan.docker_mod.runsc_available",
|
|
return_value=False,
|
|
),
|
|
):
|
|
plan = backend.prepare(spec, Path(self.tmp.name) / "stage")
|
|
|
|
require_docker.assert_called_once_with()
|
|
metadata = bottle_state.read_metadata("demo-docker")
|
|
self.assertIsNotNone(metadata)
|
|
assert metadata is not None
|
|
self.assertEqual("docker", metadata.backend)
|
|
self.assertEqual({"FORWARDED_ENV": "secret-value"}, plan.forwarded_env)
|
|
self.assertEqual("literal-value", plan.agent_provision.guest_env["LITERAL_ENV"])
|
|
self.assertNotIn("FORWARDED_ENV", plan.agent_provision.guest_env)
|
|
|
|
|
|
class TestSmolmachinesPrepare(_FakeStateMixin, unittest.TestCase):
|
|
def test_records_backend_and_builds_guest_env(self) -> None:
|
|
backend = SmolmachinesBottleBackend()
|
|
spec = _spec(Path(self.tmp.name), identity="demo-smol")
|
|
|
|
with (
|
|
patch.dict("os.environ", {"HOST_SECRET_ENV": "secret-value"}),
|
|
patch(
|
|
"bot_bottle.backend.smolmachines.resolve_plan.smolmachines_preflight",
|
|
) as preflight,
|
|
):
|
|
plan = backend.prepare(spec, Path(self.tmp.name) / "stage")
|
|
|
|
preflight.assert_called_once_with()
|
|
metadata = bottle_state.read_metadata("demo-smol")
|
|
self.assertIsNotNone(metadata)
|
|
assert metadata is not None
|
|
self.assertEqual("smolmachines", metadata.backend)
|
|
self.assertEqual("literal-value", plan.guest_env["LITERAL_ENV"])
|
|
self.assertEqual("secret-value", plan.guest_env["FORWARDED_ENV"])
|
|
self.assertEqual(
|
|
"/etc/ssl/certs/ca-certificates.crt",
|
|
plan.guest_env["SSL_CERT_FILE"],
|
|
)
|
|
|
|
|
|
class TestMintSlug(unittest.TestCase):
|
|
def _spec(self, *, label: str = "", identity: str = "") -> BottleSpec:
|
|
manifest = _manifest()
|
|
return BottleSpec(
|
|
manifest=manifest,
|
|
agent_name="demo",
|
|
copy_cwd=False,
|
|
user_cwd="/tmp",
|
|
label=label,
|
|
identity=identity,
|
|
)
|
|
|
|
def test_no_label_uses_agent_name_with_random_suffix(self) -> None:
|
|
slug = mint_slug(self._spec(label=""))
|
|
self.assertTrue(slug.startswith("demo-"), slug)
|
|
# random suffix present — slug is longer than just "demo"
|
|
self.assertGreater(len(slug), len("demo-"))
|
|
|
|
def test_label_becomes_exact_slug(self) -> None:
|
|
slug = mint_slug(self._spec(label="my-run"))
|
|
self.assertEqual("my-run", slug)
|
|
|
|
def test_label_with_spaces_slugified_no_suffix(self) -> None:
|
|
slug = mint_slug(self._spec(label="My Feature Run"))
|
|
self.assertEqual("my-feature-run", slug)
|
|
|
|
def test_identity_takes_precedence_over_label(self) -> None:
|
|
slug = mint_slug(self._spec(label="my-run", identity="fixed-id"))
|
|
self.assertEqual("fixed-id", slug)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
unittest.main()
|