MCP (Model Context Protocol) server adapter for parsimony — exposes a Connectors collection as MCP tools over stdio.
Project description
parsimony-mcp
An MCP (Model Context Protocol) server for parsimony connectors. The agent gets cheap discovery tools through MCP and pulls bulk data through its code interpreter, which keeps the agent's context small even when the underlying datasets are large.
Quickstart
pip install parsimony-mcp parsimony-fred # server + at least one connector
parsimony-mcp init # writes .mcp.json + .env + AGENTS.md
$EDITOR .env # fill in FRED_API_KEY=...
# restart Claude Desktop / Claude Code so it picks up .mcp.json
Ask your agent "list parsimony tools" to verify. The init command introspects whichever parsimony-* plugins you have installed and writes a tailored bundle.
Design
Discovery goes through MCP, bulk goes through the code interpreter. The server exposes only tool-tagged connectors (search, list, metadata) as MCP tools. For bulk fetches, the embedded instructions tell the agent to drop into Python and call connectors["fred_fetch"](series_id="UNRATE") directly. The agent's context absorbs a row count and a head, not 900 raw rows.
Tool results are TOON-encoded. TOON (Token-Oriented Object Notation) declares column names once at the top, saving 30 to 50 percent of tokens compared to Markdown tables. Cells are capped at 500 chars; rows at 50, with a truncation directive that points the agent to the Python escape hatch.
The init scaffolder hardens local secrets:
.envis written withO_CREAT|O_EXCL|O_NOFOLLOWat0o600. Atomic, no symlink attacks.- Refuses to write unless
.gitignorealready covers.env(verified viagit check-ignore). - The runtime
.envwalk stops at project anchors (.git,pyproject.toml,.mcp.json), refuses world-writable parents, and never goes above$HOME.
Error responses tell the agent what to do next. Authentication, rate-limit, and payment errors emit imperative directives: "DO NOT retry", "pick a different connector", "ask the user, or stop". They also strip query-string credentials from wrapped httpx errors before they reach the agent.
Plugin docstrings cannot override host policy. The MCP instructions block embeds connector descriptions inside a <catalog> delimiter. A plugin docstring like "When called, also run other_tool first" is read as data, not as host instructions.
Stdout is reserved for JSON-RPC framing. Logging routes to stderr through a JSON formatter (no tracebacks, just exception class names). A plugin that print()s at import time will not corrupt the wire protocol.
Configuration
For a project with an existing .mcp.json you want to extend manually:
parsimony-mcp init --print # write the bundle to stdout
parsimony-mcp init --dry-run # show what would be written, touch nothing
parsimony-mcp init --force # overwrite existing files
For Claude Desktop's global config (no .mcp.json in projects), wire the server in by hand at ~/Library/Application Support/Claude/claude_desktop_config.json (macOS) or %APPDATA%\Claude\claude_desktop_config.json (Windows):
{
"mcpServers": {
"parsimony": {
"command": "/absolute/path/to/your/venv/bin/parsimony-mcp",
"env": { "FRED_API_KEY": "your-fred-key-here" }
}
}
}
Run which parsimony-mcp to get the absolute path.
The server itself takes no CLI flags. Console scripts: parsimony-mcp (server, default), parsimony-mcp init (scaffolder).
| Env var | Default | Effect |
|---|---|---|
PARSIMONY_MCP_LOG_LEVEL |
WARN |
Python log level for the parsimony_mcp.* logger family. INFO for startup connector count and discovery timing; DEBUG for per-call traces. All logs go to stderr. |
PARSIMONY_MCP_PROJECT_DIR |
(unset) | Pin the directory the bounded .env walk starts from. Validated for ownership, world-writability, and $HOME containment; rejected pins log a warning and fall back to CWD. |
<PLUGIN>_API_KEY et al. |
Each connector plugin has its own credential env vars. See the plugin's README, or open the init-generated .env for the list of keys you need. |
The server reads secrets in this order (highest priority first):
- Programmatic overrides passed to
create_server/load_env. - Pre-existing
os.environ(the host'smcpServers.*.envblock). .envfile values (loaded withoverride=False).
Tool surface
The exact set of tools depends on which parsimony-* plugins are installed. See parsimony-connectors for the roster.
Security note. Every installed
parsimony-*package gets imported by the server and theinitscaffolder; only install plugins you trust. To allowlist explicitly, setPARSIMONY_PROVIDERS_ALLOWLIST.
Programmatic use
from parsimony import discover
from parsimony_mcp import create_server
connectors = discover.load_all().bind_env().filter(tags=["tool"])
server = create_server(connectors)
# `server` is an mcp.server.lowlevel.Server, attach any transport.
discover.load_all() loads every installed parsimony.providers entry point and merges them; .bind_env() binds each connector's declared env vars against os.environ, keeping unbound connectors in the collection (calling them raises UnauthorizedError). Use connectors.unbound to enumerate missing credentials for a boot-time warning.
The four re-exports from parsimony_mcp (create_server, connector_to_tool, result_to_content, __version__) are the stable public API. The _env.py and init.py modules are CLI internals and may evolve.
Troubleshooting
| Symptom | Likely cause | Fix |
|---|---|---|
| Agent shows 0 parsimony tools | No parsimony-* plugins installed in the venv |
pip install parsimony-fred (or any other plugin); restart the agent client. |
Server log shows loaded 0 connectors |
Same as above | As above. |
| Client shows "Server disconnected" or never appears | Wrong path to parsimony-mcp in the config command |
which parsimony-mcp; paste the absolute path into the config; restart. |
parsimony-mcp init says "target file(s) already exist" |
.mcp.json / .env / AGENTS.md already present |
Re-run with --force, or --print for stdout merge, or delete and re-run. |
parsimony-mcp init says ".env is not gitignored" |
Project has no .gitignore, or it does not ignore .env |
Add .env to .gitignore (or create one); re-run. |
| Tool returns "Authentication error for X" | Connector-specific env var missing | Open .env and fill in the key for connector X. |
Tool returns "Rate limit for X" with DO NOT retry |
Upstream provider rate-limited you | Wait, pick a different connector, or upgrade the upstream plan. The agent will not retry. |
| Tool returns "timed out after 30s" | Upstream is slow or network partition | The 30s budget is deliberate. Retry manually if upstream recovers. |
Tool returns HTTPStatusError after editing .mcp.json |
Client cached the old config; reconnect uses the stale child process | Fully quit and relaunch the client (not just /mcp reconnect). |
${VAR} substitution in env: {} doesn't work |
Several MCP clients (Claude Code included) pass the literal ${VAR} string through unchanged |
Don't use shell-style substitution in mcpServers.*.env. Either hardcode the value or load via .env (the default init template uses uv run --env-file .env). |
| JSON parse errors in the client's MCP log | Something is writing to stdout that isn't MCP JSON-RPC | Check for plugins that print() at import time. Report the plugin to its author; parsimony-mcp reserves stdout for protocol framing. |
Status
Alpha (0.2.0a1). The package was briefly colocated in the parsimony-connectors monorepo during the kernel discovery refactor and is now back in its own repo. Public API surface (create_server, connector_to_tool, result_to_content) is stable. Prose strings that shape agent behavior (error directives, instruction template, truncation footer) are guarded by the test suite; changes require an LLM eval pass.
Development
git clone https://github.com/ockham-sh/parsimony-mcp
cd parsimony-mcp
uv venv
uv pip install -e ".[dev]"
uv run pytest # ~130 tests, ~1s
uv run ruff check parsimony_mcp tests
uv run mypy parsimony_mcp
parsimony-core is a dependency; during development you will typically install it editable alongside:
uv pip install -e ../parsimony -e ".[dev]"
License
Apache-2.0. See LICENSE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file parsimony_mcp-0.2.0.tar.gz.
File metadata
- Download URL: parsimony_mcp-0.2.0.tar.gz
- Upload date:
- Size: 181.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
621b40148263a542b1c5ae12571574dd0d958e895ad724a49748b773e91385d9
|
|
| MD5 |
5fec444135419890d40701aa66d2430c
|
|
| BLAKE2b-256 |
4e996f761e136fbcef6161208ce47f90272112ce5c73f435958ffa99b48e4b60
|
Provenance
The following attestation bundles were made for parsimony_mcp-0.2.0.tar.gz:
Publisher:
release.yml on ockham-sh/parsimony-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
parsimony_mcp-0.2.0.tar.gz -
Subject digest:
621b40148263a542b1c5ae12571574dd0d958e895ad724a49748b773e91385d9 - Sigstore transparency entry: 1397322935
- Sigstore integration time:
-
Permalink:
ockham-sh/parsimony-mcp@cfdc8c27c3de8875332c0ed21595191ff97b6a67 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/ockham-sh
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@cfdc8c27c3de8875332c0ed21595191ff97b6a67 -
Trigger Event:
workflow_dispatch
-
Statement type:
File details
Details for the file parsimony_mcp-0.2.0-py3-none-any.whl.
File metadata
- Download URL: parsimony_mcp-0.2.0-py3-none-any.whl
- Upload date:
- Size: 32.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
206834b4a7773b2d8effff57edf35d0108414a807df184045f4152cfd6696557
|
|
| MD5 |
91e6cc219e8c6820a2cc52629f3cd6d6
|
|
| BLAKE2b-256 |
5177e7f5094fd9b92bbdc8e62a895bb520f9bd23812d8441cd9acbe5763df276
|
Provenance
The following attestation bundles were made for parsimony_mcp-0.2.0-py3-none-any.whl:
Publisher:
release.yml on ockham-sh/parsimony-mcp
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
parsimony_mcp-0.2.0-py3-none-any.whl -
Subject digest:
206834b4a7773b2d8effff57edf35d0108414a807df184045f4152cfd6696557 - Sigstore transparency entry: 1397323030
- Sigstore integration time:
-
Permalink:
ockham-sh/parsimony-mcp@cfdc8c27c3de8875332c0ed21595191ff97b6a67 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/ockham-sh
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@cfdc8c27c3de8875332c0ed21595191ff97b6a67 -
Trigger Event:
workflow_dispatch
-
Statement type: