PRD 0001 cli.sh integration: - Source the new lib/network.sh and lib/pipelock.sh. - During plan resolution: generate the per-bottle pipelock YAML into the existing mktemp stage dir (mode 600, hostnames only) and resolve a one-line "<N> hosts allowed (...)" summary. - Add the egress summary as a sub-bullet under the bottle in the y/N preflight, alongside the existing ssh hosts line. - After the y/N gate (and after build_image): create the per-agent --internal Docker network with a slug-derived name, then start the pipelock sidecar attached to it. - docker run argv: agent attaches to the internal network with HTTPS_PROXY / HTTP_PROXY pointing at the sidecar by service name on that network. NO_PROXY only covers loopback. The internal network has no default gateway, so any path that ignores the proxy env hits no-route-to-host rather than leaking. - Exit trap: tear down the agent container, then the sidecar (so the network is empty), then remove the network, then run the existing stage cleanup. Order matters — docker refuses to remove a network with attached containers. - --dry-run continues to exit before any docker network/run/cp/exec call; the YAML write into the mktemp dir is the only new side-effect inside the dry-run path. Verified against a temp fixture: defaults-only bottle shows "7 hosts allowed", a bottle with two extra entries shows "9 hosts allowed (api.anthropic.com, api.openai.com, claude.ai, +6 more)", and dry-run exits before any docker calls. Refs: docs/prds/0001-per-agent-egress-proxy-via-pipelock.md Assisted-by: Claude Code
claude-bottle
Spins up an isolated container for running Claude Code with a curated set of skills and env vars.
Why "claude-bottle"?
Each container is a bottle; Claude is the genie inside. The genie has
broad powers within the bottle — read, write, run anything — but it
cannot escape to the host. You uncork one bottle per agent
(./cli.sh start <agent>), many bottles run in parallel, and each
one's powers are scoped to what the manifest grants it: a curated set
of skills, env vars, and a starting prompt. When the session ends the
bottle is destroyed and the genie does not persist.
Goals
- Minimize risk of running claude with full permissions
- Allow me to easily spin up agent tasks in parallel
- Create isolated, well defined, easily updated, shareable agents
Non-goals
- Communicating between agents directly
- Self hosted VMs (v1 uses local Docker containers, not VMs)
- Advanced agent auditing (lean on git history for auditing)
Quickstart
Requires Docker on the host and a long-lived Claude Code OAuth token in your shell env.
./cli.sh start <agent> # builds the image on first run, drops you into claude
The container is removed automatically when the session ends. If the script
is killed with SIGKILL the exit trap won't fire and the container may be
left running; remove it with docker rm -f <container-name>.
Auth: OAuth token, not API key
claude-bottle authenticates claude inside the container with the same
Pro/Max subscription you already use on the host, via a long-lived OAuth
token. No ANTHROPIC_API_KEY is needed.
Why a token instead of mounting ~/.claude.json: on macOS, Claude
Code stores OAuth credentials in the encrypted Keychain, not in
~/.claude.json. Mounting that file into a Linux container does not
carry the credentials with it. Linux hosts keep credentials in
~/.claude/.credentials.json, but to keep the launcher portable
claude-bottle uses the env-var path on every host.
One-time setup on the host:
claude setup-token # browser login, prints a ~1-year OAuth token
Stash the token in your shell env (e.g. ~/.zshrc or a secret manager)
as CLAUDE_BOTTLE_OAUTH_TOKEN:
export CLAUDE_BOTTLE_OAUTH_TOKEN="<token>"
cli.sh automatically forwards it to every container as
CLAUDE_CODE_OAUTH_TOKEN via docker run -e — no manifest wiring
required, and the value is never written to disk or placed on argv.
Inside the container, claude picks up CLAUDE_CODE_OAUTH_TOKEN and
authenticates against your subscription. Caveats: the token is bound
to your subscription tier (Pro/Max/Team/Enterprise), it does not work
with claude --bare (which only reads ANTHROPIC_API_KEY), and if it
leaks, regenerate via claude setup-token again. Reference:
https://code.claude.com/docs/en/authentication.