@cyanheads/nws-weather-mcp-server
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., "@@cyanheads/nws-weather-mcp-serverWhat's the forecast for Denver, CO?"
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.
Public Hosted Server: https://nws.caseyjhand.com/mcp
Tools
Seven tools for real-time US weather data:
Tool | Description |
| 7-day or hourly forecast for coordinates. Resolves NWS grid internally. |
| Active weather alerts filtered by area, point, zone, event, severity, urgency, certainty, and status. |
| Current conditions by coordinates (nearest station) or station ID. |
| Nearby observation stations sorted by distance with bearing. |
| All valid alert event type names for filter discovery. |
| Latest narrative product (AFD, HWO, ZFP, SPS) from a Weather Forecast Office. |
| Text forecast periods for a public NWS forecast zone. |
nws_get_forecast
Get the weather forecast for a US location.
Default returns named 12-hour periods (14 total, ~7 days)
Hourly mode returns up to 156 one-hour periods with dewpoint and humidity
Coordinates resolve to NWS grid internally via
/pointsendpointFormatted timestamps use the resolved local time zone
Returns forecast zone and county zone codes for chaining into
nws_search_alerts
nws_search_alerts
Search active weather alerts with flexible filtering.
Filter by area (state/territory/marine codes), point (lat,lon), zone, event type, severity, urgency, certainty, or status
area,point, andzoneare mutually exclusive; specify at most one location filterNational search when no filters provided
Blank optional location filters are ignored so form-based clients can submit empty fields safely
Event matching is case-insensitive and partial, so
"tornado"matches both watches and warningsstatusdefaults to liveActualalerts, but can be set toExercise,System,Test, orDraftResults capped at 25 with truncation notice and guidance to narrow filters
Validates area codes and point format before API call
nws_get_observations
Current measured conditions from a weather station.
Look up by coordinates (finds nearest station) or station ID directly
Blank or whitespace-only
station_idvalues are ignored so clients can fall back to coordinates cleanlyCoordinate lookups choose the nearest station from the candidates returned by NWS
Dual-unit display: F/C, mph/km/h, inHg/hPa, mi/km
Observation timestamps use the station's local time zone when available
Warns when most measurements are unavailable from a station
nws_find_stations
Discover nearby observation stations.
Sorted by haversine distance from query point
Returns distance (km) and compass bearing
Includes zone codes, elevation, time zone
Useful for finding station IDs for
nws_get_observations
nws_list_alert_types
List all valid NWS alert event type names.
Returns the full set of event types the NWS API recognizes (e.g., "Tornado Warning", "Heat Advisory")
Use to discover valid values for the
eventfilter innws_search_alerts
nws_get_office_discussion
Get the latest narrative product from a Weather Forecast Office (WFO).
office: 3-letter WFO code (e.g.,SEWfor Seattle) — returned as theofficefield bynws_get_forecastproduct_type:AFD(Area Forecast Discussion, default),HWO(Hazardous Weather Outlook),ZFP(Zone Forecast Product),SPS(Special Weather Statement)Two-hop fetch: lists products by office/type (newest first), then retrieves full product text
Returns
productTextplusissuanceTime,issuingOffice,productName,productCode,wmoCollectiveIdUnknown office returns a clear error with recovery instructions (the NWS API returns HTTP 200 with an empty list, not a 404)
nws_get_zone_forecast
Get the text forecast for a public NWS forecast zone.
zone_id: forecast zone code (e.g.,WAZ315) — returned bynws_get_forecast(forecastZone),nws_find_stations(forecastZonecolumn), andnws_search_alerts(affectedZones)Returns named periods (e.g., "Today", "Tonight", "Monday") with narrative text from local forecasters
Completes the alert-to-forecast chain: look up alert zones, then retrieve zone forecasts
County zone codes (
XXC###) are not supported — use the forecast zone code
Related MCP server: Weather Server
Resources
URI Pattern | Description |
| Static list of all valid NWS alert event type names. |
Features
Built on @cyanheads/mcp-ts-core:
Declarative tool definitions — single file per tool, framework handles registration and validation
Unified error handling across all tools
Pluggable auth (
none,jwt,oauth)Swappable storage backends:
in-memory,filesystem,Supabase,Cloudflare KV/R2/D1Structured logging with optional OpenTelemetry tracing
Runs locally (stdio/HTTP) or on Cloudflare Workers from the same codebase
NWS-specific:
Zero-auth access to the NWS API — no API keys required
Automatic coordinate-to-grid resolution with caching (1h TTL)
Request timeouts plus retry/backoff for transient NWS API failures
Dual-unit display for observations (F/C, mph/km/h, inHg/hPa, mi/km)
Continental US, Alaska, Hawaii, and US territories coverage
Getting started
Public Hosted Instance
A public instance is available at https://nws.caseyjhand.com/mcp — no installation required. Point any MCP client at it via Streamable HTTP:
{
"mcpServers": {
"nws-weather-mcp-server": {
"type": "streamable-http",
"url": "https://nws.caseyjhand.com/mcp"
}
}
}Self-Hosted / Local
Add the following to your MCP client configuration file.
{
"mcpServers": {
"nws-weather-mcp-server": {
"type": "stdio",
"command": "bunx",
"args": ["@cyanheads/nws-weather-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info"
}
}
}
}Or with npx (no Bun required):
{
"mcpServers": {
"nws-weather-mcp-server": {
"type": "stdio",
"command": "npx",
"args": ["-y", "@cyanheads/nws-weather-mcp-server@latest"],
"env": {
"MCP_TRANSPORT_TYPE": "stdio",
"MCP_LOG_LEVEL": "info"
}
}
}
}Or with Docker:
{
"mcpServers": {
"nws-weather-mcp-server": {
"type": "stdio",
"command": "docker",
"args": ["run", "-i", "--rm", "-e", "MCP_TRANSPORT_TYPE=stdio", "ghcr.io/cyanheads/nws-weather-mcp-server:latest"]
}
}
}For Streamable HTTP, set the transport and start the server:
MCP_TRANSPORT_TYPE=http MCP_HTTP_PORT=3010 bun run start:http
# Server listens at http://localhost:3010/mcpPrerequisites
Installation
Clone the repository:
git clone https://github.com/cyanheads/nws-weather-mcp-server.gitNavigate into the directory:
cd nws-weather-mcp-serverInstall dependencies:
bun installConfiguration
Variable | Description | Default |
| User-Agent for NWS API requests. The API requires this header. |
|
| Transport: |
|
| Port for HTTP server. |
|
| Hostname for HTTP server. |
|
| Log level: |
|
See .env.example for the full list including auth, storage, and OpenTelemetry options.
Running the server
Local development
Build and run the production version:
# One-time build bun run rebuild # Run the built server bun run start:http # or bun run start:stdioRun checks and tests:
bun run devcheck # Lints, formats, type-checks bun run test # Runs test suite
Project structure
Directory | Purpose |
| Tool definitions ( |
| Resource definitions ( |
| NWS API client and response types. |
| Environment variable parsing and validation with Zod. |
Development guide
See CLAUDE.md for development guidelines and architectural rules. The short version:
Handlers throw, framework catches — no
try/catchin tool logicUse
ctx.logfor domain-specific logging,ctx.statefor storageAdd new tools/resources to the barrel exports and the
createApp()arrays insrc/index.ts
Contributing
Issues and pull requests are welcome. Run checks before submitting:
bun run devcheck
bun run testLicense
Apache-2.0 — see LICENSE for details.
This server cannot be installed
Maintenance
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/cyanheads/nws-weather-mcp-server'
If you have feedback or need assistance with the MCP directory API, please join our Discord server