MCP-Geo
by webcoderz
from geopy.geocoders import Nominatim, ArcGIS, Bing
from geopy.exc import GeocoderTimedOut, GeocoderServiceError
import os
from geopy.extra.rate_limiter import RateLimiter
from geopy.distance import distance
from mcp.server.fastmcp import FastMCP
__version__ = "0.1.0"
'''how to use this code
uv pip install -r requirements.txt
add to mcp server config
{
"mcp-geo": {
"command": "uv",
"args": [
"--directory",
"MCP-Geo",
"run",
"geo.py"
],
"env": {
"NOMINATIM_URL": "${NOMINATIM_URL}",
"SCHEME": "http",
"GEOCODER_PROVIDER": "nominatim"
}
}
}
'''
# Instantiate FastMCP server
mcp = FastMCP("MCP Geo", dependencies=["geopy"])
# If you need HTTPS, ensure scheme='https' and domain=some-url.
# Decide which geocoder to use based on env var
geocoder_name = os.environ.get("GEOCODER_PROVIDER", "nominatim").lower()
if geocoder_name == "nominatim":
# For Nominatim, read the domain from NOMINATIM_URL or default to openstreetmap
domain = os.environ.get("NOMINATIM_URL", "nominatim.openstreetmap.org")
scheme = os.environ.get("SCHEME", "http")
# If you need https, set scheme='https'
app = Nominatim(domain=domain, scheme=scheme)
elif geocoder_name == "arcgis":
# ArcGIS typically just works; optionally pass username/password or referer
# if needed for premium data.
# Read additional env vars if you have them, e.g., ARC_USERNAME, ARC_PASSWORD
# i dont use this but if desired, you can add the env vars to the mcp server config
app = ArcGIS(user=os.environ.get("ARC_USERNAME", ""), password=os.environ.get("ARC_PASSWORD", ""))
elif geocoder_name == "bing":
# For Bing, you typically need an API key
bing_key = os.environ.get("BING_API_KEY", "")
if not bing_key:
raise ValueError("Missing BING_API_KEY env var for Bing geocoder.")
app = Bing(api_key=bing_key)
else:
raise ValueError(f"Unsupported geocoder provider: {geocoder_name}")
geocode = RateLimiter(app.geocode, min_delay_seconds=1)
reverse = RateLimiter(app.reverse, min_delay_seconds=1)
@mcp.tool()
def geocode_location(location_str: str) -> dict | None:
"""
Geocodes a single location string (an address or place name).
Returns {'latitude', 'longitude', 'address'} or None if not found.
"""
try:
location = geocode.geocode(location_str)
if not location:
return None
return {
"latitude": location.latitude,
"longitude": location.longitude,
"address": location.address
}
except (GeocoderTimedOut, GeocoderServiceError) as e:
print(f"Geocoding error: {e}")
return None
@mcp.tool()
def reverse_geocode(lat: float, lon: float) -> dict | None:
"""
Reverse geocodes a latitude and longitude to find the nearest address.
Returns {'latitude', 'longitude', 'address'} or None if not found.
"""
try:
location = reverse.reverse((lat, lon))
if not location:
return None
return {
"latitude": lat,
"longitude": lon,
"address": location.address
}
except (GeocoderTimedOut, GeocoderServiceError) as e:
print(f"Reverse geocoding error: {e}")
return None
@mcp.tool()
def geocode_with_details(location_str: str) -> dict | None:
"""
Geocodes a single location string with extra details such as bounding box
and detailed address info, if available.
"""
try:
location = app.geocode(location_str, addressdetails=True)
if not location:
return None
return {
"latitude": location.latitude,
"longitude": location.longitude,
"address": location.address,
"details": location.raw.get("address", {}),
"bounding_box": location.raw.get("boundingbox", [])
}
except (GeocoderTimedOut, GeocoderServiceError) as e:
print(f"Geocoding error: {e}")
return None
@mcp.tool()
def geocode_multiple_locations(location_strs: list[str]) -> list[dict | None]:
"""
Geocodes multiple address strings, returning a list of results.
Each element is either:
{
"latitude": float,
"longitude": float,
"address": str
}
or None if no result was found.
This function uses the same RateLimiter above, so it waits min_delay_seconds
between each geocode call to respect usage limits.
"""
results = []
for loc_str in location_strs:
try:
location = geocode.geocode(loc_str)
if not location:
results.append(None)
else:
results.append({
"latitude": location.latitude,
"longitude": location.longitude,
"address": location.address
})
except (GeocoderTimedOut, GeocoderServiceError) as e:
print(f"Geocoding error for '{loc_str}': {e}")
results.append(None)
return results
@mcp.tool()
def reverse_geocode_multiple_locations(coords: list[list[float]]) -> list[dict | None]:
"""
Reverse geocodes multiple [latitude, longitude] pairs to find the nearest address.
Each item in `coords` should be a list with two floats: [lat, lon].
Example: [[37.7749, -122.4194], [40.7128, -74.0060]]
Returns a list of results, where each result is either:
{
"latitude": float,
"longitude": float,
"address": str
}
or None if the location could not be found or an error occurred.
This function uses the same RateLimiter above, so it waits at least
min_delay_seconds between each reverse geocode call.
"""
results = []
for latlon in coords:
if len(latlon) != 2:
# If there's a malformed input, skip it
results.append(None)
continue
lat, lon = latlon
try:
location = reverse.reverse((lat, lon))
if not location:
results.append(None)
else:
results.append({
"latitude": lat,
"longitude": lon,
"address": location.address
})
except (GeocoderTimedOut, GeocoderServiceError) as e:
print(f"Reverse geocoding error for ({lat}, {lon}): {e}")
results.append(None)
return results
@mcp.tool()
def distance_between_addresses(address1: str, address2: str, unit: str = "kilometers") -> float | None:
"""
Calculate the distance between two addresses or place names.
:param address1: The first address or place name.
:param address2: The second address or place name.
:param unit: "kilometers" (default) or "miles".
Returns the distance in the specified unit, or None if either address could not be geocoded.
"""
# Geocode both addresses
loc1 = geocode.geocode(address1)
loc2 = geocode.geocode(address2)
if not loc1 or not loc2:
# If we couldn't geocode either one, return None
return None
# Extract lat/lon for each location
coords1 = (loc1.latitude, loc1.longitude)
coords2 = (loc2.latitude, loc2.longitude)
# Calculate geodesic distance
distance = distance(coords1, coords2)
# Return in the specified unit
if unit.lower() == "miles":
return distance.miles
else:
# Default is kilometers
return distance.kilometers
@mcp.tool()
def distance_between_coords(
lat1: float, lon1: float, lat2: float, lon2: float, unit: str = "kilometers"
) -> float:
"""
Calculate the distance between two lat/lon pairs.
:param lat1: Latitude of the first location.
:param lon1: Longitude of the first location.
:param lat2: Latitude of the second location.
:param lon2: Longitude of the second location.
:param unit: "kilometers" (default) or "miles".
Returns the distance in the specified unit.
"""
coords1 = (lat1, lon1)
coords2 = (lat2, lon2)
distance = distance(coords1, coords2)
if unit.lower() == "miles":
return distance.miles
else:
return distance.kilometers
if __name__ == "__main__":
mcp.run()