An MCP (Model Context Protocol) server that provides access to the NIH UMLS (Unified Medical Language System) API and the NIH VSAC (Value Set Authority Center) FHIR API. This server enables AI models to search medical terminology, look up concept definitions, explore relationships between medical concepts, map codes between different medical coding systems, and work with curated clinical value sets.
The Unified Medical Language System (UMLS) integrates biomedical terminology from many sources and provides a mapping structure among these vocabularies. It includes over 200 source vocabularies and classification systems, containing millions of names for medical concepts.
The Value Set Authority Center (VSAC) is an NLM repository of curated value sets — collections of medical codes from standard code systems (SNOMED CT, ICD-10, LOINC, CPT, RxNorm, etc.) that define clinical concepts for electronic Clinical Quality Measures (eCQMs), C-CDA documents, and health IT standards.
This MCP server exposes both APIs through standardized MCP tools, making it easy for AI assistants to:
- Search for medical concepts and terminology
- Get detailed concept information and definitions
- Explore relationships between concepts (parent/child hierarchies)
- Map codes between different medical coding systems (e.g., ICD-10 ↔ SNOMED CT)
- Look up specific codes from vocabularies like ICD-10, SNOMED, RxNorm, LOINC, etc.
- Search and retrieve curated clinical value sets
- Expand value sets to get all member codes
- Validate whether a code belongs to a value set
- Test hierarchical subsumption relationships between codes
- UMLS API Key: You need a UMLS API key to use this server (the same key works for both UMLS and VSAC). To obtain one:
- Visit https://uts.nlm.nih.gov/uts/signup-login
- Create an account or sign in
- Go to "My Profile" and generate an API key
- The key is free for users who agree to the UMLS license terms
pip install nih-umls-mcp# Clone the repository
git clone https://github.com/feordin/nih-umls-mcp.git
cd nih-umls-mcp
# Install the package
pip install -e .# Using uvx (no installation needed)
uvx nih-umls-mcp# If installed with pip
pip install --upgrade nih-umls-mcp
# If installed with uv
uv tool upgrade nih-umls-mcpThe server requires your UMLS API key to be set as an environment variable:
export UMLS_API_KEY="your-api-key-here"To use this server with Claude Desktop, add the following to your Claude Desktop configuration file:
MacOS: ~/Library/Application Support/Claude/claude_desktop_config.json
Windows: %APPDATA%\Claude\claude_desktop_config.json
{
"mcpServers": {
"nih-umls": {
"command": "python",
"args": ["-m", "nih_umls_mcp.server"],
"env": {
"UMLS_API_KEY": "your-api-key-here"
}
}
}
}Or if using uvx:
{
"mcpServers": {
"nih-umls": {
"command": "uvx",
"args": ["nih-umls-mcp"],
"env": {
"UMLS_API_KEY": "your-api-key-here"
}
}
}
}Search for medical concepts, terms, or codes in the UMLS.
Parameters:
query(string, required): Search term, medical concept, or codesearch_type(string, optional): Type of search - "exact", "words" (default), "leftTruncation", "rightTruncation", "normalizedString", "approximate"page_size(integer, optional): Number of results (1-25, default: 10)
Example:
Search for "diabetes mellitus" to find relevant concepts
Get detailed information about a specific UMLS concept using its CUI (Concept Unique Identifier).
Parameters:
cui(string, required): Concept Unique Identifier (e.g., "C0009044")
Example:
Get details for concept C0009044 (Closed Fracture)
Get all definitions for a concept from various medical vocabularies.
Parameters:
cui(string, required): Concept Unique Identifier
Example:
Get definitions for diabetes mellitus concept
Get relationships between concepts (parent concepts, child concepts, related terms).
Parameters:
cui(string, required): Concept Unique Identifierpage_size(integer, optional): Number of results (1-25, default: 10)
Example:
Get parent and child concepts for diabetes mellitus
Map a medical code from one coding system to equivalent codes in other systems.
Parameters:
source(string, required): Source vocabulary (e.g., "ICD10CM", "SNOMEDCT_US", "RXNORM", "LOINC")code(string, required): The code to maptarget_source(string, optional): Specific target vocabulary to filter resultspage_size(integer, optional): Number of results (1-25, default: 10)
Common vocabulary abbreviations:
ICD10CM- ICD-10 Clinical ModificationICD9CM- ICD-9 Clinical ModificationSNOMEDCT_US- SNOMED CT US EditionRXNORM- RxNorm (medications)LOINC- Logical Observation Identifiers Names and CodesCPT- Current Procedural TerminologyHCPCS- Healthcare Common Procedure Coding SystemICD10PCS- ICD-10 Procedure Coding SystemNDC- National Drug Code
Example:
Map ICD-10 code E11.9 (Type 2 diabetes without complications) to SNOMED CT
Get information about a specific code from a particular medical vocabulary.
Parameters:
source(string, required): Source vocabulary abbreviationcode(string, required): The code in that vocabulary
Example:
Get information about ICD-10 code E11.9
Search the VSAC for curated value sets by title, keyword, publisher, or contained code.
Parameters:
title(string, optional): Search by title (e.g., "Diabetes", "Hypertension")keyword(string, optional): Search by keyword/tagpublisher(string, optional): Filter by publishing organization (e.g., "Mathematica", "NCQA")code(string, optional): Find value sets containing a specific codecount(integer, optional): Number of results (default: 10)
Example:
Search for value sets related to "Diabetes" to find eCQM-relevant code groupings
Get a value set definition by its OID (Object Identifier).
Parameters:
oid(string, required): The value set OID (e.g., "2.16.840.1.113883.3.464.1003.103.12.1001")
Example:
Get the definition of the Diabetes value set used in CMS quality measures
Expand a value set to get all its member codes. Returns every code with its code system, value, and display name.
Parameters:
oid(string, required): The value set OIDfilter(string, optional): Text to filter codes within the expansioncount(integer, optional): Number of codes to return (default: 100)offset(integer, optional): Starting offset for pagination (default: 0)
Example:
Expand the Diabetes value set to get all ICD-10 and SNOMED codes it contains
Check if a specific code is a member of a value set.
Parameters:
oid(string, required): The value set OID to check againstcode(string, required): The code to validatesystem(string, optional): Code system URI (required if the value set spans multiple code systems)
Common code system URIs:
http://snomed.info/sct- SNOMED CThttp://hl7.org/fhir/sid/icd-10-cm- ICD-10-CMhttp://loinc.org- LOINChttp://www.nlm.nih.gov/research/umls/rxnorm- RxNormhttp://www.ama-assn.org/go/cpt- CPT
Example:
Check if ICD-10 code E11.9 is in the Diabetes value set for eCQM reporting
Look up details about a specific code in a code system via VSAC.
Parameters:
system(string, required): Code system URI (see common URIs above)code(string, required): The code to look upversion(string, optional): Code system version
Example:
Look up SNOMED CT code 44054006 to get its display name and properties
Test whether one code subsumes (is an ancestor of) another in a hierarchical code system.
Parameters:
system(string, required): Code system URI (e.g., "http://snomed.info/sct")code_a(string, required): The potential ancestor/broader codecode_b(string, required): The potential descendant/narrower code
Returns: "subsumes", "subsumed-by", "equivalent", or "not-subsumed"
Example:
Check if SNOMED "Diabetes mellitus" subsumes "Type 2 diabetes mellitus"
User: "What is the UMLS concept for Type 2 Diabetes?"
AI uses search_umls:
- query: "Type 2 Diabetes Mellitus"
- search_type: "words"
Returns CUI C0011860 with name "Diabetes Mellitus, Non-Insulin-Dependent"
User: "What is the SNOMED code for ICD-10 code E11.9?"
AI uses crosswalk_codes:
- source: "ICD10CM"
- code: "E11.9"
- target_source: "SNOMEDCT_US"
Returns SNOMED code 44054006 and other equivalent codes
User: "What are the subtypes of diabetes?"
AI first uses search_umls to find diabetes CUI, then uses get_concept_relations
to explore child concepts and related terms.
User: "What codes count as Diabetes for eCQM reporting?"
AI uses search_value_sets with title "Diabetes" to find relevant value sets,
then uses expand_value_set to get all member codes (ICD-10, SNOMED CT, etc.)
from the appropriate value set.
User: "Does ICD-10 code E11.9 qualify for the Diabetes quality measure?"
AI uses validate_code_in_value_set:
- oid: "2.16.840.1.113883.3.464.1003.103.12.1001"
- code: "E11.9"
- system: "http://hl7.org/fhir/sid/icd-10-cm"
Returns true/false indicating membership.
User: "Is Type 2 diabetes a subtype of diabetes mellitus in SNOMED?"
AI uses check_code_subsumption:
- system: "http://snomed.info/sct"
- code_a: "73211009" (Diabetes mellitus)
- code_b: "44054006" (Type 2 diabetes)
Returns "subsumes" confirming the hierarchical relationship.
pip install -e ".[dev]"
pytestnih-umls-mcp/
├── src/
│ └── nih_umls_mcp/
│ ├── __init__.py
│ ├── server.py # MCP server implementation
│ ├── umls_client.py # UMLS REST API client
│ └── vsac_client.py # VSAC FHIR API client
├── examples/
│ ├── test_server.py # Tool registration verification
│ └── client_example.py # Live API usage example
├── pyproject.toml
└── README.md
Both the UMLS and VSAC APIs share the same rate limit:
- Maximum 20 requests per second per IP address
- The server doesn't currently implement client-side rate limiting, so be mindful of usage patterns
- Ensure you've set the
UMLS_API_KEYenvironment variable - Check that the key is valid by testing it at https://documentation.uts.nlm.nih.gov/umls-api-interactive.html
- Your API key may be invalid or expired
- Regenerate your API key from your UTS profile
- The CUI, code, or value set OID you're looking up doesn't exist
- Check the spelling and format of identifiers
- UMLS REST API Documentation
- UMLS Metathesaurus Browser
- VSAC FHIR API Documentation
- VSAC Home
- Model Context Protocol Documentation
- UMLS License Information
This is an independent, community-built project and is not an official product of the National Institutes of Health (NIH) or the National Library of Medicine (NLM). It simply provides a convenient interface to their publicly available APIs.
MIT License - see LICENSE file for details.
Contributions are welcome! Please feel free to submit a Pull Request.