---
title: "Weather Forcecasts on the Command-Line with the National Weather Service API"
date: 2025-03-16T15:33:49-04:00
draft: true
tags: []
math: false
medium_enabled: false
---
The United States National Weather Service (NWS) has a [JSON API](https://www.weather.gov/documentation/services-web-api) which provides weather alerts and forecasts for free without having to create an API key[^1]. We'll use this resource to write a small bash script which gives today and tomorrow's weather forecast given a latitude and longitude.
[^1]: In this day and age, it almost feels too good to be true.
My goal is to have a small bash script which I can run in the terminal to get the forecast information. We'll use [`jq`](https://jqlang.org/) for handling the JSON data.
The NWS API does not provide the forecast directly given a latitude and longitude coordinates. Instead, we need to provide the specific [Weather Forecast Office (WFO)](https://www.weather.gov/srh/nwsoffices) responsible for that area plus their grid definitions for the latitude and longitude coordinate.
Luckily, we can query for that information as well.
```bash
POINTS=$(curl -Ls "https://api.weather.gov/points/$LAT,$LON")
```
{{< unsafe >}}For 42.7289, -73.6915 it returns:
{{}}
```json
{
"@context": [
"https://geojson.org/geojson-ld/geojson-context.jsonld",
{
"@version": "1.1",
"wx": "https://api.weather.gov/ontology#",
"s": "https://schema.org/",
"geo": "http://www.opengis.net/ont/geosparql#",
"unit": "http://codes.wmo.int/common/unit/",
"@vocab": "https://api.weather.gov/ontology#",
"geometry": {
"@id": "s:GeoCoordinates",
"@type": "geo:wktLiteral"
},
"city": "s:addressLocality",
"state": "s:addressRegion",
"distance": {
"@id": "s:Distance",
"@type": "s:QuantitativeValue"
},
"bearing": {
"@type": "s:QuantitativeValue"
},
"value": {
"@id": "s:value"
},
"unitCode": {
"@id": "s:unitCode",
"@type": "@id"
},
"forecastOffice": {
"@type": "@id"
},
"forecastGridData": {
"@type": "@id"
},
"publicZone": {
"@type": "@id"
},
"county": {
"@type": "@id"
}
}
],
"id": "https://api.weather.gov/points/42.7289,-73.6915",
"type": "Feature",
"geometry": {
"type": "Point",
"coordinates": [
-73.6915,
42.7289
]
},
"properties": {
"@id": "https://api.weather.gov/points/42.7289,-73.6915",
"@type": "wx:Point",
"cwa": "ALY",
"forecastOffice": "https://api.weather.gov/offices/ALY",
"gridId": "ALY",
"gridX": 74,
"gridY": 67,
"forecast": "https://api.weather.gov/gridpoints/ALY/74,67/forecast",
"forecastHourly": "https://api.weather.gov/gridpoints/ALY/74,67/forecast/hourly",
"forecastGridData": "https://api.weather.gov/gridpoints/ALY/74,67",
"observationStations": "https://api.weather.gov/gridpoints/ALY/74,67/stations",
"relativeLocation": {
"type": "Feature",
"geometry": {
"type": "Point",
"coordinates": [
-73.707011,
42.724883
]
},
"properties": {
"city": "Watervliet",
"state": "NY",
"distance": {
"unitCode": "wmoUnit:m",
"value": 1343.4228258697
},
"bearing": {
"unitCode": "wmoUnit:degree_(angle)",
"value": 70
}
}
},
"forecastZone": "https://api.weather.gov/zones/forecast/NYZ053",
"county": "https://api.weather.gov/zones/county/NYC083",
"fireWeatherZone": "https://api.weather.gov/zones/fire/NYZ208",
"timeZone": "America/New_York",
"radarStation": "KENX"
}
}
```
{{< unsafe >}}
{{}}
That's a lot of output! However, we only need three pieces of information:
- WFO office (ex: "ALY")
- Grid-X (ex: 74)
- Grid-Y (ex: 67)
We can use the following code to store this into variables using `jq`
```bash
OFFICE=$(echo $POINTS | jq -r .properties.gridId)
X=$(echo $POINTS | jq .properties.gridX)
Y=$(echo $POINTS | jq .properties.gridY)
```
From here, we can query for the weather forecast for that grid point.
```bash
WEATHER=$(curl -Ls "https://api.weather.gov/gridpoints/$OFFICE/$X,$Y/forecast")
```
{{< unsafe >}}Show output:
{{}}
```json
{
"@context": [
"https://geojson.org/geojson-ld/geojson-context.jsonld",
{
"@version": "1.1",
"wx": "https://api.weather.gov/ontology#",
"geo": "http://www.opengis.net/ont/geosparql#",
"unit": "http://codes.wmo.int/common/unit/",
"@vocab": "https://api.weather.gov/ontology#"
}
],
"type": "Feature",
"geometry": {
"type": "Polygon",
"coordinates": [
[
[
-73.6648,
42.7172
],
[
-73.6602,
42.7386
],
[
-73.6893,
42.742
],
[
-73.694,
42.720499999999994
],
[
-73.6648,
42.7172
]
]
]
},
"properties": {
"units": "us",
"forecastGenerator": "BaselineForecastGenerator",
"generatedAt": "2025-03-16T19:59:48+00:00",
"updateTime": "2025-03-16T19:38:29+00:00",
"validTimes": "2025-03-16T13:00:00+00:00/P8D",
"elevation": {
"unitCode": "wmoUnit:m",
"value": 61.8744
},
"periods": [
{
"number": 1,
"name": "This Afternoon",
"startTime": "2025-03-16T15:00:00-04:00",
"endTime": "2025-03-16T18:00:00-04:00",
"isDaytime": true,
"temperature": 69,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": null
},
"windSpeed": "22 mph",
"windDirection": "S",
"icon": "https://api.weather.gov/icons/land/day/wind_ovc?size=medium",
"shortForecast": "Cloudy",
"detailedForecast": "Cloudy, with a high near 69. South wind around 22 mph, with gusts as high as 43 mph. New rainfall amounts less than a tenth of an inch possible."
},
{
"number": 2,
"name": "Tonight",
"startTime": "2025-03-16T18:00:00-04:00",
"endTime": "2025-03-17T06:00:00-04:00",
"isDaytime": false,
"temperature": 50,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": 100
},
"windSpeed": "5 to 21 mph",
"windDirection": "SW",
"icon": "https://api.weather.gov/icons/land/night/tsra,100?size=medium",
"shortForecast": "Showers And Thunderstorms",
"detailedForecast": "A chance of rain showers before 8pm, then showers and thunderstorms. Cloudy, with a low around 50. Southwest wind 5 to 21 mph, with gusts as high as 44 mph. Chance of precipitation is 100%. New rainfall amounts between three quarters and one inch possible."
},
{
"number": 3,
"name": "Monday",
"startTime": "2025-03-17T06:00:00-04:00",
"endTime": "2025-03-17T18:00:00-04:00",
"isDaytime": true,
"temperature": 53,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": 60
},
"windSpeed": "5 to 12 mph",
"windDirection": "NW",
"icon": "https://api.weather.gov/icons/land/day/rain_showers,60/rain_showers,50?size=medium",
"shortForecast": "Rain Showers Likely",
"detailedForecast": "Rain showers likely. Cloudy. High near 53, with temperatures falling to around 46 in the afternoon. Northwest wind 5 to 12 mph. Chance of precipitation is 60%. New rainfall amounts between a half and three quarters of an inch possible."
},
{
"number": 4,
"name": "Monday Night",
"startTime": "2025-03-17T18:00:00-04:00",
"endTime": "2025-03-18T06:00:00-04:00",
"isDaytime": false,
"temperature": 31,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": 30
},
"windSpeed": "7 to 10 mph",
"windDirection": "NW",
"icon": "https://api.weather.gov/icons/land/night/rain_showers,30/bkn?size=medium",
"shortForecast": "Chance Rain Showers then Mostly Cloudy",
"detailedForecast": "A chance of rain showers before 8pm. Mostly cloudy, with a low around 31. Northwest wind 7 to 10 mph. Chance of precipitation is 30%. New rainfall amounts between a tenth and quarter of an inch possible."
},
{
"number": 5,
"name": "Tuesday",
"startTime": "2025-03-18T06:00:00-04:00",
"endTime": "2025-03-18T18:00:00-04:00",
"isDaytime": true,
"temperature": 56,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": null
},
"windSpeed": "7 mph",
"windDirection": "NW",
"icon": "https://api.weather.gov/icons/land/day/few?size=medium",
"shortForecast": "Sunny",
"detailedForecast": "Sunny, with a high near 56. Northwest wind around 7 mph."
},
{
"number": 6,
"name": "Tuesday Night",
"startTime": "2025-03-18T18:00:00-04:00",
"endTime": "2025-03-19T06:00:00-04:00",
"isDaytime": false,
"temperature": 35,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": null
},
"windSpeed": "1 to 5 mph",
"windDirection": "NE",
"icon": "https://api.weather.gov/icons/land/night/few?size=medium",
"shortForecast": "Mostly Clear",
"detailedForecast": "Mostly clear, with a low around 35."
},
{
"number": 7,
"name": "Wednesday",
"startTime": "2025-03-19T06:00:00-04:00",
"endTime": "2025-03-19T18:00:00-04:00",
"isDaytime": true,
"temperature": 65,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": null
},
"windSpeed": "2 to 9 mph",
"windDirection": "SE",
"icon": "https://api.weather.gov/icons/land/day/sct?size=medium",
"shortForecast": "Mostly Sunny",
"detailedForecast": "Mostly sunny, with a high near 65."
},
{
"number": 8,
"name": "Wednesday Night",
"startTime": "2025-03-19T18:00:00-04:00",
"endTime": "2025-03-20T06:00:00-04:00",
"isDaytime": false,
"temperature": 45,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": null
},
"windSpeed": "9 mph",
"windDirection": "SE",
"icon": "https://api.weather.gov/icons/land/night/bkn?size=medium",
"shortForecast": "Mostly Cloudy",
"detailedForecast": "Mostly cloudy, with a low around 45."
},
{
"number": 9,
"name": "Thursday",
"startTime": "2025-03-20T06:00:00-04:00",
"endTime": "2025-03-20T18:00:00-04:00",
"isDaytime": true,
"temperature": 61,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": 50
},
"windSpeed": "8 to 14 mph",
"windDirection": "S",
"icon": "https://api.weather.gov/icons/land/day/bkn/rain_showers,50?size=medium",
"shortForecast": "Mostly Cloudy then Chance Rain Showers",
"detailedForecast": "A chance of rain showers after 2pm. Mostly cloudy, with a high near 61. Chance of precipitation is 50%."
},
{
"number": 10,
"name": "Thursday Night",
"startTime": "2025-03-20T18:00:00-04:00",
"endTime": "2025-03-21T06:00:00-04:00",
"isDaytime": false,
"temperature": 33,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": 70
},
"windSpeed": "8 to 12 mph",
"windDirection": "SW",
"icon": "https://api.weather.gov/icons/land/night/rain_showers,70/snow,70?size=medium",
"shortForecast": "Rain Showers Likely then Chance Rain And Snow Showers",
"detailedForecast": "Rain showers likely before 2am, then a chance of rain and snow showers. Mostly cloudy, with a low around 33. Chance of precipitation is 70%."
},
{
"number": 11,
"name": "Friday",
"startTime": "2025-03-21T06:00:00-04:00",
"endTime": "2025-03-21T18:00:00-04:00",
"isDaytime": true,
"temperature": 41,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": 50
},
"windSpeed": "12 to 17 mph",
"windDirection": "NW",
"icon": "https://api.weather.gov/icons/land/day/snow,50/snow,30?size=medium",
"shortForecast": "Chance Rain And Snow Showers",
"detailedForecast": "A chance of rain and snow showers. Partly sunny, with a high near 41. Chance of precipitation is 50%."
},
{
"number": 12,
"name": "Friday Night",
"startTime": "2025-03-21T18:00:00-04:00",
"endTime": "2025-03-22T06:00:00-04:00",
"isDaytime": false,
"temperature": 30,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": null
},
"windSpeed": "9 to 16 mph",
"windDirection": "NW",
"icon": "https://api.weather.gov/icons/land/night/snow/sct?size=medium",
"shortForecast": "Slight Chance Snow Showers then Partly Cloudy",
"detailedForecast": "A slight chance of snow showers before 8pm. Partly cloudy, with a low around 30."
},
{
"number": 13,
"name": "Saturday",
"startTime": "2025-03-22T06:00:00-04:00",
"endTime": "2025-03-22T18:00:00-04:00",
"isDaytime": true,
"temperature": 52,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": null
},
"windSpeed": "9 to 13 mph",
"windDirection": "NW",
"icon": "https://api.weather.gov/icons/land/day/sct?size=medium",
"shortForecast": "Mostly Sunny",
"detailedForecast": "Mostly sunny, with a high near 52."
},
{
"number": 14,
"name": "Saturday Night",
"startTime": "2025-03-22T18:00:00-04:00",
"endTime": "2025-03-23T06:00:00-04:00",
"isDaytime": false,
"temperature": 34,
"temperatureUnit": "F",
"temperatureTrend": "",
"probabilityOfPrecipitation": {
"unitCode": "wmoUnit:percent",
"value": null
},
"windSpeed": "3 to 9 mph",
"windDirection": "NW",
"icon": "https://api.weather.gov/icons/land/night/bkn?size=medium",
"shortForecast": "Mostly Cloudy",
"detailedForecast": "Mostly cloudy, with a low around 34."
}
]
}
}
```
{{< unsafe >}}
{{}}
The National Weather Service packs in so much weather information in just one API call! My favorite part is that they have a `detailedForecast` field which provides a nice human-readable description of the weather.
I don't know about you, but at any given time, I'm mostly curious about today and tomorrow's weather. Unless I'm planning a trip or need to be aware of an extreme weather event, the day's forecast mostly influences the clothes I wear.
The final part of this script is to grab the next three "periods" and display its `detailedForecast` field.
```bash
for i in 0 1 2; do
echo "$WEATHER" | jq .properties.periods[$i] | jq -r '
"\(.name)
\(.detailedForecast)
"
'
done
```
Instead of having to read an entire JSON message to get the weather, we now have a description of the forecast coming up.
```
This Afternoon
Cloudy, with a high near 69. South wind around 22 mph, with gusts as high as 43 mph. New rainfall amounts less than a tenth of an inch possible.
Tonight
A chance of rain showers before 8pm, then showers and thunderstorms. Cloudy, with a low around 50. Southwest wind 5 to 21 mph, with gusts as high as 44 mph. Chance of precipitation is 100%. New rainfall amounts between three quarters and one inch possible.
Monday
Rain showers likely. Cloudy. High near 53, with temperatures falling to around 46 in the afternoon. Northwest wind 5 to 12 mph. Chance of precipitation is 60%. New rainfall amounts between a half and three quarters of an inch possible.
```
Look how short and sweet that output is! With that, we have all the steps needed in our script to grab the upcoming forecast using the NWS API. I created an alias called `myweather` so that I don't have to memorize the latitude or longitude of where I live.
```bash
alias myweather='usweather 42.7289, -73.6915'
```
Here is the script that I call `usweather` in its entirety. I placed it in `~/.local/bin`.
```bash
#!/bin/sh
# Script to query the US National Weather Forecast Office (WFO)
set -o errexit
set -o nounset
show_usage() {
echo "Usage: usweather [lat] [lon]"
exit 1
}
# Check Argument Count
if [ "$#" -ne 2 ]; then
show_usage
fi
if ! command -v jq &> /dev/null; then
echo "jq is not installed"
exit 1
fi
# We need to figure out WFO specific parameters
POINTS=$(curl -Ls "https://api.weather.gov/points/$1,$2")
OFFICE=$(echo $POINTS | jq -r .properties.gridId)
X=$(echo $POINTS | jq .properties.gridX)
Y=$(echo $POINTS | jq .properties.gridY)
if [ "$OFFICE" = "null" ] || [ "$X" = "null" ] || [ "$Y" = "null" ]; then
echo -e "Error:\n $POINTS"
exit 1
fi
WEATHER=$(curl -Ls "https://api.weather.gov/gridpoints/$OFFICE/$X,$Y/forecast")
# Provide the detailed forecast for the next three weather periods
for i in 0 1 2; do
echo "$WEATHER" | jq .properties.periods[$i] | jq -r '
"\(.name)
\(.detailedForecast)
"
'
done
```