swiss-courts-mcp
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., "@swiss-courts-mcpSearch Swiss Federal Supreme Court decisions on data protection from 2023"
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.
Part of the Swiss Public Data MCP Portfolio
๐๏ธ swiss-courts-mcp
MCP Server for Swiss court decisions โ Federal Supreme Court (BGer), Federal Administrative Court (BVGer), Federal Criminal Court (BStGer), and all 26 cantonal courts via entscheidsuche.ch
Overview
Access Swiss court decisions from all judicial levels through a single MCP interface. Combines full-text search with structured filters for canton, court level, date range, and law references.
Source | Coverage | Data |
Federal + 26 cantons | Court decisions since ~2000 |
Synergy with fedlex-mcp: Legislation (SR) + case law = complete legal research.
Related MCP server: Entscheidsuche MCP Server
Features
Full-text search across all Swiss court decisions
Multi-stage law reference search with regex parser and Elasticsearch boost scoring
Dedicated Federal Supreme Court search with chamber filter
Canton and court level filtering
Recent decisions feed
Court taxonomy listing
Decision statistics with aggregations
Trilingual support (German, French, Italian)
No API key required
Prerequisites
Python 3.11 or higher
An MCP-compatible client (Claude Desktop, Cursor, Windsurf, etc.)
Installation
pip install swiss-courts-mcpOr install from source:
git clone https://github.com/malkreide/swiss-courts-mcp.git
cd swiss-courts-mcp
pip install -e ".[dev]"Quickstart
# Run directly
swiss-courts-mcp
# Or via Python module
python -m swiss_courts_mcpConfiguration
Claude Desktop
Add to your claude_desktop_config.json:
{
"mcpServers": {
"swiss-courts": {
"command": "python",
"args": ["-m", "swiss_courts_mcp"]
}
}
}Cloud Deployment (HTTP transport)
The HTTP transport is off by default. The default bind host is 127.0.0.1
(loopback only) โ 0.0.0.0 must be opted into explicitly (the Dockerfile does
this). Running HTTP without authentication logs a warning; only do so behind an
authenticating reverse proxy.
# Local HTTP (loopback), no auth โ development only
swiss-courts-mcp --http --port 8000
# Container (binds 0.0.0.0, auth enabled) โ see Dockerfile
docker build -t swiss-courts-mcp .
docker run -p 8000:8000 -e MCP_AUTH_SECRET="$(openssl rand -hex 32)" swiss-courts-mcpRelevant environment variables (see .env.example):
Variable | Default | Purpose |
|
| Bind host. Set to |
|
| Bind port. |
|
| Suppress the |
|
| Stateless HTTP โ horizontal scaling without sticky sessions. |
|
| Enable bearer-token auth for HTTP. |
| โ | HS256 signing key (dev). |
| โ | JWKS URL for RS256 validation (production). |
| โ | Comma-separated required scopes. |
| โ | Comma-separated allowed origins (no wildcard in prod). |
Authentication validates the user identity from the JWT sub claim only; see
ADR 0001.
MCP Protocol Version
This server pins MCP protocol version 2025-11-25 (constant
PROTOCOL_VERSION in server.py). A regression test detects drift against the
installed SDK so a protocol bump is a conscious change (version + CHANGELOG +
this section). SDK updates land monthly via Dependabot.
Project Phase
Phase 1 โ read-only (see ROADMAP.md). All tools are
readOnlyHint: true; there are no writing or destructive operations. A move to
Phase 2 (write) requires a clean re-audit and the gates listed in the roadmap.
Available Tools
Court Decision Search
Tool | Description |
| Full-text search across all court decisions with canton, court level, and date filters |
| Retrieve a single decision by its unique signature |
| Search Federal Supreme Court decisions with optional chamber filter |
| Find decisions citing a specific law article (e.g., "Art. 8 BV") |
Court Information
Tool | Description |
| List all indexed courts, optionally filtered by canton |
| Latest decisions, filterable by canton and court level |
| Statistics on indexed decisions by canton and year |
Tool Annotations
All seven tools share the same hints โ they are read-only, idempotent, non-destructive, and reach an external system:
Annotation | Value |
|
|
|
|
|
|
|
|
A rechtsrecherche prompt is also provided (a second MCP primitive
alongside tools).
Example Use Cases
Use Case | Tool Chain |
Research case law on data protection |
|
Find practice on a constitutional right |
|
Latest Federal Supreme Court rulings |
|
Combined: Law text + case law |
|
โ More use cases by audience โ
Architecture
โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโ
โ MCP Client (LLM) โ
โ Claude / Cursor / Windsurf โ
โโโโโโโโโโโโโโโโฌโโโโโโโโโโโโโโโโโโโโโโโ
โ MCP Protocol
โโโโโโโโโโโโโโโโผโโโโโโโโโโโโโโโโโโโโโโโ
โ swiss-courts-mcp โ
โ 7 tools ยท Pydantic validation โ
โ Elasticsearch query builder โ
โโโโโโโโโโโโโโโโฌโโโโโโโโโโโโโโโโโโโโโโโ
โ HTTPS (POST/GET)
โโโโโโโโโโโโโโโโผโโโโโโโโโโโโโโโโโโโโโโโ
โ entscheidsuche.ch โ
โ Elasticsearch backend โ
โ No authentication required โ
โ Federal + 26 cantonal courts โ
โโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโโSafety & Limits
Aspect | Details |
Access | Read-only ( |
Personal data | No personal data โ all decisions are public court rulings |
Rate limits | Built-in per-query caps (max 50 results per search, 50 aggregation buckets) |
Timeout | 30 seconds per API call |
Data source auth | No API keys required โ entscheidsuche.ch is publicly accessible |
HTTP transport auth | Optional bearer-token auth (JWT, |
Egress | Code-layer allow-list ( |
Error masking | Internal exceptions are logged server-side only; clients receive friendly messages |
Secrets | No secrets in code/logs; |
Licenses | Court decisions are public domain under Swiss law (BGG Art. 27) |
Terms of Service | Subject to entscheidsuche.ch usage terms โ please be kind to the server |
Project Structure
swiss-courts-mcp/
โโโ src/
โ โโโ swiss_courts_mcp/
โ โโโ __init__.py
โ โโโ __main__.py
โ โโโ server.py # MCP server, 7 tools + 1 prompt, lifespan, auth wiring
โ โโโ api_client.py # HTTP client, ES query builder, egress allow-list
โ โโโ auth.py # JWT bearer-token verifier (HTTP transport)
โ โโโ config.py # Settings object (env-driven)
โ โโโ logging_config.py # structured logging on stderr
โ โโโ models.py # structured response envelope
โโโ tests/ # unit (respx-mocked) + live + security tests
โโโ docs/ # egress, secret-management, ADRs
โโโ .github/workflows/ # ci ยท security (gitleaks) ยท live ยท publish
โโโ Dockerfile # hardened container (non-root, 0.0.0.0 only here)
โโโ ROADMAP.md
โโโ pyproject.toml ยท CHANGELOG.md ยท LICENSE
โโโ CONTRIBUTING.md ยท CONTRIBUTING.de.md
โโโ SECURITY.md ยท SECURITY.de.md
โโโ README.md ยท README.de.mdNote (single-file tools): the 7 tools live in
server.pyrather than atools/package. At this count a single module stays readable; the registry (register_tools) keeps registration declarative. This is a deliberate deviation from the "split when > 5 tools" convention and will be revisited if the tool count grows.
Known Limitations
Search is limited to decisions indexed by entscheidsuche.ch (not all decisions are publicly available)
Full-text document content is not returned โ only metadata, title, and abstract
Statistics depend on Elasticsearch aggregation support of the backend
The court taxonomy structure from
Facetten_alle.jsonmay vary
Testing
Unit tests mock all HTTP with respx; live tests hit the real API and run in a
separate nightly workflow (live.yml), never
blocking PRs.
# Unit tests (HTTP mocked) โ what CI runs
pytest tests/ -v -m "not live"
# Live API tests (real entscheidsuche.ch)
pytest tests/ -v -m live
# Linting
ruff check src/ tests/
ruff format src/ tests/Changelog
See CHANGELOG.md.
Contributing
See CONTRIBUTING.md.
Security
See SECURITY.md for the security posture and how to report a vulnerability.
License
Author
Hayal Oezkan ยท malkreide
Credits & Related Projects
entscheidsuche.ch โ Swiss court decision search engine
fedlex-mcp โ MCP Server for Swiss federal law (legislation synergy)
zurich-opendata-mcp โ MCP Server for Zurich open data
Model Context Protocol โ Open protocol for AI tool integration
Installation
Run via uv's uvx โ no clone or manual install needed. Add to your MCP client config (mcpServers for Claude Desktop, Cursor and Windsurf; use a top-level servers key for VS Code in .vscode/mcp.json):
{
"mcpServers": {
"swiss-courts-mcp": {
"command": "uvx",
"args": [
"swiss-courts-mcp"
]
}
}
}Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
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/malkreide/swiss-courts-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server