Untitled diff
38 removals
Words removed | 58 |
Total words | 2745 |
Words removed (%) | 2.11 |
593 lines
111 additions
Words added | 464 |
Total words | 3151 |
Words added (%) | 14.73 |
661 lines
---
<!-- YR-TODO:
sidebar: false
-->
---
<style>
.yr-tmp-redText {
color: red;
font-style: italic;
border: 1px solid black;
padding: 2px;
}
.yr-tmp-redText a {
color: red;
font-weight: bold;
}
</style>
<div class="yr-tmp-redText">
**NOTE** These instructions were modified slightly since they were
originally posted. This did NOT change what you need to do in any
significant way. A summary of the changes is listed below.
I'll give full credit if you did the project using either the new (see below)
or the [old instructions](assignment-api-0000100-weatherDataApi-v001.html).
- Feb. 28, 2025 - Modifications:
* Changed some wording or style to make to make the instructions
clearer without changing the meaning of what was said.
* In the [instructions for function getForecastDf](#function-getForecaseId)
elaborated more on
how to check the arguments. This doesn't change the assigment,
it just makes the instructions more explicit and easier to understand.
* In the [instructions for function createForecastFiles](#function-createForecastFiles),
changed the filename prefix to "weatherData_". (The original instructions
said "hourlyData_", however that name is misleading as you can generate
the files on any time schedule - not just hourly).
* To see exactly which lines changed (you really don't have to) you can
[click here](files/_assignment-weatherDataApi-diff-ver1-ver2.html) to see the output of <https://diffchecker.com> when applied
to the
[original Quarto .qmd](assignment-api-0000100-weatherDataApi-v001.qmd.zip)
file and the
[new Quarto .qmd file](assignment-api-0000100-weatherDataApi-v002.qmd.zip)
(for this current version of this webpage).
<span style="font-style: italic; font-size: .8em;">
(The page from diffchecker was saved with the
[yrWebSnap chrome plugin](https://github.com/y-rosenthal/yrWebSnap).)
</span>
</div>
# <yrChapterNumber>18. </yrChapterNumber> assignment - weather API
# <yrChapterNumber>18. </yrChapterNumber> assignment - weather API
<style>
<style>
h4 { font-weight: bold; font-size: 1.2em; }
h4 { font-weight: bold; font-size: 1.2em; }
</style>
</style>
The United States National Weather Service maintains a free API that returns
The United States National Weather Service maintains a free API that returns
weather information for specific latitudes and longitudes. For this assignment
weather information for specific latitudes and longitudes. For this assignment
you will call the API and store the returned info in CSV files. There are two
you will call the API and store the returned info in CSV files. There are two
different API endpoints that will be needed for this assignment.
different API endpoints that will be needed for this assignment.
## The Weather API
## The Weather API
All URLs used to call the weather api start with *https://api.weather.gov/*.
All URLs used to call the weather api start with *https://api.weather.gov/*.
This is known as the "base URL" for the API.
This is known as the "base URL" for the API.
The following describes how to use the weather API.
The following describes how to use the weather API.
### https://api.weather.gov/points/{latitude},{longitude}
### https://api.weather.gov/points/{latitude},{longitude}
The first "endpoint" we will use has the following format:
The first "endpoint" we will use has the following format:
> https://api.weather.gov/points/{latitude},{longitude}
> https://api.weather.gov/points/{latitude},{longitude}
where {latitude} and {longitude} are replaced with specific numeric values
where {latitude} and {longitude} are replaced with specific numeric values
for latitude and longitude. For example,
for latitude and longitude. For example,
the following URL returns data for info about latitude 38.8894 and
the following URL returns data for info about latitude 38.8894 and
longitude -77.0352
longitude -77.0352
*(make sure to notice the comma "," that appears between the latitude*
*(make sure to notice the comma "," that appears between the latitude*
*and longitude numbers in the URL)*
*and longitude numbers in the URL)*
To make a similar call for other locations, simply replace the values
To make a similar call for other locations, simply replace the values
with the appropriate latitude,longitude numbers.
with the appropriate latitude,longitude numbers.
> https://api.weather.gov/points/38.8894,-77.0352
> https://api.weather.gov/points/38.8894,-77.0352
This "API call" doesn't return the actual weather data. Rather, the JSON
This "API call" doesn't return the actual weather data. Rather, the JSON
it returns includes references to other
it returns includes references to other
API URLs that can be used to get the actual weather data (see below).
API URLs that can be used to get the actual weather data (see below).
Specifically, the URL that we will be interested in is identified in the
Specifically, the URL that we will be interested in is identified in the
"forecastHourly" entry in the JSON.
"forecastHourly" entry in the JSON.
Below is an outline of the JSON that is returned from this API call.
Below is an outline of the JSON that is returned from this API call.
We omitted most of the data so that we can clearly highlight
We omitted most of the data so that we can clearly highlight
where the "forecastHourly" URL is in the JSON.
where the "forecastHourly" URL is in the JSON.
<!-- YR-TODO: run the api and put a link to the full JSON here.
Click [here]() to see the full JSON as it was returned from the API.
Click [here]() to see the full JSON as it was returned from the API.
-->
```json {style="font-size: 0.75em;"}
```json {style="font-size: 0.75em;"}
// (Note - this is a comment. It starts with // Classic JSON does not
// NOTE - this is a comment. Classic JSON does NOT allow for comments,
// allow for comments, but JSONC (i.e. JSON with Comments) is recognized
// but JSONC (i.e. JSON with Comments) is recognized by some JSON parsers.
// by some JSON parsers.) The actual API does not include comments in the JSON
// A comment in JSONC is introduced with "//" (similar to how a comment
// that is returned.
// in R is introduced with "#").
//
// The actual API does not include comments in the JSON that is returned.
// I added this comment and the comments in the JSON below in order
// to highlight just the portion of the JSON I wanted you to focus on.
{
{
"@context":[
"@context":[
// SOME STUFF REMOVED FOR BREVITY
// SOME STUFF REMOVED FOR BREVITY
],
],
"id":"https://api.weather.gov/points/38.8894,-77.0352",
"id":"https://api.weather.gov/points/38.8894,-77.0352",
"type":"Feature",
"type":"Feature",
"geometry":{
"geometry":{
// SOME STUFF REMOVED FOR BREVITY
// SOME STUFF REMOVED FOR BREVITY
},
},
"properties":{
"properties":{
// SOME STUFF REMOVED FOR BREVITY
// SOME STUFF REMOVED FOR BREVITY
// THIS IS THE LINE WE ARE INTERESTED IN
// THIS IS THE LINE WE ARE INTERESTED IN
"forecastHourly":"https://api.weather.gov/gridpoints/LWX/97,71/forecast/hourly"
"forecastHourly":"https://api.weather.gov/gridpoints/LWX/97,71/forecast/hourly"
// SOME STUFF REMOVED FOR BREVITY
// SOME STUFF REMOVED FOR BREVITY
}
}
}
}
```
```
### getting the hourly forecasts
### getting the hourly forecasts
The "forecastHourly" URL is obtained from the API call described in the
The "forecastHourly" URL is obtained from the API call described in the
previous section (see above).
previous section (see above).
That JSON included the following line.
That JSON included the following line.
It is this URL that we use to get the actual hour by hour weather forecast data.
It is this URL that we use to get the actual hour by hour weather forecast data.
> "forecastHourly":"https://api.weather.gov/gridpoints/LWX/97,71/forecast/hourly",
> "forecastHourly":"https://api.weather.gov/gridpoints/LWX/97,71/forecast/hourly",
Among other data, the JSON returned by this URL includes an array of time "periods".
Among other data, the JSON returned by this URL includes an array of time "periods".
Each period describes the forecast for a single hour of time in the
Each period describes the forecast for a single hour of time in the
specified location. For this specific URL there are 156 such periods in the data
specified location. For this specific URL there are 156 such periods in the data
(the exact number of periods may change each time you visit this URL).
(the exact number of periods may change each time you visit this URL).
Below we show an example of the first 2 periods from a specific call to this
Below we show an example of the first 2 periods from a specific call to this
API. We removed much of the other data from the JSON shown below.
API. We removed much of the other data from the JSON shown below.
*(NOTE that the forecast data could potentially change every time you call the API)*
*(NOTE that the forecast data could potentially change every time you call the API)*
```json{style="font-size: 0.75em;"}
```json{style="font-size: 0.75em;"}
{
{
"@context":[
"@context":[
// SOME STUFF REMOVED FOR BREVITY
// SOME STUFF REMOVED FOR BREVITY
],
],
"type":"Feature",
"type":"Feature",
"geometry":{
"geometry":{
// SOME STUFF REMOVED FOR BREVITY
// SOME STUFF REMOVED FOR BREVITY
},
},
"properties":{
"properties":{
"units":"us",
"units":"us",
"forecastGenerator":"HourlyForecastGenerator",
"forecastGenerator":"HourlyForecastGenerator",
"generatedAt":"2025-02-12T17:34:21+00:00",
"generatedAt":"2025-02-12T17:34:21+00:00",
"updateTime":"2025-02-12T17:30:41+00:00",
"updateTime":"2025-02-12T17:30:41+00:00",
"validTimes":"2025-02-12T11:00:00+00:00/P7DT14H",
"validTimes":"2025-02-12T11:00:00+00:00/P7DT14H",
"elevation":{
"elevation":{
// SOME STUFF REMOVED FOR BREVITY
// SOME STUFF REMOVED FOR BREVITY
},
},
"periods":[
"periods":[
{
{
"number":1,
"number":1,
"name":"",
"name":"",
"startTime":"2025-02-12T12:00:00-05:00",
"startTime":"2025-02-12T12:00:00-05:00",
"endTime":"2025-02-12T13:00:00-05:00",
"endTime":"2025-02-12T13:00:00-05:00",
"isDaytime":true,
"isDaytime":true,
"temperature":34,
"temperature":34,
"temperatureUnit":"F",
"temperatureUnit":"F",
"temperatureTrend":"",
"temperatureTrend":"",
"probabilityOfPrecipitation":{
"probabilityOfPrecipitation":{
"unitCode":"wmoUnit:percent",
"unitCode":"wmoUnit:percent",
"value":12
"value":12
},
},
"dewpoint":{
"dewpoint":{
"unitCode":"wmoUnit:degC",
"unitCode":"wmoUnit:degC",
"value":-0.5555555555555556
"value":-0.5555555555555556
},
},
"relativeHumidity":{
"relativeHumidity":{
"unitCode":"wmoUnit:percent",
"unitCode":"wmoUnit:percent",
"value":89
"value":89
},
},
"windSpeed":"6 mph",
"windSpeed":"6 mph",
"windDirection":"E",
"windDirection":"E",
"icon":"https://api.weather.gov/icons/land/day/ovc?size=small",
"icon":"https://api.weather.gov/icons/land/day/ovc?size=small",
"shortForecast":"Cloudy",
"shortForecast":"Cloudy",
"detailedForecast":""
"detailedForecast":""
},
},
{
{
"number":2,
"number":2,
"name":"",
"name":"",
"startTime":"2025-02-12T13:00:00-05:00",
"startTime":"2025-02-12T13:00:00-05:00",
"endTime":"2025-02-12T14:00:00-05:00",
"endTime":"2025-02-12T14:00:00-05:00",
"isDaytime":true,
"isDaytime":true,
"temperature":34,
"temperature":34,
"temperatureUnit":"F",
"temperatureUnit":"F",
"temperatureTrend":"",
"temperatureTrend":"",
"probabilityOfPrecipitation":{
"probabilityOfPrecipitation":{
"unitCode":"wmoUnit:percent",
"unitCode":"wmoUnit:percent",
"value":13
"value":13
},
},
"dewpoint":{
"dewpoint":{
"unitCode":"wmoUnit:degC",
"unitCode":"wmoUnit:degC",
"value":-0.5555555555555556
"value":-0.5555555555555556