fromtypingimportAnyimporthttpxfrommcp.server.fastmcpimportFastMCP# Initialize FastMCP servermcp=FastMCP("weather")# ConstantsNWS_API_BASE="https://api.weather.gov"USER_AGENT="weather-app/1.0"asyncdefmake_nws_request(url:str)->dict[str,Any]|None:"""Make a request to the NWS API with proper error handling."""headers={"User-Agent":USER_AGENT,"Accept":"application/geo+json"}asyncwithhttpx.AsyncClient()asclient:try:response=awaitclient.get(url,headers=headers,timeout=30.0)response.raise_for_status()returnresponse.json()exceptException:returnNonedefformat_alert(feature:dict)->str:"""Format an alert feature into a readable string."""props=feature["properties"]returnf"""
Event: {props.get('event','Unknown')}Area: {props.get('areaDesc','Unknown')}Severity: {props.get('severity','Unknown')}Description: {props.get('description','No description available')}Instructions: {props.get('instruction','No specific instructions provided')}"""@mcp.tool()asyncdefget_alerts(state:str)->str:"""Get weather alerts for a US state.
Args:
state: Two-letter US state code (e.g. CA, NY)
"""url=f"{NWS_API_BASE}/alerts/active/area/{state}"data=awaitmake_nws_request(url)ifnotdataor"features"notindata:return"Unable to fetch alerts or no alerts found."ifnotdata["features"]:return"No active alerts for this state."alerts=[format_alert(feature)forfeatureindata["features"]]return"\n---\n".join(alerts)@mcp.tool()asyncdefget_forecast(latitude:float,longitude:float)->str:"""Get weather forecast for a location.
Args:
latitude: Latitude of the location
longitude: Longitude of the location
"""# First get the forecast grid endpointpoints_url=f"{NWS_API_BASE}/points/{latitude},{longitude}"points_data=awaitmake_nws_request(points_url)ifnotpoints_data:return"Unable to fetch forecast data for this location."# Get the forecast URL from the points responseforecast_url=points_data["properties"]["forecast"]forecast_data=awaitmake_nws_request(forecast_url)ifnotforecast_data:return"Unable to fetch detailed forecast."# Format the periods into a readable forecastperiods=forecast_data["properties"]["periods"]forecasts=[]forperiodinperiods[:5]:# Only show next 5 periodsforecast=f"""
{period['name']}:
Temperature: {period['temperature']}°{period['temperatureUnit']}Wind: {period['windSpeed']}{period['windDirection']}Forecast: {period['detailedForecast']}"""forecasts.append(forecast)return"\n---\n".join(forecasts)if__name__=="__main__":# Initialize and run the servermcp.run(transport='stdio')
@mcp.tool()defget_forecast(latitude:float,longitude:float)->str:"""Get weather forecast for a location.
Args:
latitude: Latitude of the location
longitude: Longitude of the location
"""
Claude for Desktop は、ユーザーのリクエストを処理する際に、このコメント内に書かれた情報(例えば、Get weather forecast for a location. という説明)を読み取り、それに基づいて適切な MCPサーバーの機能を選択します。
つまり、Claudeは、
関数の説明文(docstring)
引数の説明(Args)
を参照して、ユーザーのリクエストに対して最適な MCP サーバーの API を自動的に判別して使用しています。そのため、ドキュメントコメントは、明確で簡潔な記述 を心がけ、利用する際の意図を分かりやすく記載する必要があります。