medical-codes-mcp-server
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@medical-codes-mcp-serverdecode ICD-10 code I10"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
Informational, not clinical or coding advice. This server returns official code descriptions and billable/validity flags from public-domain federal releases to help you decode and look up codes. It is not medical advice, and a valid_billable result is not a coding or reimbursement decision. Always verify codes against the official source releases (CMS, CDC/NCHS, NLM) and your payer's rules before submitting a claim. The bundled data is only as current as the release baked into the build — call medcode_list_systems to see exactly which releases are active.
How it works
The code data is bundled inside the package — a single SQLite + FTS5 database (data/medical-codes.db) built at package-build time from the canonical federal source files and shipped in the npm tarball and Docker image. The server opens it read-only at startup and answers every tool call from disk.
That means the server is offline, keyless, and deterministic: no runtime network calls, no API key, no rate limit, single-tenant. The same inputs against the same bundled build always return the same output.
Bundled code systems
Only freely-redistributable, public-domain US federal code sets are bundled:
System | Source | Covers |
ICD-10-CM | CDC/NCHS — US federal, public domain | Diagnoses (billable leaf codes + non-billable category headers) |
ICD-10-PCS | CMS — US federal, public domain | Inpatient procedures (axis-based 7-character codes) |
HCPCS Level II | CMS — US federal, public domain | Supplies, drugs, and non-physician services |
RxNorm (NLM, public domain) — the drug-crosswalk layer (drug name ↔ RXCUI ↔ NDC, ingredients, brands) — is not bundled yet; it lands in a later release. Until then, medcode_map_codes' drug directions return a direction_unavailable error, and the hierarchy directions (parents/children) work today.
CPT (AMA copyright) and SNOMED CT / LOINC (UMLS-license-gated) are intentionally absent — they are not freely redistributable, so they cannot ship in an offline package.
US scope. ICD-10-CM and ICD-10-PCS are the US clinical modifications, not the WHO ICD-10/ICD-11 base or another country's national modification.
Related MCP server: autoicd-mcp
Tools
Six tools organized goal-first — one per user action, with a system discriminator instead of a per-system tool for each of the bundled code sets. All are read-only.
Tool | Description |
| Decode 1–50 codes to their official descriptions. Auto-detects the system per code; partial-success |
| Full-text search over official descriptions — go from a clinical description to the code. |
| Validate a code's existence, currency, and billability, with a |
| Crosswalk a code within its hierarchy ( |
| Walk a system's hierarchy for discovery without a search term. |
| List bundled systems with release identifiers, effective dates, and code counts (provenance). |
medcode_get_code
Decode one or more codes seen in a claim, EHR field, or another health server's output. The 80% entry point.
Accepts 1–50 codes; mixed systems are fine — each code's system is detected independently from its shape
Partial success: resolved codes in
found, unresolved innotFoundwith a per-code reason, so one bad code never fails the batchAn explicit
systemoverrides auto-detection when a value is genuinely ambiguous (an ambiguous code lists itscandidateSystems)includeHierarchyattaches each code's parent and immediate childrenThe resolved
systemis echoed on every result for chaining intomedcode_map_codesor a billability check
medcode_search_codes
Find codes whose official descriptions match a described concept — the reverse of medcode_get_code.
Every search term must appear (prefix-matched), so
"diabetic neuropathy"returns codes mentioning bothFilter by
system,billableOnly(exclude headers/categories), andchapterRanked by full-text relevance; results echo the resolved system per row
Discloses truncation when the result hits the cap, and returns an explicit notice (with the parsed query) when nothing matched
medcode_check_code
Validate whether a code is safe to submit, before a claim goes out.
Discriminated status:
valid_billable,valid_not_billable,valid_header, orterminatedA
whyNotstring explains the non-billable and terminated cases (e.g. "valid ICD-10-CM category but not billable — submit a more specific child code")Validity vs. existence is split: a non-billable or terminated code is a successful result with a
whyNot, not an error — only a code absent from every bundled system raisesunknown_code
medcode_map_codes
Crosswalk a code across systems and within a hierarchy.
Hierarchy directions (available now):
parentsandchildrenwalk a code's prefix hierarchy (ICD-10-CM / HCPCS; ICD-10-PCS codes have no prefix parent)Drug directions (
name_to_rxcui,ndc_to_rxcui,rxcui_to_ndc,rxcui_to_ingredients,rxcui_to_brands) are RxNorm-backed and returndirection_unavailableuntil RxNorm is bundled in a later releaseEvery result carries
sourceprovenance (which system or edge answered) so a chained call uses the right identifier
medcode_browse_hierarchy
Orient in an unfamiliar system or enumerate a category's specific codes, without a search term.
With no
node: top-level entries (ICD-10-CM categories, HCPCS range buckets, or ICD-10-PCS first-axis values)With a
node: its immediate childrenICD-10-CM and HCPCS use a prefix hierarchy (a shorter code is the parent of a longer one); ICD-10-PCS is axis-based — each of the 7 characters is an independent axis, so browsing returns valid next-position axis values, not prefix children
Features
Built on @cyanheads/mcp-ts-core:
Declarative tool definitions — single file per tool, framework handles registration and validation
Unified error handling — handlers throw, framework catches, classifies, and formats
Typed per-tool error contracts — capable clients preview failure modes from
tools/listStructured logging with optional OpenTelemetry tracing
STDIO and Streamable HTTP transports
Domain-specific:
Bundled SQLite + FTS5 index — offline, keyless, deterministic; no runtime network I/O, no rate limit
Code-shape auto-detection routes a code to its system; an explicit
systemdisambiguates collisionsReal billable/validity signal from the source releases — the order-file billable flag drives
medcode_check_code, not a heuristic
Agent-friendly output:
Provenance on every response — the resolved
systemis echoed for chaining, andmedcode_list_systemsreports exactly which release is baked into the running buildGraceful partial failure —
medcode_get_codereturns per-codefound/notFoundrows instead of failing the batchDiscriminated output contracts —
medcode_check_code's typed status andmedcode_map_codes'sourcelet callers branch on data, not string parsing
Getting started
This server ships with the code database bundled — there is no API key to obtain and nothing to download at runtime. Add the following to your MCP client configuration file.
{
"mcpServers": {
"medical-codes-mcp-server": {
"type": "stdio",
"command": "bunx",
"args": ["@cyanheads/medical-codes-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info"
}
}
}
}Or with npx (no Bun required):
{
"mcpServers": {
"medical-codes-mcp-server": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@cyanheads/medical-codes-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info"
}
}
}
}Or with Docker:
{
"mcpServers": {
"medical-codes-mcp-server": {
"type": "stdio",
"command": "docker",
"args": [
"run", "-i", "--rm",
"-e", "MCP_TRANSPORT_TYPE=stdio",
"ghcr.io/cyanheads/medical-codes-mcp-server:latest"
]
}
}
}For Streamable HTTP, set the transport and start the server:
MCP_TRANSPORT_TYPE=http MCP_HTTP_PORT=3010 bun run start:http
# Server listens at http://localhost:3010/mcpRefer to "your MCP client configuration file" generically — different clients use different config paths, and the server isn't client-specific.
Prerequisites
Bun v1.3 or higher (or Node.js v24+ — the server falls back to the
better-sqlite3optional dependency when not run under Bun).No API key, account, or network access required.
Installation
Clone the repository:
git clone https://github.com/cyanheads/medical-codes-mcp-server.gitNavigate into the directory:
cd medical-codes-mcp-serverInstall dependencies:
bun installConfigure environment (optional):
cp .env.example .env
# all runtime vars are optional — the server runs as-isConfiguration
The server is offline and keyless — there are no required variables. Two server-specific knobs and the standard framework vars apply:
Variable | Description | Default |
| Absolute path override for the bundled SQLite index. Set only to point at a custom-built or externally-mounted database. | packaged |
| Cap on rows returned by |
|
| Transport: |
|
| Port for the HTTP server. |
|
| Endpoint path where the MCP server is mounted. |
|
| Auth mode: |
|
| Log level (RFC 5424). |
|
| Enable OpenTelemetry instrumentation. |
|
See .env.example for the full list of optional overrides.
Running the server
Local development
Build and run:
# One-time build bun run rebuild # Run the built server bun run start:stdio # or bun run start:httpRun checks and tests:
bun run devcheck # Lint, format, typecheck, security bun run test # Vitest test suite bun run lint:mcp # Validate MCP definitions against spec
Building the bundled index
The bundled data/medical-codes.db is committed and shipped — you only rebuild it when refreshing to a new federal release:
bun run scripts/build-index.tsThe script downloads the canonical .gov source files, parses them (ICD-10-CM/PCS order files, HCPCS ANWEB.txt), and emits the single .db file. It runs at build time only — the server never downloads anything.
Docker
docker build -t medical-codes-mcp-server .
docker run --rm -e MCP_TRANSPORT_TYPE=stdio medical-codes-mcp-serverThe Dockerfile defaults to HTTP transport, stateless session mode, and logs to /var/log/medical-codes-mcp-server. It copies the bundled data/medical-codes.db into the image so the server is fully self-contained. OpenTelemetry peer dependencies are installed by default — build with --build-arg OTEL_ENABLED=false to omit them.
Project structure
Directory | Purpose |
|
|
| Server-specific environment variable parsing and validation with Zod. |
| Tool definitions ( |
| The code-index service — read-only SQLite handle, code-shape detection, FTS5 query translation. |
| Build-time ingest pipeline that bakes the federal source files into |
| The bundled SQLite + FTS5 code index, opened read-only at runtime. |
| Unit and integration tests mirroring |
Development guide
See CLAUDE.md/AGENTS.md for development guidelines and architectural rules. The short version:
Handlers throw, framework catches — no
try/catchin tool logicUse
ctx.logfor request-scoped logging; the code index is a read-only global, not tenant stateRegister new tools via the
createApp()array insrc/index.tsThe bundled DB is the source of truth — surface real billable/validity flags from the source releases; never fabricate a code or a billability decision
Contributing
Issues and pull requests are welcome. Run checks and tests before submitting:
bun run devcheck
bun run testLicense
Apache-2.0 — see LICENSE for details.
This server cannot be installed
Maintenance
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/cyanheads/medical-codes-mcp-server'
If you have feedback or need assistance with the MCP directory API, please join our Discord server