Files
bot-bottle/docs/research/remote-docker-vm-isolation.md
didericis-codex cdb1870b1c
test / unit (pull_request) Successful in 29s
test / integration (pull_request) Successful in 43s
docs(agent): clarify claude oauth env
2026-05-28 18:20:09 -04:00

14 KiB
Raw Permalink Blame History

Remote Docker VM as an isolation upgrade for bot-bottle

Note on the cheapest practical path to stronger isolation than local Docker: run bot-bottle unchanged on a remote Linux VM that has dockerd. Complements stronger-isolation-alternatives.md (which surveys runtime swaps like gVisor, Kata, Firecracker, Apple Container) and local-vs-remote-agent-execution.md (which surveys the local-vs-remote decision broadly).

Summary

If the goal is "stronger isolation than Docker-on-my-laptop without rewriting the runtime," the cleanest answer is to keep bot-bottle exactly as it is and run it on a remote Linux VM where you can install dockerd. The v1 design — pipelock as a separate container on a --internal network, ephemeral agent containers, OAuth-token forwarding — works as-is. The only thing that changes is that the "host" is now a disposable VM you provisioned for the session, not your laptop.

This is structurally equivalent to a Firecracker rewrite (Rung 3 in stronger-isolation-alternatives.md), but the cloud provider operates the runtime for you. It is also strictly cheaper than adopting a cloud sandbox SDK (Vercel Sandbox, E2B, Cloudflare Sandbox SDK) because you keep the existing Docker-shaped abstractions instead of swapping them for a vendor API.

The argument

What changes in the threat model

The agent's blast radius shrinks from "developer laptop + everything on the LAN" to "this disposable VM." Concretely, what's no longer reachable on container escape:

  • ~/.ssh/, ~/.aws/credentials, ~/.config/gh, the macOS Keychain
  • Browser cookies and session state
  • Other dev machines on the home/office LAN
  • NAS, printers, smart-home devices, anything else on the local network

What replaces it on the remote side: only what the operator chose to ship to the VM for the session. Typically the OAuth token, optional SSH keys for the bottle, the manifest, and the workspace if the agent needs one. None of which are on the laptop after the VM is destroyed.

Why the boundary is equivalent to v1, not weaker

A natural objection — raised in the design discussion that produced this note — is that running pipelock and the agent on the same VM collapses a network boundary into a kernel-namespace boundary, which sounds weaker. It is not, if you reuse Docker for the inner topology.

Docker on the remote VM gives the agent and pipelock their own network namespaces by default, with the agent attached to a --internal network and pipelock straddling it and an egress bridge. That is the same v1 topology. Bypassing pipelock from the agent requires the same class of attack as bypassing it on a laptop: a kernel-level netns escape inside the VM. The only difference is that the kernel under attack belongs to a disposable VM, not the developer's machine.

In other words: the "weaker because colocated" framing only applies if you naively run agent and pipelock as two processes in the same namespace. With Docker on the VM, you don't.

Why this is cheaper than the alternatives

Path Effort Where the VM-grade boundary comes from
gVisor (runsc) per bottle ~12 days Userspace syscall barrier; not a full VM
Kata Containers per bottle ~12 days, Linux-only Kata's microVM-per-container
Firecracker rewrite 24 weeks Self-operated Firecracker
Apple Container (macOS) ~1 week spike + integration Apple's Virtualization.framework, per-container
Cloud sandbox SDK (Vercel, E2B, …) Daysweeks of API rewrite + lock-in Provider-operated Firecracker / equivalent
Remote Docker VM (this note) 0 lines of code Cloud-provider hypervisor under the VM

The "stronger isolation alternatives" doc concludes that gVisor is the right today-step and Apple Container is probably the right v2. This note adds a third option that sits orthogonal to both: don't change the runtime, change the host. Use it when the failure mode you care about is "agent compromises my laptop" specifically, rather than "agent escapes Docker into a kernel I share with other workloads."

What the provider has to give you

Not every cloud sandbox is suitable. The minimum for this approach to work:

  • Root or rootless-Docker capability inside the VM. Rules out Fargate-style locked-down container hosts and most "function" tier FaaS. Verify before committing — Vercel Sandbox specifically may or may not allow installing dockerd depending on tier; Fly Machines, EC2, GCE, Hetzner, Linode, and self-hosted hypervisors give you full control.
  • Enough disk + RAM to host the bot-bottle image, the agent container, and the pipelock sidecar. Headroom of ~24 GB RAM and ~5 GB disk is comfortable; less works for short sessions.
  • An interactive reach path. SSH is fine. The launcher uses docker exec -it, so any TTY-capable session works.

