loading…
Search for a command to run...
loading…
A meta MCP server that orchestrates other MCP servers by lazily connecting to them and exposing their tools as JavaScript libraries. It allows users to execute
A meta MCP server that orchestrates other MCP servers by lazily connecting to them and exposing their tools as JavaScript libraries. It allows users to execute JavaScript code that programmatically interacts with multiple MCP servers within a unified environment.
jsmcp exists for cases where an agent needs to do more than a single MCP tool call.
Most MCP clients are great at one tool call at a time, but awkward when work requires:
jsmcp solves that by exposing approved MCP tools as JavaScript namespaces. Instead of forcing the model to juggle many separate tool invocations, it can discover what is available and then write a small amount of JavaScript to use those tools programmatically.
In practice, this means:
jsmcp constrains access to whatever servers and tools you allow in a presetConfig is read from $XDG_CONFIG_HOME/jsmcp/ or, if XDG_CONFIG_HOME is not set, ~/.config/jsmcp/. Exactly one of config.json, config.yaml, or config.yml must exist there.
Use jsmcp when you want agents to treat MCP tools more like a small programmable API surface than a sequence of isolated button presses.
This is especially useful when an agent needs to:
npm install -g @alesya_h/jsmcp
Or run it without installing globally:
npx @alesya_h/jsmcp run
jsmcp run
jsmcp run work
jsmcp server work --port 3000 --bind 0.0.0.0
jsmcp client --profile work --host 127.0.0.1 --port 3000
jsmcp client --profile work --port 3000 --session-id my-agent-session
jsmcp status --profile work --host 127.0.0.1 --port 3000
jsmcp status kagi --tools --profile work --port 3000
jsmcp restart --host 127.0.0.1 --port 3000
jsmcp auth
jsmcp auth firefox_devtools
If you are running from a source checkout instead of an installed package, replace jsmcp with node src/index.js, for example node src/index.js run.
run starts the meta-MCP server directly over stdio.
server starts a long-lived daemon on ws://<bind>:<port>/mcp, starts every globally enabled MCP server once, and keeps those underlying connections warm. The chosen preset becomes the default profile for connections that do not request one. It binds to 0.0.0.0 by default and accepts --bind <host> to choose another bind address.
client exposes a stdio MCP server that proxies raw MCP/JSON-RPC messages to server over WebSocket. It accepts --host <host> and --port <number> to choose which daemon to connect to, can optionally pass --profile <name> to select that daemon-side profile, and accepts --session-id <id> to reuse the same daemon-side log session across client reconnects.
status connects to the daemon over HTTP, retrying until it is reachable, and prints the configured servers in the selected profile with their startup status or startup errors. Pass a server name to show only that server, and pass --tools to include each healthy server's allowed tools and descriptions. It accepts --host <host>, --port <number>, and --profile <name>.
restart asks a running daemon over HTTP to exit with a restart-request status. Run it under a supervisor such as the included systemd user service so the daemon is started again automatically. It accepts --host <host> and --port <number>.
run, server, and client all accept an optional preset as either a positional argument or --profile <name>. The default daemon port is 41528. If client --session-id is omitted, the client generates a random session id and reuses it for reconnects during that client process.
On first server start, jsmcp creates an API key at $XDG_CONFIG_HOME/jsmcp/api-key.txt, or ~/.config/jsmcp/api-key.txt if XDG_CONFIG_HOME is not set. Daemon WebSocket and HTTP API requests must include it in the X-JSMCP-API-Key header; unauthenticated requests receive 401.
The daemon also exposes the five meta tools through one JSON HTTP endpoint, plus a restart endpoint:
POST /api/call?tool=list_servers&profile=<name>
POST /api/call?tool=list_tools&profile=<name>
POST /api/call?tool=execute_code&sessionId=<id>&profile=<name>
POST /api/call?tool=fetch_logs&sessionId=<id>
POST /api/call?tool=clear_logs&sessionId=<id>
POST /api/restart
The /api/call request body is a JSON object matching the selected MCP tool arguments. HTTP callers may include sessionId in the query string to use a stable daemon-side log session. They may include profile to select which profile filters the server and tool view for that request.
Use jsmcp auth to manage OAuth for remote servers. With no arguments it lists remote servers that have OAuth enabled. With a server name it starts the OAuth flow for that server.
If no graphical environment is detected, or if you pass --no-browser, jsmcp auth <server> prints the authorization URL and waits for either the localhost callback or a pasted callback URL/code.
This repo includes systemd/jsmcp.service, a user unit that starts jsmcp server from the globally installed CLI.
Install it with:
npm install -g .
mkdir -p ~/.config/systemd/user
ln -sfn "$PWD/systemd/jsmcp.service" ~/.config/systemd/user/jsmcp.service
systemctl --user daemon-reload
systemctl --user enable --now jsmcp.service
Useful commands:
systemctl --user status jsmcp.service
journalctl --user -u jsmcp.service -f
systemctl --user restart jsmcp.service
The checked-in unit starts the default preset on the default daemon port and resolves jsmcp through the user's actual login shell from getent passwd.
The config file may be JSON or YAML and uses these top-level keys:
servers: server definitionsjsmcp: optional jsmcp-specific settingspresets: optional overrides for which servers and tools are exposed to the agentServer names must be valid JavaScript identifiers because execute_code() exposes them directly as globals.
jsmcp accepts both OpenCode MCP config style and the overlapping Claude Code MCP style for the common fields:
type: "local" or type: "stdio"type: "remote", type: "http", or type: "sse"command: ["cmd", "arg1"] or command: "cmd" with args: ["arg1"]environment or envSupported servers.<name> fields:
type: required; one of local, stdio, remote, http, ssedescription: optional string shown in list_servers()enabled: optional boolean; defaults to truetimeout: optional number in milliseconds used for initial tool discoverystrip_tool_prefix: optional string, true, or false; strings are removed from exposed tool names, true infers a shared prefix, and false disables prefix stripping for that servernormalize_tool_names: optional boolean; converts exposed tool names to snake_case after prefix strippingblocked_tools: optional server-level deny list; a tool name string or array of exact tool names, { glob: "..." }, and { regex: "..." } selectors. Selectors match final exposed tool names after prefix stripping and normalization, and blocked tools cannot be re-enabled by presets.Supported jsmcp fields:
auto_strip_tool_prefixes: optional boolean; default false; if true, servers infer and strip shared tool-name prefixes unless overridden by servers.<name>.strip_tool_prefixnormalize_tool_names: optional boolean; default false; if true, servers expose tool names as snake_case unless overridden by servers.<name>.normalize_tool_namesFor local / stdio servers:
command: required; non-empty string or non-empty arrayargs: optional array; appended to command when command is a string, and also accepted when command is an arrayenv: optional object of environment variablesenvironment: optional object of environment variables; merged with env, and wins on duplicate keyscwd: optional working directoryFor remote / HTTP / SSE servers:
url: required stringheaders: optional object of request headersoauth: optional OAuth configSupported oauth forms:
null, or true: enable OAuth with default behaviorfalse: disable OAuth for that serverclientIdclientSecretscopeSupported value substitutions in string fields:
{env:NAME}: expand from the current environment${NAME}: Claude Code-style environment expansion${NAME:-default}: Claude Code-style expansion with fallback{file:path}: replace with file contentsFor {file:path}:
~/... resolves from the user home directoryIf presets is omitted, the default preset includes every server with enabled !== false and allows all of that server's tools.
If presets is present, it is an object of preset names. Each preset is an object of per-server overrides layered on top of the server definitions:
presets.default: optional overrides for the default presetpresets.work: additional named preset overridesIf a server strips prefixes or normalizes names, preset tool selectors match the final exposed tool names that agents see.
Server-level blocked_tools selectors are applied before preset allowlists. Use them for globally unsafe tools, and use presets for profile-specific whitelists.
Within a preset, server rules work like this:
true: include that server and allow all its toolsfalse: exclude that server from that preset"tool_name": include only that exact tool{ "regex": "..." } selectors{ "glob": "..." } selectorsIf a server has enabled: false in servers, it is globally disabled and is not started or exposed by any preset.
Example:
{
"servers": {
"math": {
"type": "stdio",
"description": "Basic arithmetic tools",
"command": "node",
"args": ["/absolute/path/to/math-server.js"],
"env": {
"LOG_LEVEL": "debug"
},
"cwd": "${PWD}"
},
"docs": {
"type": "http",
"description": "Documentation search and retrieval",
"url": "https://example.com/mcp",
"headers": {
"Authorization": "Bearer ${DOCS_TOKEN}"
},
"oauth": {
"scope": "docs.read"
}
}
},
"presets": {
"default": {
"math": ["add", { "glob": "mul_*" }],
"docs": [{ "regex": "(search|fetch)" }]
},
"work": {
"docs": true
}
}
}
Compatibility notes:
env, type: "stdio", type: "http", type: "sse", and command plus args are supportedtype: "local", type: "remote", command arrays, and environment are also supportedheadersHelper and advanced OAuth fields like callbackPort or authServerMetadataUrl are not supported yetOAuth tokens and registration state are stored in $XDG_DATA_HOME/jsmcp/oauth.json or ~/.local/share/jsmcp/oauth.json.
list_serverslist_toolsexecute_codefetch_logsclear_logsenabled !== false is started once when jsmcp startslist_servers() is the required first step so the agent can learn what capabilities are availablelist_tools(server) before using a server in execute_code() so you know the exact tool names, aliases, and schemaslist_servers() and list_tools(server) return only the servers and tools allowed in the connection's selected profileexecute_code({ code, data?, timeoutMs? }) does not manage server lifecycle; it can only use servers that are already startedexecute_code({ code, ... }) whenever the work would require more than a single tool callconsole.log, console.info, console.warn, and console.error inside execute_code() are stored for fetch_logs()fetch_logs() drains the log buffer on readexecute_codeexecute_code runs JavaScript as the body of an async function.
Started servers are injected as globals. Each allowed MCP tool becomes a function on that server object. Prefer underscore aliases when available.
If you pass data, it is exposed to the script as the global variable data. This is useful for strings or structured values that would otherwise need escaping inside the code string.
You should call list_tools(server) before using a server in execute_code(). For multi-step work, prefer writing JavaScript instead of trying to mentally chain several tool calls.
Example:
return await math.add({ a: 2, b: 5 });
With data:
return data.message;
If the MCP tool returns structuredContent, that is what the JavaScript call resolves to. So the example above can return:
{
"sum": 7
}
If a tool name is not a valid JavaScript identifier, prefer its underscore alias:
return await math.tool_name({ value: 1 });
The original tool name still works with bracket access:
return await math["tool-name"]({ value: 1 });
Run in your terminal:
claude mcp add programmatic-mcp -- npx CSA PROJECT - FZCO © 2026 IFZA Business Park, DDP, Premises Number 31174 - 001
Security
Low riskAutomated heuristic from public metadata — not a security guarantee.