@cyanheads/eur-lex-mcp-server
Search EU legislation, CJEU case law, and treaties; traverse the CELLAR relationship graph; resolve EuroVoc concepts via MCP. STDIO or Streamable HTTP.
Public Hosted Server: https://eur-lex.caseyjhand.com/mcp
Tools
Seven tools covering EU legal research — document discovery, content retrieval, citation resolution, case law, relationship graph traversal, EuroVoc thesaurus lookup, and raw SPARQL access:
| Tool | Description |
|---|---|
eurlex_search_documents | Search EU legislation, treaties, and preparatory acts across the CELLAR corpus. Filters by document type, date range, EuroVoc concept, author institution, and in-force status. |
eurlex_get_document | Fetch structured metadata and full text (HTML or Formex4 XML) for a work by CELEX number or ELI URI. |
eurlex_lookup_celex | Resolve any EU legal citation — CELEX number, ELI URI, or Official Journal reference — to the canonical CELLAR work. |
eurlex_get_cases | Search CJEU and General Court case law — judgments, orders, and Advocate General opinions — by case number, party name, subject, or date range. |
eurlex_get_relations | Traverse the CELLAR relationship graph: amendment chain, consolidated versions, legal basis, citation network, and national transposition measures. |
eurlex_browse_subjects | Search the EuroVoc multilingual thesaurus to resolve human-readable terms to EuroVoc concept IDs — required before using the eurovoc_concept filter in eurlex_search_documents. |
eurlex_query_sparql | Execute a raw SPARQL SELECT query against the CELLAR Virtuoso endpoint. Results capped at 100; use only when curated tools don't cover the needed CDM ontology traversal. |
eurlex_search_documents
Search EU legislation, treaties, preparatory acts, and more across the 2.7M+ work CELLAR corpus.
- Keyword search across work titles and CELEX string patterns
- Filter by document type (
REG,DIR,DEC,TREATY, and more) - Date range filtering (
date_from,date_to) - EuroVoc concept filtering — use
eurlex_browse_subjectsfirst to resolve concept IDs - Filter to in-force acts only
- Pagination via
offsetand configurablelimit(max 100) - Returns CELEX numbers, work URIs, document types, and dates for chaining into
eurlex_get_document
eurlex_get_document
Fetch the notice and full text of an EU legal act.
- Accepts CELEX numbers (e.g.,
32016R0679) or ELI URIs - Returns structured metadata: title, date, document type, author institution, legal basis, EuroVoc subjects, in-force flag
- Full text in HTML (default) or Formex4 XML
- Supports all 24 official EU languages; defaults to English with automatic fallback when a translation is unavailable
- Older acts and some CJEU judgments may lack English translations
eurlex_lookup_celex
Resolve EU legal identifiers to canonical CELLAR works.
- Accepts CELEX numbers, ELI URIs, and Official Journal references
- Auto-detects format with
identifier_type: "auto"(default); set explicitly when auto-detection fails - Returns work URI, confirmed CELEX number, document type, and date — the prerequisite step before
eurlex_get_documentoreurlex_get_relations
eurlex_get_cases
Search CJEU and General Court case law.
- Case-specific search: case number, keyword, court (
CJEUorGC), and case type (judgment,order,ag_opinion) - Date range filtering
- Returns case identifier, court, date, document type, and parties
- Distinct from
eurlex_search_documents— case law (CELEX sector 6) has its own search parameters and practitioner workflows
eurlex_get_relations
Traverse the CELLAR relationship graph for a given work.
- Amendment chain (what amends it, what it amends)
- Consolidated versions (the current in-force text)
- Legal basis
- Citation network (
cdm:work_cites_workin both directions) - National transposition measures
- Filter to specific relation types or retrieve all at once
- Returns one-hop relations; multi-hop traversal requires multiple calls or
eurlex_query_sparql
eurlex_browse_subjects
Resolve human-readable terms to EuroVoc concept IDs.
- Full-text search across the multilingual EuroVoc thesaurus
- Returns concept URI, preferred label, concept code, and broader/narrower hierarchy hints
- Supports all EU official languages; defaults to English
- Required before using the
eurovoc_conceptfilter ineurlex_search_documents
Resources and prompts
| Type | Name | Description |
|---|---|---|
| Resource | eurlex://document/{celexNumber} | Metadata snapshot for a CELLAR work — type, date, title, author institution, in-force flag |
| Resource | eurlex://document/{celexNumber}/relations | Relationship summary for a work: amendment chain, consolidations, legal basis, cited-by count |
| Prompt | eurlex_comparative_analysis | Frames a comparative legal analysis across EU and US law for a given policy domain |
All resource data is also reachable via tools. Resources provide stable-URI injectable context for agents that support MCP resources.
Features
Built on @cyanheads/mcp-ts-core:
- Declarative tool, resource, and prompt definitions — single file per primitive, framework handles registration and validation
- Unified error handling — handlers throw, framework catches, classifies, and formats
- Pluggable auth:
none,jwt,oauth - Swappable storage backends:
in-memory,filesystem,Supabase,Cloudflare KV/R2/D1 - Structured logging with optional OpenTelemetry tracing
- STDIO and Streamable HTTP transports
EUR-Lex-specific:
- No API key required — both CELLAR SPARQL and EUR-Lex REST content endpoints are publicly accessible
CellarSparqlServicePOSTsapplication/x-www-form-urlencodedSPARQL with CDM prefix declarations built in; server-side LIMIT enforcement (max 100) prevents Virtuoso timeout abuseEurLexContentServicefetches full HTML or Formex4 XML via the canonicallegal-content/{LANG}/TXT/URL pattern (CELLAR work URIs return 400 on direct GET — this avoids that)- Virtuoso error classification: HTTP 200 with
Virtuoso 37000 Errorbody is parsed and re-raised asServiceUnavailable(transient/timeout) orInvalidParams(syntax error) - Language fallback on document fetch: if the requested language is unavailable, retries with English; returns metadata-only with a note when English also fails
- Typed error contracts on every tool — structured
reasoncodes let agents branch on outcomes without parsing text
Agent-friendly output:
- EuroVoc prerequisite guidance in server-level instructions — agents are directed to
eurlex_browse_subjectsbefore attempting concept-filtered searches eurlex_lookup_celexsurfaces CELEX confirmation and work existence upfront, preventing downstream errors in document or relation fetches- Typed
unavailablereasons andlanguage_unavailablesignals let agents retry or explain to users with structured data, not string parsing - Relationship graph output carries relation type labels alongside CELLAR URIs and resolved CELEX numbers for human-readable downstream use
Getting started
Public Hosted Instance
A public instance is available at https://eur-lex.caseyjhand.com/mcp — no installation required. Point any MCP client at it via Streamable HTTP:
{
"mcpServers": {
"eur-lex-mcp-server": {
"type": "streamable-http",
"url": "https://eur-lex.caseyjhand.com/mcp"
}
}
}
Self-Hosted / Local
Add the following to your MCP client configuration file. No API key is required.
{
"mcpServers": {
"eur-lex-mcp-server": {
"type": "stdio",
"command": "bunx",
"args": ["@cyanheads/eur-lex-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info"
}
}
}
}
Or with npx (no Bun required):
{
"mcpServers": {
"eur-lex-mcp-server": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@cyanheads/eur-lex-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info"
}
}
}
}
Or with Docker:
{
"mcpServers": {
"eur-lex-mcp-server": {
"type": "stdio",
"command": "docker",
"args": [
"run", "-i", "--rm",
"-e", "MCP_TRANSPORT_TYPE=stdio",
"ghcr.io/cyanheads/eur-lex-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/mcp
Prerequisites
- Bun v1.3.11 or higher (or Node.js v24+).
- No API key needed — EUR-Lex and CELLAR are publicly accessible.
Installation
- Clone the repository:
git clone https://github.com/cyanheads/eur-lex-mcp-server.git
- Navigate into the directory:
cd eur-lex-mcp-server
- Install dependencies:
bun install
- Configure environment (optional):
cp .env.example .env
# All server-specific vars have sensible defaults — no required vars
Configuration
All configuration is validated at startup via Zod schemas in src/config/server-config.ts.
| Variable | Description | Default |
|---|---|---|
CELLAR_SPARQL_ENDPOINT | CELLAR SPARQL endpoint URL override (e.g., for a local Virtuoso mirror). | http://publications.europa.eu/webapi/rdf/sparql |
EURLEX_CONTENT_BASE_URL | EUR-Lex content API base URL override. | https://eur-lex.europa.eu |
SPARQL_QUERY_TIMEOUT_MS | Client-side timeout for SPARQL requests in milliseconds. | 55000 |
MAX_SPARQL_RESULTS | Enforced ceiling on LIMIT in all generated SPARQL queries. | 100 |
MCP_TRANSPORT_TYPE | Transport: stdio or http. | stdio |
MCP_HTTP_PORT | Port for HTTP server. | 3010 |
MCP_AUTH_MODE | Auth mode: none, jwt, or oauth. | none |
MCP_LOG_LEVEL | Log level (RFC 5424). | info |
LOGS_DIR | Directory for log files (Node.js only). | <project-root>/logs |
OTEL_ENABLED | Enable OpenTelemetry instrumentation. | false |
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:http -
Run 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
Docker
docker build -t eur-lex-mcp-server .
docker run --rm -p 3010:3010 eur-lex-mcp-server
The Dockerfile defaults to HTTP transport, stateless session mode, and logs to /var/log/eur-lex-mcp-server. OpenTelemetry peer dependencies are installed by default — build with --build-arg OTEL_ENABLED=false to omit them.
Project structure
| Directory | Purpose |
|---|---|
src/index.ts | createApp() entry point — registers tools, resources, and prompts; initializes services. |
src/config | Server-specific environment variable parsing and validation with Zod. |
src/services/cellar-sparql | CELLAR SPARQL service — POST client, binding mapper, LIMIT enforcement, CDM PREFIX declarations. |
src/services/eurlex-content | EUR-Lex content API service — GET client for legal-content/{LANG}/TXT/ URL pattern with language fallback. |
src/mcp-server/tools | Tool definitions (*.tool.ts). Seven tools across document search, retrieval, resolution, case law, relations, EuroVoc, and raw SPARQL. |
src/mcp-server/resources | Resource definitions (*.resource.ts). Metadata and relations resources. |
src/mcp-server/prompts | Prompt definitions (*.prompt.ts). Comparative analysis prompt. |
tests/ | Unit and integration tests mirroring src/. |
Development guide
See CLAUDE.md for development guidelines and architectural rules. The short version:
- Handlers throw, framework catches — no
try/catchin tool logic - Use
ctx.logfor request-scoped logging,ctx.statefor tenant-scoped storage - Register new tools and resources via the barrels in
src/mcp-server/*/definitions/index.ts - Wrap external API calls: validate raw → normalize to domain type → return output schema; never fabricate missing fields
Contributing
Issues and pull requests are welcome. Run checks and tests before submitting:
bun run devcheck
bun run test
License
Apache-2.0 — see LICENSE for details.