claude-mac-bridge
MCP server that lets an AI agent delegate tasks to Claude Code on a remote Mac via SSH over Tailscale. Persistent sessions, 200k context, full Mac filesystem access.
README
claude-mac-bridge
MCP server that lets an AI agent delegate tasks to Claude Code on a remote Mac via SSH over Tailscale. Persistent sessions, 200k context, full Mac filesystem access.
Agent (server) → ask_claude tool → SSH → claude -p (Mac) → JSON response + session_id
Why
Your agent runs on a Linux server. Claude Code runs on a Mac with Homebrew, Docker, Xcode, and 200k context. Sometimes you need both.
This bridge:
- Gives your agent Mac access: file system, tools, local services
- Persists sessions: pass
session_idback to continue conversations across hours - Zero config on the Mac: Claude Code runs headless via
-p, no GUI needed - Full audit trail: every delegation logged to
bridge.logwith cost + duration
Quick Start
# 1. Install
pip install mcp
# 2. Set up SSH key
ssh-keygen -t ed25519 -f ~/.ssh/mac_bridge
ssh-copy-id -i ~/.ssh/mac_bridge.pub user@<mac-tailscale-ip>
# 3. Set env vars
export CLAUDE_BRIDGE_SSH_HOST="user@100.x.x.x"
export CLAUDE_BRIDGE_CLAUDE_BIN="/opt/homebrew/bin/claude"
# 4. Run the server
python3 server.py
# 5. Wire into your MCP config (see Setup below)
Requirements
- Claude Code on the Mac (
npm install -g @anthropic-ai/claude-codeor Homebrew) - GNU
timeoutfor remote process cleanup, install viabrew install coreutilson macOS (providesgtimeout). Falls back gracefully if unavailable, but remote Claude processes may outlive timed-out SSH connections without it. - Tailscale on both machines
- Passwordless SSH (key-based auth) from server → Mac
- Python 3.10+ on the server
mcppackage (pip install mcp)
Setup
1. SSH key auth
# On server: generate a dedicated key
ssh-keygen -t ed25519 -f ~/.ssh/mac_bridge
# Copy to Mac
ssh-copy-id -i ~/.ssh/mac_bridge.pub user@<mac-tailscale-ip>
# Verify
ssh -i ~/.ssh/mac_bridge user@<mac-tailscale-ip> "echo ok"
2. Configure
| Env var | Required | Default | Description |
|---|---|---|---|
CLAUDE_BRIDGE_SSH_HOST |
✅ | , | SSH target: user@100.x.x.x |
CLAUDE_BRIDGE_CLAUDE_BIN |
❌ | claude |
Path to Claude binary on the Mac |
CLAUDE_BRIDGE_TIMEOUT |
❌ | 600 |
Max seconds per delegation (wall-clock, never pauses) |
CLAUDE_BRIDGE_LOG_PATH |
❌ | ./bridge.log |
Path for the delegation audit log |
3. Wire into MCP config
mcp_servers:
claude_bridge:
command: "python3"
args: ["/path/to/server.py"]
env:
CLAUDE_BRIDGE_SSH_HOST: "user@100.x.x.x"
CLAUDE_BRIDGE_CLAUDE_BIN: "/opt/homebrew/bin/claude"
timeout: 620 # must exceed CLAUDE_BRIDGE_TIMEOUT
API
ask_claude(task, context?, resume_session_id?)
| Parameter | Required | Description |
|---|---|---|
task |
✅ | The task or question. Be specific. |
context |
❌ | Extra context: code, error messages, background. |
resume_session_id |
❌ | From a previous call, continues that conversation. |
Returns: Claude's response with session_id, cost, and duration appended.
# Basic
result = ask_claude(task="Write a Python script that reads a CSV and...")
# With context
result = ask_claude(
task="Find the bug",
context="def foo(): ...\nError: TypeError..."
)
# Persistent session
r1 = ask_claude(task="Review this PR")
r2 = ask_claude(
task="Fix the first issue you found",
resume_session_id="<session_id from r1>"
)
Session Chaining
Every response includes a session_id. Pass it back as resume_session_id to continue the same conversation, Claude remembers all prior context, files read, and decisions made.
Rule: always chain session_ids for related tasks. Without resume_session_id, each call is a cold start.
r1 = ask_claude(task="Build the auth module")
# r1 contains: session_id = "abc123"
r2 = ask_claude(task="Add rate limiting to the auth module", resume_session_id="abc123")
r3 = ask_claude(task="Write tests for all of it", resume_session_id="abc123")
Long Tasks: tmux Blocking Loop
The MCP tool has a hard wall-clock timeout (default 600s). For tasks that may run longer, project scaffolding, npm installs, multi-file builds, use this tmux pattern instead. It blocks until Claude finishes, auto-handles confirmation prompts, and returns the full result + session_id.
#!/usr/bin/env bash
MAC="user@<mac-tailscale-ip>"
TMUX="/opt/homebrew/bin/tmux"
CLAUDE="/opt/homebrew/bin/claude"
SESSION="agent-work"
MAX_WAIT=3600 # 60 min ceiling
POLL=45
# 1. write task to file (avoids SSH quoting hell)
scp /tmp/task.md "$MAC:~/task.md"
# 2. start fresh tmux session
ssh "$MAC" "$TMUX kill-session -t $SESSION 2>/dev/null; \
$TMUX new-session -d -s $SESSION -x 220 -y 50"
# 3. launch claude
ssh "$MAC" "$TMUX send-keys -t $SESSION \
'$CLAUDE -p "Read ~/task.md and execute every step." \
--output-format json --dangerously-skip-permissions 2>&1 | tee ~/task_output.json' Enter"
# 4. blocking poll -- auto-answer confirmations, exit on shell prompt
elapsed=0
while [ $elapsed -lt $MAX_WAIT ]; do
sleep $POLL
elapsed=$((elapsed + POLL))
pane=$(ssh "$MAC" "$TMUX capture-pane -t $SESSION -p 2>/dev/null | tail -10")
# auto-approve confirmation prompts
if echo "$pane" | grep -qiE 'do you want|allow|yes.*no|\[y/n\]|approve|1\).*2\)'; then
ssh "$MAC" "$TMUX send-keys -t $SESSION '2' Enter"
sleep 3; continue
fi
# shell prompt = claude exited
echo "$pane" | grep -qE '^\s*(%|\$|>)\s*$' && break
done
# 5. grab result + session_id
result=$(ssh "$MAC" "cat ~/task_output.json 2>/dev/null")
session_id=$(echo "$result" | python3 -c \
"import json,sys; print(json.load(sys.stdin).get('session_id',''))" 2>/dev/null)
# 6. cleanup
ssh "$MAC" "$TMUX kill-session -t $SESSION 2>/dev/null; rm -f ~/task.md ~/task_output.json"
echo "done in ${elapsed}s | session_id: $session_id"
When to use tmux vs MCP tool:
| Task type | Use |
|---|---|
| Quick question, code review, research | MCP tool |
| Multi-step work with session chaining | MCP tool + resume_session_id |
Project scaffolding / npm install |
tmux blocking loop |
| Multi-file writes across a codebase | tmux blocking loop |
| Unknown duration / likely >5 min | tmux blocking loop |
If unsure → tmux. A 30s task in tmux costs nothing extra. A 12-min task in the MCP tool gets killed at 10 min.
Monitoring
python3 monitor.py # live feed (tails bridge.log)
python3 monitor.py /path/to/bridge.log
Color-coded live TUI showing start/done/error/timeout events with task previews, response length, cost, and duration.
Troubleshooting
| Symptom | Cause | Fix |
|---|---|---|
Host key verification failed |
Mac host key unknown | ssh -o StrictHostKeyChecking=accept-new user@<ip> once |
Permission denied (publickey) |
SSH key not deployed | Run ssh-copy-id, verify with ssh -i ~/.ssh/mac_bridge user@<ip> |
claude: command not found |
Wrong binary path | Set CLAUDE_BRIDGE_CLAUDE_BIN to full path (which claude on Mac) |
| Bridge times out | Task too long for CLAUDE_BRIDGE_TIMEOUT |
Increase timeout via env var, or use the tmux pattern for tasks >10 min (see Long Tasks section) |
is_error: true with no detail |
Claude returned an error | Check bridge.log for full response |
| Session not continuing | Claude process ended on Mac | Sessions live as long as the daemon on Mac, restarting Mac or Claude clears them |
TASK=$(cat) errors with special chars |
Prompt has unmatched quotes or shell metacharacters | The JSON output format should handle escaping, if hit, wrap task in a temp file |
Security
bridge.logcontains full task/response history, it is gitignored, keep it local- Never commit
.envor SSH private keys --dangerously-skip-permissionsis passed to Claude so it can use tools without interactive prompts, only use this on a trusted Mac you control- The SSH command runs as-is on your Mac, do not expose this bridge to untrusted clients
Known Limitations
- Mac must be awake + on Tailscale. If the Mac sleeps or goes offline, delegations timeout.
- Session lifetime depends on Claude process. If Claude restarts (Mac reboot, crash), session IDs are lost.
bridge.loggrows unbounded. No log rotation, monitor size or add your own.- Async server, concurrent tool calls are supported but each delegation opens its own SSH connection to the Mac.
--dangerously-skip-permissionsis hardcoded. If you want to review permissions per-call, fork and modify.
License
MIT.
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.