swisstopo-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., "@swisstopo-mcpWhat's the elevation at Uetliberg summit?"
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
swisstopo-mcp
MCP server for Swiss federal geodata -- maps, elevation, geocoding, cadastral extracts, and downloadable datasets via Swisstopo APIs
Overview
swisstopo-mcp gives AI assistants access to Switzerland's official geodata infrastructure through 13 tools across 6 API families, all without authentication:
Source | Data | API |
Swisstopo REST API | 500+ geodata layers (buildings, boundaries, land use) | REST/JSON |
Geocoding | Official addresses, place names, postal codes | REST/JSON |
Height Service | Elevation above sea level, elevation profiles | REST/JSON |
STAC Catalog | Orthophotos, elevation models, 3D buildings | STAC 0.9 |
WMTS | National maps, aerial images, zoning maps | URL builder |
OEREB Cadastre | Public-law restrictions, parcels | REST/JSON (cantonal) |
Anchor demo query: "What land-use restrictions apply to the parcel at Musterstrasse 5, Zurich? Show me the location on a map." β More use cases by audience β
Features
πΊοΈ 13 tools across 6 API families (REST, Geocoding, Height, STAC, WMTS, OEREB)
π Geocode Swiss addresses and reverse-geocode coordinates
ποΈ Query elevation and compute elevation profiles
π¦ Discover and download geodatasets (orthophotos, 3D buildings, historical maps)
ποΈ Identify map features at coordinates across 500+ Swisstopo layers
π Generate shareable map.geo.admin.ch links
π Look up cadastral property IDs (EGRID) and retrieve OEREB extracts
π No API key required for 11 of 13 tools
βοΈ Dual transport -- stdio (Claude Desktop) + Streamable HTTP (cloud)
Prerequisites
Python 3.11+
uv (recommended) or pip
Installation
# Clone the repository
git clone https://github.com/malkreide/swisstopo-mcp.git
cd swisstopo-mcp
# Install
pip install -e .
# or with uv:
uv pip install -e .Or with uvx (no permanent installation):
uvx swisstopo-mcpQuickstart
# stdio (for Claude Desktop)
python -m swisstopo_mcp.server
# Streamable HTTP (port 8000)
python -m swisstopo_mcp.server --http --port 8000Try it immediately in Claude Desktop:
"Where is Bahnhofstrasse 1, Zurich? Give me the coordinates." "What is the elevation at the Uetliberg summit?" "What buildings are at coordinates 2683500, 1247500 (LV95)?"
Configuration
Claude Desktop
Edit ~/Library/Application Support/Claude/claude_desktop_config.json (macOS) or %APPDATA%\Claude\claude_desktop_config.json (Windows):
{
"mcpServers": {
"swisstopo": {
"command": "python",
"args": ["-m", "swisstopo_mcp.server"]
}
}
}Or with uvx:
{
"mcpServers": {
"swisstopo": {
"command": "uvx",
"args": ["swisstopo-mcp"]
}
}
}Config file locations:
macOS:
~/Library/Application Support/Claude/claude_desktop_config.jsonWindows:
%APPDATA%\Claude\claude_desktop_config.json
Cloud Deployment (SSE for browser access)
For use via claude.ai in the browser (e.g. on managed workstations without local software):
Render.com (recommended):
Push/fork the repository to GitHub
On render.com: New Web Service -> connect GitHub repo
Set start command:
python -m swisstopo_mcp.server --http --port 8000In claude.ai under Settings -> MCP Servers, add:
https://your-app.onrender.com/sse
Available Tools
REST API (Layer & Feature Queries)
Tool | Description |
| Search the Swisstopo layer catalog (500+ layers) by keyword |
| Find map features at a specific coordinate (spatial query) |
| Search features by attribute value within a layer (e.g. buildings by EGID) |
| Retrieve full attributes and geometry for a feature by ID |
Geocoding
Tool | Description |
| Convert Swiss addresses, place names, or postal codes to coordinates |
| Find the nearest address for given coordinates |
Height Service
Tool | Description |
| Get elevation above sea level (m a.s.l.) at a coordinate |
| Compute an elevation profile along a line |
STAC Catalog (Geodata Downloads)
Tool | Description |
| Search the STAC catalog for downloadable geodatasets |
| Get details and download links for a STAC collection |
WMTS (Map URLs)
Tool | Description |
| Generate a map.geo.admin.ch URL for browser display |
OEREB Cadastre
Tool | Description |
| Resolve a cadastral property ID (EGRID) from coordinates |
| Retrieve public-law land-use restrictions (OEREB) for a parcel |
Example Use Cases
Query | Tool |
"Where is Bahnhofstrasse 1, Zurich?" |
|
"What is the elevation at the Uetliberg summit?" |
|
"What buildings are at coordinates 2683500, 1247500?" |
|
"Find orthophoto datasets for download" |
|
"Show me a map of Bern at zoom level 10" |
|
"What restrictions apply to parcel at Musterstrasse 5?" |
|
Architecture
βββββββββββββββββββ ββββββββββββββββββββββββββββββββ ββββββββββββββββββββββββββββ
β Claude / AI ββββββΆβ swisstopo-mcp ββββββΆβ Swisstopo REST API β
β (MCP Host) βββββββ (MCP Server) βββββββ api3.geo.admin.ch β
βββββββββββββββββββ β β ββββββββββββββββββββββββββββ€
β 13 Tools ββββββΆβ Geocoding β
β Stdio | Streamable HTTP βββββββ api3.geo.admin.ch β
β β ββββββββββββββββββββββββββββ€
β No authentication required ββββββΆβ STAC Catalog β
β (11 of 13 tools) βββββββ data.geo.admin.ch β
β β ββββββββββββββββββββββββββββ€
β ββββββΆβ OEREB Cadastre β
β βββββββ (cantonal endpoints) β
ββββββββββββββββββββββββββββββββ ββββββββββββββββββββββββββββProject Structure
swisstopo-mcp/
βββ src/swisstopo_mcp/
β βββ __init__.py # Package version
β βββ server.py # MCP server wiring (tool registrations)
β βββ api_client.py # Shared HTTP client (httpx + error handling)
β βββ geocoding.py # swisstopo_geocode, swisstopo_reverse_geocode
β βββ rest_api.py # swisstopo_search_layers, identify, find, get_feature
β βββ height.py # swisstopo_get_height, swisstopo_elevation_profile
β βββ stac.py # swisstopo_search_geodata, swisstopo_get_collection
β βββ wmts.py # swisstopo_map_url
β βββ oereb.py # swisstopo_get_egrid, swisstopo_get_oereb_extract
βββ tests/
β βββ test_api_client.py
β βββ test_geocoding.py
β βββ test_height.py
β βββ test_oereb.py
β βββ test_rest_api.py
β βββ test_stac.py
β βββ test_wmts.py
βββ .github/workflows/ci.yml # GitHub Actions (Python 3.11/3.12/3.13)
βββ pyproject.toml
βββ CHANGELOG.md
βββ CONTRIBUTING.md
βββ LICENSE
βββ README.md # This file (English)
βββ README.de.md # German versionSecurity & Compliance
Phase
This server is in Phase 1 β Read-only wrapper. All 13 tools are
readOnlyHint: true / destructiveHint: false; there are no write or send
capabilities. See docs/roadmap.md for later phases.
Lethal Trifecta assessment
Capability | Status | Rationale |
Access to private data | β No | Public Open Data only (federal/cantonal geodata) |
Exposure to untrusted content | β οΈ Limited | Reads only from a fixed allow-list of trusted geo.admin / OEREB hosts |
External communication (write/send) | β No | Read-only; no mail/webhook/write tools |
Trifecta score: at most 1 of 3 β safe by design.
Egress
Outbound requests are restricted to an explicit code-layer allow-list and redirects are disabled β see docs/network-egress.md.
Container deployment
For containerised HTTP deployments, a hardened Dockerfile and Kubernetes
manifests (non-root, read-only root filesystem, dropped capabilities, egress
NetworkPolicy) are provided β see docs/deployment.md.
MCP Protocol Version
The MCP protocol version is negotiated by the mcp SDK, which is pinned to the
1.x major in pyproject.toml so an update cannot silently change the
negotiated version. SDK bumps are proposed monthly via Dependabot and tracked
in CHANGELOG.md.
Sessions & Authentication
The server is unauthenticated by design β it serves only public open data. Over HTTP, session IDs are managed entirely by the FastMCP framework; there is no per-user state, so there is nothing user-specific to bind a session to. If an authenticated deployment is ever introduced, session IDs must be bound to the validated user identity (audit finding SEC-009).
Error handling
Execution errors (upstream failure, invalid value) are returned as a
ToolResponsewithis_error: trueand a user-friendlysummary; raw exception text is never leaked to the client (it is logged to stderr instead).Protocol errors (unknown tool, malformed/invalid arguments) are emitted by the MCP SDK as JSON-RPC errors with standard codes (e.g.
-32602invalid params). Input validation happens at the Pydantic boundary (SEC-018).
MCP Primitives
This server intentionally exposes Tools only (no Resources or Prompts): it is a Phase-1 read-only wrapper, and every result is a live, parameterised API query rather than a static addressable document. Resources/Prompts may be added in a later phase if stable URI schemes emerge.
Tool workflows
Most tools return a thought-complete result in a single call. Two domains use a short, documented discovery chain (each tool's description states the next step):
Feature query:
swisstopo_search_layers(find layer IDs) βswisstopo_identify_features/swisstopo_find_featuresβswisstopo_get_feature(full detail).Cadastre:
swisstopo_geocodeβswisstopo_get_egridβswisstopo_get_oereb_extract.Downloads:
swisstopo_search_geodataβswisstopo_get_collection.
Response Format
Every tool returns a structured ToolResponse (FastMCP emits it as structured
content with an output schema, plus a JSON text block):
Field | Meaning |
| Human-readable Markdown summary |
| Machine-readable structured records |
| Number of |
|
|
| Data attribution (OGD-CH, CC/OGD terms) |
| How and when the data was obtained |
|
|
Known Limitations
OEREB tools require a canton parameter; not all cantons expose the same API format
STAC catalog uses Swisstopo's v0.9 endpoint; some collections may lack complete metadata
Geocoding covers Swiss addresses only (no Liechtenstein)
Rate limits are enforced by Swisstopo; high-frequency usage may be throttled
Testing
# Unit tests (no network required)
pytest tests/ -m "not live"
# Integration tests (live API calls)
pytest tests/ -m "live"Changelog
See CHANGELOG.md
Contributing
See CONTRIBUTING.md
License
MIT License -- see LICENSE
Data provided by swisstopo under Open Government Data terms.
Author
Hayal Oezkan Β· malkreide
Credits & Related Projects
Swisstopo: www.swisstopo.admin.ch -- Swiss Federal Office of Topography
Swisstopo APIs: api3.geo.admin.ch / data.geo.admin.ch
Protocol: Model Context Protocol -- Anthropic / Linux Foundation
Related: zurich-opendata-mcp -- Zurich city open data
Related: swiss-transport-mcp -- Swiss public transport
Related: swiss-cultural-heritage-mcp -- Swiss cultural heritage
Portfolio: Swiss Public Data MCP Portfolio
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/swisstopo-mcp'
If you have feedback or need assistance with the MCP directory API, please join our Discord server