README.md

senamhiR: A collection of functions to obtain Peruvian climate data in R

The package provides an automated solution for the acquisition of archived Peruvian climate and hydrology data directly within R. The data was compiled from the Senamhi website, and contains all of the data that was available as of April 10, 2018. This data was originally converted from HTML, and is now accessible via an API hosted by the package author.

It is important to note that the info on the Senamhi website has not undergone quality control, however, this package includes a helper function to perform the most common quality control operations for the temperature variables. More functions will be added in the future.

Installing

This package is under active development, and is not available from the official Comprehensive R Archive Network (CRAN). To make installation easier, I have written a script that should facilitate the installation of the package and its dependencies. Use the following command to run this script:

source("https://gitlab.com/snippets/1793256/raw");install("senamhiR")

Note: It is always a good idea to review code before you run it. Click the URL in the above command to see the commands that we will run to install.

Once the packages have installed, load senamhiR by:

library(senamhiR)

Basic workflow

The functions contained in the senamhiR functions allow for the discovery and visualization of meteorological and hydrological stations, and the acquisition of daily climate data from these stations.

station_search()

To search for a station by name, use the station_search() function. For instance, to search for a station with the word ‘Santa’ in the station name, use the following code:

station_search("Santa")
## # A tibble: 43 x 14
##    Station StationID Type  Configuration `Data Start` `Data End`
##    <chr>   <chr>     <chr> <chr>         <chr>        <chr>     
##  1 SANTA … 000256    CON   M             1951         2018      
##  2 SANTA   000433    CON   M             1964         1986      
##  3 SANTA … 158201    CON   M             1964         1982      
##  4 SANTA … 000829    CON   M             1977         1992      
##  5 SANTA … 000834    CON   M             1963         1973      
##  6 SANTA … 113248    SUT   M             2015         2016      
##  7 SANTA … 153200    CON   M             1963         1983      
##  8 SANTA … 000351    CON   M             1963         2018      
##  9 HACIEN… 000766    CON   M             1954         1955      
## 10 SANTA … 000515    CON   M             1942         1947      
## # ... with 33 more rows, and 8 more variables: `Period (Yr)` <dbl>,
## #   `Station Status` <fct>, Latitude <dbl>, Longitude <dbl>,
## #   Altitude <int>, Region <chr>, Province <chr>, District <chr>

Note that the tbl_df object (a special sort of data.frame) won’t print more than the first 10 rows by default. To see all of the results, you can wrap the command in View() so that it becomes View(find_station("Santa")).

Note that you can also use wildcards as supported by the glob2rx() from the utils package by passing the argument glob = TRUE, as in the following example.

station_search("San*", glob = TRUE)
## # A tibble: 139 x 14
##    Station StationID Type  Configuration `Data Start` `Data End`
##    <chr>   <chr>     <chr> <chr>         <chr>        <chr>     
##  1 SAN RA… 152222    CON   M             1965         1966      
##  2 SANTA … 000256    CON   M             1951         2018      
##  3 SAN PE… 211404    CON   H             2009         2018      
##  4 SANTIA… 000426    CON   M             1998         2018      
##  5 SAN DI… 000420    CON   M             1959         1962      
##  6 SAN LO… 000430    CON   M             1966         1972      
##  7 SAN JA… 000424    CON   M             1956         1968      
##  8 SAN JA… 201901    CON   H             1947         1990      
##  9 SANTA   000433    CON   M             1964         1986      
## 10 SANTA … 158201    CON   M             1964         1982      
## # ... with 129 more rows, and 8 more variables: `Period (Yr)` <dbl>,
## #   `Station Status` <fct>, Latitude <dbl>, Longitude <dbl>,
## #   Altitude <int>, Region <chr>, Province <chr>, District <chr>

You can filter your search results by region, by station type, by a given period, and by proximity to another station or a vector of coordinates. You can use any combination of these four filters in your search. The function is fully documented, so take a look at ?station_search. Let’s see some examples.

Find all stations in the San Martín Region

station_search(region = "SAN MARTIN")
## # A tibble: 72 x 14
##    Station StationID Type  Configuration `Data Start` `Data End`
##    <chr>   <chr>     <chr> <chr>         <chr>        <chr>     
##  1 CUZCO   000389    CON   M             1966         1970      
##  2 DOS DE… 153225    CON   M             1963         2018      
##  3 CUZCO-… 153345    CON   M             1996         2018      
##  4 BIAVO   221804    CON   H             1969         2018      
##  5 LA UNI… 000384    CON   M             1970         2018      
##  6 NUEVO … 153312    CON   M             1963         2018      
##  7 BELLAV… 000382    CON   M             1963         2018      
##  8 SAN PA… 153307    CON   M             1967         2018      
##  9 SISA    000381    CON   M             1964         1988      
## 10 ALAO    003308    CON   M             1972         2018      
## # ... with 62 more rows, and 8 more variables: `Period (Yr)` <dbl>,
## #   `Station Status` <fct>, Latitude <dbl>, Longitude <dbl>,
## #   Altitude <int>, Region <chr>, Province <chr>, District <chr>

