Skip to content

Commit

Permalink
Add missing DataArray.dt.total_seconds() method
Browse files Browse the repository at this point in the history
  • Loading branch information
maresb committed Nov 9, 2023
1 parent feba698 commit bc18b5b
Show file tree
Hide file tree
Showing 4 changed files with 29 additions and 0 deletions.
1 change: 1 addition & 0 deletions doc/api.rst
Original file line number Diff line number Diff line change
Expand Up @@ -557,6 +557,7 @@ Datetimelike properties
DataArray.dt.seconds
DataArray.dt.microseconds
DataArray.dt.nanoseconds
DataArray.dt.total_seconds

**Timedelta methods**:

Expand Down
2 changes: 2 additions & 0 deletions doc/whats-new.rst
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,8 @@ New Features

- Use `opt_einsum <https://optimized-einsum.readthedocs.io/en/stable/>`_ for :py:func:`xarray.dot` by default if installed.
By `Deepak Cherian <https://github.com/dcherian>`_. (:issue:`7764`, :pull:`8373`).
- Add ``DataArray.dt.total_seconds()`` method to match the Pandas API. (:pull:`8435`).
By `Ben Mares <https://github.com/maresb>`_.

Breaking changes
~~~~~~~~~~~~~~~~
Expand Down
14 changes: 14 additions & 0 deletions xarray/core/accessor_dt.py
Original file line number Diff line number Diff line change
Expand Up @@ -74,6 +74,8 @@ def _access_through_series(values, name):
if name == "season":
months = values_as_series.dt.month.values
field_values = _season_from_months(months)
elif name == "total_seconds":
field_values = values_as_series.dt.total_seconds().values
elif name == "isocalendar":
# special NaT-handling can be removed when
# https://github.com/pandas-dev/pandas/issues/54657 is resolved
Expand Down Expand Up @@ -574,6 +576,13 @@ class TimedeltaAccessor(TimeAccessor[T_DataArray]):
43200, 64800])
Coordinates:
* time (time) timedelta64[ns] 1 days 00:00:00 ... 5 days 18:00:00
>>> ts.dt.total_seconds()
<xarray.DataArray 'total_seconds' (time: 20)>
array([ 86400., 108000., 129600., 151200., 172800., 194400., 216000.,
237600., 259200., 280800., 302400., 324000., 345600., 367200.,
388800., 410400., 432000., 453600., 475200., 496800.])
Coordinates:
* time (time) timedelta64[ns] 1 days 00:00:00 ... 5 days 18:00:00
"""

@property
Expand All @@ -596,6 +605,11 @@ def nanoseconds(self) -> T_DataArray:
"""Number of nanoseconds (>= 0 and less than 1 microsecond) for each element"""
return self._date_field("nanoseconds", np.int64)

# Not defined as a property in order to match the Pandas API
def total_seconds(self) -> T_DataArray:
"""Total duration of each element expressed in seconds."""
return self._date_field("total_seconds", np.float64)


class CombinedDatetimelikeAccessor(
DatetimeAccessor[T_DataArray], TimedeltaAccessor[T_DataArray]
Expand Down
12 changes: 12 additions & 0 deletions xarray/tests/test_accessor_dt.py
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,18 @@ def test_field_access(self, field) -> None:
assert expected.dtype == actual.dtype
assert_identical(expected, actual)

def test_total_seconds(self) -> None:
# Subtract a value in the middle of the range to ensure that some values
# are negative
delta = self.data.time - np.datetime64("2000-01-03")
actual = delta.dt.total_seconds()
expected = xr.DataArray(
np.arange(-48, 52, dtype=np.float64) * 3600,
name="total_seconds",
coords=[self.data.time],
)
assert_identical(expected, actual)

Check failure on line 113 in xarray/tests/test_accessor_dt.py

View workflow job for this annotation

GitHub Actions / ubuntu-latest py3.9 bare-minimum

TestDatetimeAccessor.test_total_seconds AssertionError: Left and right DataArray objects are not identical Differing values: L array([-172800., -169200., -165600., -162000., -158400., -154800., -151200., -147600., -144000., -140400., -136800., -133200., -129600., -126000., -122400., -118800., -115200., -111600., -108000., -104400., -100800., -97200., -93600., -90000., -86400., -82800., -79200., -75600., -72000., -68400., -64800., -61200., -57600., -54000., -50400., -46800., -43200., -39600., -36000., -32400., -28800., -25200., -21600., -18000., -14400., -10800., -7200., -3600., 0., 3600., 7200., 10800., 14400., 18000., 21600., 25200., 28800., 32400., 36000., 39600., 43200., 46800., 50400., 54000., 57600., 61200., 64800., 68400., 72000., 75600., 79200., 82800., 86400., 90000., 93600., 97200., 100800., 104400., 108000., 111600., 115200., 118800., 122400., 126000., 129600., 133200., 136800., 140400., 144000., 147600., 151200., 154800., 158400., 162000., 165600., 169200., 172800., 176400., 180000., 183600.]) R array([-172800., -169200., -165600., -162000., -158400., -154800., -151200., -147600., -144000., -140400., -136800., -133200., -129600., -126000., -122400., -118800., -115200., -111600., -108000., -104400., -100800., -97200., -93600., -90000., -86400., -82800., -79200., -75600., -72000., -68400., -64800., -61200., -57600., -54000., -50400., -46800., -43200., -39600., -36000., -32400., -28800., -25200., -21600., -18000., -14400., -10800., -7200., -3600., 0., 3600., 7200., 10800., 14400., 18000., 21600., 25200., 28800., 32400., 36000., 39600., 43200., 46800., 50400., 54000., 57600., 61200., 64800., 68400., 72000., 75600., 79200., 82800., 86400., 90000., 93600., 97200., 100800., 104400., 108000., 111600., 115200., 118800., 122400., 126000., 129600., 133200., 136800., 140400., 144000., 147600., 151200., 154800., 158400., 162000., 165600., 169200., 172800., 176400., 180000., 183600.])

Check failure on line 113 in xarray/tests/test_accessor_dt.py

View workflow job for this annotation

GitHub Actions / ubuntu-latest py3.9 min-all-deps

TestDatetimeAccessor.test_total_seconds AssertionError: Left and right DataArray objects are not identical Differing values: L array([-172800., -169200., -165600., -162000., -158400., -154800., -151200., -147600., -144000., -140400., -136800., -133200., -129600., -126000., -122400., -118800., -115200., -111600., -108000., -104400., -100800., -97200., -93600., -90000., -86400., -82800., -79200., -75600., -72000., -68400., -64800., -61200., -57600., -54000., -50400., -46800., -43200., -39600., -36000., -32400., -28800., -25200., -21600., -18000., -14400., -10800., -7200., -3600., 0., 3600., 7200., 10800., 14400., 18000., 21600., 25200., 28800., 32400., 36000., 39600., 43200., 46800., 50400., 54000., 57600., 61200., 64800., 68400., 72000., 75600., 79200., 82800., 86400., 90000., 93600., 97200., 100800., 104400., 108000., 111600., 115200., 118800., 122400., 126000., 129600., 133200., 136800., 140400., 144000., 147600., 151200., 154800., 158400., 162000., 165600., 169200., 172800., 176400., 180000., 183600.]) R array([-172800., -169200., -165600., -162000., -158400., -154800., -151200., -147600., -144000., -140400., -136800., -133200., -129600., -126000., -122400., -118800., -115200., -111600., -108000., -104400., -100800., -97200., -93600., -90000., -86400., -82800., -79200., -75600., -72000., -68400., -64800., -61200., -57600., -54000., -50400., -46800., -43200., -39600., -36000., -32400., -28800., -25200., -21600., -18000., -14400., -10800., -7200., -3600., 0., 3600., 7200., 10800., 14400., 18000., 21600., 25200., 28800., 32400., 36000., 39600., 43200., 46800., 50400., 54000., 57600., 61200., 64800., 68400., 72000., 75600., 79200., 82800., 86400., 90000., 93600., 97200., 100800., 104400., 108000., 111600., 115200., 118800., 122400., 126000., 129600., 133200., 136800., 140400., 144000., 147600., 151200., 154800., 158400., 162000., 165600., 169200., 172800., 176400., 180000., 183600.])

@pytest.mark.parametrize(
"field, pandas_field",
[
Expand Down

0 comments on commit bc18b5b

Please sign in to comment.