MCP server exposing ContextAI's static code-graph API to LLM agents
Project description
ConnectContext
An MCP server that exposes ContextAI's code-graph engine as tools an LLM agent can call: build a knowledge graph of a Python project (functions/classes as nodes, calls/relationships as edges), query it, and confirm it against real runtime behavior.
Stability
| Tier | Tools | Notes |
|---|---|---|
| Stable | build_graph, load_graph, find_node, get_context, get_edge_path, list_gaps |
Read-only static analysis. Never executes code. Tested against real multi-thousand-node codebases. |
| Experimental | run_trace, merge_trace |
run_trace executes the target's code (in a subprocess, in a chosen interpreter). Tested on synchronous, dependency-light Python. Async/threaded/network-dependent targets, and cross-platform/cross-version behavior, are not yet validated. Use on trusted code only. |
Tools
| Tool | What it does |
|---|---|
build_graph(project_root, output="graph.json") |
Statically analyze a project; save the graph. Run this first. |
load_graph(graph_path) |
Load/validate a graph; report node/edge counts. |
find_node(graph_path, query) |
Substring search for nodes; returns slim records with ids. |
get_context(graph_path, node_id, depth=2, direction="in", include_code=True) |
Focal node + neighbors + edges. Focal and (by default) each neighbor include their source code, so an agent can read the whole neighborhood in one call; pass include_code=false for a slim response. |
get_edge_path(graph_path, from_id, to_id) |
Direct edges between two nodes. |
list_gaps(graph_path, node_id) |
Unresolved calls / dynamic-dispatch gaps for a node. |
run_trace(project_root, target, static_graph, entry=None, python=None, timeout_seconds=120, ...) |
Execute a target under the tracer in a subprocess and return the ordered execution path with each step's source code, mapped onto the graph. On a crash it returns the path up to the failure plus the error and traceback — trace-to-context debugging in one call. Pass python=".venv/bin/python" to trace a project in its own venv (that interpreter must have contextai installed). Pass entry=<function> to get an exact, lossless split between import-time setup and real execution — without it, a long path falls back to a head/tail approximation. Always merges onto a fresh static graph; refuses a static_graph that already carries runtime data, so traced actions never silently accumulate. |
merge_trace(base_graph, call_log, project_root, ...) |
Fold a previously captured call log onto a pristine graph (no code execution). Same no-accumulation guarantee as run_trace. |
The long-running-session tracers (start_trace / stop_trace) are intentionally not
exposed: they hook the process they're called in, so driven across separate MCP tool calls
they capture nothing useful — they're meant to be embedded inside a long-running app
(e.g. a FastAPI startup/shutdown hook), not called remotely.
Setup
pip install -r requirements.txt # mcp[cli]<2 + contextai
Run
contextai-mcp # or: python server.py — stdio transport
For interactive inspection: mcp dev server.py.
Using with an MCP client
The server is the same regardless of client — only where each client looks for its config differs.
Once published to PyPI, any client can run it with zero manual install via
uvx:
| Client | Config file | Format |
|---|---|---|
| Claude Code | .mcp.json (project) |
JSON |
| Claude Desktop | claude_desktop_config.json |
JSON |
| Cursor | .cursor/mcp.json (project) or ~/.cursor/mcp.json (global) |
JSON, same shape as Claude |
| Codex CLI | .codex/config.toml (project) or ~/.codex/config.toml (global) |
TOML |
Claude Code / Claude Desktop / Cursor (JSON):
{ "mcpServers": { "contextai-graph": { "command": "uvx", "args": ["contextai-mcp"] } } }
Codex CLI (TOML):
[mcp_servers.contextai-graph]
command = "uvx"
args = ["contextai-mcp"]
Before publishing (local development), point at the installed console script directly instead — see .mcp.json, .cursor/mcp.json, and .codex/config.toml in this repo for working examples against this workspace's interpreter.
Typical flow
build_graph("/path/to/project")→ writesgraph.json.find_node("graph.json", "process_data")→ grab a nodeid.get_context("graph.json", "<id>")→ see callers/callees.list_gaps/get_edge_pathfor deeper inspection.run_trace(..., entry="the_function")→ confirm what actually executes, or get the exact path to a bug.
Layout
server.py # entrypoint: create_server().run()
contextai_mcp/
├── app.py # create_server() factory
├── _helpers.py # GraphStore cache, JSON coercion, no-accumulation guard
└── tools/ # one file per tool, each with register(mcp)
├── __init__.py # register_all(mcp)
├── build_graph.py load_graph.py find_node.py
├── get_context.py get_edge_path.py list_gaps.py
└── run_trace.py merge_trace.py
.mcp.json .cursor/mcp.json .codex/config.toml # client configs for this workspace
See docs/PROBLEMS_AND_SOLUTIONS.md for the full build log: every issue hit, why, and how it was fixed.
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 contextai_mcp-0.1.1.tar.gz.
File metadata
- Download URL: contextai_mcp-0.1.1.tar.gz
- Upload date:
- Size: 16.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7622c5603fef6216c25edbc23719b493ca1d2af106f53ea59c00ced096630da7
|
|
| MD5 |
ff1ccddb47ccd991058d4aa7c5e28b06
|
|
| BLAKE2b-256 |
04a7b6bc5b81bd2702a6c0402000c29940e99b1e3d0fb19b122c28b1b02ca847
|
File details
Details for the file contextai_mcp-0.1.1-py3-none-any.whl.
File metadata
- Download URL: contextai_mcp-0.1.1-py3-none-any.whl
- Upload date:
- Size: 18.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.1
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
78ca39829326385abc0064d6f2a80da09bad526a5729c7bdf2188771aff039cd
|
|
| MD5 |
c0fcab4728182e860331272d5e4b60ff
|
|
| BLAKE2b-256 |
2d51e9d79ff2edb87d1731f695d31953c9ac807f86ee4b7684d6be3d52545cf4
|