Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

coerce and rotate pvgis TMY data to desired tz and year #2138

Merged
merged 26 commits into from
Aug 16, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
883cfbe
coerce and rotate pvgis TMY data to desired tz and year
mikofski Jul 19, 2024
7e6e4c0
test get_pvgis_tmy_coerce_year
mikofski Jul 19, 2024
84d4695
fix flake8 in test_pvgis_coerce_year
mikofski Jul 19, 2024
3e42075
remove iloc for index in test pvgis coerce
mikofski Jul 19, 2024
d0bd0a0
deal with leap year in pvgis when coercing
mikofski Jul 19, 2024
ec482ac
fix space around operator in coerce pvgis
mikofski Jul 19, 2024
883ced8
fix pd.Timestamp in pvgis coerce year
mikofski Jul 19, 2024
dd2c85f
Update v0.11.1 what's new for coerce pvgis tmy
mikofski Jul 20, 2024
31800e9
replace year and tzinfo in pvgis_tmy coerce year
mikofski Jul 27, 2024
72c55f6
remove unused imports from pvgis.py for flake8
mikofski Jul 27, 2024
0735183
change private function name to _coerce_and_roll_pvgis_tmy
mikofski Jul 27, 2024
2059dc6
spot check rolled pvgis TMY values after converting tz
mikofski Jul 27, 2024
a38cf9a
Update utc_offset description
mikofski Jul 31, 2024
440e780
change coerce_year and utc_offset defaults to None in pvgis TMY
mikofski Jul 31, 2024
4fd05fb
rename roll_utc_offset in get_pvgis_tmy
mikofski Jul 31, 2024
1b92d40
Update pvlib/iotools/pvgis.py with suggestions
mikofski Aug 1, 2024
43ce3c4
Update docs/sphinx/source/whatsnew/v0.11.1.rst
mikofski Aug 1, 2024
76c8ec6
Merge branch 'main' into coerce-and-rotate-pvgis
AdamRJensen Aug 5, 2024
d77a3d3
Merge branch 'main' into coerce-and-rotate-pvgis
mikofski Aug 5, 2024
91c169e
rename _coerce_and_roll_tmy, remove 'pvgis'
mikofski Aug 6, 2024
1e9ee64
rename index with new tz in coerce pvgis tmy
mikofski Aug 6, 2024
d24e4f3
allow tz of None in _coerce_and_roll_tmy
mikofski Aug 6, 2024
f12571d
Merge branch 'main' into coerce-and-rotate-pvgis
mikofski Aug 13, 2024
fb2813d
clarify input tmy_data is UTC...
mikofski Aug 14, 2024
ac9dbb0
fix flake8 in _coerce_and_roll_tmy
mikofski Aug 14, 2024
9875521
Merge branch 'main' into coerce-and-rotate-pvgis
mikofski Aug 16, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions docs/sphinx/source/whatsnew/v0.11.1.rst
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,10 @@ Enhancements
* Add new parameters for min/max absolute air mass to
:py:func:`pvlib.spectrum.spectral_factor_firstsolar`.
(:issue:`2086`, :pull:`2100`)
* Add ``roll_utc_offset`` and ``coerce_year`` arguments to
:py:func:`pvlib.iotools.get_pvgis_tmy` to allow user to specify time zone,
rotate indices of TMY to begin at midnight, and force indices to desired
year. (:issue:`2139`, :pull:`2138`)
* Restructured the pvlib/spectrum folder by breaking up the contents of
pvlib/spectrum/mismatch.py into pvlib/spectrum/mismatch.py,
pvlib/spectrum/irradiance.py, and
Expand Down Expand Up @@ -62,5 +66,6 @@ Contributors
* Leonardo Micheli (:ghuser:`lmicheli`)
* Echedey Luis (:ghuser:`echedey-ls`)
* Rajiv Daxini (:ghuser:`RDaxini`)
* Mark A. Mikofski (:ghuser:`mikofski`)
* Ben Pierce (:ghuser:`bgpierc`)
* Jose Meza (:ghuser:`JoseMezaMendieta`)
42 changes: 39 additions & 3 deletions pvlib/iotools/pvgis.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,10 @@
import json
from pathlib import Path
import requests
import numpy as np
import pandas as pd
import pytz
from pvlib.iotools import read_epw, parse_epw
import warnings
from pvlib._deprecation import pvlibDeprecationWarning

URL = 'https://re.jrc.ec.europa.eu/api/'

Expand Down Expand Up @@ -390,9 +390,33 @@ def read_pvgis_hourly(filename, pvgis_format=None, map_variables=True):
raise ValueError(err_msg)


def _coerce_and_roll_tmy(tmy_data, tz, year):
"""
Assumes ``tmy_data`` input is UTC, converts from UTC to ``tz``, rolls
dataframe so timeseries starts at midnight, and forces all indices to
``year``. Only works for integer ``tz``, but ``None`` and ``False`` are
re-interpreted as zero / UTC.
"""
if tz:
tzname = pytz.timezone(f'Etc/GMT{-tz:+d}')
else:
tz = 0
tzname = pytz.timezone('UTC')
new_index = pd.DatetimeIndex([
timestamp.replace(year=year, tzinfo=tzname)
for timestamp in tmy_data.index],
name=f'time({tzname})')
new_tmy_data = pd.DataFrame(
np.roll(tmy_data, tz, axis=0),
columns=tmy_data.columns,
index=new_index)
return new_tmy_data


