Skip to content

Commit 72ced3c

Browse files
Add tests for time conversions in tools package (#2341)
* Add tests for tools.localize_to_utc * Add tests for datetime_to_djd and djd_to_datetime * Update what's new * Appease the linter * Fix pandas equality tests for Python 3.9 * Fix pandas equality tests for Python 3.9 more * Fix pandas equality tests for Python 3.9 more more * Bump miniimum pandas to fix bad test failure * Try alternative pandas test fix * Revert change in minimum pandas version * Fix test * Type Location's tz and pytz attributes as advertised * Add timezone type checks to Location init test * Don't parameterize repetitive tests * Update whatsnew for Location bugfix * Update docstring * Improve whatsnew formatting * Support non-fractional int and float and pytz and zoneinfo time zones * Appease the linter * Use zoneinfo as single source of truth and tz as interface point * Add zoneinfo asserts in tests * Try to fix asv ci * See if newer asv works with newer conda * Remove comments no longer needed * Remove addition of zoneinfo attribute * Revise whatsnew bugfix * Revise whatsnew bugfix more * Spell my name correctly * The linter strikes back again * Fix whatsnew after main merge * Address Cliff's comment * Adjust Location documentation * Fix indent * More docstring tweaks * Try to fix bad parens * Rearrange docstring * Appease the linter * Document pytz attribute as read only * Consistent read only * Update pvlib/location.py per review comment Co-authored-by: Cliff Hansen <[email protected]> * Add breaking change to whatsnew and fix linting * Clarify breaking change in whatsnew * Update whatsnew ordering * Implement review comments on documentation * Missed saving changes and appease the linter * Apply suggestions from code review --------- Co-authored-by: Cliff Hansen <[email protected]>
1 parent afc90f6 commit 72ced3c

File tree

5 files changed

+259
-56
lines changed

5 files changed

+259
-56
lines changed

docs/sphinx/source/whatsnew/v0.11.3.rst

+14-1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,11 @@ Bug fixes
1313
* :py:class:`~pvlib.modelchain.ModelChain` now requires only a minimal set of
1414
parameters to run the SAPM electrical model. (:issue:`2369`, :pull:`2393`)
1515
* Correct keys for First Solar modules in `~pvlib.spectrum.spectral_factor_pvspec` (:issue:`2398`, :pull:`2400`)
16+
* Ensure proper tz and pytz types in pvlib.location.Location. To ensure that
17+
the time zone in pvlib.location.Location remains internally consistent
18+
if/when the time zone is updated, the tz attribute is now the single source
19+
of time-zone truth, is the single time-zone setter interface, and its getter
20+
returns an IANA string. (:issue:`2340`, :pull:`2341`)
1621

1722

1823
Deprecations
@@ -40,6 +45,9 @@ Testing
4045
* Moved tests folder to `/tests` and data exclusively used for testing to `/tests/data`.
4146
(:issue:`2271`, :pull:`2277`)
4247
* Added Python 3.13 to test suite. (:pull:`2258`)
48+
* Add tests for all input types for the pvlib.location.Location.tz attribute.
49+
(:issue:`2340`, :pull:`2341`)
50+
* Add tests for time-conversion functions in pvlib.tools. (:issue:`2340`, :pull:`2341`)
4351

4452

4553
Requirements
@@ -53,15 +61,20 @@ Maintenance
5361
* asv 0.4.2 upgraded to asv 0.6.4 to fix CI failure due to pinned older conda.
5462
(:pull:`2352`)
5563

64+
Breaking Changes
65+
~~~~~~~~~~~~~~~~
66+
* The pvlib.location.Location.pytz attribute is now read only. The
67+
pytz attribute is now set internally to be consistent with the
68+
pvlib.location.Location.tz attribute. (:issue:`2340`, :pull:`2341`)
5669

5770
Contributors
5871
~~~~~~~~~~~~
5972
* Rajiv Daxini (:ghuser:`RDaxini`)
60-
* Mark Campanelli (:ghuser:`markcampanelli`)
6173
* Cliff Hansen (:ghuser:`cwhanse`)
6274
* Jason Lun Leung (:ghuser:`jason-rpkt`)
6375
* Manoj K S (:ghuser:`manojks1999`)
6476
* Kurt Rhee (:ghuser:`kurt-rhee`)
6577
* Ayush jariyal (:ghuser:`ayushjariyal`)
6678
* Kevin Anderson (:ghuser:`kandersolar`)
6779
* Echedey Luis (:ghuser:`echedey-ls`)
80+
* Mark Campanelli (:ghuser:`markcampanelli`)

pvlib/location.py

+71-31
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66

77
import pathlib
88
import datetime
9+
import zoneinfo
910

1011
import pandas as pd
1112
import pytz
@@ -18,13 +19,16 @@
1819
class Location:
1920
"""
2021
Location objects are convenient containers for latitude, longitude,
21-
timezone, and altitude data associated with a particular
22-
geographic location. You can also assign a name to a location object.
22+
time zone, and altitude data associated with a particular geographic
23+
location. You can also assign a name to a location object.
2324
24-
Location objects have two timezone attributes:
25+
Location objects have two time-zone attributes:
2526
26-
* ``tz`` is a IANA timezone string.
27-
* ``pytz`` is a pytz timezone object.
27+
* ``tz`` is an IANA time-zone string.
28+
* ``pytz`` is a pytz-based time-zone object (read only).
29+
30+
The read-only ``pytz`` attribute will stay in sync with any changes made
31+
using ``tz``.
2832
2933
Location objects support the print method.
3034
@@ -38,12 +42,16 @@ class Location:
3842
Positive is east of the prime meridian.
3943
Use decimal degrees notation.
4044
41-
tz : str, int, float, or pytz.timezone, default 'UTC'.
42-
See
43-
http://en.wikipedia.org/wiki/List_of_tz_database_time_zones
44-
for a list of valid time zones.
45-
pytz.timezone objects will be converted to strings.
46-
ints and floats must be in hours from UTC.
45+
tz : time zone as str, int, float, or datetime.tzinfo, default 'UTC'.
46+
See http://en.wikipedia.org/wiki/List_of_tz_database_time_zones for a
47+
list of valid name strings. An `int` or `float` must be a whole-number
48+
hour offsets from UTC that can be converted to the IANA-supported
49+
'Etc/GMT-N' format. (Note the limited range of the offset N and its
50+
sign-change convention.) Time zones from the pytz and zoneinfo packages
51+
may also be passed here, as they are subclasses of datetime.tzinfo.
52+
53+
The `tz` attribute is represented as a valid IANA time zone name
54+
string.
4755
4856
altitude : float, optional
4957
Altitude from sea level in meters.
@@ -54,43 +62,75 @@ class Location:
5462
name : string, optional
5563
Sets the name attribute of the Location object.
5664
65+
Raises
66+
------
67+
ValueError
68+
when the time zone ``tz`` cannot be converted.
69+
70+
zoneinfo.ZoneInfoNotFoundError
71+
when the time zone ``tz`` is not recognizable as an IANA time zone by
72+
the ``zoneinfo.ZoneInfo`` initializer used for internal time-zone
73+
representation.
74+
5775
See also
5876
--------
5977
pvlib.pvsystem.PVSystem
6078
"""
6179

62-
def __init__(self, latitude, longitude, tz='UTC', altitude=None,
63-
name=None):
64-
80+
def __init__(
81+
self, latitude, longitude, tz='UTC', altitude=None, name=None
82+
):
6583
self.latitude = latitude
6684
self.longitude = longitude
67-
68-
if isinstance(tz, str):
69-
self.tz = tz
70-
self.pytz = pytz.timezone(tz)
71-
elif isinstance(tz, datetime.timezone):
72-
self.tz = 'UTC'
73-
self.pytz = pytz.UTC
74-
elif isinstance(tz, datetime.tzinfo):
75-
self.tz = tz.zone
76-
self.pytz = tz
77-
elif isinstance(tz, (int, float)):
78-
self.tz = tz
79-
self.pytz = pytz.FixedOffset(tz*60)
80-
else:
81-
raise TypeError('Invalid tz specification')
85+
self.tz = tz
8286

8387
if altitude is None:
8488
altitude = lookup_altitude(latitude, longitude)
8589

8690
self.altitude = altitude
87-
8891
self.name = name
8992

9093
def __repr__(self):
9194
attrs = ['name', 'latitude', 'longitude', 'altitude', 'tz']
95+
# Use None as getattr default in case __repr__ is called during
96+
# initialization before all attributes have been assigned.
9297
return ('Location: \n ' + '\n '.join(
93-
f'{attr}: {getattr(self, attr)}' for attr in attrs))
98+
f'{attr}: {getattr(self, attr, None)}' for attr in attrs))
99+
100+
@property
101+
def tz(self):
102+
"""The location's IANA time-zone string."""
103+
return str(self._zoneinfo)
104+
105+
@tz.setter
106+
def tz(self, tz_):
107+
# self._zoneinfo holds single source of time-zone truth as IANA name.
108+
if isinstance(tz_, str):
109+
self._zoneinfo = zoneinfo.ZoneInfo(tz_)
110+
elif isinstance(tz_, int):
111+
self._zoneinfo = zoneinfo.ZoneInfo(f"Etc/GMT{-tz_:+d}")
112+
elif isinstance(tz_, float):
113+
if tz_ % 1 != 0:
114+
raise TypeError(
115+
"Floating-point tz has non-zero fractional part: "
116+
f"{tz_}. Only whole-number offsets are supported."
117+
)
118+
119+
self._zoneinfo = zoneinfo.ZoneInfo(f"Etc/GMT{-int(tz_):+d}")
120+
elif isinstance(tz_, datetime.tzinfo):
121+
# Includes time zones generated by pytz and zoneinfo packages.
122+
self._zoneinfo = zoneinfo.ZoneInfo(str(tz_))
123+
else:
124+
raise TypeError(
125+
f"invalid tz specification: {tz_}, must be an IANA time zone "
126+
"string, a whole-number int/float UTC offset, or a "
127+
"datetime.tzinfo object (including subclasses)"
128+
)
129+
130+
@property
131+
def pytz(self):
132+
"""The location's pytz time zone (read only)."""
133+
return pytz.timezone(str(self._zoneinfo))
94134

