Skip to content

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

yaml
# 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:

    bash
    mcp-hangar serve
  2. Check status -- MCP server is COLD (not yet started):

    bash
    mcp-hangar status
    math    subprocess    cold    tools=0    idle
  3. Invoke 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}"}]}}
  4. Check status again -- MCP server is now READY:

    bash
    mcp-hangar status
    math    subprocess    ready    tools=5    idle_ttl=300s
  5. Wait 5 minutes (or set idle_ttl_s: 10 for testing) and watch it stop:

    bash
    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

KeyTypeDefaultDescription
modestring--Set to subprocess
commandlist[string]--Command and arguments to start the MCP server
idle_ttl_sint300Seconds of inactivity before auto-stop
envdict{}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

MCP Hangar · Released under MIT License.