Docker-Native Install¶
Run Headroom without installing Python or Node.js on the host. The install scripts add a native headroom wrapper that keeps Headroom itself in Docker while orchestrating the rest of your workflow on the host OS.
One-line install¶
Linux¶
macOS (bash 4.3+)¶
curl -fsSL https://raw.githubusercontent.com/chopratejas/headroom/main/scripts/install.sh | "$(brew --prefix bash)/bin/bash"
Stock /bin/bash on macOS is 3.2, so install a newer bash first (for example via Homebrew) and run the installer with that shell. The installed wrapper pins that same bash interpreter so later invocations stay on the supported runtime.
Windows PowerShell¶
What the installer does¶
- Verifies Docker is installed and available.
- Pulls
ghcr.io/chopratejas/headroom:latestby default, or reuses / pullsHEADROOM_DOCKER_IMAGEwhen you set a custom image override. - Installs a
headroomwrapper into~/.local/binor~/bin. - Updates shell startup files so the wrapper directory is on
PATH.
The wrapper keeps Headroom inside Docker and mounts host state back into the container so native behavior stays consistent:
- project workspace ->
/workspace ~/.headroom~/.claude~/.codex~/.gemini
Port 8787 stays the default, so http://localhost:8787 works the same way as a native install.
Published releases also push versioned GHCR tags such as ghcr.io/chopratejas/headroom:0.5.26, and those images are built with the same synced package version used for the matching PyPI and npm release.
How the wrapper behaves¶
Native Headroom commands¶
These run directly inside the container:
For proxy, the wrapper publishes the selected port back to the host:
docker run --rm -it \
-p 8787:8787 \
-v "$PWD:/workspace" \
-w /workspace \
ghcr.io/chopratejas/headroom:latest \
headroom proxy --host 0.0.0.0 --port 8787
wrap commands¶
wrap is host-oriented in Docker-native mode:
- the wrapper starts the Headroom proxy in Docker
- container-side prep writes Headroom config, memory, and
rtkguidance into mounted host files - the target CLI itself is launched on the host by the wrapper
Supported host wrap flows:
headroom wrap claudeheadroom wrap codexheadroom wrap aiderheadroom wrap cursorheadroom wrap openclawheadroom unwrap openclaw
OpenClaw remains host-native in Docker-native mode:
- the host must already have the
openclawCLI installed headroom wrap openclawinstalls/configures the Headroom plugin through the hostopenclawCLI- plugin auto-start still launches the installed host
headroomwrapper fromPATH, which then runs Headroom in Docker - local plugin source mode (
--plugin-path) is also supported, but it may require hostnpmwhen build steps are needed
Persistent Docker lifecycle from the native wrapper¶
The Docker-native headroom wrapper now exposes the persistent Docker lifecycle directly:
headroom install apply --profile default --preset persistent-docker
headroom install status
headroom install restart
headroom install remove
In Docker-native mode this surface is intentionally scoped to persistent-docker:
- supported:
apply,status,start,stop,restart,remove - supported flags:
--profile,--port,--backend,--anyllm-provider,--region,--mode,--memory,--no-telemetry,--image - not supported:
persistent-service,persistent-task, or provider/user/system mutation flags such as--scope,--providers, and--target
Those broader lifecycle and config-mutation flows still belong to the Python-native headroom install ... command.
Persistent Docker deployments launched by the wrapper also tag the proxy process with deployment metadata, so /health reports the active profile, preset, runtime, supervisor, and scope the same way the Python install subsystem does.
Docker Compose support¶
Use docker/docker-compose.native.yml when you want an explicit compose-managed proxy or CLI shell, or when you prefer compose over the native wrapper's headroom install ... surface.
Persistent Docker runtime¶
The proxy service now uses restart: unless-stopped, so compose can act as the always-on Docker runtime for Headroom:
export HEADROOM_HOST_HOME="$HOME"
export HEADROOM_WORKSPACE="$PWD"
docker compose -f docker/docker-compose.native.yml up -d proxy
$env:HEADROOM_HOST_HOME = $HOME
$env:HEADROOM_WORKSPACE = (Get-Location).Path
docker compose -f docker/docker-compose.native.yml up -d proxy
This remains a supported persistent-Docker path when you want the proxy managed explicitly through Compose instead of the installed wrapper.
HEADROOM_WORKSPACE vs HEADROOM_WORKSPACE_DIR¶
These are two different variables — both are set by the compose file, and both are retained for backward compatibility:
HEADROOM_WORKSPACE(host-side) is the directory the compose file bind-mounts into the container as/workspace. It behaves like CWD in a native (non-Docker) run.HEADROOM_WORKSPACE_DIR(inside-the-container) is the canonical Headroom state root — part of the filesystem contract introduced in issue #175. The compose file sets it to/tmp/headroom-home/.headroomso the proxy resolves savings, logs, TOIN, and memory under the bind-mounted${HOME}/.headroom.
You do not need to set HEADROOM_WORKSPACE_DIR manually when using the
shipped compose file — it is already in the environment: block.
macOS / Linux¶
export HEADROOM_HOST_HOME="$HOME"
export HEADROOM_WORKSPACE="$PWD"
docker compose -f docker/docker-compose.native.yml up proxy
Windows PowerShell¶
$env:HEADROOM_HOST_HOME = $HOME
$env:HEADROOM_WORKSPACE = (Get-Location).Path
docker compose -f docker/docker-compose.native.yml up proxy
You can also run one-off CLI commands through compose:
docker compose -f docker/docker-compose.native.yml run --rm cli learn
docker compose -f docker/docker-compose.native.yml run --rm cli mcp install
Environment passthrough¶
The wrapper forwards Headroom and provider environment variables into the container, including common prefixes such as:
HEADROOM_ANTHROPIC_OPENAI_GEMINI_AWS_GOOGLE_/GOOGLE_CLOUD_AZURE_OTEL_
That keeps provider auth and runtime config working without maintaining a separate env file for the container.
Notes¶
- Docker is the only required Headroom runtime dependency on the host.
- Wrapped tools like Claude Code, Codex CLI, Aider, and Cursor still run on the host when you use
headroom wrap .... - The install scripts are idempotent: rerunning them refreshes the wrapper and image without duplicating shell profile blocks.
- For persistent service and task installs, use the Python-native
headroom install ...workflow described in Persistent Installs. - For Docker-native
headroom install ..., the wrapper persists its profile manifest under~/.headroom/deploy/<profile>/.