openclaw-docker-setup
name: openclaw-docker-setup
by chunhualiao · published 2026-03-22
$ claw add gh:chunhualiao/chunhualiao-openclaw-docker-setup---
name: openclaw-docker-setup
description: Install and configure a fully operational Dockerized OpenClaw instance on macOS from scratch. Includes browser pairing, Discord channel setup, and optional Gmail/Google Drive integration. Use when user asks to "install openclaw docker", "set up dockerized openclaw", "openclaw in docker", or "isolated openclaw instance".
triggers:
- install openclaw docker
- set up dockerized openclaw
- openclaw in docker
- isolated openclaw instance
- dockerized openclaw
- run openclaw in docker
metadata: {"openclaw": {"os": ["darwin"]}}
---
# openclaw-docker-setup
Install a fully isolated, production-ready OpenClaw instance inside Docker on macOS.
One session, zero to running. All common pitfalls are handled inline.
**Supports multiple instances on the same machine.** Each instance gets a unique name and port.
**What you end up with:**
---
Step 0: Pick Your Instance Name and Port
Run this auto-detect script. It scans existing OpenClaw containers, finds the next free port, and suggests a name. Confirm or override.
# Auto-detect existing instances and suggest next available name+port
python3 - << 'AUTODETECT'
import subprocess, re
# Find all running openclaw containers
result = subprocess.run(
["docker", "ps", "-a", "--format", "{{.Names}} {{.Ports}}"],
capture_output=True, text=True
)
existing = {}
for line in result.stdout.strip().splitlines():
parts = line.split(' ')
name = parts[0]
ports = parts[1] if len(parts) > 1 else ""
if 'openclaw' in name.lower() or '18789' in ports:
m = re.search(r'0\.0\.0\.0:(\d+)->18789', ports)
port = int(m.group(1)) if m else None
existing[name] = port
# Find next free port starting from 19002
used_ports = set(p for p in existing.values() if p)
port = 19002
while port in used_ports:
port += 1
# Suggest name
count = len(existing) + 1
names = ["openclaw-main", "openclaw-work", "openclaw-demo", "openclaw-test", "openclaw-lab"]
suggested_name = names[min(count - 1, len(names) - 1)]
print("\n=== Existing OpenClaw instances ===")
if existing:
for n, p in existing.items():
print(f" {n} → port {p}")
else:
print(" (none found)")
print(f"\n=== Suggested for new instance ===")
print(f" INSTANCE={suggested_name}")
print(f" HOST_PORT={port}")
print(f"\nTo accept, run:")
print(f" export INSTANCE={suggested_name}")
print(f" export HOST_PORT={port}")
print(f"\nTo override, replace the values and run the export commands with your chosen values.")
AUTODETECTReview the output, then set your variables:
# Accept suggestion (paste the export lines from the output above)
export INSTANCE=openclaw-main
export HOST_PORT=19002
# Or override with your own values:
export INSTANCE=openclaw-demo
export HOST_PORT=19003**All subsequent commands in this guide use `$INSTANCE` and `$HOST_PORT`.** Keep this terminal session open, or re-export the variables if you open a new one.
**Multiple instances example:**
| Instance | Host port | Purpose |
|----------|-----------|---------|
| `openclaw-main` | 19002 | Primary personal assistant |
| `openclaw-demo` | 19003 | Public demo / lecture |
| `openclaw-work` | 19004 | Work projects |
Each instance has its own volumes (`$INSTANCE-data`, `$INSTANCE-home`) — data is fully isolated.
---
Prerequisites
Verify Docker is running:
docker --version
docker psBoth commands must succeed before continuing.
---
Step 1: Pull the Image
The official image is on GitHub Container Registry (GHCR), **not Docker Hub**.
docker pull ghcr.io/openclaw/openclaw:latest> **Pitfall:** `openclaw/openclaw` on Docker Hub does not exist. Always use `ghcr.io/openclaw/openclaw:latest`.
**Success:** Pull completes without error and `docker images | grep openclaw` shows the image.
---
Step 2: Generate a Claude Setup Token
**Skip this step if you have a raw Anthropic API key** — you will pass it via `-e ANTHROPIC_API_KEY=sk-ant-api03-...` in Step 3 instead.
If you have a Claude Max or Pro subscription, generate a setup token on the host:
claude setup-tokenCopy the token (format: `sk-ant-oat01-...`). You will paste it into the container in Step 5.
> **Pitfall:** This is a **setup token**, not an API key. The two are different. A setup token lets the container authenticate using your subscription. An API key charges per token.
---
Step 3: Launch the Container
Use the `$INSTANCE` and `$HOST_PORT` variables you set in Step 0. Do not reduce the memory — 512 MB and 1024 MB are insufficient and cause crash loops.
docker run -d \
--name $INSTANCE \
--restart unless-stopped \
-p $HOST_PORT:18789 \
-m 2048m \
--cpus=2 \
--cap-drop=ALL \
--cap-add=NET_BIND_SERVICE \
--security-opt no-new-privileges \
-v ${INSTANCE}-data:/app/data \
-v ${INSTANCE}-home:/home/node \
-e NODE_OPTIONS="--max-old-space-size=1024" \
ghcr.io/openclaw/openclaw:latestIf using a raw API key instead of a setup token, add `-e ANTHROPIC_API_KEY=sk-ant-api03-...` before the image name.
Wait 10 seconds, then verify:
docker ps --filter name=$INSTANCE**Success:** Status shows `Up X seconds` and the container is not restarting.
> **Pitfall — OOM crash loop:** If the container keeps restarting, check logs:
> `docker logs --tail 20 $INSTANCE`
> If you see `JavaScript heap out of memory`, the container needs `-m 2048m` AND `-e NODE_OPTIONS="--max-old-space-size=1024"`. Recreate with the full command above.
> **Pitfall — port conflict:** If the port is in use, you chose the wrong `HOST_PORT` in Step 0. Re-run the conflict check: `lsof -i :$HOST_PORT`. Pick a free port and relaunch.
---
Step 4: Configure the Gateway
The gateway binds to `127.0.0.1` (loopback) inside the container by default. Docker port-forwarding sends traffic to the container's network interface, not its loopback. You must switch to LAN mode.
4a. Set bind to LAN
docker exec $INSTANCE node /app/openclaw.mjs config set gateway.bind lan4b. Set allowed origins
Non-loopback bind requires explicitly allowed origins or the gateway refuses to start:
docker exec $INSTANCE node /app/openclaw.mjs config set \
gateway.controlUi.allowedOrigins '["http://127.0.0.1:$HOST_PORT"]' --json4c. Restart to apply
docker restart $INSTANCEWait 10 seconds, then verify:
curl -s -o /dev/null -w "%{http_code}" http://127.0.0.1:$HOST_PORT/**Success:** Returns `200`.
> **Pitfall — crash after setting LAN bind:** If you set `gateway.bind lan` but forget the `allowedOrigins` step, the container crash-loops with `non-loopback Control UI requires gateway.controlUi.allowedOrigins`. Run step 4b, then restart.
---
Step 5: Register Authentication
> **Important:** Do NOT paste your token directly in the command line — it would be stored in shell history. The command below prompts interactively.
docker exec -it $INSTANCE node /app/openclaw.mjs models auth paste-token --provider anthropicWhen prompted, paste the setup token from Step 2 (or your API key if using one).
**Success:**
Updated ~/.openclaw/openclaw.json
Auth profile: anthropic:manual (anthropic/token)> **Pitfall — `openclaw` command not found:** The CLI binary is NOT installed as a global command in this image. Always use `node /app/openclaw.mjs` inside the container:
> ```bash
> docker exec $INSTANCE node /app/openclaw.mjs <command>
> ```
> For interactive commands (pasting tokens, TTY prompts), add `-it`:
> ```bash
> docker exec -it $INSTANCE node /app/openclaw.mjs <command>
> ```
---
Step 6: Verify Authentication and Gateway
# Check model auth
docker exec $INSTANCE node /app/openclaw.mjs models status**Success:** Output includes `Providers w/ OAuth/tokens (1): anthropic (1)`
# Check gateway
docker exec $INSTANCE node /app/openclaw.mjs gateway status**Success:** Output includes `RPC probe: ok`
---
Step 7: Access the Dashboard and Pair Your Browser
Get the gateway auth token
The CLI redacts secrets in output. Read the raw config file instead:
docker exec $INSTANCE cat /home/node/.openclaw/openclaw.jsonFind `gateway.auth.token` in the output. Copy it.
Open the dashboard
Open http://127.0.0.1:$HOST_PORT/ in your browser. Enter the gateway token to log in.
The browser will show "Pairing required." This is expected on first access.
Approve the pairing request
docker exec $INSTANCE node /app/openclaw.mjs devices listFind the pending request ID, then approve it:
docker exec $INSTANCE node /app/openclaw.mjs devices approve <REQUEST_ID>Refresh the browser.
**Success:** Dashboard loads and shows the OpenClaw interface.
---
Step 8: Configure Discord
Create a Discord bot
1. Go to https://discord.com/developers/applications
2. Click **New Application** — name it (e.g. "OpenClaw Isolated")
3. Go to **Bot** → set a username → click **Reset Token** → copy the token
4. Under **Privileged Gateway Intents**, enable:
- **Message Content Intent** (required)
- **Server Members Intent** (recommended)
5. Go to **OAuth2 > URL Generator**:
- Scopes: `bot` AND `applications.commands` (both required)
- Bot Permissions: View Channels, Send Messages, Read Message History, Embed Links, Attach Files
6. Copy the generated URL, open it in a browser, and add the bot to your server
> **Pitfall — slash commands say "not authorized":** If you invite the bot without `applications.commands`, slash commands will not work. Re-authorize using this URL (replace `BOT_CLIENT_ID`):
> ```
> https://discord.com/oauth2/authorize?client_id=BOT_CLIENT_ID&scope=bot+applications.commands&permissions=274877991936
> ```
Collect Discord IDs
Enable **Developer Mode**: Discord → User Settings → Advanced → Developer Mode.
Configure Discord in the container
Replace `YOUR_DISCORD_BOT_TOKEN`, `YOUR_SERVER_ID`, and user IDs with real values.
# Enable Discord
docker exec $INSTANCE node /app/openclaw.mjs config set \
channels.discord.enabled true --json
# Set bot token
docker exec $INSTANCE node /app/openclaw.mjs config set \
channels.discord.token '"YOUR_DISCORD_BOT_TOKEN"' --json
# Set access policy to allowlist
docker exec $INSTANCE node /app/openclaw.mjs config set \
channels.discord.groupPolicy '"allowlist"' --json
# Configure the guild with authorized users
docker exec $INSTANCE node /app/openclaw.mjs config set \
channels.discord.guilds \
'{"YOUR_SERVER_ID":{"requireMention":false,"users":["USER_ID_1","USER_ID_2"]}}' \
--json
# Restart to apply
docker restart $INSTANCEVerify Discord is connected
docker exec $INSTANCE node /app/openclaw.mjs channels status --probe**Success:** Output includes `Discord default: enabled, configured, running, ... works`
Confirm users resolved:
docker logs --tail 10 $INSTANCELook for: `[discord] channel users resolved: USER_ID_1→USER_ID_1`
> **Pitfall — guilds config fails with "expected record, received array":** Always set the full `guilds` object at once. Setting individual guild keys fails:
> ```bash
> # WRONG
> docker exec $INSTANCE node /app/openclaw.mjs config set \
> 'channels.discord.guilds.SERVER_ID' '{"requireMention":false}' --json
>
> # CORRECT
> docker exec $INSTANCE node /app/openclaw.mjs config set \
> channels.discord.guilds '{"SERVER_ID":{"requireMention":false,"users":["UID"]}}' --json
> ```
> **Pitfall — adding users later:** `config set guilds` replaces the entire object. Always include ALL existing user IDs when adding new ones.
---
Optional: Gmail Integration
**Skip this section if you do not need Gmail.** The rest of the setup works without it.
See `references/gmail-setup.md` for the complete guide (uses Himalaya — the only reliable option for attachment download).
The key insight: OAuth browser auth does NOT work inside Docker because the callback URL points to localhost inside the container, which the host browser cannot reach. The solution is to authenticate on the host Mac first, then copy tokens into the container.
---
Optional: Google Drive Integration
**Skip this section if you do not need Google Drive.** Google Drive uses gog (gogcli) — independent of the Gmail/Himalaya setup. You can set up Drive without setting up Gmail.
See `references/google-drive-setup.md` for the complete guide.
---
Maintenance Reference
Daily operations
# Stop the container
docker stop $INSTANCE
# Start the container
docker start $INSTANCE
# View logs
docker logs --tail 50 $INSTANCE
# Monitor resource usage
docker stats --no-stream $INSTANCEUpdate OpenClaw to a new version
docker pull ghcr.io/openclaw/openclaw:latest
docker stop $INSTANCE
docker rm $INSTANCE
# Re-run the same launch command from Step 3
# Named volumes retain all config, auth, and workspace data
docker run -d \
--name $INSTANCE \
--restart unless-stopped \
-p $HOST_PORT:18789 \
-m 2048m \
--cpus=2 \
--cap-drop=ALL \
--cap-add=NET_BIND_SERVICE \
--security-opt no-new-privileges \
-v ${INSTANCE}-data:/app/data \
-v ${INSTANCE}-home:/home/node \
-e NODE_OPTIONS="--max-old-space-size=1024" \
ghcr.io/openclaw/openclaw:latestTransfer files
# Host to container
docker cp /path/to/local/file.txt $INSTANCE:/home/node/.openclaw/workspace/
# Container to host
docker cp $INSTANCE:/home/node/.openclaw/workspace/file.txt ~/Desktop/
# Backup entire .openclaw directory
docker exec $INSTANCE tar -czf - -C /home/node .openclaw \
> ~/Desktop/${INSTANCE}-backup.tar.gz
# Restore from backup
cat ~/Desktop/${INSTANCE}-backup.tar.gz | \
docker exec -i $INSTANCE tar -xzf - -C /home/nodeCompletely remove everything
docker rm -f $INSTANCE
docker volume rm ${INSTANCE}-data ${INSTANCE}-home**Warning:** This permanently deletes all config, auth, and workspace data. Backup first.
---
Troubleshooting
| Symptom | Cause | Fix |
|---------|-------|-----|
| `pull access denied for openclaw/openclaw` | Wrong image registry | Use `ghcr.io/openclaw/openclaw:latest` |
| Container keeps restarting | OOM crash | Use `-m 2048m` + `-e NODE_OPTIONS="--max-old-space-size=1024"` |
| `curl 127.0.0.1:$HOST_PORT` returns 000 or connection refused | Gateway on loopback | Set `gateway.bind lan` + `allowedOrigins`, restart |
| Container crash-loops after setting LAN bind | Missing `allowedOrigins` | Set `gateway.controlUi.allowedOrigins`, restart |
| `exec: "openclaw": executable file not found` | No global CLI binary | Use `node /app/openclaw.mjs` |
| Dashboard shows "Pairing required" | Browser not approved | `devices list` then `devices approve <ID>` |
| `config get gateway.auth.token` returns `__OPENCLAW_REDACTED__` | CLI redacts secrets | `cat /home/node/.openclaw/openclaw.json` |
| Discord slash commands say "not authorized" | Missing `applications.commands` scope or user not in allowlist | Re-authorize bot; check `guilds` config |
| Data gone after container recreation | Data on ephemeral overlay | Mount `/home/node` as named volume (the launch command above already does this) |
For full pitfall details, see `references/pitfalls.md`.
---
Container Reference
| Setting | Value |
|---------|-------|
| Container name | `$INSTANCE` |
| Image | `ghcr.io/openclaw/openclaw:latest` |
| Host port | `19002` |
| Container port | `18789` |
| Dashboard URL | http://127.0.0.1:$HOST_PORT/ |
| Memory limit | 2048 MB |
| CPU limit | 1 core |
| Node.js heap | 1024 MB |
| Restart policy | `unless-stopped` |
| CLI prefix inside container | `node /app/openclaw.mjs` |
| Volume | Mounted at | Contains |
|--------|-----------|----------|
| `${INSTANCE}-home` | `/home/node` | Config, auth, workspace |
| `${INSTANCE}-data` | `/app/data` | App-level data |
---
Do You Need to Be at Your Mac?
**Short answer: Yes for initial setup. No for ongoing use.**
Why initial setup requires your Mac
Three steps require direct Mac access (physical or SSH with port forwarding):
| Step | Why Mac access is needed |
|------|--------------------------|
| **Step 2: Claude setup token** | `claude setup-token` opens a browser auth flow on `localhost`. Cannot be done remotely without a browser on the Mac. Skip if using a raw API key — that can be set from anywhere. |
| **Step 4: Browser pairing** | The OpenClaw dashboard runs at `http://127.0.0.1:$HOST_PORT/` — only reachable from the Mac. You must open a browser there to pair. |
| **Optional Gmail/Drive OAuth** | Google OAuth callback points to `localhost` on the Mac. Must authenticate from a browser running on the Mac. |
Remote setup is possible via SSH port forwarding
If you SSH into your Mac from another machine, you can forward the container port to your local browser:
# From your remote machine — forward Mac's port 19002 to your local 19002
ssh -L 19002:localhost:19002 your-mac.local
# Then open http://127.0.0.1:19002/ in your local browserThis lets you complete the browser pairing step remotely.
After setup is complete
Once the container is running and paired, you do **not** need to be at your Mac. OpenClaw runs as a background service (`--restart unless-stopped`). You interact with it entirely through your configured channel (Discord, Telegram, etc.) — from your phone, any browser, anywhere.
---
Configuration
No persistent configuration required. All settings are chosen interactively in Step 0 and set as shell variables (`$INSTANCE`, `$HOST_PORT`).
**Optional integrations require additional setup:**
| Integration | Guide | Requires |
|-------------|-------|---------|
| Gmail (email + attachments) | `references/gmail-setup.md` | Gmail App Password |
| Google Drive / Docs / Sheets / Calendar | `references/google-drive-setup.md` | Google Cloud OAuth credentials |
**System dependencies:**
| Dependency | Purpose | Check |
|------------|---------|-------|
| Docker Desktop | Container runtime | `docker --version` |
| Python 3 | Auto-detect script in Step 0 | `python3 --version` |
| Claude Code CLI (`claude`) | Generate setup token (Claude Max/Pro only) | `claude --version` |
More tools from the same signal band
Order food/drinks (点餐) on an Android device paired as an OpenClaw node. Uses in-app menu and cart; add goods, view cart, submit order (demo, no real payment).
Sign plugins, rotate agent credentials without losing identity, and publicly attest to plugin behavior with verifiable claims and authenticated transfers.
The philosophical layer for AI agents. Maps behavior to Spinoza's 48 affects, calculates persistence scores, and generates geometric self-reports. Give your...