Skip to main content
Glama

map-mcp

Drive and perceive an existing, live MapLibre GL map from an AI agent (over MCP) or a human CLI — query the rendered features, read the viewport, click and read popups, navigate, toggle layers. The agent and the CLI act on the same map a person is looking at, with parity by construction.

It does not generate maps. Other geo-MCP servers (gis-mcp, Mapbox, CARTO) create maps or call GIS operations; map-mcp reaches into a map that's already on screen. Think of it as the agent-native counterpart to MapGrab: same live-map access, but conversational over MCP instead of written as test code.

Status: v1, MapLibre GL only. Cooperation-required (your app adds a one-line hook). A no-cooperation path and other map libraries are future work.

Install

uvx map-mcp --help          # or: pip install map-mcp

Related MCP server: QGIS MCP

Quickstart

  1. Start the bridge + MCP server. It prints a WebSocket URL and a per-session token.

    map-mcp serve
  2. Add the hook to your MapLibre page (map is your existing maplibregl.Map):

    <script src="map-mcp-hook.js"></script>
    <script>
      mapMcp.register(map, { url: "ws://127.0.0.1:8765", token: "PASTE_TOKEN" });
    </script>
  3. Point your agent at the MCP server (stdio by default; --transport http for HTTP/SSE). Or drive it yourself from the terminal:

    map-mcp call get_viewport
    map-mcp call query_rendered_features --params '{"point":[12.5,41.9]}'

There's a runnable example in examples/sample_app/.

Tools (the operation surface)

The agent's MCP tools and the CLI's call operations are exactly the same set:

Operation

What it does

get_viewport

center [lng,lat], zoom, bearing, pitch, bounds

query_rendered_features

features currently rendered (optionally at a point or within a bbox)

get_features_at

features rendered at a [lng,lat] point

click_at

fire the map's click at a point (runs your popup handlers), return features + popup

read_popup

text of any open popup(s)

set_view

center+zoom (and bearing/pitch), or fit a bbox

list_layers

the style's layers + visibility

set_layer_visibility

show/hide a layer

screenshot

a PNG data URL of the current map*

Perception returns structured feature properties (GeoJSON-shaped) — agents reason over properties, not pixels. screenshot is optional.

* needs the map created with preserveDrawingBuffer: true (see hook/snippet.md).

How it works

The hook connects out to a loopback WebSocket the map-mcp process runs. The MCP tools and the CLI are thin frontends over one shared core-operations layer, so any operation one can do, the other can too.

Security model (local-only). The bridge binds 127.0.0.1 only, so nothing off your machine can reach it. Browsers do not apply same-origin policy to WebSocket connections, so the per-session token is the security boundary: only a page that presents it can drive your map. Treat the token like a secret — the convenience ?token= pattern in the example leaks it via browser history and server logs, so for anything sensitive paste the token into the page rather than the URL. A hardened Origin allowlist is future work.

Scope (v1)

  • In: MapLibre GL; the operations above; stdio + HTTP/SSE; a human CLI with parity.

  • Out: generating maps; a hosted service; non-map visualizations; other map libraries (Leaflet/deck.gl) and a no-cooperation (Playwright) path are future work.

License

MIT

A
license - permissive license
-
quality - not tested
B
maintenance

Maintenance

Maintainers
Response time
Release cycle
Releases (12mo)
Commit activity

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/dkedar7/map-mcp'

If you have feedback or need assistance with the MCP directory API, please join our Discord server