A unified MCP gateway for running Claude Code, Codex, Gemini, Grok, and Mistral CLI tools from a single endpoint. Ships sync and async job tools with persistence, session continuity, and cache-aware prompting including Claude's cache_control blocks. Built for cross-LLM validation workflows where you want second opinions or consensus checks without reconfiguring clients. Includes git worktree isolation for multi-agent review loops, HTTP transport with bearer auth for remote clients, and a Go bootstrapper with setup, doctor, and upgrade commands. Targets single-user personal appliance deployments with Sigstore-signed releases and npm provenance. Reach for this when you're orchestrating multi-model code review or testing assistant outputs across providers from one MCP integration point.
"Without consultation, plans are frustrated, but with many counselors they succeed." — Proverbs 15:22 (LSB)
A Model Context Protocol (MCP) gateway for running Claude Code, Codex, Gemini, Grok, and Mistral (Vibe) CLIs from one MCP endpoint, with durable async jobs, session continuity, cache-aware prompting, observability, and personal-appliance setup tooling.
Why developers try it: one local MCP endpoint for cross-LLM validation, multi-agent coding workflows, and repeatable assistant-led setup across five provider CLIs.
Current signals: CI and security workflows pass on main, OpenSSF Scorecard is published, OpenSSF Best Practices is passing, releases use Sigstore signing, and the package is MIT licensed.
npm install -g llm-cli-gateway
Or use directly with npx from an MCP client:
{
"mcpServers": {
"llm-gateway": {
"command": "npx",
"args": ["-y", "llm-cli-gateway"]
}
}
}
llm-cli-gateway is a single-user MCP gateway for cross-LLM validation and multi-agent coding workflows. It is more than a thin CLI wrapper:
promptParts, including explicit Claude cache_control when opted in.doctor --json, setup UI artifacts, provider setup snippets, Docker fallback, and checked release bundles.workspace; every HTTP/tunnel request must use a registered alias, session workspace, or [workspaces].default before provider execution. Local unrestricted filesystem access is the stdio transport.The repo ships agent-ready workflow skills under .agents/skills for async orchestration, session continuity, multi-LLM review, implement-review-fix loops, and secure approval-gated dispatch. Machine-readable DAG-TOML plans live under docs/plans and setup/install-plan.dag.toml for workflows that need deterministic sequencing and verification gates.
The next documentation focus is provider-specific skill and DAG-TOML pairs for each outbound CLI: Claude, Codex, Gemini, Grok, and Mistral Vibe. The implementation plan is tracked in docs/plans/provider-workflow-assets.dag.toml, with each provider asset expected to cover install/login checks, session behavior, approval modes, cache/telemetry surfaces, failure modes, and a smoke-test gate.
npm-shrinkwrap.json so registry installs resolve the audited release tree. Release gates regenerate it from package-lock.json, compare for parity, and run a registry-fidelity consumer install before publishing.socket.yml and under "Security Considerations" below. shellAccess and shrinkwrap are reviewed package capabilities/configuration for this CLI appliance, not hidden install behaviour.The personal-appliance contract keeps that surface intentionally narrow: one trusted user runs the gateway on a machine or volume they own, connects one MCP endpoint, and asks any connected client for cross-LLM validation.
The product contract is documented in docs/personal-mcp/PRODUCT_CONTRACT.md. It defines the single-user scope, security posture, target support matrix, and provider-support verification gates. Public setup guides must not claim ChatGPT, Claude web, Claude Desktop, Codex, Gemini CLI, Gemini web, or Grok inbound support until the corresponding provider/client path has been verified.
This project does not provide hosted multi-tenant credential custody. Provider credentials stay on the user's machine or user-owned deployment volume.
Release-readiness history is tracked in docs/personal-mcp/RELEASE_READINESS.md. Dogfooding evidence (which target LLMs guided setup, what unsafe suggestions were captured, and which findings were deferred from the initial personal-appliance rollout) is in docs/personal-mcp/DOGFOODING_RESULTS.md.
Current personal-appliance artifacts include:
LLM_GATEWAY_AUTH_TOKEN=<token> npm run start:httpnpm run doctorinstaller/ with setup, doctor --json, start, stop, status, repair, upgrade, uninstall, print-client-config, and verified bundle download commands.validate_with_models, second_opinion, compare_answers, red_team_review, consensus_check, ask_model, synthesize_validation, job_status, and job_result.Windows PowerShell:
$Version = '<version>'
$Base = "https://github.com/verivus-oss/llm-cli-gateway/releases/download/v$Version"
$InstallDir = Join-Path (Join-Path $env:LOCALAPPDATA 'Programs') 'llm-cli-gateway'
$ExeName = "llm-cli-gateway-$Version-windows-amd64.exe"
$BundleName = "llm-cli-gateway-bundle-$Version-windows-amd64.tar.gz"
$Exe = Join-Path $InstallDir 'llm-cli-gateway.exe'
$Checksums = Join-Path $InstallDir 'SHA256SUMS'
$ChecksumBundle = Join-Path $InstallDir 'SHA256SUMS.sigstore.json'
New-Item -ItemType Directory -Force $InstallDir | Out-Null
Invoke-WebRequest -UseBasicParsing "$Base/$ExeName" -OutFile $Exe
Invoke-WebRequest -UseBasicParsing "$Base/SHA256SUMS" -OutFile $Checksums
Invoke-WebRequest -UseBasicParsing "$Base/SHA256SUMS.sigstore.json" -OutFile $ChecksumBundle
cosign verify-blob $Checksums --bundle $ChecksumBundle --certificate-identity "https://github.com/verivus-oss/llm-cli-gateway/.github/workflows/release-installer.yml@refs/tags/v$Version" --certificate-oidc-issuer "https://token.actions.githubusercontent.com"
if ($LASTEXITCODE -ne 0) { throw "Sigstore verification failed for SHA256SUMS" }
function Get-ReleaseSha256($Name) {
$line = Select-String -Path $Checksums -Pattern "^[a-fA-F0-9]{64}\s+$([regex]::Escape($Name))$" | Select-Object -First 1
if (-not $line) { throw "No SHA256SUMS entry found for $Name" }
return (($line.Line -split "\s+")[0]).ToLowerInvariant()
}
if ((Get-FileHash $Exe -Algorithm SHA256).Hash.ToLowerInvariant() -ne (Get-ReleaseSha256 $ExeName)) { throw "Checksum mismatch for $ExeName" }
$env:RVWR_GATEWAY_BUNDLE_URL = "$Base/$BundleName"
$env:RVWR_GATEWAY_BUNDLE_SHA256 = Get-ReleaseSha256 $BundleName
& $Exe setup
& $Exe stop
& $Exe install-bundle
& $Exe start
& $Exe status
& $Exe doctor
The Windows installer keeps a stable llm-cli-gateway.exe command in
%LOCALAPPDATA%\Programs\llm-cli-gateway and adds that directory to the user
PATH. Do not script against release-versioned exe names after install.
# After downloading the binary that matches your OS/arch from a release:
cosign verify-blob SHA256SUMS --bundle SHA256SUMS.sigstore.json \
--certificate-identity "https://github.com/verivus-oss/llm-cli-gateway/.github/workflows/release-installer.yml@refs/tags/v<version>" \
--certificate-oidc-issuer "https://token.actions.githubusercontent.com"
sha256sum --check SHA256SUMS # verify before run (or `shasum -a 256 --check` on macOS)
chmod +x llm-cli-gateway-<ver>-<os>-<arch>
./llm-cli-gateway-<ver>-<os>-<arch> setup
./llm-cli-gateway-<ver>-<os>-<arch> install-bundle # uses the platform bundle URL/SHA256
./llm-cli-gateway-<ver>-<os>-<arch> start
./llm-cli-gateway-<ver>-<os>-<arch> doctor
# Upgrade: replace the binary, set the new bundle env vars, run upgrade.
./llm-cli-gateway-<new>-<os>-<arch> upgrade
# Uninstall: dry-run first, then run with --yes.
./llm-cli-gateway-<ver>-<os>-<arch> uninstall
./llm-cli-gateway-<ver>-<os>-<arch> uninstall --yes
Docker fallback:
LLM_GATEWAY_AUTH_TOKEN=$(openssl rand -hex 32) \
docker compose -f docker/personal.compose.yml up -d
docker compose -f docker/personal.compose.yml run --rm doctor
~/.llm-cli-gateway/logs.db with correlation IDs, token usage, duration, retry counts, and circuit breaker state. Browse with Datasette: datasette ~/.llm-cli-gateway/logs.dbstructuredContent (model, cli, correlationId, sessionId, durationMs, token counts)cache-state://global, cache-state://session/{id}, and cache-state://prefix/{hash} MCP resources return aggregate cache hit/miss/savings — tokens and hashes only, no prompt text. session_get includes a cacheState block when the session has prior requests.provider_tool_capabilities and provider-tools://catalog expose the gateway request fields, supported/degraded provider controls, local skill/tool discovery, and safe config-surface hints for Claude Code, Codex CLI, Gemini/Antigravity, Grok CLI/API, and Mistral Vibe. doctor --json includes a compact provider_capabilities summary for setup assistants.Every *_request and *_request_async tool accepts an optional promptParts field that structures the prompt for better cache hit rates. The gateway concatenates the parts in canonical order (system → tools → context → task) so that the stable prefix bytes precede the volatile task tail unchanged across calls, letting each provider's automatic prompt-caching land on the same content hash each time.
{
"promptParts": {
"system": "You are a helpful code reviewer.",
"tools": "You have access to Read, Grep, Bash.",
"context": "<long stable context block — file dumps, etc.>",
"task": "Review the changes in src/foo.ts for security issues."
}
}
prompt and promptParts are mutually exclusive — pass exactly one.
Per-CLI capability matrix (prefix discipline is automatic via promptParts for all; explicit levers are provider-specific):
| CLI | Prefix discipline | Explicit lever(s) |
|---|---|---|
| claude | yes | promptParts.cacheControl + outputFormat: "stream-json" (Anthropic cache_control breakpoints on stable blocks; ttl="1h" forced) |
| codex | yes | none (OpenAI implicit) |
| gemini | yes | none (implicit server-side) |
| grok | yes | compactionMode / compactionDetail (context compaction: `summary |
| mistral | yes | none (implicit) |
Claude example (explicit cacheControl)
claude_request({
promptParts: {
system: "You are a helpful code reviewer.",
context: "<long stable file dump>",
task: "Review the diff.",
cacheControl: { system: true, context: true } // task is never marked
},
outputFormat: "stream-json"
})
Gateway emits the stream-json stdin path with cache_control: {type:"ephemeral", ttl:"1h"} on marked blocks only.
Grok example (compaction)
grok_request({
promptParts: { system: "...", context: "...", task: "..." },
compactionMode: "segments",
compactionDetail: "balanced"
})
Emits --compaction-mode segments --compaction-detail balanced.
See docs/personal-mcp/PROVIDER_CACHE_SURFACES.md for full surfaces, telemetry differences (e.g. Grok -p vs ACP), exact stream-json payload shapes, and cross-LLM review notes.
Opt-in flags (all default off) live under [cache_awareness] in ~/.llm-cli-gateway/config.toml.
*_request_async + polling tools.github/workflows/security.yml runs actionlint, zizmor, shellcheck, typos, osv-scanner, gitleaks, and lychee on every push and PR (see SECURITY.md for the threat model)Node.js >= 24.4.0 is required (engines.node in package.json). The gateway uses Node's built-in node:sqlite module for persistence — there is no native binding to compile and no install scripts run. The 24.4 floor is where allowBareNamedParameters defaults to true, which the persistence layer relies on.
Before using this gateway, you need to install the CLI tools you want to use:
# Installation instructions for Claude Code
# Visit: https://docs.anthropic.com/claude-code
npm install -g @anthropic-ai/claude-code
npm install -g @openai/codex
codex login
The Gemini provider runs through Google Antigravity CLI (agy).
curl -fsSL https://antigravity.google/cli/install.sh | bash
# Docs: https://antigravity.google/docs/cli-overview
curl -fsSL https://x.ai/cli/install.sh | bash
grok login # OAuth flow; for headless auth, set XAI_API_KEY
# Docs: https://docs.x.ai/build/overview
# Pick one — the gateway's cli_upgrade auto-detects which one you used.
curl -LsSf https://mistral.ai/vibe/install.sh | bash
pip install mistral-vibe
uv tool install mistral-vibe
brew install mistral-vibe
vibe auth login
# Current Vibe defaults session logging to enabled. If an older config disabled it,
# edit ~/.vibe/config.toml and set:
# [session_logging]
# enabled = true
Vibe-specific notes:
VIBE_ACTIVE_MODEL environment variable —
Vibe has no --model flag. The gateway discovers ~/.vibe/config.toml /
VIBE_MODELS, injects VIBE_ACTIVE_MODEL only when a model is explicitly
requested or Vibe config needs recovery, and retries once after a
model-not-found failure with refreshed discovery.permissionMode is the Vibe --agent name. Builtins are
default | plan | accept-edits | auto-approve; Vibe also accepts install-gated
builtins (e.g. lean) and custom agents from ~/.vibe/agents, so any name is
passed through and Vibe validates availability. The gateway's
programmatic-mode default is auto-approve; pick a stricter mode
explicitly if you need approval gates.allowedTools is allow-list only — the gateway emits one
--enabled-tools <tool> flag per entry. disallowedTools is accepted in
the schema for caller-side parity but is silently ignored at the CLI
boundary (a logger.info warning records the no-op).cli_upgrade --cli mistral detects whether you used
pip / uv / brew and dispatches the matching upgrade command. Running
vibe update is not a thing.npm install -g llm-cli-gateway
Or use directly with npx:
{
"mcpServers": {
"llm-gateway": {
"command": "npx",
"args": ["-y", "llm-cli-gateway"]
}
}
}
git clone https://github.com/verivus-oss/llm-cli-gateway.git
cd llm-cli-gateway
npm install
npm run build
For clients that already support local stdio MCP servers, add a configuration like:
{
"mcpServers": {
"llm-cli-gateway": {
"command": "node",
"args": ["/path/to/llm-cli-gateway/dist/index.js"]
}
}
}
Stdio is the recommended path for unrestricted machine-local development access. HTTP MCP, including localhost HTTP and tunneled HTTPS, is treated as remote-capable for provider execution: provider tools must resolve a registered workspace alias, a session workspace, or [workspaces].default before spawning a CLI. Remote clients should pass relative workingDir, addDir, and include-directory values inside the selected workspace; disabling auth or using a no-auth connector path is not a filesystem bypass.
This generic stdio example is not provider-support verification for the Personal MCP Appliance. Client-specific setup guides for ChatGPT, Claude web, Claude Desktop, Codex, Gemini CLI, Gemini web, and Grok remain gated by the provider-support matrix in docs/personal-mcp/PRODUCT_CONTRACT.md.
The personal-appliance surface exposes simplified validation tools for non-developer clients. These tools start provider CLI jobs through the durable async job manager and return normalized provider status plus raw job references.
validate_with_models: ask two or more providers to independently validate a question.second_opinion: ask one provider to review an answer.red_team_review: challenge a plan, answer, or document for risks and failure modes.consensus_check: check whether providers agree with a claim.ask_model: ask one provider through the simplified surface.synthesize_validation: run an explicit judge model after provider results have been collected.list_available_models: list the models each provider CLI exposes through the simplified surface.job_status and job_result: poll and collect validation job outputs.The validation report preserves per-provider disagreement. Optional judge synthesis is explicit about which provider produced the judge job.
claude_requestExecute a Claude Code request with optional session management.
Parameters:
prompt (string, optional*): The prompt to send (1-100,000 chars). *Exactly one of prompt or promptParts is required (mutually exclusive)model (string, optional): Model name or alias (use list_models for available values; supports latest)outputFormat (string, optional): Output format (text|json|stream-json), default: stream-json — the gateway parses NDJSON usage events for token/cost observability; override to text only when you want unparsed stdoutsessionId (string, optional): Specific session ID to usecontinueSession (boolean, optional): Continue the active sessioncreateNewSession (boolean, optional): Always create a new sessionforkSession (boolean, optional): Fork the resumed session instead of appending to itallowedTools (string[], optional): Restrict Claude tools to this allow-listdisallowedTools (string[], optional): Explicitly deny listed Claude toolspermissionMode (string, optional): Claude permission mode (default|acceptEdits|plan|auto|dontAsk|bypassPermissions); preferred over dangerouslySkipPermissionsdangerouslySkipPermissions (boolean, optional): Deprecated — maps to permissionMode: "bypassPermissions"; permissionMode wins when both are setagent (string, optional): Named sub-agent to run asagents (string, optional): Inline agent definitions JSONsystemPrompt / appendSystemPrompt (string, optional): Replace or extend the system promptmaxBudgetUsd (number, optional): Budget cap in USD for the requestmaxTurns (integer, optional): Agent-loop turn capeffort (string, optional): Reasoning effort (low|medium|high|xhigh|max)fallbackModel (string, optional): Auto-fallback model when the default is overloadedjsonSchema (string, optional): JSON Schema literal constraining structured outputaddDir (string[], optional): Additional workspace directoriesnoSessionPersistence (boolean, optional): Ephemeral session (not persisted to disk)settingSources / settings / tools (optional): Setting sources to load, settings JSON path/literal, built-in tool restrictionexcludeDynamicSystemPromptSections (boolean, optional): Trim dynamic system prompt sectionsapprovalStrategy (string, optional): "legacy" (default) or "mcp_managed"approvalPolicy (string, optional): "strict", "balanced", or "permissive"mcpServers (string[], optional): Names of MCP servers to expose to Claude (default: none). The gateway resolves each name to a launch command from its local registry / Codex MCP config; unknown names are reported as unavailable. Configure the servers your deployment uses in the gateway environment.strictMcpConfig (boolean, optional): Require Claude to use only supplied MCP config, default: true (request fails if any requested server is unavailable)optimizePrompt (boolean, optional): Optimize prompt for token efficiency (44% reduction), default: falseoptimizeResponse (boolean, optional): Optimize response for token efficiency (37% reduction), default: falsecorrelationId (string, optional): Request trace ID (auto-generated if omitted)idleTimeoutMs (integer, optional): Kill a stuck process after output inactivity; 30,000 to 3,600,000 msworktree (boolean|object, optional): Run inside a gateway-owned git worktree (slice λ)promptParts (object, optional): Cache-aware structured prompt { system?, tools?, context?, task }; mutually exclusive with promptforceRefresh (boolean, optional): Bypass dedup and force a fresh CLI run, default: falseWorkspace boundary: stdio callers may use machine-local paths directly. HTTP/tunnel callers must pass workspace or rely on a configured default/session workspace; path fields are then validated relative to that workspace. [workspaces].allow_unregistered_working_dir is a stdio/local legacy setting and does not allow arbitrary HTTP working directories or additional directories.
Response extras:
approval: Approval decision record when approvalStrategy="mcp_managed"mcpServers: Requested/enabled/missing MCP servers for this callExample:
{
"prompt": "Write a Python function to calculate fibonacci numbers",
"model": "sonnet",
"continueSession": true,
"optimizePrompt": true,
"optimizeResponse": true
}
codex_requestExecute a Codex request with optional session tracking.
Parameters:
prompt (string, optional*): The prompt to send (1-100,000 chars). *Exactly one of prompt or promptParts is required (mutually exclusive)model (string, optional): Model name or alias (use list_models for available values; supports latest, recommended: gpt-5.5)fullAuto (boolean, optional): Deprecated — expands to --sandbox workspace-write only (current Codex no longer accepts approval-policy flags); prefer sandboxModesandboxMode (string, optional): Codex sandbox (read-only|workspace-write|danger-full-access)dangerouslyBypassApprovalsAndSandbox (boolean, optional): Request Codex bypass flagsapprovalStrategy (string, optional): "legacy" (default) or "mcp_managed"approvalPolicy (string, optional): "strict", "balanced", or "permissive"mcpServers (string[], optional): MCP servers expected for Codex execution contextsessionId (string, optional): Session identifier for trackingresumeLatest (boolean, optional): Resume the most recent Codex session in the current cwd (codex exec resume --last); ignored if sessionId is setcreateNewSession (boolean, optional): Always create a new sessionforceRefresh (boolean, optional): Bypass dedup and force a fresh CLI run, default: falseoutputFormat (string, optional): text (default) or json (--json JSONL events for token usage extraction)outputSchema (string|object, optional): Codex --output-schema — path or inline JSON SchemaworkingDir (string, optional): Working root for this session (-C/--cd; new sessions only)addDir (string[], optional): Additional writable workspace directories (one --add-dir per entry; new sessions only)ephemeral (boolean, optional): Codex --ephemeral (no session persistence)images (string[], optional): Image attachments (one -i <path> per entry)profile (string, optional): Codex --profile <name> (new sessions only; ignored with a logged warning on resume)configOverrides (object, optional): Codex -c key=value overridesignoreRules / ignoreUserConfig (boolean, optional): Codex --ignore-rules / --ignore-user-configworktree (boolean|object, optional): Run inside a gateway-owned git worktree (slice λ)promptParts (object, optional): Cache-aware structured prompt { system?, tools?, context?, task }; mutually exclusive with promptoptimizePrompt (boolean, optional): Optimize prompt for token efficiency, default: falseoptimizeResponse (boolean, optional): Optimize response for token efficiency, default: falsecorrelationId (string, optional): Request trace ID (auto-generated if omitted)idleTimeoutMs (integer, optional): Kill a stuck Codex process after output inactivity; 30,000 to 3,600,000 msResponse extras:
approval: Approval decision record when approvalStrategy="mcp_managed"mcpServers: Requested MCP servers for this callExample:
{
"prompt": "Create a REST API endpoint",
"model": "gpt-5.5",
"sandboxMode": "workspace-write",
"optimizePrompt": true
}
codex_fork_sessionFork an existing Codex session into a new branch (codex fork <SESSION_ID|--last> <prompt>), preserving the original session's history while the fork diverges.
Parameters:
prompt (string, required): Prompt text for the forked session (1-100,000 chars)sessionId (string, optional): Codex session UUID to fork from (mutually exclusive with forkLast)forkLast (boolean, optional): Fork the most recent Codex session instead of naming onemodel (string, optional): Model name or alias (e.g. gpt-5.5, latest)sandboxMode (string, optional): Codex sandbox (read-only|workspace-write|danger-full-access)correlationId (string, optional): Request trace ID (auto-generated if omitted)idleTimeoutMs (number, optional): Idle timeout in ms (30s-1h, omit for CLI default)gemini_requestExecute a Google Antigravity CLI (agy) request with session support.
Parameters:
prompt (string, optional*): The prompt to send (1-100,000 chars). *Exactly one of prompt or promptParts is required (mutually exclusive)model (string, optional): Model name or alias (use list_models for available values; supports latest, pro, flash)sessionId (string, optional): Session ID to resumeresumeLatest (boolean, optional): Resume the latest session automaticallycreateNewSession (boolean, optional): Always create a new sessionapprovalMode (string, optional): Antigravity approval mode in legacy mode. Only default (prompted execution) and yolo (emits --dangerously-skip-permissions) are accepted; auto_edit and plan are rejected with an error.approvalStrategy (string, optional): "legacy" (default) or "mcp_managed"approvalPolicy (string, optional): "strict", "balanced", or "permissive"includeDirs (string[], optional): Additional workspace directories (passed as --add-dir)sandbox (boolean, optional): Run Antigravity in sandbox mode (--sandbox)outputFormat (string, optional): text only. Antigravity print mode emits text; json and stream-json are rejected.mcpServers, allowedTools, policyFiles, adminPolicyFiles, attachments (string[], optional) and skipTrust (boolean, optional): Unsupported by Antigravity CLI — non-empty values (or skipTrust: true) are rejected with an explanatory error. Retained in the schema for caller parity.yolo (boolean, optional): Auto-approve all; equivalent to approvalMode: "yolo". Emits --dangerously-skip-permissionsworktree (boolean|object, optional): Run inside a gateway-owned git worktree (slice λ)promptParts (object, optional): Cache-aware structured prompt { system?, tools?, context?, task }; mutually exclusive with promptoptimizePrompt (boolean, optional): Optimize prompt for token efficiency, default: falseoptimizeResponse (boolean, optional): Optimize response for token efficiency, default: falsecorrelationId (string, optional): Request trace ID (auto-generated if omitted)idleTimeoutMs (integer, optional): Kill a stuck process after output inactivity; 30,000 to 3,600,000 msforceRefresh (boolean, optional): Bypass dedup and force a fresh CLI run, default: falseResponse extras:
approval: Approval decision record when approvalStrategy="mcp_managed"mcpServers: Requested MCP servers for this callExample:
{
"prompt": "Explain quantum computing",
"model": "latest",
"resumeLatest": true,
"optimizePrompt": true
}
grok_requestExecute a Grok CLI (xAI) request with session support.
Parameters:
prompt (string, optional*): The prompt to send (1-100,000 chars). *Exactly one of prompt or promptParts is required (mutually exclusive)model (string, optional): Model name or alias (e.g. grok-build, latest)outputFormat (string, optional): "plain" (default), "json", or "streaming-json"sessionId (string, optional): Session ID to resume (--resume <id>)resumeLatest (boolean, optional): Resume the most recent session in the current cwd (--continue)createNewSession (boolean, optional): Always create a new sessionalwaysApprove (boolean, optional): Auto-approve all tool executions (--always-approve) in legacy modepermissionMode (string, optional): default|acceptEdits|auto|dontAsk|bypassPermissions|planeffort (string, optional): low|medium|high|xhigh|maxreasoningEffort (string, optional): Reasoning effort for reasoning modelsapprovalStrategy (string, optional): "legacy" (default) or "mcp_managed"approvalPolicy (string, optional): "strict", "balanced", or "permissive"mcpServers (string[], optional): MCP server names tracked for approvals (Grok manages its own MCP config via grok mcp)allowedTools (string[], optional): Allowed built-in tools (passed as --tools comma list)disallowedTools (string[], optional): Disallowed built-in tools (passed as --disallowed-tools comma list)maxTurns (integer, optional): Agent-loop iteration cap (--max-turns)workingDir (string, optional): Working directory for this invocation (--cwd)sandbox (string, optional): Sandbox profile for filesystem/network access (--sandbox, freeform; also via GROK_SANDBOX)rules (string, optional): Extra rules appended to the system prompt (--rules; supports @file prefix)systemPromptOverride (string, optional): Replace the agent's system prompt entirelyallow / deny (string[], optional): Permission allow/deny rules (one --allow/--deny per entry)compactionMode (string, optional): summary (default) |transcript|segmentscompactionDetail (string, optional): none|minimal|balanced|verbose (segments mode only)agent (string, optional): Agent name or definition file pathagents (string|object, optional): Inline subagent definitions JSONbestOfN (integer, optional): Run the task N ways in parallel and pick the best (headless only)check (boolean, optional): Append a self-verification loop (headless only)disableWebSearch (boolean, optional): Disable web search and remote retrieval toolstodoGate (boolean, optional): Enable runtime turn-end TodoGate (session-scoped)verbatim (boolean, optional): Send the prompt exactly as given (also skips gateway prompt optimisation)promptFile / promptJson / single (optional): Single-turn prompt from a file / JSON blocks / literalexperimentalMemory / noMemory (boolean, optional): Enable/disable cross-session memorynoAltScreen / noPlan / noSubagents (boolean, optional): Disable alt screen / plan mode / subagent spawningoauth (boolean, optional): Use OAuth during authenticationrestoreCode (boolean, optional): Check out the original session commit when resumingleaderSocket (string, optional): Custom leader socket path (--leader-socket, Grok 0.2.32+; default ~/.grok/leader.sock) — targets an isolated leader process, e.g. a local/branch Grok buildnativeWorktree (boolean|string, optional): Grok's own --worktree flag (true → bare, string → named); distinct from the gateway worktree optionworktree (boolean|object, optional): Run inside a gateway-owned git worktree (slice λ)promptParts (object, optional): Cache-aware structured prompt { system?, tools?, context?, task }; mutually exclusive with promptoptimizePrompt (boolean, optional): Optimize prompt for token efficiency, default: falseoptimizeResponse (boolean, optional): Optimize response for token efficiency, default: falsecorrelationId (string, optional): Request trace ID (auto-generated if omitted)idleTimeoutMs (integer, optional): Kill a stuck process after output inactivity; 30,000 to 3,600,000 msforceRefresh (boolean, optional): Bypass dedup and force a fresh CLI run, default: falseExample:
{
"prompt": "Summarize the latest commit message in 1 sentence",
"model": "grok-build",
"effort": "low"
}
Every async job is persisted to a job store as it transitions through running → completed/failed/canceled. This makes the gateway a durable collection layer:
*_request / *_request_async calls within the dedup window (default 1 hour) short-circuit onto the existing running or completed job — the caller gets back the same job ID instead of starting a duplicate run. This directly fixes the "agent times out polling, re-issues, and the whole job starts over" failure mode.llm_job_status and llm_job_result work across gateway restarts. Job rows live for 30 days by default; callers can collect results long after the in-memory cache has evicted them.orphaned on the next gateway boot (the detached child can't be reattached to). Their captured partial output remains readable.forceRefresh: true on any request tool to bypass dedup and force a fresh CLI run.The job-store backend is configured by ~/.llm-cli-gateway/config.toml (override with LLM_GATEWAY_CONFIG=/path/to/config.toml). Example:
[persistence]
backend = "sqlite" # "sqlite" | "memory" | "postgres" | "none"
path = "~/.llm-cli-gateway/logs.db" # for sqlite
# dsn = "postgresql://user:pw@host/db" # for postgres (interface only — impl not yet shipped)
retentionDays = 30
dedupWindowMs = 3600000
acknowledgeEphemeral = false # required to enable async tools with memory backend
Backends:
sqlite (default) — durable, file-backed. Safe for single-instance deployments.memory — in-process Map. Lost on gateway exit. Requires acknowledgeEphemeral = true to be loaded. Suitable for tests and ephemeral CI gateways.postgres — interface only, implementation not yet shipped. Selecting this backend throws at startup.none — no store. *_request_async, llm_job_status, llm_job_result, and llm_job_cancel are NOT registered on the gateway. This is a structural invariant: agents that try to call async tools against a gateway with backend = "none" get a clean "tool not found" at connect time instead of silent in-memory loss after the 1-hour TTL. Use llm_process_health to inspect the resolved persistence state programmatically.Legacy environment variables (deprecated; emit a warning at startup):
LLM_GATEWAY_LOGS_DB / LLM_GATEWAY_JOBS_DB — none selects backend = "none"; any other value selects backend = "sqlite" with that path.LLM_GATEWAY_JOB_RETENTION_DAYS — overrides retentionDays.LLM_GATEWAY_DEDUP_WINDOW_MS — overrides dedupWindowMs.LLM_GATEWAY_ACKNOWLEDGE_EPHEMERAL — 1/true/yes sets acknowledgeEphemeral = true.By default, all gateway data is global per user, not per project. With no overrides, every Claude Code window — across every repo — spawns its own gateway subprocess but they all read and write the same files:
~/.llm-cli-gateway/logs.db (async jobs + flight recorder)~/.llm-cli-gateway/sessions.json (CLI sessions)~/.llm-cli-gateway/config.toml (resolved config)This is usually what you want — session_list from repo A shows sessions from repo B, an async job started in window A can be polled from window B, and the 1-hour dedup window catches re-issues across windows. SQLite WAL mode makes concurrent access from multiple gateway subprocesses safe.
If you instead want per-project isolation (e.g. unrelated repos shouldn't share session lists or risk false dedup hits), point each project at its own config file. In .claude/settings.local.json for the project:
{
"mcpServers": {
"llm-gateway": {
"env": {
"LLM_GATEWAY_CONFIG": "${workspaceFolder}/.gateway/config.toml"
}
}
}
}
…and put a per-project config.toml in the repo:
[persistence]
backend = "sqlite"
path = "/srv/repos/.../my-repo/.gateway/logs.db"
Now every gateway subprocess spawned for this repo's Claude Code window reads its own config and writes to its own SQLite file; sessions, jobs, and dedup state are scoped to the repo. Other repos keep using the global default. llm_process_health.persistence.sources.configFile lets an agent confirm which config it's actually running under.
If you want an LLM agent to perform this setup deterministically — rather than reading the prose above and guessing — copy the following DAG-TOML into the repo (e.g. docs/planning/per-project-gateway-isolation.toml) and point your agent at it. The schema is agent-assurance template_kind = "implementation-dag". The agent MUST execute units in layer order, must not skip the verification unit, and must treat any failed gate as blocking.
[meta]
schema_version = "1.0.0"
template_kind = "implementation-dag"
docs = "https://github.com/verivus-oss/agent-assurance/blob/main/SPEC.md"
confidentiality = "public"
title = "Per-project llm-cli-gateway persistence isolation"
spec = "https://github.com/verivus-oss/llm-cli-gateway#per-project-isolation"
created = "YYYY-MM-DD"
total_units = 5
tier1_units = ["U01","U02","U03","U04","U05"]
tier2_units = []
tier3_units = []
# ============================================================================
# [policy.agent] — persona for the agent performing the configuration.
# ============================================================================
[policy.agent]
name = "Gateway Persistence Isolator"
role = "Configuration Engineer"
purpose = "Configure the llm-cli-gateway MCP server so its async job store, sessions, dedup state, and flight recorder are scoped to THIS repository instead of the per-user default at ~/.llm-cli-gateway/."
validation_type = "Structural + Runtime Verification"
workflow_initiator = false
description = "Writes a repo-local config.toml, registers an LLM_GATEWAY_CONFIG override in .claude/settings.local.json, restarts the MCP server, and confirms via llm_process_health that the gateway is now reading the repo-local config and writing to the repo-local SQLite path."
[policy.agent.orchestration]
consumes_events = ["PerProjectIsolationRequested"]
produces_events = ["PerProjectIsolationComplete"]
[policy.agent.responsibilities]
items = [
"Create the repo-local gateway data directory and add it to .gitignore.",
"Write a config.toml that pins backend=sqlite to a repo-local path.",
"Register the LLM_GATEWAY_CONFIG env override in .claude/settings.local.json (NOT .mcp.json — that file is committed and shared).",
"Trigger an MCP server reconnect.",
"Verify via llm_process_health that the resolved configFile and dbPath are the repo-local values.",
]
# ============================================================================
# [policy.instance] — concrete paths the agent fills in for THIS repo.
# Agent MUST replace <REPO_ABS_PATH> with the absolute path to the repo
# before emitting any artefact. Relative paths in config.toml MUST be
# expanded to absolute — the gateway does not re-resolve them per cwd.
# ============================================================================
[policy.instance]
repo_abs_path = "<REPO_ABS_PATH>" # e.g. /srv/repos/me/my-project
gateway_data_dir_relative = ".gateway" # repo-relative directory
config_toml_relative = ".gateway/config.toml"
sqlite_db_relative = ".gateway/logs.db"
claude_local_settings_relative = ".claude/settings.local.json"
gitignore_relative = ".gitignore"
mcp_server_name = "llm-gateway" # must match the entry in .mcp.json
# ============================================================================
# [policy.gates] — blocking checks. Any failure stops the workflow.
# ============================================================================
[policy.gates]
gate_repo_abs_path_resolved = "policy.instance.repo_abs_path must NOT be the literal string '<REPO_ABS_PATH>' when U01 starts."
gate_config_is_committed = "policy.instance.config_toml_relative MAY be committed. policy.instance.claude_local_settings_relative MUST NOT be committed (it is per-developer). Agent MUST verify .gitignore covers .claude/settings.local.json if absent."
gate_no_legacy_env_leak = "Agent MUST grep the shell init files for LLM_GATEWAY_LOGS_DB / LLM_GATEWAY_JOBS_DB. If set, the legacy env var will override the new config and the deprecation warning will fire at every gateway boot. The agent reports this as a finding and asks the operator to unset before proceeding."
gate_health_confirms_isolation = "U05 MUST observe llm_process_health.persistence.sources.configFile == policy.instance.repo_abs_path + '/' + policy.instance.config_toml_relative AND llm_process_health.persistence.path == policy.instance.repo_abs_path + '/' + policy.instance.sqlite_db_relative. Anything else means the override did not take effect."
# ============================================================================
# [policy.evidence] — what each unit must emit so the work is auditable.
# ============================================================================
[policy.evidence]
per_unit_required_fields = [
"unit_id", # U01..U05
"status", # "completed" | "failed"
"artefact_paths", # files written / modified
"stdout_tail", # last 20 lines of any command output
"verification_quote", # for U05, the verbatim llm_process_health.persistence block
]
findings_required_fields = [
"gate_id", # which gate failed
"observed",
"expected",
"remediation",
]
# ============================================================================
# Units. Execute in layer order. U01..U03 modify the working tree; U04
# triggers a reconnect; U05 is the verification gate that decides success.
# ============================================================================
[units.U01]
name = "create-repo-local-data-dir"
summary = "mkdir -p <repo>/.gateway and append /.gateway/ to .gitignore (creating .gitignore if missing). The gateway will write logs.db, logs.db-wal, logs.db-shm here — none should be committed."
layer = 0
tier = 1
status = "pending"
depends_on = []
blocks = ["U02"]
estimated_loc = 5
files_modify = [".gitignore"]
produces = ["ART:gateway-data-dir"]
consumes = []
[units.U02]
name = "write-config-toml"
summary = "Write <repo>/.gateway/config.toml with [persistence] backend='sqlite' and path=<absolute-path-to-repo>/.gateway/logs.db. Path MUST be absolute. Do NOT use ~ — the gateway expands ~ but [persistence].path is read literally if not prefixed with ~/, and Claude Code may launch the gateway with a HOME that surprises you."
layer = 1
tier = 1
status = "pending"
depends_on = ["U01"]
blocks = ["U03"]
estimated_loc = 10
files_modify = [".gateway/config.toml"]
produces = ["ART:gateway-config"]
consumes = ["ART:gateway-data-dir"]
[units.U03]
name = "register-llm-gateway-config-env-in-claude-local-settings"
summary = "Add (or merge) an mcpServers.<mcp_server_name>.env entry in .claude/settings.local.json that sets LLM_GATEWAY_CONFIG to the absolute path of .gateway/config.toml. Do NOT modify .mcp.json — that file is committed and the path would be wrong for every other developer. If .claude/settings.local.json already has an mcpServers.<mcp_server_name> entry, the agent MUST merge into the existing env map (preserving other keys), not overwrite the whole entry."
layer = 2
tier = 1
status = "pending"
depends_on = ["U02"]
blocks = ["U04"]
estimated_loc = 20
files_modify = [".claude/settings.local.json"]
produces = ["ART:claude-local-settings"]
consumes = ["ART:gateway-config"]
[units.U04]
name = "trigger-mcp-reconnect"
summary = "Ask the operator to run /mcp in Claude Code (or restart Claude Code) so the gateway subprocess is re-spawned under the new env. The agent cannot do this itself — MCP server lifecycle is owned by the host."
layer = 3
tier = 1
status = "pending"
depends_on = ["U03"]
blocks = ["U05"]
estimated_loc = 0
files_modify = []
produces = ["OUT:mcp-reconnected"]
consumes = ["ART:claude-local-settings"]
[units.U05]
name = "verify-via-llm-process-health"
summary = "Call llm_process_health and assert the returned persistence block satisfies policy.gates.gate_health_confirms_isolation. Quote the verbatim persistence block in evidence. If the assertion fails, the agent MUST NOT mark the workflow complete — it must emit a finding under policy.evidence.findings_required_fields, naming the observed vs. expected configFile/path, and stop."
layer = 4
tier = 1
status = "pending"
depends_on = ["U04"]
blocks = []
estimated_loc = 5
files_modify = []
produces = ["ART:isolation-verification","OUT:per-project-isolation-complete"]
consumes = ["OUT:mcp-reconnected"]
Why this matters for agents: the gateway has multiple configuration surfaces (TOML file, env-var overrides, two different MCP settings files) and one easy mistake — editing the committed .mcp.json instead of the local-only .claude/settings.local.json — will silently break the per-project scope for every other developer on the repo. The DAG above encodes the correct sequence, the verification gate, and the failure modes explicitly so an agent can execute it without inference.
mistral_requestRun a Mistral Vibe agentic coding request. Like grok_request in shape, but with Vibe's specific surface:
model (string, optional): Vibe model alias (for example mistral-medium-3.5 or latest). The resolved value is injected via the VIBE_ACTIVE_MODEL environment variable; omit it to let the gateway discover Vibe config and avoid stale hardcoded defaults.permissionMode: the Vibe --agent name — builtins default | plan | accept-edits | auto-approve, or any install-gated/custom agent. Emitted as --agent <name>. Defaults to auto-approve in programmatic mode.allowedTools (string[], optional): One --enabled-tools <tool> flag per entry (allow-list only).disallowedTools (string[], optional): Accepted for parity with the other providers; ignored at the CLI boundary with a logged warning.outputFormat (string, optional): Vibe 2.x values are "text", "json", or "streaming"; legacy aliases "plain" and "stream-json" are accepted and normalized before spawn.sessionId / resumeLatest / createNewSession: standard session controls. Current Vibe defaults session logging to enabled; if an older config has [session_logging] enabled = false, doctor --json surfaces an actionable next-action.trust (boolean, optional): Emit --trust so Vibe trusts the cwd for this invocation only (not persisted; skips the interactive trust prompt)maxTurns (integer, optional): Agent-loop iteration cap (--max-turns, programmatic mode only)maxPrice (number, optional): Interrupt when cumulative cost crosses this USD cap (--max-price, programmatic mode only)maxTokens (integer, optional): Cap cumulative prompt + completion tokens (--max-tokens, programmatic mode only)workingDir (string, optional): Change to this directory before running (--workdir)addDir (string[], optional): Additional writable workspace directories (one --add-dir per entry)approvalStrategy (string, optional): "legacy" (default) or "mcp_managed"approvalPolicy (string, optional): "strict", "balanced", or "permissive"mcpServers (string[], optional): MCP server names tracked for approvals (Vibe manages its own MCP config via vibe mcp)worktree (boolean|object, optional): Run inside a gateway-owned git worktree (slice λ)promptParts (object, optional): Cache-aware structured prompt { system?, tools?, context?, task }; mutually exclusive with promptoptimizePrompt / optimizeResponse (boolean, optional): Token-efficiency optimisation, default: falsecorrelationId (string, optional): Request trace ID (auto-generated if omitted)idleTimeoutMs (integer, optional): Kill a stuck process after output inactivity; 30,000 to 3,600,000 msforceRefresh (boolean, optional): Bypass dedup and force a fresh CLI run, default: falseclaude_request_async / codex_request_async / gemini_request_async / grok_request_async / mistral_request_asyncStart a long-running Claude, Codex, Gemini, Grok, or Mistral request without waiting for completion in the same MCP call.
Use this flow when analysis/runtime can exceed client tool-call limits:
*_request_asyncllm_job_statusllm_job_resultllm_job_cancelAsync request tools accept the same approval strategy fields as their sync variants:
approvalStrategy: "legacy" (default) or "mcp_managed"approvalPolicy: "strict"|"balanced"|"permissive" overridemcpServers: Names of requested MCP servers, resolved against the gateway's local registry / Codex MCP configclaude_request_async also supports strictMcpConfig and fails fast when requested servers are unavailablellm_job_statusReturn lifecycle status (running, completed, failed, canceled) and metadata for an async job.
llm_job_resultReturn captured stdout/stderr for an async job (with configurable max chars per stream).
llm_job_cancelCancel a running async job.
approval_listList recent MCP-managed approval decisions recorded by the gateway.
Parameters:
limit (number, optional): Max records (1-500), default: 50cli (string, optional): Filter by "claude", "codex", "gemini", "grok", or "mistral"Approval records are persisted to ~/.llm-cli-gateway/approvals.jsonl.
llm_request_resultRead back any persisted request — sync or async — by its correlation ID. Every response echoes its ID in structuredContent.correlationId; pass it here to recover the persisted prompt/response after the inline result is gone. Reads the flight recorder, so it works independently of async-job persistence (returns "not found" when flight recording is disabled).
Parameters:
correlationId (string, required): Correlation ID from a prior requestmaxChars (number, optional): Max chars of the persisted response to return (1,000-2,000,000)includePrompt (boolean, optional): Include the full persisted prompt text, default: falsellm_process_healthReport gateway process health: async-job manager state plus the resolved persistence block (backend, dbPath, config sources). Use it to confirm which config file and SQLite paths the gateway is actually running under.
upstream_contractsReturn the gateway's declared provider CLI contracts, optionally probing the installed binaries for drift.
Parameters:
cli (string, optional): Filter (claude|codex|gemini|grok|mistral)probeInstalled (boolean, optional, default false): Run local --help probes and compare advertised flags against the declared contract — strongly recommended after any provider CLI upgrade. The probe reports missingFlags, extraFlags, acknowledgedExtraFlags (known upstream-only flags filtered from extraFlags), discoveredFlags, and stale-marker warnings.session_createCreate a new session for a specific CLI.
Parameters:
cli (string, required): CLI to create session for ("claude", "codex", "gemini", "grok", "mistral")description (string, optional): Description for the sessionsetAsActive (boolean, optional): Set as active session, default: trueExample:
{
"cli": "claude",
"description": "Code review session",
"setAsActive": true
}
session_listList all sessions, optionally filtered by CLI.
Parameters:
cli (string, optional): Filter by CLI ("claude", "codex", "gemini", "grok", "mistral")Response includes:
session_set_activeSet the active session for a specific CLI.
Parameters:
cli (string, required): CLI to set active session forsessionId (string, required): Session ID to activate (or null to clear)session_getRetrieve details for a specific session.
Parameters:
sessionId (string, required): Session ID to retrievesession_deleteDelete a specific session.
Parameters:
sessionId (string, required): Session ID to deletesession_clear_allClear all sessions, optionally for a specific CLI.
Parameters:
cli (string, optional): Clear sessions for specific CLI onlylist_modelsList available models for each CLI.
Parameters:
cli (string, optional): Specific CLI to list models for ("claude", "codex", "gemini", "grok", "mistral")Response includes:
defaultModel and defaultModelSource when a default is explicitly configuredmodelMetadata with source/confidence (fallback, config, env, observed)aliases and warnings when configured or when discovery degrades gracefullyThe registry treats explicit configuration as authoritative. Bundled fallback models are low-confidence hints, and Gemini models observed in local session history are merged as low-confidence entries only; they do not become the default model.
Model registry environment overrides:
# Explicit defaults
CLAUDE_DEFAULT_MODEL=haiku
CODEX_DEFAULT_MODEL=<codex-model-id>
GEMINI_DEFAULT_MODEL=gemini-2.5-flash
# Additional models: comma/newline list, JSON array, or JSON object of model->description
GEMINI_MODELS='{"gemini-team-default":"Team-approved Gemini model"}'
# Aliases
GEMINI_MODEL_ALIASES='team=gemini-team-default'
LLM_GATEWAY_MODEL_ALIASES='codex.fast=gpt-5.3-codex-spark,gemini.fast=gemini-team-default'
# Deterministic config/discovery paths
CODEX_CONFIG_PATH=/path/to/config.toml
CLAUDE_SETTINGS_PATH=/path/to/settings.json
CLAUDE_SETTINGS_LOCAL_PATH=/path/to/settings.local.json
GEMINI_SETTINGS_PATH=/path/to/settings.json
GEMINI_HISTORY_ROOT=/path/to/.gemini/tmp
# Disable local model-history discovery
LLM_GATEWAY_DISABLE_MODEL_DISCOVERY=1
provider_tool_capabilitiesReport the provider tool and feature capability catalog. Use this before orchestrating provider-specific requests so callers can distinguish supported controls, provider-owned configuration, ignored parity fields, and unsupported inputs.
Parameters:
cli (string, optional): Provider filter ("claude", "codex", "gemini", "grok", "grok_api", or "mistral")includeSkills (boolean, default true): Include bounded local skill discoveryincludeProviderTools (boolean, default true): Include provider-native tools extracted from discovered skillsincludeUnsupported (boolean, default true): Include explicit unsupported/degraded input recordsincludePaths (boolean, default false): Include raw local filesystem paths in discovery outputrefresh (boolean, default false): Bypass the short-lived capability cacheThe response schema is provider-tool-capabilities.v2. Capability discovery is
read-only and bounded; raw local paths are redacted unless includePaths is
explicitly true, and secret-bearing auth files are not read.
Equivalent MCP resources:
provider-tools://catalog: full provider catalogprovider-tools://claudeprovider-tools://codexprovider-tools://geminiprovider-tools://grokprovider-tools://grok_apiprovider-tools://mistraldoctor --json also emits a compact provider_capabilities block with the
same schema version, per-provider request tool names, supported feature names,
unsupported input names, config-surface counts, discovery counts, and resource
URIs. This block is intended for setup assistants that need a concise capability
summary without local skill bodies or raw paths.
cli_versionsReport installed CLI versions.
Parameters:
cli (string, optional): Specific CLI to inspect ("claude", "codex", "gemini", "grok", "mistral")cli_upgradePlan or run an upgrade for one CLI.
Parameters:
cli (string, required): CLI to upgrade ("claude", "codex", "gemini", "grok", "mistral")target (string, optional): Package tag/version/target, default: latestdryRun (boolean, optional): Return the upgrade plan without running it, default: truetimeoutMs (number, optional): Upgrade timeout when dryRun=falseUpgrade strategies:
claude updateclaude install <target>codex updatenpm install -g @openai/codex@<target>agy update (Antigravity self-update; explicit version targets are unsupported)grok updategrok update --version <target>pip/uv/brew); errors with guidance when none is detected (Vibe ships no self-update command)Example dry run:
{
"cli": "gemini",
"target": "latest",
"dryRun": true
}
~/.llm-cli-gateway/sessions.json// 1. Create a new session
await callTool("session_create", {
cli: "claude",
description: "Debugging session",
setAsActive: true,
});
// 2. Make requests (automatically uses active session)
await callTool("claude_request", {
prompt: "What's the bug in this code?",
// sessionId is automatically used
});
// 3. Continue the conversation
await callTool("claude_request", {
prompt: "Can you explain that fix in more detail?",
continueSession: true,
});
// 4. List all sessions
await callTool("session_list", { cli: "claude" });
// 5. Switch to a different session
await callTool("session_set_active", {
cli: "claude",
sessionId: "some-other-session-id",
});
// 6. Delete when done
await callTool("session_delete", {
sessionId: "session-id-to-delete",
});
DEBUG: Enable debug logging (set to any value)
DEBUG=1 node dist/index.js
LLM_GATEWAY_APPROVAL_POLICY: Default approval policy when request does not pass approvalPolicy (strict, balanced, permissive)
LLM_GATEWAY_APPROVAL_POLICY=strict node dist/index.js
LLM_GATEWAY_APPROVAL_ALLOW_BYPASS: Under approvalStrategy:"mcp_managed", a full permission / sandbox bypass request (e.g. dangerouslyBypassApprovalsAndSandbox, dangerouslySkipPermissions) is denied by default regardless of approval score, and mcp_managed no longer force-bypasses any provider — each defaults to an auto-accept-edits-level mode (auto-accept file edits, still gate Bash and other dangerous tools) instead of full auto-approve:
--permission-mode acceptEdits (was bypassPermissions)--permission-mode acceptEdits (was --always-approve)--agent accept-edits (was --agent auto-approve)default / prompted, i.e. no --dangerously-skip-permissions (the agy CLI has no accept-edits middle rung, so the safe default is prompted execution; without the opt-in, Gemini cannot auto-approve mutating tools under mcp_managed)Set to 1/true to let the operator opt back in: this permits bypass requests through the approval gate and restores each provider's full auto-approve mode under mcp_managed (Claude bypassPermissions, Grok --always-approve, Mistral --agent auto-approve, Gemini --dangerously-skip-permissions). Sandboxed auto modes (e.g. codex --sandbox workspace-write) are unaffected.
LLM_GATEWAY_APPROVAL_ALLOW_BYPASS=1 node dist/index.js
LLM_GATEWAY_TRUSTED_PRINCIPAL_HEADER: Name of an HTTP header carrying the authenticated user identity asserted by a trusted front door (any identity-aware reverse proxy / IdP). When set, the gateway adopts that header value as the request's ownership principal — but only for requests authenticated with the gateway's own static bearer token (i.e. the trusted upstream proxy), never from an arbitrary remote client. Off by default; IdP-agnostic. Lets a proxy-fronted multi-user deployment carry per-user identity into the gateway.
LLM_GATEWAY_TRUSTED_PRINCIPAL_HEADER=x-gateway-principal node dist/index.js
LLM_GATEWAY_OAUTH_REQUIRE_CONSENT / LLM_GATEWAY_OAUTH_CONSENT_SECRET: Opt-in human-consent gate for the built-in OAuth server. When enabled (REQUIRE_CONSENT=1, or implied by setting CONSENT_SECRET), /oauth/authorize renders an operator approval page (CSRF-protected) and issues an authorization code only after the dedicated consent password is entered — instead of auto-issuing. CONSENT_SECRET is the plaintext password (hashed in memory; or persist a consent_secret_hash in [http.oauth]). Off by default; remote OAuth refuses to enable consent without a secret to verify.
LLM_GATEWAY_OAUTH_REQUIRE_CONSENT=1 LLM_GATEWAY_OAUTH_CONSENT_SECRET='choose-a-strong-code' node dist/index.js
LLM_GATEWAY_CONFIG: Path to the gateway TOML config (default: ~/.llm-cli-gateway/config.toml). See Persistence configuration above for the [persistence] schema.
LLM_GATEWAY_LOGS_DB: Deprecated — overrides [persistence].path and selects backend = "sqlite" (or backend = "none" when set to none). Emits a deprecation warning at startup; migrate to config.toml.
# Custom path
LLM_GATEWAY_LOGS_DB=/var/log/gateway/logs.db node dist/index.js
# Disable durable persistence (also disables *_request_async tools)
LLM_GATEWAY_LOGS_DB=none node dist/index.js
LLM_GATEWAY_REDACT_LOGGED_SECRETS: Redact recognisable secrets (provider/cloud/VCS keys, bearer tokens, JWTs, PEM private keys, key=value secret assignments) from the prompt/system/response copies written to the flight-recorder log. Enabled by default; set to 0/false/off/no to store content verbatim. Only the audit log is affected — live sync responses and async llm_job_result output are never altered.
# Opt out of flight-recorder secret redaction
LLM_GATEWAY_REDACT_LOGGED_SECRETS=0 node dist/index.js
Each CLI can be configured through its own configuration files:
~/.claude/config.json~/.codex/config.toml~/.gemini/config.jsonllm-cli-gateway/
├── src/
│ ├── index.ts # Main MCP server and tool definitions
│ ├── executor.ts # CLI execution with timeout support
│ ├── session-manager.ts # Session management logic
│ └── __tests__/
│ ├── executor.test.ts # Unit tests for executor
│ └── integration.test.ts # Integration tests
├── dist/ # Compiled JavaScript
├── package.json
├── tsconfig.json
└── vitest.config.ts
# Run all tests
npm test
# Run unit tests only
npm run test:unit
# Run integration tests only
npm run test:integration
# Watch mode
npm run test:watch
npm run build
npm start
The gateway provides detailed error messages for common issues:
Error executing claude CLI:
spawn claude ENOENT
The 'claude' command was not found. Please ensure claude CLI is installed and in your PATH.
Error executing codex CLI: Command timed out
Process timed out after 120000ms
Prompt cannot be empty
Prompt too long (max 100k chars)
Logs are written to stderr (stdout is reserved for MCP protocol):
[INFO] 2026-01-24T05:00:00.000Z - Starting llm-cli-gateway MCP server
[INFO] 2026-01-24T05:00:01.000Z - claude_request invoked with model=sonnet, prompt length=150
[INFO] 2026-01-24T05:00:05.000Z - claude_request completed successfully in 4523ms, response length=2048
[ERROR] 2026-01-24T05:00:10.000Z - codex CLI execution failed: spawn codex ENOENT
Enable debug logging:
DEBUG=1 node dist/index.js
Make sure the CLIs are installed and in your PATH:
which claude
which codex
which agy
The gateway extends PATH to include common locations:
~/.local/bin/usr/local/bin/usr/bin~/.nvm/versions/node/*/bin directoriesIf you encounter permission errors, ensure the CLI tools have proper permissions:
chmod +x $(which claude)
chmod +x $(which codex)
chmod +x $(which agy)
Sessions are stored in ~/.llm-cli-gateway/sessions.json. If you encounter issues:
ls -la ~/.llm-cli-gateway/
rm ~/.llm-cli-gateway/sessions.json
cat ~/.llm-cli-gateway/sessions.json
The gateway does not enforce a default execution timeout for LLM CLI requests.
If your MCP client/runtime enforces per-tool-call deadlines, use async tools (*_request_async + llm_job_status/llm_job_result) so long-running jobs can complete outside a single call window.
The gateway supports concurrent requests across different CLIs. Each request spawns a separate process.
spawn with separate arguments (not shell execution)ajv codegen case)SHA256SUMS.sigstore.json before trusting the checksum fileIf you're vetting llm-cli-gateway through Socket or a similar supply-chain scanner, you'll see behavioural alerts and some dependency-ownership alerts. They are accurate descriptions of what the package does and what it depends on. The reviewed shellAccess and shrinkwrap entries are configured in socket.yml for repository/PR policy surfaces, but Socket's public package page may still display them for the published npm artifact; the rationale remains documented here and in the package.
The currently flagged surfaces are not new in 2.6.x: the 2.3.0, 2.4.0, 2.5.0, and 2.6.3 npm tarballs all include npm-shrinkwrap.json, and all include the same dist/executor.js child-process spawn surface used to run provider CLIs. The socket.yml policy for 2.4.0, 2.5.0, 2.6.0, and 2.6.3 is materially the same for shellAccess; this README now adds the missing shrinkwrap disclosure as well.
| Alert | Where | Why it's bounded |
|---|---|---|
| Network access | src/http-transport.ts opens an HTTP MCP transport when started via npm run start:http. src/endpoint-exposure.ts issues a HEAD probe to verify configured public/tunnel URLs. Socket also flagged dist/upstream-contracts.js in v1.17.2 from descriptive text, not a network call. | The transport binds to 127.0.0.1 by default and requires LLM_GATEWAY_AUTH_TOKEN to be set. The default stdio MCP entry point (npm start) opens no sockets. src/upstream-contracts.ts stores provider CLI metadata and imports no HTTP client APIs. |
| Shell access | src/executor.ts uses child_process.spawn(cmd, args, …) to invoke the underlying LLM CLIs. | spawn is called with an argument array and never shell: true, so there is no shell interpolation path for caller input. The command name is restricted to an allow-list of known CLI binaries (claude, codex, agy, grok, vibe). |
| Published shrinkwrap | The npm artifact includes npm-shrinkwrap.json; package.json#files includes it and scripts/make-prod-shrinkwrap.mjs generates it from package-lock.json. | This is a CLI/application package. npm documents the shrinkwrap use case for applications, daemons, and command-line tools published through the registry. Our shrinkwrap is a prod-only projection, not a committed full dev lockfile: scripts/release-security-audit.sh verifies parity with the audited lockfile, and scripts/verify-registry-install.sh proves fresh registry consumers receive no better-sqlite3/prebuild-install/tar-fs/tar-stream production chain. |
| Uses eval | None in our source. Transitive: @modelcontextprotocol/sdk → ajv@8 uses new Function(...) in ajv/dist/compile/index.js to compile JSON Schema validators. | This is ajv's standard codegen path. Only known schemas (defined in our source and the MCP SDK) flow into it; no caller-supplied data ever reaches the compiled function body. |
| SQLite adapter isolation | Persistence uses Node's built-in node:sqlite module (no native binding, no install scripts) through a single adapter, src/sqlite-driver.ts. | node:sqlite is touched by exactly one production module (the adapter); every other module talks to SQLite through its typed surface. We never call any db.pragma() helper (it does not exist on node:sqlite); SQLite setup uses fixed literal db.exec("PRAGMA ...") statements. npm run security:audit fails the release if production code references node:sqlite outside the adapter or reintroduces a .pragma() call. |
| Dependency ownership | A handful of small transitive packages (e.g. media-typer via @modelcontextprotocol/sdk) trip Socket's "unstable ownership" or "obfuscated code" heuristics. | These are pinned, well-known micro-deps in the Node ecosystem with no known issues. We pin direct override versions of content-type and type-is in package.json#overrides. As of 2.0.0 the prod graph carries no native module (better-sqlite3 moved to devDependencies; node:sqlite is built into Node), eliminating the entire prebuild-install/tar-fs/tar-stream install-time chain. Our earlier direct dependency on toml@3.0.0 was replaced with smol-toml. |
See socket.yml for the same context in machine-readable form.
npm testnpm run buildMIT. See LICENSE for details.
For issues and questions:
See CHANGELOG.md for detailed release history.
io.github.ericm1018/skillfm-llm-cost-optimizer-openai-anthropic-usage
io.github.mikerawsonnz/llm-orchestration-agent
io.github.mikerawsonnz/authenticated-llm-agent
labforgedev/copilot-memory-mcp
csoai-org/agent-prompt-injection-firewall-mcp
io.github.mikerawsonnz/authenticated-multi-llm-agent