Find stations named “Santa”, with data available between 1971 to 2000

station_search("Santa", period = 1971:2000)
## # A tibble: 10 x 14
##    Station StationID Type  Configuration `Data Start` `Data End`
##    <chr>   <chr>     <chr> <chr>         <chr>        <chr>     
##  1 SANTA … 000256    CON   M             1951         2018      
##  2 SANTA … 000351    CON   M             1963         2018      
##  3 SANTA … 155202    CON   M             1963         2018      
##  4 SANTA … 155213    CON   M             1963         2018      
##  5 SANTA … 000536    CON   M             1967         2006      
##  6 SANTA … 152303    CON   M             1963         2008      
##  7 SANTA … 152401    CON   M             1963         2018      
##  8 SANTA … 152409    CON   M             1963         2018      
##  9 SANTA … 000177    CON   M             1963         2017      
## 10 SANTA … 000823    CON   M             1956         2017      
## # ... with 8 more variables: `Period (Yr)` <dbl>, `Station Status` <fct>,
## #   Latitude <dbl>, Longitude <dbl>, Altitude <int>, Region <chr>,
## #   Province <chr>, District <chr>

Find all stations between 0 and 100 km from Station No. 000401

station_search(target = "000401", dist = 0:100)
## # A tibble: 58 x 15
##    Station StationID Type  Configuration `Data Start` `Data End`
##    <chr>   <chr>     <chr> <chr>         <chr>        <chr>     
##  1 TARAPO… 000401    CON   M             1998         2018      
##  2 CUNUMB… 153311    CON   M             1963         2018      
##  3 CUMBAZA 221801    CON   H             1968         2018      
##  4 LAMAS   000383    CON   M             1963         2018      
##  5 SAN AN… 153314    CON   M             1963         2018      
##  6 SHANAO  221802    CON   H             1965         2018      
##  7 SHANAO  210006    SUT   H             2016         2018      
##  8 SHANAO  153328    CON   M             2002         2018      
##  9 TABALO… 000322    CON   M             1963         2018      
## 10 EL POR… 000310    CON   M             1964         2018      
## # ... with 48 more rows, and 9 more variables: `Period (Yr)` <dbl>,
## #   `Station Status` <fct>, Latitude <dbl>, Longitude <dbl>,
## #   Altitude <int>, Region <chr>, Province <chr>, District <chr>,
## #   Dist <dbl>

Find all stations that are within 50 km of Machu Picchu

station_search(target = c(-13.163333, -72.545556), dist = 0:50)
## # A tibble: 20 x 15
##    Station StationID Type  Configuration `Data Start` `Data End`
##    <chr>   <chr>     <chr> <chr>         <chr>        <chr>     
##  1 MACHU … 000679    CON   M             1964         2018      
##  2 HUYRO   000678    CON   M             1964         1981      
##  3 CHILCA  472A9204  SUT   M             2015         2018      
##  4 ECHARA… 000716    CON   M             1981         1982      
##  5 MARANU… 000676    CON   M             1970         1978      
##  6 OLLANT… 47295014  SUT   M             2011         2014      
##  7 QUILLA… 4729B3E6  SUT   M1            2000         2018      
##  8 QUILLA… 000606    CON   M             1964         2018      
##  9 OCOBAM… 000681    CON   M             1964         1983      
## 10 MOLLEP… 000680    CON   M             1963         1978      
## 11 CUNYAC  156224    CON   M             2002         2018      
## 12 ECHARA… 156300    CON   M             1963         1981      
## 13 PUENTE… 472C44A2  SUT   H             2016         2018      
## 14 PUENTE… 230503    CON   H             1995         2018      
## 15 ZURITE  000682    CON   M             1963         1983      
## 16 CURAHU… 000677    CON   M             1963         2018      
## 17 URUBAM… 113131    DAV   M             2006         2008      
## 18 URUBAM… 000683    CON   M             1963         2018      
## 19 ANTA A… 000684    CON   M             1964         2018      
## 20 HUACHI… 156303    CON   M             1963         1978      
## # ... with 9 more variables: `Period (Yr)` <dbl>, `Station Status` <fct>,
## #   Latitude <dbl>, Longitude <dbl>, Altitude <int>, Region <chr>,
## #   Province <chr>, District <chr>, Dist <dbl>

Acquire data: senamhiR()

