09 -- Subprocess MCP servers
Prerequisite: 01 -- HTTP GatewayYou will need: Running Hangar, Python 3.11+ Time: 5 minutes Adds: Run MCP servers as local subprocesses via stdin/stdout
The Problem
You have a Python MCP server package. You don't want to run it as a separate HTTP service -- you want Hangar to manage its lifecycle directly. Start it on demand, stop it when idle.
The Config
# config.yaml -- Recipe 09: Subprocess MCP servers
mcp_servers:
math:
mode: subprocess # NEW: subprocess mode
command: [python, -m, math_server] # NEW: command to run
idle_ttl_s: 300 # NEW: stop after 5min idle
health_check_interval_s: 60
max_consecutive_failures: 3
env: # NEW: environment variables
PYTHONUNBUFFERED: "1"Try It
Start Hangar:
bashmcp-hangar serveCheck status -- MCP server is COLD (not yet started):
bashmcp-hangar statusmath subprocess cold tools=0 idleInvoke a tool -- this triggers a cold start. Use the JSON-RPC protocol via stdio:
bash( echo '{"jsonrpc":"2.0","method":"initialize","params":{"protocolVersion":"2024-11-05","capabilities":{},"clientInfo":{"name":"test","version":"1.0"}},"id":1}' sleep 0.5 echo '{"jsonrpc":"2.0","method":"notifications/initialized","params":{}}' sleep 0.5 echo '{"jsonrpc":"2.0","method":"tools/call","params":{"name":"hangar_call","arguments":{"calls":[{"mcp_server":"math","tool":"add","arguments":{"a":1,"b":2}}]}},"id":2}' sleep 2 ) | mcp-hangar serve 2>/dev/null | grep '"id":2'{"jsonrpc":"2.0","id":2,"result":{"content":[{"type":"text","text":"{\"result\": 3}"}]}}Check status again -- MCP server is now READY:
bashmcp-hangar statusmath subprocess ready tools=5 idle_ttl=300sWait 5 minutes (or set
idle_ttl_s: 10for testing) and watch it stop:bashmcp-hangar statusmath subprocess cold tools=5 idle
What Just Happened
Subprocess MCP servers communicate via JSON-RPC over stdin/stdout. Hangar starts the process on first tool call, keeps it running while active, and stops it after the idle TTL expires. The StdioClient manages message correlation, timeouts, and process lifecycle.
Stderr output is captured into a ring buffer and available via the Log Streaming API.
Key Config Reference
| Key | Type | Default | Description |
|---|---|---|---|
mode | string | -- | Set to subprocess |
command | list[string] | -- | Command and arguments to start the MCP server |
idle_ttl_s | int | 300 | Seconds of inactivity before auto-stop |
env | dict | {} | Environment variables for the subprocess |
What's Next
Subprocesses are great for development. For isolation in production, run MCP servers in containers.