"""start: boot a sandboxed container for a named agent and attach an interactive claude-code session. The container is torn down when the session ends. The launch core is shared with `cli.py resume `: see _launch_bottle below. """ from __future__ import annotations import argparse import os import shutil import sys import tempfile from pathlib import Path from ..backend import BottleSpec, get_bottle_backend from ..backend.docker.bottle_state import ( cleanup_state, is_preserved, mark_preserved, ) from ..backend.docker.capability_apply import snapshot_transcript from ..log import info from ..manifest import Manifest from ._common import PROG, USER_CWD, read_tty_line def cmd_start(argv: list[str]) -> int: parser = argparse.ArgumentParser(prog=f"{PROG} start", add_help=True) parser.add_argument("--dry-run", action="store_true") parser.add_argument("--cwd", action="store_true", help="copy host cwd into a derived image") parser.add_argument("--remote-control", action="store_true") parser.add_argument("name", help="agent name defined in claude-bottle.json") args = parser.parse_args(argv) dry_run = args.dry_run or os.environ.get("CLAUDE_BOTTLE_DRY_RUN") == "1" manifest = Manifest.resolve(USER_CWD) spec = BottleSpec( manifest=manifest, agent_name=args.name, copy_cwd=args.cwd, user_cwd=USER_CWD, ) return _launch_bottle( spec, dry_run=dry_run, remote_control=args.remote_control, ) def _launch_bottle( spec: BottleSpec, *, dry_run: bool, remote_control: bool, ) -> int: """Shared launch core for `start` and `resume`. Builds the plan, prints / dry-runs / prompts as appropriate, brings the bottle up, attaches claude, and prints the resume hint on session end.""" stage_dir = Path(tempfile.mkdtemp(prefix="claude-bottle-stage.")) try: backend = get_bottle_backend() plan = backend.prepare(spec, stage_dir=stage_dir) plan.print(remote_control=remote_control) if dry_run: info("dry-run requested; not starting container.") return 0 sys.stderr.write("claude-bottle: launch this agent? [y/N] ") sys.stderr.flush() reply = read_tty_line() if reply not in ("y", "Y", "yes", "YES"): info("aborted by user") return 0 identity = _identity_from_plan(plan) with backend.launch(plan) as bottle: info( "attaching interactive claude session " "(Ctrl-D or 'exit' to leave; container will be removed)" ) claude_args = ["--dangerously-skip-permissions"] if remote_control: claude_args.append("--remote-control") exit_code = bottle.exec_claude(claude_args, tty=True) info( f"session ended (exit {exit_code}); " f"container {bottle.name} will be removed" ) # While the container is still alive: always snapshot the # transcript and — if the agent exited non-zero — mark # the state for preservation. Capability-block already # did both before triggering teardown from the dashboard; # this picks up crashes / Ctrl-Cs / OOM kills the same # way. snapshot_transcript is best-effort so the # capability-block path's prior snapshot isn't clobbered # when the container is already gone. _capture_session_state(identity, exit_code) # Context exited → containers + networks gone. Now decide # what to do with the per-bottle state dir on the host: any # preserve marker (capability-block OR crash) keeps it; a # clean exit cleans it up so ~/.claude-bottle/state/ doesn't # accumulate per-launch debris. _settle_state(identity) return 0 finally: shutil.rmtree(stage_dir, ignore_errors=True) def _capture_session_state(identity: str, exit_code: int) -> None: """Inside the launch context, while the container is still alive: snapshot the transcript and mark for preservation if claude crashed. Pure-function-ish; tests stub the helpers.""" if not identity: return snapshot_transcript(identity) if exit_code != 0: mark_preserved(identity) def _settle_state(identity: str) -> None: if not identity: return if is_preserved(identity): info(f"to resume this bottle: ./cli.py resume {identity}") return cleanup_state(identity) def _identity_from_plan(plan: object) -> str: """Backend-specific: the docker plan exposes the identity as `.slug`. Other backends in the future would expose their own identity attribute; for now we duck-type to keep this layer backend-agnostic.""" return getattr(plan, "slug", "")