Once you have found your station of interest, you can download the daily data using the eponymous senamhiR() function. The function takes two arguments, station and year. If year is left blank, the function will return all available archived data.

If I wanted to download data for Requena (station no. 000280) from 1981 to 2010, I could use:

requ <- senamhiR("000280", 1981:2010)

Note: Since the StationID numbers contain leading zeros, any station that is less than six characters long will be padded with zeroes. i.e. 280 becomes 000280.

requ
## # A tibble: 10,957 x 13
##    Fecha      `Tmax (C)` `Tmin (C)` `TBS07 (C)` `TBS13 (C)` `TBS19 (C)`
##    <date>          <dbl>      <dbl>       <dbl>       <dbl>       <dbl>
##  1 1981-01-01       35         23          24.8        30.2        24.4
##  2 1981-01-02       34         22.2        24.2        30          25.6
##  3 1981-01-03       29         23.2        24.6        25.2        24  
##  4 1981-01-04       30.2       22          24.6        28          25  
##  5 1981-01-05       33         22.4        24          25          23.4
##  6 1981-01-06       35.2       23          25          30.8        27.8
##  7 1981-01-07       33.6       23          25.4        30.8        26  
##  8 1981-01-08       37.4       22.8        25.4        35          24  
##  9 1981-01-09       35         23          27          35          26  
## 10 1981-01-10       35.6       22.4        24.8        34.4        28  
## # ... with 10,947 more rows, and 7 more variables: `TBH07 (C)` <dbl>,
## #   `TBH13 (C)` <dbl>, `TBH19 (C)` <dbl>, `Prec07 (mm)` <dbl>, `Prec19
## #   (mm)` <dbl>, `Direccion del Viento` <chr>, `Velocidad del Viento
## #   (m/s)` <dbl>

Make sure to use the assignment operator (<-) to save the data into an R object, otherwise the data will just print out to the console, and won’t get saved anywhere in the memory.

For easier station visualization

map_stations()

Sometimes a long list of stations is hard to visualize spatially. The map_stations() function helps to overcome this. This function takes a list of stations and shows them on a map powered by the Leaflet library. Like the previous function, the map function is even smart enough to take a search as its list of stations as per the example below. Note that this mapping functionality requires the leaflet package to be installed, and it is not included as a dependency of senamhiR.

Show a map of all stations that are between 30 and 50 km of Machu Picchu

map_stations(station_search(target = c(-13.163333, -72.545556), dist = 30:50), zoom = 7)

Quality control functions

There are two functions included to perform some basic quality control.

quick_audit()

The quick_audit() function will return a tibble listing the percentage or number of missing values for a station. For instance, the following command will return the percentage of missing values in our 30-year Requena data set:

quick_audit(requ, c("Tmax", "Tmin"))
## # A tibble: 1 x 2
##   `Tmax (C) pct NA` `Tmin (C) pct NA`
##               <dbl>             <dbl>
## 1              3.55              3.80

Use report = "n" to show the number of missing values. Use by = "month" or by = "year" to show missing data by month or year. For instance, the number of days for which Mean Temperature was missing at Tocache in 1980:

toca <- senamhiR("000463", year = 1980)
quick_audit(toca, "Tmean", by = "month", report = "n")
## # A tibble: 12 x 5
##    Year  Month `Year-month`  `Tmean (C) consec NA` `Tmean (C) tot NA`
##    <chr> <chr> <S3: yearmon>                 <int>              <int>
##  1 1980  01    Jan 1980                          0                  0
##  2 1980  02    Feb 1980                          0                  0
##  3 1980  03    Mar 1980                          2                  3
##  4 1980  04    Apr 1980                          4                  4
##  5 1980  05    May 1980                          0                  0
##  6 1980  06    Jun 1980                          0                  0
##  7 1980  07    Jul 1980                          0                  0
##  8 1980  08    Aug 1980                          0                  0
##  9 1980  09    Sep 1980                          1                  1
## 10 1980  10    Oct 1980                          0                  0
## 11 1980  11    Nov 1980                          1                  1
## 12 1980  12    Dec 1980                          0                  0

qc()

There is an incomplete and experimental function to perform automated quality control on data acquired through this package. Fow now, the package tests temperature and river level only. The logic used between these two types of data is different. Note that these methods are not necessarily statistically robust, and have not been subjected to rigourous testing. Your mileage may vary. In all cases, the original values are archived in an “observations” column, so you can always restore the original values manually.

Temperature variables

Case 1: Missing decimal point

Any number above 100 °C or below -100 °C is tested:

If the number appears to have missed a decimal place (e.g. 324 -> 32.4; 251 -> 25.1), we try to divide that number by 10. If the result is within 1.5 standard deviations of all values 30 days before and after the day in question, we keep the result, otherwise, we discard it.

