Skip to content

09 -- Subprocess MCP servers

Prerequisite: 01 -- HTTP Gateway You 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

  1. Start Hangar:
mcp-hangar serve
  1. Check status -- MCP server is COLD (not yet started):
mcp-hangar status
math    subprocess    cold    tools=0    idle
  1. Invoke a tool -- this triggers a cold start. Use the JSON-RPC protocol via stdio:
(
  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}"}]}}
  1. Check status again -- MCP server is now READY:
mcp-hangar status
math    subprocess    ready    tools=5    idle_ttl=300s
  1. Wait 5 minutes (or set idle_ttl_s: 10 for testing) and watch it stop:
mcp-hangar status
math    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.

--> 10 -- Discovery: Docker