mcp-broker
mcp-broker
mcp-broker is a local Model Context Protocol process broker for MCP clients.
Think PgBouncer for MCP: one stable local endpoint in front of many upstream MCP servers. The broker owns upstream startup, reuse, cleanup, profile exposure, status, and safe tool routing.
The core idea is simple: do not make every agent session load every upstream tool definition before the user asks a task.
Why this exists
AI coding sessions with many MCP servers tend to accumulate the same problems:
every client config repeats the same MCP server list
every new session can start duplicate upstream processes
OAuth, browser state, local files, and database handles spread across tools
raw tool lists consume context before the task begins
hosted connector caches can duplicate local MCP tools
orphaned MCP processes survive after client sessions exit
mcp-broker puts a small broker facade in front of those upstreams. It is not a hosted workflow builder; it is local infrastructure for keeping MCP clients small, predictable, and under one config contract.
Client profile
|
| one local MCP entry
v
mcp-broker-client
|
| Unix socket
v
mcp-broker-daemon
|
| profile gates, namespace routing, status, cleanup
v
upstream MCP serversThe client sees a small set of broker tools:
broker.search_tools
broker.describe_tool
broker.call_tool
broker.statusThe upstream MCPs still exist. They are discovered and called through the broker when a task needs them.
Measured context reduction
On 2026-05-24, the measured Codex setup went from many raw MCP and hosted app tool definitions to one broker facade plus a pruned codex_apps cache.
Surface | Before | After | Reduction |
Direct Codex MCP server entries | 11 | 1 | 90.91% |
MCP tool definitions | 414 | 4 | 99.03% |
Hosted | 195 | 39 | 80.00% |
Combined always-loaded tool definitions | 609 | 43 | 92.94% |
Combined serialized tool payload bytes | 1,026,171 | 185,877 | 81.89% |
Combined | 276,989 | 45,281 | 83.65% |
The 92.94% number is a tool-definition count reduction. The 83.65% number is a token reduction for canonical serialized tool payloads measured with tiktoken o200k_base.
See docs/context-reduction-measurement.md for evidence and caveats.
What it does
Runs one local broker daemon over a Unix socket.
Exposes one lightweight stdio client shim to MCP clients.
Starts upstream MCP servers on demand.
Reuses shared upstreams across sessions when configured.
Isolates per-session upstreams when state must not be shared.
Maps upstream tools into stable namespaces.
Exposes compact search, describe, call, and status tools.
Enforces profile-level tool budgets and exposure gates.
Blocks mutating upstream exposure unless a profile allowlist grants it.
Stores runtime state under
$HOME/mcp/mcp-broker, outside the repo.Renders MCP client config entries with dry-run, backup, and rollback.
Provides LaunchAgent install and uninstall flows for macOS.
Provides Linux systemd user-service render, install, unload, and removal flows.
Provides Windows PowerShell Scheduled Task render, install, and removal flows.
Includes unit, journey, live, and e2e tests through Makefile targets.
Core differentiators:
Profile-scoped exposure: each MCP client gets a configured view of upstreams instead of every tool by default.
Mutating-tool gates: mutating upstreams stay hidden until a profile allowlist grants access.
Lifecycle ownership: shared and per-session upstreams are started, watched, stopped, and reaped by the broker.
Client parity checks: rendered profiles can be validated through the same compact broker facade before config is applied.
Who this is for
Use mcp-broker if you:
use Codex, Claude Code, Gemini CLI, or other MCP clients
have more MCP tools than you want in every session
need shared local MCP servers without duplicate process startup
want one place for OAuth state, browser state, sockets, logs, and cleanup
need per-client profiles instead of the same tool list everywhere
want a small broker facade instead of raw upstream tool dumps
This repo is not an enterprise MCP control plane. It is local desktop infrastructure for developer-agent workflows.
Current status
Implemented:
YAML config loading from
config/broker.private.yaml, created fromconfig/broker.example.yaml.Strict YAML contract validation for runtime, clients, profiles, upstreams, and policy blocks.
Public JSON Schema validation through
config/broker.schema.json.Runtime path derivation from
runtime.root.Tool namespace mapping from configured upstream prefixes.
Local upstream subprocess lifecycle management and process-group cleanup.
Broker daemon over Unix socket.
MCP client shim and renderers for configured MCP client profiles, including Codex, Claude, and Gemini.
Gemini profile rendering to
.gemini/settings.json, including its MCP allowed-server policy.Dry-run client config rendering, apply-time backups, and rollback.
LaunchAgent render and install scripts with dry-run defaults.
Compact broker facade for search, describe, call, and status.
Profile validation from YAML smoke probes.
Discovery parity checks between compact client profiles.
Public and maintainer quality gates through Makefile targets.
Wiring status:
Codex is wired through the broker.
Claude is wired through the broker after profile validation and manual
/mcpacceptance.Gemini is wired through the broker by rendering
.gemini/settings.json.
Public release status:
The repo is designed to stay public-safe.
Private upstream inventory, account paths, OAuth state, secrets, sockets, logs, and generated client configs stay outside git.
Stable release
1.1.0is published to PyPI, Homebrew, NPM, Docker Hub, GHCR, and the MCP Registry; publication proof is tracked indocs/distribution.md.Docker image support is available for container-friendly configs. Docker MCP Catalog submission still requires Docker review.
MCPB metadata is present at
mcpb/manifest.jsonfor local directory review.
See ROADMAP.md for public-facing release work.
Architecture
mcp-broker has three runtime layers:
Layer | Responsibility |
Client shim | Presents one stdio MCP server entry to each MCP client and forwards JSON-RPC over the broker socket. |
Broker daemon | Owns profile gates, namespace routing, upstream lifecycle, status, logging, and cleanup. |
Upstream MCP servers | Run as configured |
The config file is the contract. Profiles decide exposure, upstreams define transport and lifecycle behavior, and smoke probes define safe read calls for validation.
Comparison
Approach | Best fit | Tradeoff |
Raw MCP client config | Small setups with a few tools. | Every session loads the full tool list and each client repeats config. |
Simple MCP proxy | Forwarding one server to one client. | Does not own upstream lifecycle, profile budgets, or cross-client cleanup. |
Hosted app connectors | SaaS tools managed by the client provider. | Local MCP state and cross-client parity remain outside user control. |
| Local developers with many upstream MCPs across MCP clients. | Adds a local daemon and config contract that must be installed and monitored. |
Screenshots Or GIF
The quickstart flow should look like this:
make config-init
make config-validate
make broker-status
make codex-facade-smokeIn an MCP client, /mcp should show one mcp-broker entry. Use broker.status
to inspect profile-visible upstream state.
Quickstart
Prerequisites:
macOS with
launchctlfor LaunchAgent use.Python 3.10 or newer available as
python3.make.Node.js and
npxfor npm-based upstream MCP servers.A clone of this repo.
Package installs:
pipx install mcp-broker
uv tool install mcp-broker
brew tap NavinAgrawal/tap
brew install mcp-brokerHomebrew installs the same console scripts as the Python package. Package installs do not write MCP client config; client wiring stays an explicit Makefile action.
Docker is for container-friendly configs:
docker build -t mcp-broker:local .
docker run --rm -i mcp-broker:localLocal stdio clients and MCPB-style installs use the package-owned lifecycle:
mcp-broker stdio --init-if-missingCreate the local venv, install dependencies, and verify runtime layout:
make setupCreate private config from the public template:
make config-initconfig-init creates the destination directory when needed and copies the public
template as the starting point. It does not import local MCP inventory, user
paths, or secrets.
Edit config/broker.private.yaml for local upstreams. Keep secret values out of config. Use environment variable names or files under:
$HOME/mcp/mcp-broker/secrets/Run the quality gate:
make quality-gateValidate the configured YAML contract:
make config-validateStart the broker:
make broker-startCheck status:
make broker-statusFor the full install flow, see docs/install.md.
Runtime layout
Default runtime root:
$HOME/mcp/mcp-broker/
|- backups/
|- logs/
|- renders/
|- run/
|- secrets/
|- sockets/
`- state/
`- upstreams/Runtime files are not repo files. Upstream OAuth state, browser state, secret files, sockets, logs, rendered client configs, backups, and daemon state belong under the runtime root.
Client wiring
Back up a client config:
make config-backup CLIENT=codexDry-run render:
make config-render CLIENT=codex CONFIG_RENDER_APPLY=0Apply after reviewing the rendered file under $HOME/mcp/mcp-broker/renders/:
make config-render CLIENT=codex CONFIG_RENDER_APPLY=1Rollback:
make config-rollback CLIENT=codexUse CLIENT=claude or CLIENT=gemini after that profile smoke passes and that
client is intended to use the broker. For new JSON-based MCP clients, generate a
starter block:
make profile-snippet NEW_PROFILE=local-client NEW_CLIENT_FORMAT=mcp-settings-jsonSee docs/add-profile.md for the full new-profile flow.
Compact broker facade
The compact facade keeps chat-facing profiles small:
Tool | Purpose |
| Search configured upstream tools by query. |
| Return schema and metadata for one upstream tool. |
| Call one upstream tool through broker routing. |
| Show profile-visible upstream state, passive auth probes, and last errors without starting tools. |
Codex /mcp shows the single mcp-broker entry by design. Per-upstream visibility comes from broker.status.
Profiles and safety
Profiles decide which upstreams a client can see and call.
Supported concepts:
max_toolsprotects clients from huge tool lists.compact_tools_enabledexposes broker facade tools instead of raw upstream tools.broker_tool_name_styleadapts broker facade names for clients that cannot surface dotted tool names.mcp_allowed_serversrenders client settings for MCP clients that require an explicit server allowlist.allow_mutating_upstreamsis required before a mutating upstream can be exposed.sharedmode reuses one upstream process where shared account state is acceptable.per_sessionmode isolates upstream state per client session.disabledmode keeps compatibility records without exposing the upstream.
Protected surfaces such as OAuth, browser state, filesystem roots, and databases require explicit config and validation. Public examples stay disabled or placeholder-based.
See docs/security-review.md and docs/upstream-compatibility-matrix.md. For a deeper safety checklist, see docs/safety.md.
Config contract
The public template is config/broker.example.yaml. The matching JSON Schema is config/broker.schema.json.
Supported top-level sections:
schema_version: 1
runtime: {}
broker: {}
profiles: {}
clients: {}
upstreams: {}The loader rejects unknown keys. Runtime placeholders such as {runtime.root}, {runtime.state_dir}, and {runtime.secrets_dir} can be used in upstream command, args, working directory, and env file paths.
make config-validate checks the selected CONFIG_PATH against the public JSON Schema first, then runs the runtime loader so semantic rules are enforced from the same code path the broker uses.
Each enabled upstream exposed to a profile should define a safe smoke probe:
smoke:
query: read example graph
tool: example-store.read_graph
arguments: {}
call: truemake profile-validation PROFILE=<profile> validates every enabled upstream visible to that profile through broker.status, broker.search_tools, broker.describe_tool, and the configured safe broker.call_tool.
Codex operator acceptance
Repo-owned tests validate broker behavior through the local client shim. The last Codex-specific check has to run inside an active Codex session because that is where the deferred MCP wrapper tools exist.
Generate the current acceptance steps from YAML:
make codex-deferred-acceptanceThe target reads the configured smoke probes and prints the exact
mcp__mcp_broker__ wrapper calls for search, describe, and safe call. It does
not invoke Codex, does not call an external LLM session, and is not part of
make quality-gate.
See docs/codex-deferred-tool-acceptance.md.
LaunchAgent
Render without writing:
make launchagent-installApply and load:
make launchagent-install LAUNCHAGENT_APPLY=1
make launchagent-load
make broker-statusUnload or remove:
make launchagent-unload
make launchagent-uninstall LAUNCHAGENT_APPLY=1systemd
Linux user-service install uses the same runtime root and config path contract:
make systemd-install
make systemd-install SYSTEMD_APPLY=1
make systemd-loadFor package installs, set MCP_BROKER_DAEMON_COMMAND to the installed daemon
path before applying the service.
Windows
Windows startup uses PowerShell Scheduled Task commands with the same runtime root and config path contract:
make windows-install
make windows-install WINDOWS_APPLY=1
make windows-loadRemove it with:
make windows-unload
make windows-uninstall WINDOWS_APPLY=1Test and release gates
Run all test tiers:
make testRun the public quality gate:
make quality-gateThe coverage gate uses line and branch coverage for Python source.
Run the release gate when preparing a tag:
make release-gaterelease-gate runs package, smoke, and mutation checks. Mutation
runs public unit and journey tests last and writes
var/quality/mutation_stats.json with total counts, score, and ranked
blocked_by_file entries. On macOS, the release gate runs mutation inside a
Linux container to avoid local mutmut fork failures. E2E tests remain in make
quality-gate.
Run smoke and runtime cleanup checks:
make config-validate
make broker-smoke
make broker-stop
make broker-reap
make doctor
make release-smokeRelease or client config apply should wait for:
make quality-gatemake config-validatemake broker-smokedry-run config render for each intended client
rollback test
make release-smokemake release-gatebefore taggingmake doctorwith no stale broker-owned resources
See docs/release-checklist.md.
Public commands
These targets use this repo plus declared Python and Node prerequisites:
make setup
make config-init
make test
make test-unit
make test-journey
make test-live
make test-e2e
make test-cov
make precommit
make quality-gate
make release-gate
make config-validate
make broker-smoke
make broker-start
make broker-status
make broker-stop
make broker-reap
make doctor
make config-backup
make codex-app-policy
make config-render
make config-rollback
make tools-count
make facade-smoke
make codex-facade-smoke
make claude-facade-smoke
make gemini-facade-smoke
make profile-validation
make codex-profile-validation
make claude-profile-validation
make gemini-profile-validation
make discovery-parity
make codex-claude-discovery-parity
make codex-deferred-acceptance
make launchagent-install
make launchagent-load
make launchagent-unload
make launchagent-uninstall
make systemd-install
make systemd-load
make systemd-unload
make systemd-uninstall
make windows-install
make windows-load
make windows-unload
make windows-uninstall
make linux-container-smoke
make windows-powershell-smoke
make release-smoke
make mutation
make mutation-linuxmake quality-gate is repo-local. It does not call personal scripts outside this repo.
make codex-deferred-acceptance is maintainer-only. It does not invoke Codex
or an external LLM session. It reads the same YAML smoke probes and prints the
exact mcp__mcp_broker__ deferred wrapper calls to run inside an active Codex
session. See docs/codex-deferred-tool-acceptance.md.
Project tree
mcp-broker/
|- .gitignore
|- Makefile
|- README.md
|- pyproject.toml
|- requirements.txt
|- config/
| |- broker.example.yaml
| |- broker.private.yaml # local, ignored by git
| `- broker.schema.json
|- docs/
|- registry/
|- scripts/
| `- check_mutation_stats.py
|- src/
| `- mcp_broker/
|- tests/
| |- unit/
| |- journey/
| |- live/
| |- e2e/
| `- support/
`- var/ # tracked skeleton; generated contents ignoredGenerated reports stay under var/, especially var/coverage/, var/test-logs/, and var/quality/.
Docs
Design rules
Keep upstream definitions in central config.
Keep runtime state under
$HOME/mcp/mcp-broker.Keep private upstream inventory in
config/broker.private.yaml, which is ignored by git.Keep secret values out of config and source.
Do not hardcode personal paths in source, tests, docs, or public config.
Run build, test, runtime, and config operations through the Makefile.
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/NavinAgrawal/mcp-broker'
If you have feedback or need assistance with the MCP directory API, please join our Discord server