def get_pvgis_tmy(latitude, longitude, outputformat='json', usehorizon=True,
userhorizon=None, startyear=None, endyear=None,
map_variables=True, url=URL, timeout=30):
map_variables=True, url=URL, timeout=30,
roll_utc_offset=None, coerce_year=None):
"""
Get TMY data from PVGIS.
Expand Down Expand Up @@ -424,6 +448,13 @@ def get_pvgis_tmy(latitude, longitude, outputformat='json', usehorizon=True,
base url of PVGIS API, append ``tmy`` to get TMY endpoint
timeout : int, default 30
time in seconds to wait for server response before timeout
roll_utc_offset: int, optional
Use to specify a time zone other than the default UTC zero and roll
dataframe by ``roll_utc_offset`` so it starts at midnight on January
1st. Ignored if ``None``, otherwise will force year to ``coerce_year``.
coerce_year: int, optional
Use to force indices to desired year. Will default to 1990 if
``coerce_year`` is not specified, but ``roll_utc_offset`` is specified.
mikofski marked this conversation as resolved.
Show resolved Hide resolved
Returns
-------
Expand Down Expand Up @@ -510,6 +541,11 @@ def get_pvgis_tmy(latitude, longitude, outputformat='json', usehorizon=True,
if map_variables:
data = data.rename(columns=VARIABLE_MAP)

if not (roll_utc_offset is None and coerce_year is None):
# roll_utc_offset is specified, but coerce_year isn't
coerce_year = coerce_year or 1990
data = _coerce_and_roll_tmy(data, roll_utc_offset, coerce_year)

return data, months_selected, inputs, meta


Expand Down
47 changes: 47 additions & 0 deletions pvlib/tests/iotools/test_pvgis.py
Original file line number Diff line number Diff line change
Expand Up @@ -435,6 +435,53 @@ def _compare_pvgis_tmy_basic(expected, meta_expected, pvgis_data):
assert np.allclose(data[outvar], expected[outvar])


@pytest.mark.remote_data
@pytest.mark.flaky(reruns=RERUNS, reruns_delay=RERUNS_DELAY)
def test_get_pvgis_tmy_coerce_year():
"""test utc_offset and coerce_year work as expected"""
base_case, _, _, _ = get_pvgis_tmy(45, 8) # Turin
assert str(base_case.index.tz) == 'UTC'
assert base_case.index.name == 'time(UTC)'
noon_test_data = [
base_case[base_case.index.month == m].iloc[12]
for m in range(1, 13)]
cet_tz = 1 # Turin time is CET
cet_name = 'Etc/GMT-1'
# check indices of rolled data after converting timezone
pvgis_data, _, _, _ = get_pvgis_tmy(45, 8, roll_utc_offset=cet_tz)
jan1_midnight = pd.Timestamp('1990-01-01 00:00:00', tz=cet_name)
dec31_midnight = pd.Timestamp('1990-12-31 23:00:00', tz=cet_name)
assert pvgis_data.index[0] == jan1_midnight
assert pvgis_data.index[-1] == dec31_midnight
assert pvgis_data.index.name == f'time({cet_name})'
# spot check rolled data matches original
for m, test_case in enumerate(noon_test_data):
expected = pvgis_data[pvgis_data.index.month == m+1].iloc[12+cet_tz]
assert all(test_case == expected)
# repeat tests with year coerced
test_yr = 2021
pvgis_data, _, _, _ = get_pvgis_tmy(
45, 8, roll_utc_offset=cet_tz, coerce_year=test_yr)
jan1_midnight = pd.Timestamp(f'{test_yr}-01-01 00:00:00', tz=cet_name)
dec31_midnight = pd.Timestamp(f'{test_yr}-12-31 23:00:00', tz=cet_name)
assert pvgis_data.index[0] == jan1_midnight
assert pvgis_data.index[-1] == dec31_midnight
assert pvgis_data.index.name == f'time({cet_name})'
for m, test_case in enumerate(noon_test_data):
expected = pvgis_data[pvgis_data.index.month == m+1].iloc[12+cet_tz]
assert all(test_case == expected)
# repeat tests with year coerced but utc offset none or zero
pvgis_data, _, _, _ = get_pvgis_tmy(45, 8, coerce_year=test_yr)
jan1_midnight = pd.Timestamp(f'{test_yr}-01-01 00:00:00', tz='UTC')
dec31_midnight = pd.Timestamp(f'{test_yr}-12-31 23:00:00', tz='UTC')
assert pvgis_data.index[0] == jan1_midnight
assert pvgis_data.index[-1] == dec31_midnight
assert pvgis_data.index.name == 'time(UTC)'
for m, test_case in enumerate(noon_test_data):
expected = pvgis_data[pvgis_data.index.month == m+1].iloc[12]
assert all(test_case == expected)


@pytest.mark.remote_data
@pytest.mark.flaky(reruns=RERUNS, reruns_delay=RERUNS_DELAY)
def test_get_pvgis_tmy_csv(expected, month_year_expected, inputs_expected,
Expand Down
Loading