"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-gatewayOr 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:
- Runs five provider CLIs through consistent sync and async MCP tools.
- Persists long-running jobs, supports restart-safe result collection, deduplication, cancellation, and sync-to-async deferral.
- Tracks sessions, real CLI resume paths, structured response metadata, and cache telemetry.
- Supports cache-aware
promptParts, including explicit Claudecache_controlwhen opted in. - Can run requests inside gateway-managed git worktrees for isolated multi-agent review and implementation loops.
- Ships personal-appliance setup surfaces: HTTP transport with bearer-token auth,
doctor --json, setup UI artifacts, provider setup snippets, Docker fallback, and checked release bundles. - Remote web connectors use MCP OAuth discovery and authorization-code setup with static client or shared-secret gates. Client secrets are generated locally, stored only as hashes, and printed only by explicit copy-once commands.
- Provider CLI requests can select registered workspaces by alias via
workspace; every HTTP/tunnel request must use a registered alias, session workspace, or[workspaces].defaultbefore 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.
- CI runs build, lint, format, tests, package checks, and npm audit.
- Security CI runs actionlint, zizmor, shellcheck, typos, osv-scanner, gitleaks, and lychee.
- GitHub release installer artifacts are checksummed and signed with Sigstore keyless signing.
- npm releases use provenance through OIDC trusted publishing.
- The npm package intentionally ships a generated, prod-only
npm-shrinkwrap.jsonso registry installs resolve the audited release tree. Release gates regenerate it frompackage-lock.json, compare for parity, and run a registry-fidelity consumer install before publishing. - Socket behavioural alerts are documented in
socket.ymland under "Security Considerations" below.shellAccessandshrinkwrapare 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:
- Streamable HTTP startup:
LLM_GATEWAY_AUTH_TOKEN=<token> npm run start:http - Machine-readable diagnostics:
npm run doctor - Go bootstrapper:
installer/withsetup,doctor --json,start,stop,status,repair,upgrade,uninstall,print-client-config, and verified bundle download commands. - Release packaging: the release workflow builds Linux binaries on the local self-hosted runner, builds Windows/macOS binaries on GitHub-hosted runners, then publishes checksummed platform bundles with the gateway, production dependencies, and a managed Node runtime; see installer/packaging/README.md.
- Docker Compose fallback: docker/personal.compose.yml + docker/Dockerfile.personal for users who already manage containers.
- Local setup UI artifact: setup/ui/index.html
- Provider setup snippets: setup/providers/
- Cross-validation tools:
validate_with_models,second_opinion,compare_answers,red_team_review,consensus_check,ask_model,synthesize_validation,job_status, andjob_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 doctorThe 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 --yesDocker 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- Multi-LLM Orchestration: Unified interface for Claude Code, Codex, Gemini, Grok, and Mistral (Vibe) CLIs
- Session Management: Track and resume conversations across all CLIs with persistent storage
- Gateway-owned worktrees: Run any sync or async provider request inside a managed git worktree, with per-session reuse and cleanup
- Token Optimization: Automatic 44% reduction on prompts, 37% on responses (opt-in)
- Correlation ID Tracking: Full request tracing across all LLM interactions
- Cross-Tool Collaboration: LLMs can use each other via MCP (validated through dogfooding)
- SQLite Flight Recorder: Every request/response logged to
~/.llm-cli-gateway/logs.dbwith correlation IDs, token usage, duration, retry counts, and circuit breaker state. Browse with Datasette:datasette ~/.llm-cli-gateway/logs.db - Structured Metadata: Tool responses include machine-readable
structuredContent(model, cli, correlationId, sessionId, durationMs, token counts) - Cache observability resources:
cache-state://global,cache-state://session/{id}, andcache-state://prefix/{hash}MCP resources return aggregate cache hit/miss/savings — tokens and hashes only, no prompt text.session_getincludes acacheStateblock when the session has prior requests. - Provider capability inventory:
provider_tool_capabilitiesandprovider-tools://catalogexpose 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 --jsonincludes a compactprovider_capabilitiessummary 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.
- Retry Logic: Exponential backoff with circuit breaker for transient failures
- Atomic File Writes: Process-specific temp files with fsync for data integrity
- Memory Limits: 50MB cap on CLI output prevents DoS attacks
- NVM Path Caching: Eliminates I/O overhead on every request
- Long-Running Jobs: Non-time-bound async execution via
*_request_async+ polling tools
- Comprehensive Testing: 1,000+ tests covering unit, integration, and regression scenarios with real CLI execution
- Input Validation: Zod schemas prevent injection attacks
- No Secret Leakage: Generic session descriptions only (file permissions 0o600)
- No ReDoS: Bounded regex patterns prevent catastrophic backtracking
- Type Safety: Strict TypeScript with comprehensive error handling
- Supply-chain hardening: a dedicated
.github/workflows/security.ymlruns actionlint, zizmor, shellcheck, typos, osv-scanner, gitleaks, and lychee on every push and PR (seeSECURITY.mdfor 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-codenpm install -g @openai/codex
codex loginThe Gemini provider runs through Google Antigravity CLI (agy).
curl -fsSL https://antigravity.google/cli/install.sh | bash
# Docs: https://antigravity.google/docs/cli-overviewcurl -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 = trueVibe-specific notes:
- Model selection is via the
VIBE_ACTIVE_MODELenvironment variable — Vibe has no--modelflag. The gateway discovers~/.vibe/config.toml/VIBE_MODELS, injectsVIBE_ACTIVE_MODELonly when a model is explicitly requested or Vibe config needs recovery, and retries once after a model-not-found failure with refreshed discovery. permissionModeis the Vibe--agentname. Builtins aredefault | 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 isauto-approve; pick a stricter mode explicitly if you need approval gates.allowedToolsis allow-list only — the gateway emits one--enabled-tools <tool>flag per entry.disallowedToolsis accepted in the schema for caller-side parity but is silently ignored at the CLI boundary (alogger.infowarning records the no-op).- No self-update:
cli_upgrade --cli mistraldetects whether you used pip / uv / brew and dispatches the matching upgrade command. Runningvibe updateis not a thing.
npm install -g llm-cli-gatewayOr 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 buildFor 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_statusandjob_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.
Execute a Claude Code request with optional session management.
Parameters:
prompt(string, optional*): The prompt to send (1-100,000 chars). *Exactly one ofpromptorpromptPartsis required (mutually exclusive)model(string, optional): Model name or alias (uselist_modelsfor available values; supportslatest)outputFormat(string, optional): Output format (text|json|stream-json), default:stream-json— the gateway parses NDJSON usage events for token/cost observability; override totextonly 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 overdangerouslySkipPermissionsdangerouslySkipPermissions(boolean, optional): Deprecated — maps topermissionMode: "bypassPermissions";permissionModewins 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 withpromptforceRefresh(boolean, optional): Bypass dedup and force a fresh CLI run, default: false
Workspace 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 whenapprovalStrategy="mcp_managed"mcpServers: Requested/enabled/missing MCP servers for this call
Example:
{
"prompt": "Write a Python function to calculate fibonacci numbers",
"model": "sonnet",
"continueSession": true,
"optimizePrompt": true,
"optimizeResponse": true
}Execute a Codex request with optional session tracking.
Parameters:
prompt(string, optional*): The prompt to send (1-100,000 chars). *Exactly one ofpromptorpromptPartsis required (mutually exclusive)model(string, optional): Model name or alias (uselist_modelsfor available values; supportslatest, recommended:gpt-5.5)fullAuto(boolean, optional): Deprecated — expands to--sandbox workspace-writeonly (current Codex no longer accepts approval-policy flags); prefersandboxModesandboxMode(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 ifsessionIdis setcreateNewSession(boolean, optional): Always create a new sessionforceRefresh(boolean, optional): Bypass dedup and force a fresh CLI run, default: falseoutputFormat(string, optional):text(default) orjson(--jsonJSONL 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-dirper 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=valueoverridesignoreRules/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 withpromptoptimizePrompt(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 ms
Response extras:
approval: Approval decision record whenapprovalStrategy="mcp_managed"mcpServers: Requested MCP servers for this call
Example:
{
"prompt": "Create a REST API endpoint",
"model": "gpt-5.5",
"sandboxMode": "workspace-write",
"optimizePrompt": true
}Fork 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 withforkLast)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)
Execute a Google Antigravity CLI (agy) request with session support.
Parameters:
prompt(string, optional*): The prompt to send (1-100,000 chars). *Exactly one ofpromptorpromptPartsis required (mutually exclusive)model(string, optional): Model name or alias (uselist_modelsfor available values; supportslatest,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. Onlydefault(prompted execution) andyolo(emits--dangerously-skip-permissions) are accepted;auto_editandplanare 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):textonly. Antigravity print mode emits text;jsonandstream-jsonare rejected.mcpServers,allowedTools,policyFiles,adminPolicyFiles,attachments(string[], optional) andskipTrust(boolean, optional): Unsupported by Antigravity CLI — non-empty values (orskipTrust: true) are rejected with an explanatory error. Retained in the schema for caller parity.yolo(boolean, optional): Auto-approve all; equivalent toapprovalMode: "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 withpromptoptimizePrompt(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: false
Response extras:
approval: Approval decision record whenapprovalStrategy="mcp_managed"mcpServers: Requested MCP servers for this call
Example:
{
"prompt": "Explain quantum computing",
"model": "latest",
"resumeLatest": true,
"optimizePrompt": true
}Execute a Grok CLI (xAI) request with session support.
Parameters:
prompt(string, optional*): The prompt to send (1-100,000 chars). *Exactly one ofpromptorpromptPartsis 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 viagrok mcp)allowedTools(string[], optional): Allowed built-in tools (passed as--toolscomma list)disallowedTools(string[], optional): Disallowed built-in tools (passed as--disallowed-toolscomma 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 viaGROK_SANDBOX)rules(string, optional): Extra rules appended to the system prompt (--rules; supports@fileprefix)systemPromptOverride(string, optional): Replace the agent's system prompt entirelyallow/deny(string[], optional): Permission allow/deny rules (one--allow/--denyper 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--worktreeflag (true→ bare, string → named); distinct from the gatewayworktreeoptionworktree(boolean|object, optional): Run inside a gateway-owned git worktree (slice λ)promptParts(object, optional): Cache-aware structured prompt{ system?, tools?, context?, task }; mutually exclusive withpromptoptimizePrompt(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: false
Example:
{
"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:
- Re-issuing a request is safe. Identical
*_request/*_request_asynccalls 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_statusandllm_job_resultwork across gateway restarts. Job rows live for 30 days by default; callers can collect results long after the in-memory cache has evicted them.- Jobs running at shutdown are marked
orphanedon the next gateway boot (the detached child can't be reattached to). Their captured partial output remains readable. - Pass
forceRefresh: trueon 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 backendBackends:
sqlite(default) — durable, file-backed. Safe for single-instance deployments.memory— in-process Map. Lost on gateway exit. RequiresacknowledgeEphemeral = trueto 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, andllm_job_cancelare NOT registered on the gateway. This is a structural invariant: agents that try to call async tools against a gateway withbackend = "none"get a clean "tool not found" at connect time instead of silent in-memory loss after the 1-hour TTL. Usellm_process_healthto inspect the resolved persistence state programmatically.
Legacy environment variables (deprecated; emit a warning at startup):
LLM_GATEWAY_LOGS_DB/LLM_GATEWAY_JOBS_DB—noneselectsbackend = "none"; any other value selectsbackend = "sqlite"with that path.LLM_GATEWAY_JOB_RETENTION_DAYS— overridesretentionDays.LLM_GATEWAY_DEDUP_WINDOW_MS— overridesdedupWindowMs.LLM_GATEWAY_ACKNOWLEDGE_EPHEMERAL—1/true/yessetsacknowledgeEphemeral = 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.
Run a Mistral Vibe agentic coding request. Like grok_request in shape, but with Vibe's specific surface:
model(string, optional): Vibe model alias (for examplemistral-medium-3.5orlatest). The resolved value is injected via theVIBE_ACTIVE_MODELenvironment variable; omit it to let the gateway discover Vibe config and avoid stale hardcoded defaults.permissionMode: the Vibe--agentname — builtinsdefault | plan | accept-edits | auto-approve, or any install-gated/custom agent. Emitted as--agent <name>. Defaults toauto-approvein 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 --jsonsurfaces an actionable next-action.trust(boolean, optional): Emit--trustso 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-dirper 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 viavibe 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 withpromptoptimizePrompt/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: false
claude_request_async / codex_request_async / gemini_request_async / grok_request_async / mistral_request_async
Start 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:
- Start job with
*_request_async - Poll with
llm_job_status - Read output with
llm_job_result - Optionally stop with
llm_job_cancel
Async 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_asyncalso supportsstrictMcpConfigand fails fast when requested servers are unavailable
Return lifecycle status (running, completed, failed, canceled) and metadata for an async job.
Return captured stdout/stderr for an async job (with configurable max chars per stream).
Cancel a running async job.
List 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.
Read 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: false
Report 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.
Return 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, defaultfalse): Run local--helpprobes and compare advertised flags against the declared contract — strongly recommended after any provider CLI upgrade. The probe reportsmissingFlags,extraFlags,acknowledgedExtraFlags(known upstream-only flags filtered fromextraFlags),discoveredFlags, and stale-markerwarnings.
Create 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: true
Example:
{
"cli": "claude",
"description": "Code review session",
"setAsActive": true
}List all sessions, optionally filtered by CLI.
Parameters:
cli(string, optional): Filter by CLI ("claude", "codex", "gemini", "grok", "mistral")
Response includes:
- Total session count
- Session details (ID, CLI, description, timestamps, active status)
- Active session IDs for each CLI
Set 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)
Retrieve details for a specific session.
Parameters:
sessionId(string, required): Session ID to retrieve
Delete a specific session.
Parameters:
sessionId(string, required): Session ID to delete
Clear all sessions, optionally for a specific CLI.
Parameters:
cli(string, optional): Clear sessions for specific CLI only
List available models for each CLI.
Parameters:
cli(string, optional): Specific CLI to list models for ("claude", "codex", "gemini", "grok", "mistral")
Response includes:
- Model names and descriptions
- Best use cases for each model
- CLI-specific information
defaultModelanddefaultModelSourcewhen a default is explicitly configuredmodelMetadatawith source/confidence (fallback,config,env,observed)aliasesandwarningswhen configured or when discovery degrades gracefully
The 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=1Report 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, defaulttrue): Include bounded local skill discoveryincludeProviderTools(boolean, defaulttrue): Include provider-native tools extracted from discovered skillsincludeUnsupported(boolean, defaulttrue): Include explicit unsupported/degraded input recordsincludePaths(boolean, defaultfalse): Include raw local filesystem paths in discovery outputrefresh(boolean, defaultfalse): Bypass the short-lived capability cache
The 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://mistral
doctor --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.
Report installed CLI versions.
Parameters:
cli(string, optional): Specific CLI to inspect ("claude", "codex", "gemini", "grok", "mistral")
Plan 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 whendryRun=false
Upgrade strategies:
- Claude latest:
claude update - Claude explicit target:
claude install <target> - Codex latest:
codex update - Codex explicit target:
npm install -g @openai/codex@<target> - Gemini latest:
agy update(Antigravity self-update; explicit version targets are unsupported) - Grok latest:
grok update - Grok explicit target:
grok update --version <target> - Mistral (Vibe): dispatches to the detected installer (
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
}- Automatic Session Tracking: By default, the gateway automatically tracks sessions for each CLI
- Active Sessions: Each CLI can have one active session that's used by default
- Persistent Storage: Sessions are stored in
~/.llm-cli-gateway/sessions.json - Context Reuse: Using sessions maintains conversation history and context
// 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 passapprovalPolicy(strict,balanced,permissive)LLM_GATEWAY_APPROVAL_POLICY=strict node dist/index.js
-
LLM_GATEWAY_APPROVAL_ALLOW_BYPASS: UnderapprovalStrategy:"mcp_managed", a full permission / sandbox bypass request (e.g.dangerouslyBypassApprovalsAndSandbox,dangerouslySkipPermissions) is denied by default regardless of approval score, andmcp_managedno 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:- Claude →
--permission-mode acceptEdits(wasbypassPermissions) - Grok →
--permission-mode acceptEdits(was--always-approve) - Mistral (Vibe) →
--agent accept-edits(was--agent auto-approve) - Gemini (Antigravity) →
default/ prompted, i.e. no--dangerously-skip-permissions(theagyCLI has no accept-edits middle rung, so the safe default is prompted execution; without the opt-in, Gemini cannot auto-approve mutating tools undermcp_managed)
Set to
1/trueto let the operator opt back in: this permits bypass requests through the approval gate and restores each provider's full auto-approve mode undermcp_managed(ClaudebypassPermissions, 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
- Claude →
-
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 settingCONSENT_SECRET),/oauth/authorizerenders an operator approval page (CSRF-protected) and issues an authorization code only after the dedicated consent password is entered — instead of auto-issuing.CONSENT_SECRETis the plaintext password (hashed in memory; or persist aconsent_secret_hashin[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].pathand selectsbackend = "sqlite"(orbackend = "none"when set tonone). Emits a deprecation warning at startup; migrate toconfig.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=valuesecret assignments) from the prompt/system/response copies written to the flight-recorder log. Enabled by default; set to0/false/off/noto store content verbatim. Only the audit log is affected — live sync responses and asyncllm_job_resultoutput 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 Code:
~/.claude/config.json - Codex:
~/.codex/config.toml - Gemini:
~/.gemini/config.json
llm-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:watchnpm run buildnpm startThe 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.jsMake sure the CLIs are installed and in your PATH:
which claude
which codex
which agyThe gateway extends PATH to include common locations:
~/.local/bin/usr/local/bin/usr/bin- All
~/.nvm/versions/node/*/bindirectories
If 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:
- Check file permissions:
ls -la ~/.llm-cli-gateway/- Reset sessions:
rm ~/.llm-cli-gateway/sessions.json- Or manually edit the session file:
cat ~/.llm-cli-gateway/sessions.jsonThe 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.
- Input Validation: All prompts are validated (min 1 char, max 100k chars)
- Command Execution: Uses
spawnwith separate arguments (not shell execution) - No Eval: No dynamic code evaluation in our source (see "Socket alerts" below for the transitive
ajvcodegen case) - Sandboxing: Consider running in containers for production use
- Provenance: Releases are published with npm provenance via OIDC trusted publishing from GitHub Actions
- Release signing: GitHub release installer artifacts are signed with Sigstore keyless signing; verify
SHA256SUMS.sigstore.jsonbefore trusting the checksum file
If 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.
- Fork the repository
- Create a feature branch
- Make your changes
- Run tests:
npm test - Build:
npm run build - Submit a pull request
MIT. See LICENSE for details.
For issues and questions:
- Open an issue on GitHub
- Check existing issues and documentation
- Review CLI-specific documentation for CLI-related problems
See CHANGELOG.md for detailed release history.