claude-code-mcp
MCP server that wraps Claude Code (Claude Agent SDK) as tools, enabling any MCP client to invoke Claude Code for autonomous coding tasks.
Inspired by the Codex MCP design philosophy — minimum tools, maximum capability.
Features
- 4 tools covering the full agent lifecycle: start, continue, check/poll, manage
- Session management with resume and fork support
- Local settings loaded by default — automatically reads
~/.claude/settings.json,.claude/settings.json,.claude/settings.local.json, andCLAUDE.mdso the agent behaves like your local Claude Code CLI - Async permissions — allow/deny lists + explicit approvals via
claude_code_check - Custom subagents — define specialized agents per session
- Cost tracking — per-session turn and cost accounting
- Session cancellation via AbortController
- Auto-cleanup — 30-minute idle timeout for expired sessions
- Security — callers control tool permissions via allow/deny lists + explicit permission decisions
Prerequisites
- Node.js >= 18 is required.
This MCP server uses the @anthropic-ai/claude-agent-sdk package, which bundles its own Claude Code CLI (cli.js). It does not use the claude binary from your system PATH.
- The SDK's bundled CLI version is determined by the SDK package version (e.g. SDK 0.2.38 = Claude Code 2.1.38)
- Configuration is shared — the bundled CLI reads API keys and settings from
~/.claude/, same as the system-installedclaude - All local settings are loaded by default — unlike the raw SDK (which defaults to isolation mode), this MCP server loads
user,project, andlocalsettings automatically, includingCLAUDE.mdproject context. Passadvanced.settingSources: []to opt out - You must have Claude Code configured (API key set up) before using this MCP server: see Claude Code documentation
Note: The bundled CLI version may differ from your system-installed
claude. To check:claude --version(system) vsnpm ls @anthropic-ai/claude-agent-sdk(SDK).
Quick Start
As an MCP server (recommended)
Install globally or use npx (no install needed):
npm install -g @leo000001/claude-code-mcp
Add to your MCP client configuration (Claude Desktop, Cursor, etc.):
{
"mcpServers": {
"claude-code": {
"command": "npx",
"args": ["-y", "@leo000001/claude-code-mcp"]
}
}
}
OpenAI Codex CLI
codex mcp add claude-code -- npx -y @leo000001/claude-code-mcp
Or manually add to ~/.codex/config.toml:
[mcp_servers.claude-code]
command = "npx"
args = ["-y", "@leo000001/claude-code-mcp"]
Codex supports both user-level (~/.codex/config.toml) and project-level (.codex/config.toml) configuration. See Codex config reference for advanced options like tool_timeout_sec and enabled_tools.
From source
git clone https://github.com/xihuai18/claude-code-mcp.git
cd claude-code-mcp
npm install
npm run build
npm start
Tools
claude_code — Start a new session
Start a new Claude Code session. The agent autonomously performs coding tasks: reading/writing files, running shell commands, searching code, managing git, and interacting with APIs.
| Parameter | Type | Required | Description |
|---|---|---|---|
prompt | string | Yes | Task or question for Claude Code |
cwd | string | No | Working directory (defaults to server cwd) |
allowedTools | string[] | No | Auto-approved tool names. Default: [] (none). Tools not in allowedTools/disallowedTools may surface permission requests via claude_code_check. Example: ["Bash", "Read", "Write", "Edit"] |
disallowedTools | string[] | No | Forbidden tool names. Default: [] (none). SDK behavior: disallowed tools are removed from the model's context. Takes precedence over allowedTools and will be denied even if later approved interactively |
maxTurns | number | No | Maximum number of agent reasoning steps. Each step may involve one or more tool calls. Default: SDK/Claude Code default |
model | string | No | Model to use (e.g. "claude-sonnet-4-5-20250929"). Default: SDK/Claude Code default |
systemPrompt | string | object | No | Override the agent's system prompt. Default: SDK/Claude Code default. Pass a string for full replacement, or { type: "preset", preset: "claude_code", append?: "..." } to extend the default prompt |
permissionRequestTimeoutMs | number | No | Timeout in milliseconds waiting for permission decisions. Default: 60000 |
advanced | object | No | Advanced/low-frequency parameters (see below) |
advanced object parameters (22 low-frequency parameters)
| Parameter | Type | Description |
|---|---|---|
advanced.tools | string[] | object | Define the base tool set. Default: SDK/Claude Code default toolset. Array of tool name strings, or { type: "preset", preset: "claude_code" } for the default toolset. allowedTools/disallowedTools further filter on top of this |
advanced.persistSession | boolean | Persist session history to disk (~/.claude/projects/). Default: true. Set false to disable. |
advanced.sessionInitTimeoutMs | number | Timeout in milliseconds waiting for system/init. Default: 10000 |
advanced.agents | object | Define custom sub-agents the main agent can delegate tasks to. Default: none. SDK default: if a sub-agent omits tools, it inherits all tools from the parent. |
advanced.agent | string | Name of a custom agent (defined in agents) to use as the primary agent. Default: omitted |
advanced.maxBudgetUsd | number | Maximum budget in USD. Default: SDK/Claude Code default |
advanced.effort | string | Effort level: "low", "medium", "high", "max". Default: SDK/Claude Code default |
advanced.betas | string[] | Beta features (e.g. ["context-1m-2025-08-07"]). Default: none |
advanced.additionalDirectories | string[] | Additional directories the agent can access beyond cwd. Default: none |
advanced.outputFormat | object | Structured output: { type: "json_schema", schema: {...} }. Default: omitted (plain text) |
advanced.thinking | object | Thinking mode: { type: "adaptive" }, { type: "enabled", budgetTokens: N }, or { type: "disabled" }. Default: SDK/Claude Code default |
advanced.pathToClaudeCodeExecutable | string | Path to the Claude Code executable. Default: SDK-bundled Claude Code (cli.js) |
advanced.mcpServers | object | MCP server configurations (key: server name, value: server config). Default: none |
advanced.sandbox | object | Sandbox configuration for isolating shell command execution (e.g., Docker container settings). Default: SDK/Claude Code default |
advanced.fallbackModel | string | Fallback model if the primary model fails or is unavailable. Default: none |
advanced.enableFileCheckpointing | boolean | Enable file checkpointing to track file changes during the session. Default: false |
advanced.includePartialMessages | boolean | When true, includes intermediate streaming messages in the response. Useful for real-time progress monitoring. Default: false |
advanced.strictMcpConfig | boolean | Enforce strict validation of MCP server configurations. Default: false |
advanced.settingSources | string[] | Which filesystem settings to load. Defaults to ["user", "project", "local"] (loads all settings and CLAUDE.md). Pass [] for SDK isolation mode |
advanced.debug | boolean | Enable debug mode for verbose logging. Default: false |
advanced.debugFile | string | Write debug logs to a specific file path (implicitly enables debug mode). Default: omitted |
advanced.env | object | Environment variables to merge with process.env and pass to the Claude Code process (user values take precedence). Default: inherit process.env |
Returns: { sessionId, status: "running", pollInterval, resumeToken? }
Notes:
resumeTokenis omitted by default, and is only returned whenCLAUDE_CODE_MCP_RESUME_SECRETis set on the server.- On error:
{ sessionId: "", status: "error", error }
Use claude_code_check to poll events and obtain the final result.
Notes:
- Subagents require the
Tasktool to be available to the primary agent. If you useallowedTools, include"Task"or the agent will be unable to invoke subagents.- If you configure
advanced.mcpServersand want the agent to auto-use tools from those servers without approvals, include the exact tool names inallowedTools(e.g.["mcp__my_server__tools/list"]). Otherwise you will see permission requests viaclaude_code_check.advanced.includePartialMessagesaffects the underlying SDK event stream; intermediate messages are captured as events and returned viaclaude_code_check(theclaude_codecall itself does not stream).
Security: Configure permissions based on your own scope. Callers (MCP clients / orchestrating agents) MUST set
allowedToolsanddisallowedToolsaccording to their own permission boundaries. Only pre-approve tools that you yourself are authorized to perform — do not grant the agent broader permissions than you have. For example, if you lack write access to a directory, do not includeWrite/EditinallowedTools. When in doubt, leave both lists empty and review each permission request individually viaclaude_code_check.
claude_code_reply — Continue a session
Continue an existing session by sending a follow-up message. The agent retains full context from previous turns including files read, code analysis, and conversation history.
| Parameter | Type | Required | Description |
|---|---|---|---|
sessionId | string | Yes | Session ID from a previous claude_code call |
prompt | string | Yes | Follow-up prompt |
forkSession | boolean | No | Create a branched copy of this session. Default: false |
permissionRequestTimeoutMs | number | No | Timeout in milliseconds waiting for permission decisions. Default: 60000 |
sessionInitTimeoutMs | number | No | Timeout in milliseconds waiting for fork system/init. Default: 10000 |
diskResumeConfig | object | No | Disk resume parameters (see below). Used when CLAUDE_CODE_MCP_ALLOW_DISK_RESUME=1 and in-memory session is missing |
diskResumeConfig object parameters (28 disk-resume-only parameters)
| Parameter | Type | Description |
|---|---|---|
diskResumeConfig.resumeToken | string | Resume token returned by claude_code / claude_code_reply. Required for disk resume fallback |
diskResumeConfig.cwd | string | Working directory. Required for disk resume. |
diskResumeConfig.allowedTools | string[] | Auto-approved tool names (see claude_code). Default: [] |
diskResumeConfig.disallowedTools | string[] | Forbidden tool names (see claude_code). Default: [] |
diskResumeConfig.tools | string[] | object | Base tool set (see claude_code). Default: SDK/Claude Code default |
diskResumeConfig.persistSession | boolean | Persist session history to disk. Default: true |
diskResumeConfig.maxTurns | number | Maximum number of agent reasoning steps. Default: SDK/Claude Code default |
diskResumeConfig.model | string | Model to use. Default: SDK/Claude Code default |
diskResumeConfig.systemPrompt | string | object | Override the agent's system prompt. Default: SDK/Claude Code default |
diskResumeConfig.agents | object | Custom sub-agent definitions (see claude_code). Default: none |
diskResumeConfig.agent | string | Primary agent name (see claude_code tool). Default: omitted |
diskResumeConfig.maxBudgetUsd | number | Maximum budget in USD. Default: SDK/Claude Code default |
diskResumeConfig.effort | string | Effort level. Default: SDK/Claude Code default |
diskResumeConfig.betas | string[] | Beta features. Default: none |
diskResumeConfig.additionalDirectories | string[] | Additional directories. Default: none |
diskResumeConfig.outputFormat | object | Structured output format. Default: omitted (plain text) |
diskResumeConfig.thinking | object | Thinking mode. Default: SDK/Claude Code default |
diskResumeConfig.resumeSessionAt | string | Resume only up to and including a specific message UUID. Default: omitted |
diskResumeConfig.pathToClaudeCodeExecutable | string | Path to Claude Code executable. Default: SDK-bundled Claude Code (cli.js) |
diskResumeConfig.mcpServers | object | MCP server configurations. Default: none |
diskResumeConfig.sandbox | object | Sandbox config for command isolation. Default: SDK/Claude Code default |
diskResumeConfig.fallbackModel | string | Fallback model. Default: none |
diskResumeConfig.enableFileCheckpointing | boolean | Enable file checkpointing. Default: false |
diskResumeConfig.includePartialMessages | boolean | Include intermediate streaming messages. Default: false |
diskResumeConfig.strictMcpConfig | boolean | Strict MCP config validation. Default: false |
diskResumeConfig.settingSources | string[] | Which filesystem settings to load. Default: ["user", "project", "local"] |
diskResumeConfig.debug | boolean | Debug mode. Default: false |
diskResumeConfig.debugFile | string | Debug log file path. Default: omitted |
diskResumeConfig.env | object | Environment variables. Default: inherit process.env (user values override) |
Returns: { sessionId, status: "running", pollInterval, resumeToken? }
Notes:
resumeTokenis omitted by default, and is only returned whenCLAUDE_CODE_MCP_RESUME_SECRETis set on the server.- On error:
{ sessionId, status: "error", error }
Use claude_code_check to poll events and obtain the final result.
Disk resume (optional): By default, claude_code_reply requires the session to exist in the MCP server's in-memory Session Manager. If you set CLAUDE_CODE_MCP_ALLOW_DISK_RESUME=1, it can attempt to resume using the Claude Code CLI's on-disk transcript even when the in-memory session is missing (e.g. after a restart / TTL cleanup). For safety, disk resume fallback requires CLAUDE_CODE_MCP_RESUME_SECRET to be set on the server and requires callers to pass diskResumeConfig.resumeToken (returned by claude_code / claude_code_reply when CLAUDE_CODE_MCP_RESUME_SECRET is set).
claude_code_session — Manage sessions
List, inspect, or cancel sessions.
| Parameter | Type | Required | Description |
|---|---|---|---|
action | string | Yes | "list", "get", or "cancel" |
sessionId | string | For get/cancel | Target session ID |
includeSensitive | boolean | No | Include cwd/systemPrompt/agents/additionalDirectories (default: false) |
Returns: { sessions, message?, isError? }
claude_code_check — Poll events and respond to permission requests
Poll session events/results and approve/deny pending permission requests.
| Parameter | Type | Required | Description |
|---|---|---|---|
action | string | Yes | "poll" or "respond_permission" |
sessionId | string | Yes | Target session ID |
cursor | number | No | Event cursor for incremental polling (poll only). Default: omitted (starts from the beginning of the buffer) |
responseMode | string | No | "minimal" (default) or "full" — controls payload size and redaction behavior |
maxEvents | number | No | Max events per poll (pagination via nextCursor). Default: 200 in "minimal"; unlimited in "full" |
requestId | string | For respond_permission | Permission request ID |
decision | string | For respond_permission | "allow" or "deny" |
denyMessage | string | No | Deny reason shown to Claude (deny only). Default: "Permission denied by caller" |
interrupt | boolean | No | When true, denying also interrupts the whole agent (deny only). Default: false |
pollOptions | object | No | Fine-grained poll control options (see below) |
permissionOptions | object | No | Advanced permission response options (see below) |
pollOptions object parameters (9 fine-grained poll controls)
| Parameter | Type | Description |
|---|---|---|
pollOptions.includeTools | boolean | When true, includes availableTools (poll only). Default: false (omitted until session init is received) |
pollOptions.includeEvents | boolean | When false, omits events (but nextCursor still advances). Default: true |
pollOptions.includeActions | boolean | When false, omits actions[] even if waiting_permission. Default: true |
pollOptions.includeResult | boolean | When false, omits top-level result even when idle/error. Default: true |
pollOptions.includeUsage | boolean | Include result.usage (default: true in full mode, false in minimal mode) |
pollOptions.includeModelUsage | boolean | Include result.modelUsage (default: true in full mode, false in minimal mode) |
pollOptions.includeStructuredOutput | boolean | Include result.structuredOutput (default: true in full mode, false in minimal mode) |
pollOptions.includeTerminalEvents | boolean | When true, keeps terminal result/error events in events even if top-level result is included. Default: false in "minimal", true in "full" |
pollOptions.includeProgressEvents | boolean | When true, includes progress events (tool_progress, auth_status) in the events stream. Default: false in "minimal", true in "full" |
permissionOptions object parameters (2 advanced permission response options)
| Parameter | Type | Description |
|---|---|---|
permissionOptions.updatedInput | object | Modified tool input to run (allow only). Default: none |
permissionOptions.updatedPermissions | array | Permission rule updates suggested/applied (allow only). Default: none |
Returns (poll and respond_permission): { sessionId, status, pollInterval?, cursorResetTo?, truncated?, truncatedFields?, events, nextCursor?, availableTools?, actions?, result?, cancelledAt?, cancelledReason?, cancelledSource?, lastEventId?, lastToolUseId? }
Notes:
- On error (e.g. invalid arguments, missing/expired session):
{ sessionId, isError: true, error } - Always treat
cursoras an incremental position: storenextCursorand pass it back on the next poll to avoid replaying old events. - If
cursorResetTois present, yourcursorwas too old (events were evicted); reset your cursor tocursorResetTo. - For safety, de-duplicate events by
event.idon the client side. - If
truncated=true, the server intentionally limited the payload (e.g.maxEvents) — continue polling withnextCursor. - In
"minimal"mode (default): assistant message events are slimmed (stripsusage,model,id,cache_controlfrom content blocks); noisy progress events (tool_progress,auth_status) are filtered out;lastEventId/lastToolUseIdare omitted;AgentResultomitsdurationApiMs/sessionTotalTurns/sessionTotalCostUsd. UseresponseMode: "full"or individualinclude*flags to restore any of these.
Usage Example
# 1) Start a new session (async start)
start = await mcp.call_tool("claude_code", {
"prompt": "Fix the authentication bug in src/auth.ts",
"cwd": "/path/to/project",
"allowedTools": ["Read", "Edit", "Bash", "Glob", "Grep"],
"advanced": {
"effort": "high",
"maxBudgetUsd": 5.0
}
})
session_id = json.loads(start)["sessionId"]
cursor = None
# 2) Poll until idle/error/cancelled
while True:
polled = await mcp.call_tool("claude_code_check", {
"action": "poll",
"sessionId": session_id,
"cursor": cursor,
"pollOptions": {
"includeProgressEvents": True
}
})
data = json.loads(polled)
cursor = data.get("nextCursor", cursor)
# If permission is needed, approve/deny via respond_permission
for action in data.get("actions", []) or []:
if action.get("type") == "permission":
await mcp.call_tool("claude_code_check", {
"action": "respond_permission",
"sessionId": session_id,
"requestId": action["requestId"],
"decision": "allow"
})
# Final result is available when status becomes idle/error
if data.get("status") in ["idle", "error", "cancelled"]:
final_result = data.get("result")
break
# 3) Manage sessions (list/get/cancel)
result = await mcp.call_tool("claude_code_session", {"action": "list"})
Windows Support
The Claude Code CLI bundled in the SDK requires Git for Windows (which includes bash.exe). If you run this MCP server on Windows and see:
Claude Code on Windows requires git-bash (https://git-scm.com/downloads/win).
This means the spawned CLI process cannot locate bash.exe. Your locally installed claude command may work fine — the issue is that the MCP server's child process may not inherit your shell environment.
Fix: set CLAUDE_CODE_GIT_BASH_PATH in your MCP server config.
For JSON-based MCP clients (Claude Desktop, Cursor, etc.):
{
"mcpServers": {
"claude-code": {
"command": "npx",
"args": ["-y", "@leo000001/claude-code-mcp"],
"env": {
"CLAUDE_CODE_GIT_BASH_PATH": "C:\\Program Files\\Git\\bin\\bash.exe"
}
}
}
}
For OpenAI Codex CLI (~/.codex/config.toml):
[mcp_servers.claude-code]
command = "npx"
args = ["-y", "@leo000001/claude-code-mcp"]
[mcp_servers.claude-code.env]
CLAUDE_CODE_GIT_BASH_PATH = "C:\\Program Files\\Git\\bin\\bash.exe"
Replace the path with your actual
bash.exelocation. Common paths:
C:\Program Files\Git\bin\bash.exe(default installer)To find yours:
where gitin CMD/PowerShell, then look forbash.exeunder the same Git root'sbin\folder.
Alternatively, set the environment variable system-wide so all processes inherit it:
# PowerShell (permanent, requires new terminal)
setx CLAUDE_CODE_GIT_BASH_PATH "C:\Program Files\Git\bin\bash.exe"
Security
- Async permission approvals — when a tool call needs approval, the session transitions to
waiting_permissionand surfaces requests viaclaude_code_check(actions[]). - No runtime privilege escalation tool — permission decisions are per-session (allow/deny lists + explicit approvals), and the server does not expose a
claude_code_configurebypass switch. - Environment variables are inherited — the spawned Claude Code process inherits all environment variables (including
ANTHROPIC_API_KEY) from the parent process by default. Theadvanced.envparameter merges withprocess.env(user-provided values take precedence), so you can safely add or override individual variables without losing existing ones. - Tool visibility vs approvals:
- Use
advanced.toolsto restrict which tools the agent can see (hidden tools cannot be called). - Use
allowedToolsto auto-approve specific tools without prompting (the SDK may still prompt for path-based restrictions likeblockedPath). - Use
disallowedToolsto hard-block tools; they are denied even if later approved viaclaude_code_check.
- Use
maxTurnsandadvanced.maxBudgetUsdprevent runaway execution.- Sessions auto-expire after 30 minutes of inactivity.
Environment Variables
All environment variables are optional. They are set on the MCP server process (not on the Claude Code child process — for that, use the env tool parameter).
| Variable | Description | Default |
|---|---|---|
CLAUDE_CODE_GIT_BASH_PATH | Path to bash.exe on Windows (see Windows Support) | Auto-detected |
CLAUDE_CODE_MCP_ALLOW_DISK_RESUME | Set to 1 to allow claude_code_reply to resume from on-disk transcripts when the in-memory session is missing | 0 (disabled) |
CLAUDE_CODE_MCP_RESUME_SECRET | HMAC secret used to validate resumeToken for disk resume fallback (recommended if disk resume is enabled) | (unset) |
How to configure
JSON-based MCP clients (Claude Desktop, Cursor, etc.) — add an "env" block:
{
"mcpServers": {
"claude-code": {
"command": "npx",
"args": ["-y", "@leo000001/claude-code-mcp"],
"env": {
"CLAUDE_CODE_MCP_ALLOW_DISK_RESUME": "1",
"CLAUDE_CODE_MCP_RESUME_SECRET": "change-me"
}
}
}
}
OpenAI Codex CLI — add an [mcp_servers.claude-code.env] section in ~/.codex/config.toml:
[mcp_servers.claude-code]
command = "npx"
args = ["-y", "@leo000001/claude-code-mcp"]
[mcp_servers.claude-code.env]
CLAUDE_CODE_MCP_ALLOW_DISK_RESUME = "1"
CLAUDE_CODE_MCP_RESUME_SECRET = "change-me"
System-wide — set via your shell profile or OS settings so all processes inherit them:
# bash / zsh
export CLAUDE_CODE_MCP_ALLOW_DISK_RESUME=1
export CLAUDE_CODE_MCP_RESUME_SECRET=change-me
# PowerShell (permanent, requires new terminal)
setx CLAUDE_CODE_MCP_ALLOW_DISK_RESUME 1
setx CLAUDE_CODE_MCP_RESUME_SECRET change-me
Development
npm install # Install dependencies
npm run build # Build with tsup
npm run typecheck # Type check with tsc
npm test # Run tests with vitest
npm run dev # Watch mode build
Architecture
MCP Client ←→ (stdio/JSON-RPC) ←→ MCP Server
├── Session Manager (Map<id, state>)
└── Claude Agent SDK (query())
Session persistence: The MCP server's Session Manager holds in-memory session metadata, a snapshot of session options (tool config, limits, cwd, allow/deny lists, etc.), and an event buffer used by claude_code_check. This metadata is not persisted to disk by the MCP server. The actual conversation history is persisted to disk by the Claude Code CLI (under ~/.claude/projects/) — this is managed by the SDK, not by this MCP server. By default, if the MCP server restarts or the session expires from memory, claude_code_reply will return SESSION_NOT_FOUND even though the CLI transcript may still exist on disk. You can opt into disk-resume behavior by setting CLAUDE_CODE_MCP_ALLOW_DISK_RESUME=1.
Sessions are automatically cleaned up after 30 minutes of idle time, or after 4 hours of continuous running.
Turn/Cost semantics: numTurns and totalCostUsd are per-call increments. For cumulative per-session totals, use sessionTotalTurns and sessionTotalCostUsd. When forkSession=true, the returned sessionId (and sessionTotal*) refer to the forked session; the original session totals are preserved.
Error Codes
MCP server validation/policy errors are returned as Error [CODE]: message where CODE is one of:
INVALID_ARGUMENT— invalid inputs (e.g. missing sessionId, empty cwd)SESSION_NOT_FOUND— session not found in memory (expired or server restarted)SESSION_BUSY— session currently runningPERMISSION_DENIED— operation not allowed by server policyPERMISSION_REQUEST_NOT_FOUND— permission request ID not found (already finished or expired)TIMEOUT— operation timed outCANCELLED— session was cancelledINTERNAL— unexpected error or protocol mismatch
For Claude Agent SDK execution failures, also check errorSubtype (e.g. error_max_turns, error_max_budget_usd, error_during_execution) and the returned result text.
License
MIT — see LICENSE
Contributing
See CONTRIBUTING.md for development setup and guidelines.