What you give up

  • Typing latency. Interactive Claude sessions over SSH have visible per-keystroke latency; usually fine on wired/fiber, less fine on Wi-Fi-to-cloud. Mosh helps if it's bothersome.
  • Token shipping. BOT_BOTTLE_CLAUDE_OAUTH_TOKEN has to live on the remote box for the launcher to forward it into containers. Use the provider's secret-injection path (cloud-init user-data, flyctl secrets, Tailscale-served local file, etc.). Never echo the token onto the SSH command line; it ends up in the local shell history and possibly the SSH server's auth log.
  • Idle cost. Unless the VM is torn down between sessions, you pay for it sitting idle. Ephemeral provisioning (one VM per session, destroyed on exit) is the cheaper and more secure pattern; see local-vs-remote-agent-execution.md on why ephemeral is also recommended for credential-concentration reasons.
  • Source code goes to the VM. Same as any remote-execution topology. If the project is under NDA, the VM provider matters.
  • Provider trust. Multi-tenancy side channels, supply-chain compromise of the provider, insider risk. Generally smaller than laptop-kernel-CVE risk, but the failure mode (provider-wide breach) is correlated across all your sandboxes.

Operational shape

The minimum-viable workflow, no bot-bottle code changes:

  1. terraform apply / flyctl machine run / gcloud compute instances create — provision a fresh Linux VM.
  2. Install dockerd via the provider's image or a one-liner (curl -fsSL https://get.docker.com | sh).
  3. SSH in.
  4. git clone bot-bottle on the VM, drop a manifest in place, inject BOT_BOTTLE_CLAUDE_OAUTH_TOKEN via the provider's secrets path.
  5. ./cli.py start <agent> — the existing launcher handles the rest.
  6. On exit: destroy the VM. No host artifacts persist.

For the "VPN pivot" failure mode, see local-vs-remote-agent-execution.md. Short version: never VPN the remote VM back to your LAN. If the agent needs LAN resources, expose those through a narrow API instead.

Case study: Fly Machines

Fly.io's Machines product is a useful concrete worked example because it satisfies all the provider requirements (root, Firecracker-backed isolation, scriptable lifecycle, per-second billing) and surfaces the gotchas the abstract pattern leaves implicit.

Image strategy

Build a custom OCI image FROM docker:dind that bakes in:

  • The bot-bottle repository checkout.
  • A pre-built bot-bottle-claude:latest image, saved via docker save on your laptop and loaded in at image-build time (RUN docker load < bot-bottle.tar) or pushed as a layer into the dind storage. Without this step, the first in-VM docker build runs apt-get and a global npm install -g @anthropic-ai/claude-code, which adds 3090 s to every cold start.
  • A flyctl secrets-injected BOT_BOTTLE_CLAUDE_OAUTH_TOKEN, exposed to the VM's PID 1 as an env var.
  • An entrypoint that starts dockerd, waits for it to be healthy, then either drops into a shell or directly runs cli.py start <agent>.

Deploy with flyctl deploy or flyctl machine run --image ….

Boot-to-first-prompt timing

Three scenarios, all assuming the custom image above (bot-bottle image baked in, token injected, no in-VM rebuild):

Phase Cold (image not cached on Fly host) Warm (image cached, machine run fresh) Hot (machine stopped, machine start)
Fly schedule + image fetch 1030 s 23 s ~1 s
Firecracker kernel boot ~1 s ~1 s ~1 s (resume)
dockerd-in-VM startup 24 s 24 s 0 s (already running)
cli.py start <agent> housekeeping (network creates, pipelock sidecar, agent container, skill copy) 46 s 46 s 46 s
Claude prints first prompt 13 s 13 s 13 s
End-to-end ~2045 s ~1017 s ~711 s

For interactive sessions the warm path is the realistic baseline once the custom image is registered. The hot path trims only a few extra seconds — the question of whether to keep stopped Machines on standby is mostly about cost, not speed.

Cost of standby vs. create-per-session

Stopped Fly Machines stop billing CPU/RAM but continue to bill for storage and any allocated IPv4. A reasonable bot-bottle Machine size (2 vCPU / 2 GB / ~3 GB rootfs) costs roughly:

Item While stopped Monthly
CPU + RAM not billed $0
Rootfs storage ~$0.15/GB-month ~$0.45
Dedicated IPv4 (if allocated) $2/month flat $2.00
Dedicated IPv6 free $0
Bandwidth usage-based $0

So roughly $0.50$2.50/month per standby Machine, with the IPv4 line dominating. Drop the dedicated v4 (use IPv6 or Fly's shared v4 via WireGuard) and standby falls under $1/month.

For comparison, running the same Machine 24/7 lands in the $15$40/month range depending on size, and the create-and-destroy pattern (one Machine per session, destroyed on exit) is effectively $0 since you only pay for the seconds it ran.

Practical pattern

Two reasonable workflows, plus one that's tempting but worse:

  1. Pure ephemeral. flyctl machine run at session start, flyctl machine destroy on exit. ~2045 s cold start, $0 idle. Maximally isolated; nothing persists between sessions. Best fit when sessions are infrequent or when state continuity across sessions is itself a concern.
  2. Standby pool. A small fleet of pre-built Machines that get started fresh and destroyed (or wiped) per session. The Machine identity is short-lived but the image is pre-cached on Fly's hosts, keeping warm-path latency at ~1017 s. ~$0.50$1/month per Machine in the pool without dedicated v4.
  3. Permanently stopped Machine, just start/stop. Saves a few extra seconds (~711 s hot) but is the weakest of the three on the isolation axis — the rootfs persists across sessions, so anything a previous session wrote is still there. Avoid unless the saved seconds matter more than the state-continuity concern.

Fly-specific caveats

  • DinD requires kernel features. Fly Machines historically had some namespacing quirks for nested Docker; verify on a smoke-test Machine before committing. The pattern is supported (Fly's own Remote Builders use it), but kernel/runtime updates have shifted the requirements over time.
  • The launcher's interactive y/N preflight blocks automated remote start. cli.py start waits on /dev/tty. For an automated entry point you need to pipe y\n into stdin, drive it from a pty, or add a --yes/--non-interactive flag (a small patch). The --remote=user@host ergonomics direction below would handle this in passing.
  • Pricing has been re-tariffed multiple times. The structure (per-second compute, GB-month storage, $2/v4) has been stable; specific rates may have moved. Verify against fly.io/docs/about/pricing before committing numbers to any planning doc.

Optional ergonomics direction

A future addon — not architecturally necessary, just nicer:

  • cli.py start --remote=user@host <agent> that:
    • rsyncs the manifest and (optionally) cwd to the remote
    • SSHes in with the OAuth token forwarded via SendEnv
    • runs cli.py start <agent> on the remote
    • forwards the TTY for the interactive session
    • on exit, optionally tears down the remote VM via a provider hook (flyctl machine destroy, terraform destroy, etc.)

This is roughly a day of work and would make the remote pattern feel like a single launcher invocation. It is the only piece of remote support that would benefit from being upstreamed; everything else is operator workflow.

Recommendation

For users who want stronger isolation than local Docker without rewriting the runtime, this is probably the right answer. Cleaner than gVisor (which only adds a syscall barrier on the same kernel), cleaner than a Firecracker rewrite (which is weeks of work), cleaner than adopting a cloud-sandbox SDK (which trades the v1 design for a vendor API). The pre-existing local-vs-remote-agent-execution.md decision heuristics still apply for whether this is worth the operational overhead in any given setting.

If we wanted to land this as a real project direction:

  1. Add a short "Running bot-bottle on a remote Docker VM" section to the README pointing at this doc.
  2. Optionally: prototype the --remote=user@host launcher subcommand.
  3. Update stronger-isolation-alternatives.md to mention the remote Docker VM as a fourth path, since the survey is otherwise incomplete.

Caveats

  • "Just install Docker" isn't free on every provider; some lock down what kernel modules and caps the VM has. Spike-test before committing.
  • Multi-tenant cloud hypervisors (EC2, GCE, Vercel) have their own side-channel and supply-chain risk surfaces, separately bounded from the laptop-kernel risk this approach addresses.
  • The remote-VM topology still does not protect source code or secrets from the cloud provider — it protects them from a kernel exploit reaching the developer's laptop. Different fear, different fix.
  • Research conducted 2026-05-10.