programmatic-mcp
Provides an execution environment for JavaScript code, allowing users to programmatically orchestrate and interact with multiple MCP servers and their tools through a unified interface.
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@programmatic-mcpRun a script to add 123 and 456 using the math server tools."
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
jsmcp
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:
several related tool calls
branching logic based on earlier results
loops, retries, or result aggregation
transforming tool output before the next call
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:
the agent first learns what servers and tools are available, while
jsmcpconstrains access to whatever servers and tools you allow in a presetthe agent can then write JavaScript for multi-step work
logs stay separate from return values so the code stays easier to reason about
Config 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.
Why
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:
combine results from several MCP tools
script workflows across one or more MCP servers
make decisions in code instead of repeatedly re-planning between tool calls
keep tool access constrained to a reviewed preset
Install
npm install -g @alesya_h/jsmcpOr run it without installing globally:
npx @alesya_h/jsmcp runRun
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 auth
jsmcp auth firefox_devtoolsIf 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 a running daemon over HTTP 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>.
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:
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>The 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.
systemd User Service
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.serviceUseful commands:
systemctl --user status jsmcp.service
journalctl --user -u jsmcp.service -f
systemctl --user restart jsmcp.serviceThe 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.
Config
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 agent
Server 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:
local servers:
type: "local"ortype: "stdio"remote servers:
type: "remote",type: "http", ortype: "sse"commands: either
command: ["cmd", "arg1"]orcommand: "cmd"withargs: ["arg1"]environment variables: either
environmentorenv
Supported servers.<name> fields:
type: required; one oflocal,stdio,remote,http,ssedescription: optional string shown inlist_servers()enabled: optional boolean; defaults totruetimeout: optional number in milliseconds used for initial tool discoverystrip_tool_prefix: optional string,true, orfalse; strings are removed from exposed tool names,trueinfers a shared prefix, andfalsedisables prefix stripping for that servernormalize_tool_names: optional boolean; converts exposed tool names tosnake_caseafter 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; defaultfalse; iftrue, servers infer and strip shared tool-name prefixes unless overridden byservers.<name>.strip_tool_prefixnormalize_tool_names: optional boolean; defaultfalse; iftrue, servers expose tool names assnake_caseunless overridden byservers.<name>.normalize_tool_names
For local / stdio servers:
command: required; non-empty string or non-empty arrayargs: optional array; appended tocommandwhencommandis a string, and also accepted whencommandis an arrayenv: optional object of environment variablesenvironment: optional object of environment variables; merged withenv, and wins on duplicate keyscwd: optional working directory
For remote / HTTP / SSE servers:
url: required stringheaders: optional object of request headersoauth: optional OAuth config
Supported oauth forms:
omitted,
null, ortrue: enable OAuth with default behaviorfalse: disable OAuth for that serverobject with any of:
clientIdclientSecretscope
Supported 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 contents
For {file:path}:
relative paths are resolved relative to the config file directory
~/...resolves from the user home directoryabsolute paths are used as-is
If 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 presetany other preset name, such as
presets.work: additional named preset overrides
If 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:
omitted server rule: use the server definition as-is
true: include that server and allow all its toolsfalse: exclude that server from that preset"tool_name": include only that exact toolarray entries may be:
exact tool name strings
{ "regex": "..." }selectors{ "glob": "..." }selectors
If 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:
Claude Code-style
env,type: "stdio",type: "http",type: "sse", andcommandplusargsare supportedOpenCode-style
type: "local",type: "remote", command arrays, andenvironmentare also supportedClaude Code-specific features such as
headersHelperand advanced OAuth fields likecallbackPortorauthServerMetadataUrlare not supported yet
OAuth tokens and registration state are stored in $XDG_DATA_HOME/jsmcp/oauth.json or ~/.local/share/jsmcp/oauth.json.
Exposed Tools
list_serverslist_toolsexecute_codefetch_logsclear_logs
Behavior
every server with
enabled !== falseis started once whenjsmcpstartslist_servers()is the required first step so the agent can learn what capabilities are availableyou must call
list_tools(server)before using a server inexecute_code()so you know the exact tool names, aliases, and schemaslist_servers()andlist_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 startedprefer
execute_code({ code, ... })whenever the work would require more than a single tool callconsole.log,console.info,console.warn, andconsole.errorinsideexecute_code()are stored forfetch_logs()fetch_logs()drains the log buffer on read
execute_code
execute_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 });This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/alesya-h/jsmcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server