A runtime harness for Claude Code — the agentic coding CLI from Anthropic — running in a fully isolated Docker container with every dev tool pre-installed, passwordless sudo, docker-in-docker support, and --dangerously-skip-permissions enabled by default.
claudebox wraps Claude Code with several distinct interfaces:
- Interactive CLI — a drop-in replacement for the native
claudecommand, with persistent containers and automatic session resumption across runs - Programmatic CLI — non-interactive mode for scripts, CI/CD pipelines, and automation; pass a prompt, get structured output, pipe it wherever you need
- HTTP API server — a full REST API with workspace management, file operations, structured output formats, and workspace isolation for multi-tenant deployments
- OpenAI-compatible endpoint — a
chat/completionsadapter that lets LiteLLM, OpenAI SDKs, and any OpenAI-compatible client talk to Claude Code, complete with streaming SSE, multi-turn conversations, and multimodal image handling - MCP server — a Model Context Protocol endpoint over streamable HTTP so other AI agents and tools (Claude Desktop, other Claude Code instances, etc.) can use Claude Code as a tool
- Telegram bot — a conversational interface with per-chat workspaces, configurable models and effort levels, file sharing, shell access, and group chat support
- Cron scheduler — yaml-defined Claude jobs running on cron schedules with per-job activity history, sub-minute resolution, and overlap protection
Beyond just running Claude Code in Docker, claudebox adds skill injection (auto-load SKILL.md files into every session), init hooks, custom script directories, structured JSON logging, and a workspace management layer that handles multi-tenant isolation with automatic busy/idle tracking.
Renamed from
docker-claude-code: This project was previously calleddocker-claude-codewith the Docker image atpsyb0t/claude-code. Starting with v1.0.0, it isclaudebox— the Docker image is nowpsyb0t/claudebox, the default binary name isclaudebox, the GitHub repository ispsyb0t/docker-claudebox, and the SSH key directory defaults to~/.ssh/claudebox. If you were using the old names, update your image references, wrapper scripts, and SSH paths accordingly.
- Requirements
- Quick Start
- Image Variants
- What's Inside (Full Image)
- Authentication
- Modes
- Configuration
- Gotchas
- License
Docker installed and running. That's it.
The install script pulls the Docker image, generates SSH keys for git operations inside the container, downloads the wrapper script, and installs it as a command on your system.
# full image (recommended — all dev tools pre-installed)
curl -fsSL https://raw.githubusercontent.com/psyb0t/docker-claudebox/master/install.sh | bash
# minimal image (just the essentials — Claude installs what it needs on the fly)
CLAUDEBOX_MINIMAL=1 curl -fsSL https://raw.githubusercontent.com/psyb0t/docker-claudebox/master/install.sh | bash
# custom binary name (e.g. if you want to call it 'claude' instead of 'claudebox')
curl -fsSL https://raw.githubusercontent.com/psyb0t/docker-claudebox/master/install.sh | bash -s -- claude
# or: CLAUDEBOX_BIN_NAME=claude curl -fsSL .../install.sh | bashIf you prefer not to pipe scripts to bash:
# 1. create the data directory
mkdir -p ~/.claude
# 2. create SSH keys for git operations inside the container
mkdir -p "$HOME/.ssh/claudebox"
ssh-keygen -t ed25519 -C "claude@claude.ai" -f "$HOME/.ssh/claudebox/id_ed25519" -N ""
# then add the public key to GitHub/GitLab/wherever you push code
# 3. pull the image
docker pull psyb0t/claudebox:latest
# or: docker pull psyb0t/claudebox:latest-minimal
# 4. grab the wrapper script and install it
# see install.sh for exactly how the wrapper is set upEverything pre-installed. Go, Python, Node.js, C/C++ toolchains, Terraform, kubectl, database clients, linters, formatters — the works. Large image, but Claude wakes up and gets to work immediately with zero wait time. This is the recommended variant for most users.
curl -fsSL https://raw.githubusercontent.com/psyb0t/docker-claudebox/master/install.sh | bashJust enough to run Claude: Ubuntu, git, curl, Node.js, and Docker. Claude has passwordless sudo, so it will install whatever else it needs on the fly via apt-get, pip, npm, etc. Smaller image to pull, but the first run takes longer as Claude sorts out its dependencies.
CLAUDEBOX_MINIMAL=1 curl -fsSL https://raw.githubusercontent.com/psyb0t/docker-claudebox/master/install.sh | bashUse ~/.claude/init.d/*.sh hooks (see Init Hooks) to pre-install your tools on first container create so Claude doesn't burn tokens figuring out package management.
latest (full) |
latest-minimal |
|
|---|---|---|
| Ubuntu 24.04 | yes | yes |
| git, curl, wget, jq | yes | yes |
| Node.js LTS + npm | yes | yes |
| Docker CE + Compose | yes | yes |
| Claude Code CLI | yes | yes |
| Go 1.26.1 + tools | yes | - |
| Python 3.12.11 + tools | yes | - |
| Node.js dev tools | yes | - |
| C/C++ tools | yes | - |
| DevOps (terraform, kubectl, helm, gh) | yes | - |
| Database clients | yes | - |
| Shell utilities (ripgrep, bat, etc.) | yes | - |
Languages and runtimes:
- Go 1.26.1 with the full toolchain — golangci-lint, gopls, delve, staticcheck, gofumpt, gotests, impl, gomodifytags
- Python 3.12.11 via pyenv — flake8, black, isort, autoflake, pyright, mypy, vulture, pytest, poetry, pipenv, plus common libraries (requests, beautifulsoup4, lxml, pyyaml, toml)
- Node.js LTS — eslint, prettier, typescript, ts-node, yarn, pnpm, nodemon, pm2, framework CLIs (React, Vue, Angular), newman, http-server, serve, lighthouse, storybook
- C/C++ — gcc, g++, make, cmake, clang-format, valgrind, gdb, strace, ltrace
DevOps and infrastructure:
- Docker CE with Docker Compose (docker-in-docker support)
- Terraform, kubectl, helm, GitHub CLI (
gh)
Database clients:
- sqlite3, postgresql-client (
psql), mysql-client, redis-tools (redis-cli)
Shell and system utilities:
- jq, tree, ripgrep, bat, exa, fd-find, ag (silversearcher), htop, tmux, shellcheck, shfmt, httpie, vim, nano
- Archive tools (zip, unzip, tar), networking (net-tools, iputils-ping, dnsutils)
Container automation:
- Auto-generated
CLAUDE.mdin each workspace listing all available tools, so Claude knows what it has access to - Git identity auto-configured from environment variables
- Claude Code CLI with auto-updates disabled by default (opt in with
--update) - Workspace trust dialog pre-accepted — no interactive prompts
- Custom scripts via
~/.claude/bin(added to PATH automatically) - Init hooks via
~/.claude/init.d/*.sh(run once on first container create) - Always-active skills via
~/.claude/.always-skills/(injected into every invocation) - Session continuity via
--continue/--no-continue/--resume <session_id> - Structured JSON debug logging with
DEBUG=true
You need either an Anthropic API key or an OAuth token. Set up once, use everywhere:
# interactive OAuth token setup (one-time)
claudebox setup-token
# then use the token for programmatic and headless runs
CLAUDE_CODE_OAUTH_TOKEN=sk-ant-oat01-xxx claudebox "do stuff"
# or use an API key directly
ANTHROPIC_API_KEY=sk-ant-api03-xxx claudebox "do stuff"claudebox can run in several modes — pick the one that matches how you want to use Claude Code. Each has its own page with full setup, env vars, and examples.
Drop-in replacement for claude. Persistent per-workspace container, automatic session resumption, plus utility commands like claudebox doctor, claudebox mcp list, claudebox stop, and claudebox clear-session.
claudeboxNon-interactive prompt → response for scripts, pipelines, and automation. Plain text, JSON, JSON-verbose (with full tool-call history), and stream-json output formats. Model selection, system prompt overrides, JSON-schema-constrained output, session continuation.
claudebox "explain this codebase" --output-format json --model haikuRun as a long-lived HTTP server. Full REST API for prompts and file ops with workspace isolation, async runs with run-id polling, OpenAI-compatible chat/completions endpoint (streaming + multimodal + LiteLLM compatible), and an MCP endpoint over streamable HTTP so other agents can use Claude Code as a tool.
environment:
- CLAUDEBOX_MODE_API=1
- CLAUDEBOX_MODE_API_TOKEN=your-secret-tokenTalk to Claude from Telegram. Per-chat isolated workspaces, configurable models/effort/system-prompts per chat, allowed-chats and per-chat allowed-users gating, file/photo/video/voice ingestion, /bash, /fetch, /cancel, /status, /config, /reload commands, and [SEND_FILE: path] for Claude to send files back.
environment:
- CLAUDEBOX_MODE_TELEGRAM=1
- CLAUDEBOX_TELEGRAM_BOT_TOKEN=...YAML-defined scheduled jobs. Standard 5-field cron or 6-field for sub-minute resolution. Per-job stream-json history under ~/.claude/cron/history/<workspace-slug>/<ts>-<job>/, foreground process so docker logs shows every tick, overlap protection. Set model at the root of the YAML as a default for all jobs; override per-job as needed.
environment:
- CLAUDEBOX_MODE_CRON=1
- CLAUDEBOX_MODE_CRON_FILE=/home/claude/.claude/cron.yaml- Environment variables → — full table of
CLAUDEBOX_*settings the wrapper and entrypoint understand, plusCLAUDEBOX_ENV_*(forward arbitrary vars into the container) andCLAUDEBOX_MOUNT_*(extra volume mounts). - Customization → — extend Claude's container with custom scripts (
~/.claude/bin), one-time init hooks (~/.claude/init.d), always-active skills auto-injected into every session (~/.claude/.always-skills), and MCP server definitions (project.mcp.jsonor global~/.claude.json).
--dangerously-skip-permissionsis always enabled. Claude has full, unrestricted access to the container. That's the entire point.- SSH keys are mounted from the host for git push/pull inside the container. Do not share your container or image with untrusted parties.
- Host paths are preserved — your project at
/home/you/projectis mounted at the same path inside the container. This means Docker volume mounts that Claude creates from within the container resolve correctly against host paths. - UID/GID matching — the container's
claudeuser UID/GID is automatically adjusted to match the host directory owner on startup. File permissions should just work without manualchown. - Docker-in-Docker — the Docker socket is mounted into the container. Claude can build images and run containers from within its container. This is by design.
- Two containers per workspace — the wrapper creates
claude-<path>for interactive (TTY) sessions andclaude-<path>_progfor programmatic (no TTY) sessions. Both share the same mounted volumes and data. - Workspace busy tracking — in API mode, each workspace can only have one active Claude process at a time. Concurrent requests to the same workspace return a 409 Conflict response. Use different workspace subpaths for parallel work.
- Telegram config is required — the Telegram bot will not start without a
telegram.ymlconfig file. This is intentional to prevent accidentally exposing Claude to the public. - Auto-updates disabled — Claude Code CLI auto-updates are disabled by default inside the container to ensure reproducible behavior. Opt in with
claudebox --updatewhen you want to update.
WTFPL — do what the fuck you want to.