mcp-v8 is a Model Context Protocol server,
written in Rust, that lets an AI agent run JavaScript and TypeScript in a
sandboxed V8 isolate. Instead of wiring up dozens of narrow tools, you give the
agent one tool — run_js — and it writes code: looping, branching, transforming
data, and calling other tools, often with far fewer tokens than equivalent
tool-call chains.
In its default stateful mode the V8 heap is saved as a content-addressed snapshot, so an agent can build up state across many turns. Host capabilities (network, filesystem, subprocess, WebAssembly, module imports, and calls to other MCP servers) are all off by default and unlocked only by explicit OPA/Rego policies.
- One tool, unbounded capability. The agent runs a program, not a fixed menu of tools.
- Durable state. Heap snapshots persist variables and objects across calls.
- Secure by default.
fetch, filesystem, subprocess, and external imports are denied until you grant them via policy. - Production-ready. stdio / Streamable HTTP / SSE transports, a REST sidecar, async execution with pagination, JWKS auth, and Raft-replicated clustering.
Full documentation lives at https://r33drichards.github.io/mcp-js/ (built
from site-docs/) — tutorials, how-to guides, concept
explanations, and complete reference for the CLI flags,
HTTP API, and
MCP tools.
# Server
curl -fsSL https://raw.githubusercontent.com/r33drichards/mcp-js/main/install.sh | sudo bash
# Optional CLI client
curl -fsSL https://raw.githubusercontent.com/r33drichards/mcp-js/main/install-cli.sh | sudo bashInstalls to /usr/local/bin. Supported platforms: Linux x86_64/arm64 and macOS
Apple Silicon. You can also nix run github:r33drichards/mcp-js, use Docker (see
the docker-compose.*.yml stacks), or build from source.
# Claude Code (stdio)
claude mcp add mcp-v8 -- mcp-v8 --directory-path /tmp/mcp-v8-heaps # stateful
claude mcp add mcp-v8 -- mcp-v8 --stateless # statelessFor Claude Desktop / Cursor, add to the client's mcpServers config:
{ "mcpServers": { "js": { "command": "mcp-v8", "args": ["--stateless"] } } }Then ask the agent: "Run this JavaScript: console.log([1,2,3].map(x => x*2))".
mcp-v8 --stateless --http-port 8080
# MCP endpoint: POST http://localhost:8080/mcp
# REST sidecar: POST http://localhost:8080/api/execSee the Quick Start tutorials and the transports guide for more.
- JavaScript & TypeScript in an isolated V8 engine (via
deno_core); TypeScript types are stripped with SWC (type removal, not type checking). - Async/await & timers — Promises and the event loop, plus
setTimeout/clearTimeout. - Console capture —
console.log/info/warn/error/debug/trace, streamed to storage and readable with line- or byte-based pagination. - Async execution model —
run_jsreturns an execution ID; poll status and stream output; cancel running work. - Content-addressed heap snapshots — persist/restore V8 state across calls (local FS, S3, or S3 + write-through cache), or run stateless.
- WebAssembly — the standard
WebAssemblyAPI, plus pre-loaded modules (--wasm-module) exposed as globals and advertised to clients asrunjs__wasm__<name>stub tools. - ES module imports — optional
npm:,jsr:, and URL imports fetched at runtime (policy-gated). - Policy-gated capabilities —
fetch, filesystem (fs), and subprocess access, each checked against a Rego policy per operation; plus header/OAuth injection forfetch. - Compose other MCP servers — connect upstream MCP servers and call them from JS via
mcp.callTool()/mcp.listTools(). - Customizable surface — override the server
instructionsand therun_jsdescription (--instructions,--run-js-description). - Auth & clustering — JWKS-based JWT verification, and optional Raft clustering with replicated session metadata and horizontal scaling.
- Multiple transports — stdio, Streamable HTTP (MCP 2025-03-26+), and SSE, with a REST sidecar and OpenAPI spec.
These globals are available inside run_js (capability globals require a policy):
| Global | Purpose | Gated by |
|---|---|---|
console, setTimeout |
Output & timers | — |
fetch(url, opts?) |
HTTP requests (Fetch API) | fetch policy |
fs.* |
File I/O (readFile, writeFile, …) |
filesystem policy |
child_process / Deno.Command |
Run subprocesses | subprocess policy |
import (npm: / jsr: / URL) |
External ES modules | --allow-external-modules + modules policy |
WebAssembly, __wasm_<name> |
Run/instantiate WASM | — |
mcp.callTool/listTools/servers |
Call upstream MCP servers | mcp_tools policy |
See Concepts → Security policies for the policy model.
| Tool | Mode | Description |
|---|---|---|
run_js |
both | Stateful: queue execution → {execution_id}. Stateless: run and return {output, error?}. |
get_execution |
stateful | Poll status/result of an execution. |
get_execution_output |
stateful | Read paginated console output (line or byte). |
cancel_execution |
stateful | Terminate a running execution. |
list_executions |
stateful | List executions and their status. |
list_sessions, list_session_snapshots |
stateful | Browse named sessions and history. |
get_heap_tags, set_heap_tags, delete_heap_tags, query_heaps_by_tags |
stateful | Tag and search heap snapshots. |
Full parameters: MCP tools reference.
mcp-v8 is configured entirely through CLI flags — storage backend, transport,
execution limits, policies, fetch-header injection, WASM modules, clustering, JWKS
auth, and the prompt/tool-description overrides. The complete, always-current list
is the generated CLI flags reference.
mcp-v8 --help # all flags
mcp-v8 --print-openapi # print the REST OpenAPI specA fully-typed client for the REST API, generated from the OpenAPI spec via progenitor:
mcp-v8 --stateless --http-port 3000 &
mcp-v8-cli exec "console.log('hello'); 1 + 1"
mcp-v8-cli executions get <execution_id>
mcp-v8-cli executions output <execution_id>
export MCP_V8_URL=https://my-server.example.com # point at a remote server[dependencies]
mcp-v8-client = { git = "https://github.com/r33drichards/mcp-js" }use mcp_v8_client::Client;
let client = Client::new("http://localhost:3000");
let body = mcp_v8_client::types::ExecRequest {
code: "1 + 1".to_string(),
heap: None, session: None,
heap_memory_max_mb: None, execution_timeout_secs: None, tags: None,
};
let resp = client.exec_handler(&body).await?;
println!("execution_id: {}", resp.into_inner().execution_id);The repo is a Nix flake (it wires up the prefetched V8 archive so the build stays offline-friendly):
nix build github:r33drichards/mcp-js # → ./result/bin/server
# or for development:
nix develop # then: cargo build -p serverA plain cargo build --release inside server/ also works if your toolchain can
build deno_core/V8.
setIntervalis not available — use a loop with awaitedsetTimeout.- No DOM or browser APIs — there is no
window/document. - TypeScript is type-stripped, not type-checked — invalid types are removed, not reported. JSX/TSX is not supported (it parses to a clear error).
Comparison of single-node vs 3-node cluster at various request rates.
ran on railway gha runners on pr
| Topology | Target Rate | Actual Iter/s | HTTP Req/s | Exec Avg (ms) | Exec p95 (ms) | Exec p99 (ms) | Success % | Dropped | Max VUs |
|---|---|---|---|---|---|---|---|---|---|
| cluster-stateful | 100/s | 99.5 | 99.5 | 44.9 | 196.88 | 416.99 | 100% | 31 | 41 |
| cluster-stateful | 200/s | 199.6 | 199.6 | 23.22 | 79.32 | 131.13 | 100% | 13 | 33 |
| cluster-stateless | 1000/s | 999.9 | 999.9 | 3.82 | 7.72 | 13.09 | 100% | 0 | 100 |
| cluster-stateless | 100/s | 100 | 100 | 3.67 | 5.65 | 8.03 | 100% | 0 | 10 |
| cluster-stateless | 200/s | 200 | 200 | 3.56 | 5.9 | 8.61 | 100% | 0 | 20 |
| cluster-stateless | 500/s | 500 | 500 | 3.42 | 5.85 | 9.2 | 100% | 0 | 50 |
| single-stateful | 100/s | 99.1 | 99.1 | 215.12 | 362.5 | 376.6 | 100% | 32 | 42 |
| single-stateful | 200/s | 97.8 | 97.8 | 1948.82 | 2212.55 | 2960.96 | 100% | 5939 | 200 |
| single-stateless | 1000/s | 977.1 | 977.1 | 60.98 | 482.98 | 602.38 | 100% | 843 | 561 |
| single-stateless | 100/s | 100 | 100 | 3.71 | 5.73 | 8.73 | 100% | 0 | 10 |
| single-stateless | 200/s | 200 | 200 | 3.61 | 5.43 | 7.74 | 100% | 0 | 20 |
| single-stateless | 500/s | 500 | 500 | 4.67 | 8.49 | 27.98 | 100% | 0 | 50 |
| Topology | Rate | P95 (ms) | |
|---|---|---|---|
| cluster-stateful | 100/s | 196.88 | █████████████████████ |
| cluster-stateful | 200/s | 79.32 | █████████████████ |
| cluster-stateless | 100/s | 5.65 | ███████ |
| cluster-stateless | 200/s | 5.9 | ███████ |
| cluster-stateless | 500/s | 5.85 | ███████ |
| cluster-stateless | 1000/s | 7.72 | ████████ |
| single-stateful | 100/s | 362.5 | ███████████████████████ |
| single-stateful | 200/s | 2212.55 | ██████████████████████████████ |
| single-stateless | 100/s | 5.73 | ███████ |
| single-stateless | 200/s | 5.43 | ██████ |
| single-stateless | 500/s | 8.49 | ████████ |
| single-stateless | 1000/s | 482.98 | ████████████████████████ |
- Target Rate: The configured constant-arrival-rate (requests/second k6 attempts)
- Actual Iter/s: Achieved iterations per second (each iteration = 1 POST /api/exec)
- HTTP Req/s: Total HTTP requests per second (1 per iteration)
- Dropped: Iterations k6 couldn't schedule because VUs were exhausted (indicates server saturation)
- Topology:
single= 1 MCP-V8 node;cluster= 3 MCP-V8 nodes with Raft