A lightweight shell wrapper that runs Claude Code inside a Docker container, keeping your host system clean while persisting your Claude configuration across sessions.
Requirements
- Docker with Buildx plugin
- dmenu — mode selection prompt (not required when using
-f) - fzf — directory picker; also used for mode selection with
-f - Current user must be a member of the
dockergroup
Installation
git clone https://github.com/shirozuki/claude-cli chmod +x claude-cli/claude-cli.sh ln -s "$PWD/claude-cli/claude-cli.sh" ~/.local/bin/claude-cli
Usage
claude-cli [-b] [-c] [-f] [-h] [DIR...]
| Option | Description |
|---|---|
-b |
Build (or rebuild) the claude-cli:latest Docker image |
-c |
Remove all claude-cli containers; optionally remove the image |
-f |
Use fzf instead of dmenu for the mode selection prompt |
-h |
Show help |
Running without any option or argument launches a dmenu prompt to choose a mode. Pass -f to use fzf instead — useful on setups without dmenu or when you prefer a terminal picker.
Passing directories as arguments
You can skip the menu and pickers entirely by passing directories directly on the command line:
claude-cli ~/projects/app # single-dir mode claude-cli ~/projects/app ~/shared # multi-dir mode (first arg is the working dir)
- One argument runs
single-dirmode with that directory. - Two or more arguments run
multi-dirmode, mounting every directory and using the first as the working directory.
Relative paths are resolved against the current working directory, so claude-cli . mounts $PWD. When directories are passed this way, dmenu and fzf are not required.
Modes
no-dir
Launches Claude without mounting any project directories. Useful for general questions, quick tasks, or exploring Claude's capabilities without exposing local files.
current-dir
Mounts the current working directory $PWD directly.
single-dir
Opens an fzf picker to select one directory from your home tree (up to 3 levels deep). The selected directory is mounted and set as the working directory inside the container.
multi-dir
Opens fzf in multi-select mode to pick several directories (use Tab to select), then asks you to designate one of them as the working directory. All selected directories are mounted simultaneously — useful when working across multiple repos or sharing common config directories.
Configuration persistence
Claude's configuration (~/.claude/ and ~/.claude.json) is bind-mounted into the container on every run, so your account, settings, and session history survive container restarts.
The mount source is resolved in this order:
$XDG_CONFIG_HOME/claude/— if$XDG_CONFIG_HOMEis set$HOME/— fallback
Configuration variables
The following environment variables can be set to override defaults without editing the script:
| Variable | Default | Description |
|---|---|---|
CLAUDE_IMAGE |
claude-cli:latest |
Docker image name to use |
NOTIFY_ERROR_ICON |
$XDG_CONFIG_HOME/dunst/critical.png |
Icon used in desktop error notifications |
CLAUDE_CLI_FLAGS |
(none) | Extra flags passed through to the claude binary inside the container |
CLAUDE_CLI_DOCKERFILE |
(see below) | Path to a custom Dockerfile used to build the image (highest precedence). |
Examples:
CLAUDE_IMAGE=my-claude:dev claude-cli # Pass flags through to Claude Code itself CLAUDE_CLI_FLAGS="--resume $session_id --dangerously-skip-permissions" claude-cli # Build the image from a custom Dockerfile CLAUDE_CLI_DOCKERFILE=~/my-claude.dockerfile claude-cli -b
CLAUDE_CLI_FLAGS is word-split, so each space-separated token becomes a
separate argument. Quote arguments that contain spaces is not supported — pass
only flags whose values have no spaces.
How it works
On first run (or after -b), the script builds a Docker image based on node:lts with @anthropic-ai/claude-code installed globally. The container user is created with the same UID/GID as the host user to avoid file permission issues on bind-mounted volumes.
The Dockerfile is selected in order of precedence:
- Custom Dockerfile path (
CLAUDE_CLI_DOCKERFILE) - Dockerfile next to the script (
claude-cli-dockerfile) - Inline Dockerfile baked into the script (the default)
The customization is purely additive: with none of the above set, the script writes its built-in Dockerfile to /tmp and builds from that, so claude-cli.sh stays self-contained and can be fetched and run entirely on its own.
To customize the image, drop a claude-cli-dockerfile next to the script and it will be preferred when present:
# create claude-cli-dockerfile to taste, then rebuild
claude-cli -bAlternatively, point CLAUDE_CLI_DOCKERFILE at a Dockerfile anywhere on disk.
The image is tagged claude-cli:latest and reused on subsequent runs until you explicitly rebuild with -b.