95135
@classmethod
96136
def from_tmy(cls, tmy_metadata, tmy_data=None, **kwargs):

pvlib/tools.py

+6-5
Original file line numberDiff line numberDiff line change
@@ -3,10 +3,11 @@
33
"""
44

55
import datetime as dt
6+
import warnings
7+
68
import numpy as np
79
import pandas as pd
810
import pytz
9-
import warnings
1011

1112

1213
def cosd(angle):
@@ -119,21 +120,21 @@ def atand(number):
119120

120121
def localize_to_utc(time, location):
121122
"""
122-
Converts or localizes a time series to UTC.
123+
Converts ``time`` to UTC, localizing if necessary using location.
123124
124125
Parameters
125126
----------
126127
time : datetime.datetime, pandas.DatetimeIndex,
127128
or pandas.Series/DataFrame with a DatetimeIndex.
128-
location : pvlib.Location object
129+
location : pvlib.Location object (unused if ``time`` is localized)
129130
130131
Returns
131132
-------
132-
pandas object localized to UTC.
133+
datetime.datetime or pandas object localized to UTC.
133134
"""
134135
if isinstance(time, dt.datetime):
135136
if time.tzinfo is None:
136-
time = pytz.timezone(location.tz).localize(time)
137+
time = location.pytz.localize(time)
137138
time_utc = time.astimezone(pytz.utc)
138139
else:
139140
try:

tests/test_location.py

+57-16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import datetime
22
from unittest.mock import ANY
3+
import zoneinfo
34

45
import numpy as np
56
from numpy import nan
@@ -9,7 +10,6 @@
910
import pytest
1011

1112
import pytz
12-
from pytz.exceptions import UnknownTimeZoneError
1313

1414
import pvlib
1515
from pvlib import location
@@ -27,22 +27,63 @@ def test_location_all():
2727
Location(32.2, -111, 'US/Arizona', 700, 'Tucson')
2828

2929

30-
@pytest.mark.parametrize('tz', [
31-
pytz.timezone('US/Arizona'), 'America/Phoenix', -7, -7.0,
32-
datetime.timezone.utc
33-
])
34-
def test_location_tz(tz):
35-
Location(32.2, -111, tz)
36-
37-
38-
def test_location_invalid_tz():
39-
with pytest.raises(UnknownTimeZoneError):
40-
Location(32.2, -111, 'invalid')
41-
42-
43-
def test_location_invalid_tz_type():
30+
@pytest.mark.parametrize(
31+
'tz,tz_expected', [
32+
pytest.param('UTC', 'UTC'),
33+
pytest.param('Etc/GMT+5', 'Etc/GMT+5'),
34+
pytest.param('US/Mountain', 'US/Mountain'),
35+
pytest.param('America/Phoenix', 'America/Phoenix'),
36+
pytest.param('Asia/Kathmandu', 'Asia/Kathmandu'),
37+
pytest.param('Asia/Yangon', 'Asia/Yangon'),
38+
pytest.param(datetime.timezone.utc, 'UTC'),
39+
pytest.param(zoneinfo.ZoneInfo('Etc/GMT-7'), 'Etc/GMT-7'),
40+
pytest.param(pytz.timezone('US/Arizona'), 'US/Arizona'),
41+
pytest.param(-6, 'Etc/GMT+6'),
42+
pytest.param(-11.0, 'Etc/GMT+11'),
43+
pytest.param(12, 'Etc/GMT-12'),
44+
],
45+
)
46+
def test_location_tz(tz, tz_expected):
47+
loc = Location(32.2, -111, tz)
48+
assert isinstance(loc.pytz, datetime.tzinfo) # Abstract base class.
49+
assert isinstance(loc.pytz, pytz.tzinfo.BaseTzInfo)
50+
assert type(loc.tz) is str
51+
assert loc.tz == tz_expected
52+
53+
54+
def test_location_tz_update():
55+
loc = Location(32.2, -111, -11)
56+
assert loc.tz == 'Etc/GMT+11'
57+
assert loc.pytz == pytz.timezone('Etc/GMT+11') # Deprecated attribute.
58+
59+
# Updating Location's tz updates read-only time-zone attributes.
60+
loc.tz = 7
61+
assert loc.tz == 'Etc/GMT-7'
62+
assert loc.pytz == pytz.timezone('Etc/GMT-7') # Deprecated attribute.
63+
64+
65+
@pytest.mark.parametrize(
66+
'tz', [
67+
'invalid',
68+
'Etc/GMT+20', # offset too large.
69+
20, # offset too large.
70+
]
71+
)
72+
def test_location_invalid_tz(tz):
73+
with pytest.raises(zoneinfo.ZoneInfoNotFoundError):
74+
Location(32.2, -111, tz)
75+
76+
77+
@pytest.mark.parametrize(
78+
'tz', [
79+
-9.5, # float with non-zero fractional part.
80+
b"bytes not str",
81+
[5],
82+
]
83+
)
84+
def test_location_invalid_tz_type(tz):
4485
with pytest.raises(TypeError):
45-
Location(32.2, -111, [5])
86+
Location(32.2, -111, tz)
4687

4788

4889
def test_location_print_all():

0 commit comments

Comments
 (0)