Branch Diagnostics MCP
An MCP server that enables AI assistants to diagnose network and service problems by walking a structured funnel of cURL timing metrics, identifying which hop is at fault, and providing targeted recommendations.
README
Branch Diagnostics MCP
An MCP server that turns a vague network complaint into a disciplined, AI-driven investigation.
"The branch office is slow." → a structured triage that pinpoints which hop is to blame — DNS, TCP, TLS, the server, or the link — and says what to do about it.
This server gives an AI assistant a set of tools to investigate service and network problems the way a seasoned engineer would: not by guessing, but by walking a deliberate funnel of evidence over cURL timing metrics.
The problem (why this exists)
When a user reports "the app is slow" or "the branch can't connect," the complaint is vague but the cost is real: time-to-resolution. Good triage is slow, inconsistent, and locked in the heads of a few senior engineers — everyone else gathers the wrong data, reads it the wrong way, and escalates.
The expertise that makes triage fast is actually quite structured: for this kind of symptom, look at these specific signals, in this order, and here's what "bad" looks like. That structure can be encoded once and handed to an AI assistant — so anyone, at any hour, runs the same rigorous investigation. That's what this project does.
What's MCP? The Model Context Protocol is an open standard (introduced by Anthropic in late 2024) for giving AI assistants real tools and data through a uniform interface. An MCP server like this one exposes capabilities; any MCP client (Claude Desktop, IDEs, agents) can use them. This project was first built in June 2025, in MCP's earliest months — see Evolution below.
What it does — the funnel
A single coherent workflow, each step a tool the assistant can call:
flowchart LR
S["Symptom<br/>(free text)"] --> C["1 · Categorize<br/>diagnostic_categorize"]
C --> M["2 · Pick metrics<br/>find_metrics"]
M --> D["3 · Collect data<br/>get_data_metrics"]
D --> A["4 · Analyse<br/>analyse"]
A --> R["Severity + anomalies<br/>+ recommendations"]
The assistant is also given a guidance prompt that teaches it how to run the funnel, and two resources it can browse: the catalog of diagnostic categories and the catalog of metrics.
Architecture & design decisions
Why cURL metrics are the right signal
CURLINFO_* values are libcurl's per-request timing and outcome
breakdown — the same data you can see with curl -w. Every HTTP request passes through ordered
phases, and libcurl reports a cumulative timestamp at each one. The power is in the differences
between adjacent phases: each gap isolates one stage of the request, so a single slow request tells
you exactly which hop is at fault.
| Phase gap | cURL metric math | What it isolates |
|---|---|---|
| DNS resolution | NAMELOOKUP_TIME |
Name servers / resolver |
| TCP connect | CONNECT_TIME − NAMELOOKUP_TIME |
Network path, routing, latency |
| TLS handshake | PRETRANSFER_TIME − CONNECT_TIME |
Certificates, TLS negotiation |
| Server think-time (TTFB) | STARTTRANSFER_TIME − PRETRANSFER_TIME |
The application / backend |
| Content download | TOTAL_TIME − STARTTRANSFER_TIME |
Throughput, payload size, link |
Alongside timing, outcome metrics (RESPONSE_CODE, SSL_VERIFYRESULT, OS_ERRNO,
NUM_CONNECTS, …) catch failures rather than slowness. Together they cover the two questions every
triage starts with: is it slow, or is it broken — and where?
The MCP surface
Tools (all read-only, annotated as such):
| Tool | Funnel step | In → Out |
|---|---|---|
diagnostic_categorize |
1 · classify | symptom → best category + confidence + all scores |
find_metrics |
2 · select | symptom, category → the metrics to collect, each with why |
get_data_metrics |
3 · collect | metrics, location → current values, rolling stats, thresholds |
analyse |
4 · evaluate | symptom, category, data → severity, anomalies, recommendations |
Resources (browsable JSON, with parameterized lookups):
branch://categories, branch://categories/{name}, branch://metrics, branch://metrics/{name}.
Prompt: branch_diagnostics_guidance — reusable system guidance that teaches a client to drive
the funnel (the diagnostic methodology, not just the tool list).
Design decisions worth calling out
- Structured, typed tool output. Tools return typed dataclasses, so the server emits machine-
readable
structuredContentwith an auto-generatedoutputSchema— clients get data, not prose to re-parse. (The original prototype returned hand-formatted Markdown; this is the meaningful upgrade.) - A registered guidance prompt. The diagnostic methodology ships with the server as a first-class MCP prompt, instead of living in a comment.
- Read-only by contract. Every tool is annotated
readOnlyHint, so clients know it's safe to call. - A pluggable data layer.
MetricsDataSourceis isolated behind one seam. It simulates realistic data today (so the server runs out of the box); a real backend drops in without touching any diagnostic logic — see Going to production. - Vendor-neutral by design. Pure libcurl + observability vocabulary; nothing tied to any product.
Worked example
Driving the funnel for "branch office VPN connectivity problems" (actual server output):
1 · diagnostic_categorize("branch office vpn connectivity problems")
→ recommended_category: "Branch Office Issue" (confidence 3)
2 · find_metrics("branch office vpn connectivity problems", "Branch Office Issue")
→ CURLINFO_NAMELOOKUP_TIME, CURLINFO_CONNECT_TIME, CURLINFO_LOCAL_IP,
CURLINFO_PRIMARY_IP, CURLINFO_TOTAL_TIME (each with a relevance note)
3 · get_data_metrics([...], "branch-paris-01")
→ { "CURLINFO_CONNECT_TIME": { current: 0.125, threshold_warning: 0.5, ... }, ... }
simulated: true
4 · analyse("branch office vpn connectivity problems", "Branch Office Issue", <data>)
→ overall_severity: "NORMAL"
analysis_summary: "No significant anomalies detected ..."
next_steps: [ "Monitor the identified metrics over time ...", ... ]
Feed analyse data where, say, CURLINFO_CONNECT_TIME exceeds its critical threshold and the verdict
flips to CRITICAL with a targeted recommendation — the network hop, not the server, is implicated.
Install & run
Requires Python ≥ 3.13 and uv.
uv venv
uv pip install -e .
Run it (stdio is the default transport, ideal for local MCP clients):
uv run python branch_diagnostics_server.py
# or via the FastMCP CLI:
uv run fastmcp run branch_diagnostics_server.py
Run it over Streamable HTTP instead:
MCP_HTTP=1 uv run python branch_diagnostics_server.py # serves on http://127.0.0.1:8000/mcp
Register it with an MCP client (e.g. Claude Desktop) by adding to the client's config:
{
"mcpServers": {
"branch-diagnostics": {
"command": "uv",
"args": ["run", "python", "branch_diagnostics_server.py"],
"cwd": "/path/to/branch_mcp_v2"
}
}
}
Smoke-test the whole funnel in-memory (no network):
uv run python tests/smoke_test.py
Simulated data — going to production
get_data_metrics returns simulated values by default (the response carries simulated: true), so
the server is useful immediately. The data layer is deliberately isolated in a single class,
MetricsDataSource. To go live, implement one that reads real measurements — from a synthetic-probe /
active-test result store, a time-series database, or an observability backend — and the four tools, the
analysis, and the schemas all keep working unchanged.
Evolution
This is the 2026 modernized successor to a prototype I built in June 2025, during MCP's first
months: branch_MCP (its commit history dates the work).
The diagnostic idea held up; the platform moved on. v2 brings it current:
| v1 (Jun 2025) | v2 (2026) | |
|---|---|---|
| Framework | FastMCP 2.8 (now EOL) | FastMCP 3.4 |
| Tool output | hand-formatted Markdown strings | typed, structured outputSchema |
| Guidance prompt | a dead variable, never registered | a registered MCP prompt |
| Tool metadata | none | read-only annotations |
| Resources | two flat JSON blobs | + parameterized templates |
| Data layer | inline simulator | pluggable MetricsDataSource seam |
Taken together, the pair is a small, honest record of spotting a protocol early, shipping a real solution to a real triage problem, and keeping the craft current as the ecosystem matured.
License
MIT.
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.