If the number seems to be the result of some other typographical error (e.g. 221.2), we discard the data point.

Case 2: Tmax \< Tmin

We perform the same tests for both Tmax and Tmin. If the number is within 1.5 standard deviations of all values 30 days before and after the day in question, we leave the number alone. (Note: this is often the case for Tmin but seldom the case for Tmax). If the number does not fall within 1.5 standard deviations, we perform an additional level of testing to check if the number is the result of a premature decimal point (e.g. 3.4 -> 34.0; 3 -> 30.0). In this case, we try to multiply the number by 10. If this new result is within 1.5 standard deviations of all values 30 days before and after the day in question, we keep the result, otherwise, we discard it.

I have less confidence in this solution than I do for Case 1.

Example:
requ_dirty <- senamhiR("000280") #1960 to 2018
requ_qc <- qc(requ_dirty)
requ_qc %>% filter(Observations != "") %>% select(Fecha, `Tmax (C)`, `Tmin (C)`, `Tmean (C)`, Observations)
## # A tibble: 5 x 5
##   Fecha      `Tmax (C)` `Tmin (C)` `Tmean (C)` Observations                
##   <date>          <dbl>      <dbl>       <dbl> <chr>                       
## 1 2013-02-27       34         22.2        28.1 "Tmax dps: 3.4 -> 34 (1.4) "
## 2 2013-05-08       31.4       20.8        26.1 Tmax dps: 314 -> 31.4 (0.25)
## 3 2013-07-18       30.8       NA          NA   Tmin err: 221.2 -> NA       
## 4 2013-10-28       33.4       23.2        28.3 Tmin dps: 232 -> 23.2 (1.03)
## 5 2013-12-24       30         23.6        26.8 "Tmax dps: 3 -> 30 (0.77) "
Cases that are currently missed:
Cases where this function is plain wrong:

River level:

Case 1: Suspected decimal place shift

The function first calculates the daily range in river level across the four daily observations. If any range is greater than the (somewhat arbitrary) value of ten times the average range, then we extract a slice of the level observations corresponding to two days before and two days after the day in question. We standardize the slice of data; if any single standardized value is above 1 (below -1), we try to multiply (divide) the value by 10. If the new value falls within 1.5 standard deviations of the mean of the “good” values, then we keep the modified value and call it a decimal place error, otherwise, we set the value to missing and label it as an error.

Example:
pico_dirty <- senamhiR("230715") #2003 to 2018
pico_qc <- qc(pico_dirty)
pico_qc %>% filter(Observations != "") %>% select(Fecha, starts_with("Nivel"), Observations)
## # A tibble: 2 x 6
##   Fecha      `Nivel06 (m)` `Nivel10 (m)` `Nivel14 (m)` `Nivel18 (m)`
##   <date>             <dbl>         <dbl>         <dbl>         <dbl>
## 1 2011-03-25          17.1          17.1          NA            17.5
## 2 2011-12-06          15.8          15.8          15.8          NA  
##   Observations         
##   <chr>                
## 1 Level err: 12.2 -> NA
## 2 Level err: 1.83 -> NA
Cases that are currently missed:
Cases where this function is plain wrong:

Variables controlled for:

No other variables are currently tested; hydrological data is not tested. This data should not be considered “high quality”, use of the data is your responsibility. Note that all values that are modified from their original values will be recorded in a new “Observations” column in the resultant tibble.

Disclaimer

The package outlined in this document is published under the GNU General Public License, version 3 (GPL-3.0). The GPL is an open source, copyleft license that allows for the modification and redistribution of original works. Programs licensed under the GPL come with NO WARRANTY. In our case, a simple R package isn’t likely to blow up your computer or kill your cat. Nonetheless, it is always a good idea to pay attention to what you are doing, to ensure that you have downloaded the correct data, and that everything looks ship-shape.

What to do if something doesn’t work

If you run into an issue while you are using the package, you can email me and I can help you troubleshoot the issue. However, if the issue is related to the package code and not your own fault, you should contribute back to the open source community by reporting the issue. You can report any issues to me here on GitLab.

If that seems like a lot of work, just think about how much work it would have been to do all the work this package does for you, or how much time went in to writing these functions … it is more than I’d like to admit!

Senamhi terms of use

Senamhi’s terms of use are here, but as of writing that link was redirecting to the Senamhi home page. An archived version is available here. The terms allow for the free and public access to information on the Senamhi website, in both for-profit and non-profit applications. However, Senamhi stipulates that any use of the data must be accompanied by a disclaimer that Senamhi is the proprietor of the information. The following text is recommended (official text in Spanish):

A message similar to the English message above is printed to the R console whenever the package is loaded.



ConorIA/senamhiR documentation built on May 6, 2019, 12:51 p.m.