Skip to content

Commit

Permalink
num2date: return datetimes by default if possible (#165)
Browse files Browse the repository at this point in the history
* make proper datetimes default

* replace cftime.utime.num2date with cftime.num2date

* attempt fixes

* pass both utime and Unit to _num2date_to_nearest_second

* revert _num2date_to_nearest_second calling sequence

* dummy commit to kick travis

* appease stickler

* license headers

* contain only_use_cftime_datetimes within utime object

* revert test__num2date_to_nearest_second.py

* actually pass flag to where it is needed

* reinstate type check

* Reinstate _num2date_to_nearest_second changes

This reverts commit 0b89777.

* reinstate test changes

* make sure option is passed down num2date
  • Loading branch information
rcomer authored May 11, 2021
1 parent 4cc94b1 commit 855077c
Show file tree
Hide file tree
Showing 3 changed files with 96 additions and 23 deletions.
73 changes: 56 additions & 17 deletions cf_units/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -484,7 +484,7 @@ def _discard_microsecond(date):
dates = np.asarray(date)
shape = dates.shape
dates = dates.ravel()
# Create date objects of the same type returned by utime.num2date()
# Create date objects of the same type returned by cftime.num2date()
# (either datetime.datetime or cftime.datetime), discarding the
# microseconds
dates = np.array([d and d.__class__(d.year, d.month, d.day,
Expand All @@ -494,7 +494,7 @@ def _discard_microsecond(date):
return result


def num2date(time_value, unit, calendar):
def num2date(time_value, unit, calendar, only_use_cftime_datetimes=False):
"""
Return datetime encoding of numeric time value (resolution of 1 second).
Expand All @@ -508,7 +508,7 @@ def num2date(time_value, unit, calendar):
unit = 'days since 001-01-01 00:00:00'}
calendar = 'proleptic_gregorian'.
The datetime instances returned are 'real' python datetime
By default, the datetime instances returned are 'real' python datetime
objects if the date falls in the Gregorian calendar (i.e.
calendar='proleptic_gregorian', or calendar = 'standard' or 'gregorian'
and the date is after 1582-10-15). Otherwise, they are 'phony' datetime
Expand All @@ -535,6 +535,13 @@ def num2date(time_value, unit, calendar):
* calendar (string):
Name of the calendar, see cf_units.CALENDARS.
Kwargs:
* only_use_cftime_datetimes (bool):
If True, will always return cftime datetime objects, regardless of
calendar. If False, returns datetime.datetime instances where
possible. Defaults to False.
Returns:
datetime, or numpy.ndarray of datetime object.
Expand All @@ -561,10 +568,12 @@ def num2date(time_value, unit, calendar):
if unit_string.endswith(" since epoch"):
unit_string = unit_string.replace("epoch", EPOCH)
unit_inst = Unit(unit_string, calendar=calendar)
return unit_inst.num2date(time_value)
return unit_inst.num2date(
time_value, only_use_cftime_datetimes=only_use_cftime_datetimes)


def _num2date_to_nearest_second(time_value, utime):
def _num2date_to_nearest_second(time_value, utime,
only_use_cftime_datetimes=False):
"""
Return datetime encoding of numeric time value with respect to the given
time reference units, with a resolution of 1 second.
Expand All @@ -574,6 +583,11 @@ def _num2date_to_nearest_second(time_value, utime):
* utime (cftime.utime):
cftime.utime object with which to perform the conversion/s.
* only_use_cftime_datetimes (bool):
If True, will always return cftime datetime objects, regardless of
calendar. If False, returns datetime.datetime instances where
possible. Defaults to False.
Returns:
datetime, or numpy.ndarray of datetime object.
"""
Expand All @@ -582,7 +596,7 @@ def _num2date_to_nearest_second(time_value, utime):
time_values = time_values.ravel()

# We account for the edge case where the time is in seconds and has a
# half second: utime.num2date() may produce a date that would round
# half second: cftime.num2date() may produce a date that would round
# down.
#
# Note that this behaviour is different to the num2date function in version
Expand All @@ -592,7 +606,9 @@ def _num2date_to_nearest_second(time_value, utime):
# later versions, if one wished to do so for the sake of consistency.
has_half_seconds = np.logical_and(utime.units == 'seconds',
time_values % 1. == 0.5)
dates = utime.num2date(time_values)
dates = cftime.num2date(
time_values, utime.unit_string, calendar=utime.calendar,
only_use_cftime_datetimes=only_use_cftime_datetimes)
try:
# We can assume all or none of the dates have a microsecond attribute
microseconds = np.array([d.microsecond if d else 0 for d in dates])
Expand All @@ -603,7 +619,10 @@ def _num2date_to_nearest_second(time_value, utime):
if time_values[ceil_mask].size > 0:
useconds = Unit('second')
second_frac = useconds.convert(0.75, utime.units)
dates[ceil_mask] = utime.num2date(time_values[ceil_mask] + second_frac)
dates[ceil_mask] = cftime.num2date(
time_values[ceil_mask] + second_frac, utime.unit_string,
calendar=utime.calendar,
only_use_cftime_datetimes=only_use_cftime_datetimes)
dates[round_mask] = _discard_microsecond(dates[round_mask])
result = dates[0] if shape is () else dates.reshape(shape)
return result
Expand Down Expand Up @@ -1878,7 +1897,7 @@ def convert(self, value, other, ctype=FLOAT64, inplace=False):
raise ValueError("Unable to convert from '%r' to '%r'." %
(self, other))

def utime(self):
def utime(self, only_use_cftime_datetimes=False):
"""
Returns a cftime.utime object which performs conversions of
numeric time values to/from datetime objects given the current
Expand All @@ -1888,6 +1907,13 @@ def utime(self):
'<time-unit> since <time-origin>'
i.e. 'hours since 1970-01-01 00:00:00'
Kwargs:
* only_use_cftime_datetimes (bool):
If True, num2date method will always return cftime datetime
objects, regardless of calendar. If False, returns
datetime.datetime instances where possible. Defaults to False.
Returns:
cftime.utime.
Expand Down Expand Up @@ -1915,7 +1941,9 @@ def utime(self):
# ensure to strip out non-parsable 'UTC' postfix, which
# is generated by UDUNITS-2 formatted output
#
return cftime.utime(str(self).rstrip(" UTC"), self.calendar)
return cftime.utime(
str(self).rstrip(" UTC"), self.calendar,
only_use_cftime_datetimes=only_use_cftime_datetimes)

def date2num(self, date):
"""
Expand Down Expand Up @@ -1956,7 +1984,7 @@ def date2num(self, date):
date = _discard_microsecond(date)
return cdf_utime.date2num(date)

def num2date(self, time_value):
def num2date(self, time_value, only_use_cftime_datetimes=False):
"""
Returns a datetime-like object calculated from the numeric time
value using the current calendar and the unit time reference.
Expand All @@ -1965,8 +1993,8 @@ def num2date(self, time_value):
'<time-unit> since <time-origin>'
i.e. 'hours since 1970-01-01 00:00:00'
The datetime objects returned are 'real' Python datetime objects
if the date falls in the Gregorian calendar (i.e. the calendar
By default, the datetime objects returned are 'real' Python datetime
objects if the date falls in the Gregorian calendar (i.e. the calendar
is 'standard', 'gregorian', or 'proleptic_gregorian' and the
date is after 1582-10-15). Otherwise a 'phoney' datetime-like
object (cftime.datetime) is returned which can handle dates
Expand All @@ -1977,8 +2005,15 @@ def num2date(self, time_value):
Args:
* time_value (float): Numeric time value/s. Maximum resolution
is 1 second.
* time_value (float):
Numeric time value/s. Maximum resolution is 1 second.
Kwargs:
* only_use_cftime_datetimes (bool):
If True, will always return cftime datetime objects, regardless of
calendar. If False, returns datetime.datetime instances where
possible. Defaults to False.
Returns:
datetime, or numpy.ndarray of datetime object.
Expand All @@ -1995,5 +2030,9 @@ def num2date(self, time_value):
['1970-01-01 06:00:00', '1970-01-01 07:00:00']
"""
cdf_utime = self.utime()
return _num2date_to_nearest_second(time_value, cdf_utime)
cdf_utime = self.utime(
only_use_cftime_datetimes=only_use_cftime_datetimes)

return _num2date_to_nearest_second(
time_value, cdf_utime,
only_use_cftime_datetimes=only_use_cftime_datetimes)
30 changes: 27 additions & 3 deletions cf_units/tests/integration/test__num2date_to_nearest_second.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
# (C) British Crown Copyright 2016 - 2020, Met Office
# (C) British Crown Copyright 2016 - 2021, Met Office
#
# This file is part of cf-units.
#
Expand Down Expand Up @@ -32,10 +32,12 @@ def setup_units(self, calendar):
self.uhours = cftime.utime('hours since 1970-01-01', calendar)
self.udays = cftime.utime('days since 1970-01-01', calendar)

def check_dates(self, nums, utimes, expected):
def check_dates(self, nums, utimes, expected, only_cftime=False):
for num, utime, exp in zip(nums, utimes, expected):
res = _num2date_to_nearest_second(num, utime)
res = _num2date_to_nearest_second(
num, utime, only_use_cftime_datetimes=only_cftime)
self.assertEqual(exp, res)
self.assertIsInstance(res, type(exp))

def check_timedelta(self, nums, utimes, expected):
for num, utime, exp in zip(nums, utimes, expected):
Expand All @@ -51,6 +53,7 @@ def test_scalar(self):
exp = datetime.datetime(1970, 1, 1, 0, 0, 5)
res = _num2date_to_nearest_second(num, utime)
self.assertEqual(exp, res)
self.assertIsInstance(res, datetime.datetime)

def test_sequence(self):
utime = cftime.utime('seconds since 1970-01-01', 'gregorian')
Expand Down Expand Up @@ -103,6 +106,27 @@ def test_simple_gregorian(self):

self.check_dates(nums, utimes, expected)

def test_simple_gregorian_cftime_type(self):
self.setup_units('gregorian')
nums = [20., 40.,
75., 150.,
8., 16.,
300., 600.]
utimes = [self.useconds, self.useconds,
self.uminutes, self.uminutes,
self.uhours, self.uhours,
self.udays, self.udays]
expected = [cftime.DatetimeGregorian(1970, 1, 1, 0, 0, 20),
cftime.DatetimeGregorian(1970, 1, 1, 0, 0, 40),
cftime.DatetimeGregorian(1970, 1, 1, 1, 15),
cftime.DatetimeGregorian(1970, 1, 1, 2, 30),
cftime.DatetimeGregorian(1970, 1, 1, 8),
cftime.DatetimeGregorian(1970, 1, 1, 16),
cftime.DatetimeGregorian(1970, 10, 28),
cftime.DatetimeGregorian(1971, 8, 24)]

self.check_dates(nums, utimes, expected, only_cftime=True)

def test_fractional_gregorian(self):
self.setup_units('gregorian')
nums = [5./60., 10./60.,
Expand Down
16 changes: 13 additions & 3 deletions cf_units/tests/test_unit.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
# -*- coding: utf-8 -*-
# (C) British Crown Copyright 2010 - 2020, Met Office
# (C) British Crown Copyright 2010 - 2021, Met Office
#
# This file is part of cf-units.
#
Expand Down Expand Up @@ -29,6 +29,7 @@
from operator import truediv

import numpy as np
import cftime

import cf_units as unit
from cf_units import suppress_errors
Expand Down Expand Up @@ -262,7 +263,7 @@ def test_add_float_offset(self):
def test_not_numerical_offset(self):
u = Unit('meter')
with self.assertRaisesRegex(TypeError,
'unsupported operand type'):
'unsupported operand type'):
operator.add(u, 'not_a_number')

def test_unit_unknown(self):
Expand Down Expand Up @@ -986,7 +987,16 @@ class TestNumsAndDates(unittest.TestCase):
def test_num2date(self):
u = Unit('hours since 2010-11-02 12:00:00',
calendar=unit.CALENDAR_STANDARD)
self.assertEqual(str(u.num2date(1)), '2010-11-02 13:00:00')
res = u.num2date(1)
self.assertEqual(str(res), '2010-11-02 13:00:00')
self.assertIsInstance(res, datetime.datetime)

def test_num2date_cftime_type(self):
u = Unit('hours since 2010-11-02 12:00:00',
calendar=unit.CALENDAR_STANDARD)
res = u.num2date(1, only_use_cftime_datetimes=True)
self.assertEqual(str(res), '2010-11-02 13:00:00')
self.assertIsInstance(res, cftime.DatetimeGregorian)

def test_date2num(self):
u = Unit('hours since 2010-11-02 12:00:00',
Expand Down

0 comments on commit 855077c

Please sign in to comment.