Repository URL to install this package:
|
Version:
0.36.2 ▾
|
"""
Test np.datetime64 and np.timedelta64 support.
"""
# NOTE: datetime64 and timedelta64 ufuncs are tested in test_ufuncs.
from __future__ import print_function
import contextlib
import itertools
import warnings
import numpy as np
import numba.unittest_support as unittest
from numba import config, jit, npdatetime, types, vectorize, numpy_support
from numba.errors import TypingError
from .support import TestCase, tag
def value_unit(val):
ty = numpy_support.from_dtype(val.dtype)
return ty.unit
date_units = ('Y', 'M')
time_units = ('W', 'D', 'h', 'm', 's', 'ms', 'us', 'ns', 'ps', 'fs', 'as')
# All except generic ("")
all_units = date_units + time_units
def add_usecase(x, y):
return x + y
def sub_usecase(x, y):
return x - y
def mul_usecase(x, y):
return x * y
def div_usecase(x, y):
return x / y
def floordiv_usecase(x, y):
return x // y
def eq_usecase(x, y):
return x == y
def ne_usecase(x, y):
return x != y
def lt_usecase(x, y):
return x < y
def le_usecase(x, y):
return x <= y
def gt_usecase(x, y):
return x > y
def ge_usecase(x, y):
return x >= y
def pos_usecase(x):
return +x
def neg_usecase(x):
return -x
def abs_usecase(x):
return abs(x)
def make_add_constant(const):
def add_constant(x):
return x + const
return add_constant
class TestModuleHelpers(TestCase):
"""
Test the various helpers in numba.npdatetime.
"""
def test_can_cast_timedelta(self):
f = npdatetime.can_cast_timedelta_units
for a, b in itertools.product(date_units, time_units):
self.assertFalse(f(a, b), (a, b))
self.assertFalse(f(b, a), (a, b))
for unit in all_units:
self.assertFalse(f(unit, ''))
self.assertTrue(f('', unit))
for unit in all_units + ('',):
self.assertTrue(f(unit, unit))
def check_units_group(group):
for i, a in enumerate(group):
for b in group[:i]:
# large into smaller is ok
self.assertTrue(f(b, a))
# small into larger is not
self.assertFalse(f(a, b))
check_units_group(date_units)
check_units_group(time_units)
def test_timedelta_conversion(self):
f = npdatetime.get_timedelta_conversion_factor
for unit in all_units + ('',):
self.assertEqual(f(unit, unit), 1)
for unit in all_units:
self.assertEqual(f('', unit), 1)
for a, b in itertools.product(time_units, date_units):
self.assertIs(f(a, b), None)
self.assertIs(f(b, a), None)
def check_units_group(group):
for i, a in enumerate(group):
for b in group[:i]:
self.assertGreater(f(b, a), 1, (b, a))
self.assertIs(f(a, b), None)
check_units_group(date_units)
check_units_group(time_units)
# Check some hand-picked values
self.assertEqual(f('Y', 'M'), 12)
self.assertEqual(f('W', 'h'), 24 * 7)
self.assertEqual(f('W', 'm'), 24 * 7 * 60)
self.assertEqual(f('W', 'us'), 24 * 7 * 3600 * 1000 * 1000)
def test_datetime_timedelta_scaling(self):
f = npdatetime.get_datetime_timedelta_conversion
def check_error(dt_unit, td_unit):
with self.assertRaises(RuntimeError):
f(dt_unit, td_unit)
# Cannot combine a Y or M timedelta64 with a finer-grained datetime64
for dt_unit, td_unit in itertools.product(time_units, date_units):
check_error(dt_unit, td_unit)
# Sanity check that all other unit pairs can be converted, we'll
# check individual results below
for dt_unit, td_unit in itertools.product(time_units, time_units):
f(dt_unit, td_unit)
for dt_unit, td_unit in itertools.product(date_units, time_units):
f(dt_unit, td_unit)
for dt_unit, td_unit in itertools.product(date_units, date_units):
f(dt_unit, td_unit)
# No-op conversions
for unit in all_units:
self.assertEqual(f(unit, unit), (unit, 1, 1))
self.assertEqual(f(unit, ''), (unit, 1, 1))
self.assertEqual(f('', unit), ('', 1, 1))
self.assertEqual(f('', ''), ('', 1, 1))
# "Regular" values
self.assertEqual(f('Y', 'M'), ('M', 12, 1))
self.assertEqual(f('M', 'Y'), ('M', 1, 12))
self.assertEqual(f('W', 'D'), ('D', 7, 1))
self.assertEqual(f('D', 'W'), ('D', 1, 7))
self.assertEqual(f('W', 's'), ('s', 7 * 24 * 3600, 1))
self.assertEqual(f('s', 'W'), ('s', 1, 7 * 24 * 3600))
self.assertEqual(f('s', 'as'), ('as', 1000 ** 6, 1))
self.assertEqual(f('as', 's'), ('as', 1, 1000 ** 6))
# "Interesting" values
self.assertEqual(f('Y', 'D'), ('D', 97 + 400 * 365, 400))
self.assertEqual(f('Y', 'W'), ('W', 97 + 400 * 365, 400 * 7))
self.assertEqual(f('M', 'D'), ('D', 97 + 400 * 365, 400 * 12))
self.assertEqual(f('M', 'W'), ('W', 97 + 400 * 365, 400 * 12 * 7))
self.assertEqual(f('Y', 's'), ('s', (97 + 400 * 365) * 24 * 3600, 400))
self.assertEqual(f('M', 's'), ('s', (97 + 400 * 365) * 24 * 3600, 400 * 12))
def test_combine_datetime_timedelta_units(self):
f = npdatetime.combine_datetime_timedelta_units
for unit in all_units:
self.assertEqual(f(unit, unit), unit)
self.assertEqual(f('', unit), unit)
self.assertEqual(f(unit, ''), unit)
self.assertEqual(f('', ''), '')
for dt_unit, td_unit in itertools.product(time_units, date_units):
self.assertIs(f(dt_unit, td_unit), None)
for dt_unit, td_unit in itertools.product(date_units, time_units):
self.assertEqual(f(dt_unit, td_unit), td_unit)
def test_same_kind(self):
f = npdatetime.same_kind
for u in all_units:
self.assertTrue(f(u, u))
A = ('Y', 'M', 'W', 'D')
B = ('h', 'm', 's', 'ms', 'us', 'ns', 'ps', 'fs', 'as')
for a, b in itertools.product(A, A):
self.assertTrue(f(a, b))
for a, b in itertools.product(B, B):
self.assertTrue(f(a, b))
for a, b in itertools.product(A, B):
self.assertFalse(f(a, b))
self.assertFalse(f(b, a))
TD = np.timedelta64
DT = np.datetime64
class TestMiscCompiling(TestCase):
def test_jit_explicit_signature(self):
def _check_explicit_signature(sig):
f = jit(sig, nopython=True)(add_usecase)
# Just a sanity check
args = DT(1, 'ms'), TD(2, 'us')
expected = add_usecase(*args)
self.assertPreciseEqual(f(*args), expected)
# Test passing the signature in object form
sig = types.NPDatetime('us')(types.NPDatetime('ms'), types.NPTimedelta('us'))
_check_explicit_signature(sig)
# Same with the signature in string form
sig = "NPDatetime('us')(NPDatetime('ms'), NPTimedelta('us'))"
_check_explicit_signature(sig)
def test_vectorize_explicit_signature(self):
def _check_explicit_signature(sig):
f = vectorize([sig], nopython=True)(mul_usecase)
# This isn't really right but we can't do better than this,
# since Numpy's ufuncs don't store the metadata of return types.
# Related to https://github.com/numpy/numpy/issues/5429
self.assertPreciseEqual(f(TD(2), 3), TD(6))
# Test passing the signature in object form (issue #917)
sig = types.NPTimedelta('s')(types.NPTimedelta('s'), types.int64)
_check_explicit_signature(sig)
# Same with the signature in string form
sig = "NPTimedelta('s')(NPTimedelta('s'), int64)"
_check_explicit_signature(sig)
def test_constant_datetime(self):
def check(const):
pyfunc = make_add_constant(const)
f = jit(nopython=True)(pyfunc)
x = TD(4, 'D')
expected = pyfunc(x)
self.assertPreciseEqual(f(x), expected)
check(DT('2001-01-01'))
check(DT('NaT', 'D'))
def test_constant_timedelta(self):
def check(const):
pyfunc = make_add_constant(const)
f = jit(nopython=True)(pyfunc)
x = TD(4, 'D')
expected = pyfunc(x)
self.assertPreciseEqual(f(x), expected)
check(TD(4, 'D'))
check(TD(-4, 'D'))
check(TD('NaT', 'D'))
class TestTimedeltaArithmetic(TestCase):
jitargs = dict(forceobj=True)
def jit(self, pyfunc):
return jit(**self.jitargs)(pyfunc)
@tag('important')
def test_add(self):
f = self.jit(add_usecase)
def check(a, b, expected):
self.assertPreciseEqual(f(a, b), expected)
self.assertPreciseEqual(f(b, a), expected)
check(TD(1), TD(2), TD(3))
check(TD(1, 's'), TD(2, 's'), TD(3, 's'))
# Implicit unit promotion
if not numpy_support.strict_ufunc_typing:
check(TD(1), TD(2, 's'), TD(3, 's'))
check(TD(1), TD(2, 'ms'), TD(3, 'ms'))
check(TD(1, 's'), TD(2, 'us'), TD(1000002, 'us'))
check(TD(1, 'W'), TD(2, 'D'), TD(9, 'D'))
# NaTs
check(TD('NaT'), TD(1), TD('NaT'))
check(TD('NaT', 's'), TD(1, 'D'), TD('NaT', 's'))
check(TD('NaT', 's'), TD(1, 'ms'), TD('NaT', 'ms'))
# Cannot add days and months
with self.assertRaises((TypeError, TypingError)):
f(TD(1, 'M'), TD(1, 'D'))
@tag('important')
def test_sub(self):
f = self.jit(sub_usecase)
def check(a, b, expected):
self.assertPreciseEqual(f(a, b), expected)
self.assertPreciseEqual(f(b, a), -expected)
check(TD(3), TD(2), TD(1))
check(TD(3, 's'), TD(2, 's'), TD(1, 's'))
# Implicit unit promotion
if not numpy_support.strict_ufunc_typing:
check(TD(3), TD(2, 's'), TD(1, 's'))
check(TD(3), TD(2, 'ms'), TD(1, 'ms'))
check(TD(3, 's'), TD(2, 'us'), TD(2999998, 'us'))
check(TD(1, 'W'), TD(2, 'D'), TD(5, 'D'))
# NaTs
check(TD('NaT'), TD(1), TD('NaT'))
check(TD('NaT', 's'), TD(1, 'D'), TD('NaT', 's'))
check(TD('NaT', 's'), TD(1, 'ms'), TD('NaT', 'ms'))
# Cannot sub days to months
with self.assertRaises((TypeError, TypingError)):
f(TD(1, 'M'), TD(1, 'D'))
def test_mul(self):
f = self.jit(mul_usecase)
def check(a, b, expected):
self.assertPreciseEqual(f(a, b), expected)
self.assertPreciseEqual(f(b, a), expected)
# non-int64 int * timedelta64
check(TD(3), np.uint32(2), TD(6))
# int * timedelta64
check(TD(3), 2, TD(6))
check(TD(3, 'ps'), 2, TD(6, 'ps'))
check(TD('NaT', 'ps'), 2, TD('NaT', 'ps'))
# float * timedelta64
check(TD(7), 1.5, TD(10))
check(TD(-7), 1.5, TD(-10))
check(TD(7, 'ps'), -1.5, TD(-10, 'ps'))
check(TD(-7), -1.5, TD(10))
check(TD('NaT', 'ps'), -1.5, TD('NaT', 'ps'))
check(TD(7, 'ps'), float('nan'), TD('NaT', 'ps'))
# wraparound on overflow
check(TD(2**62, 'ps'), 16, TD(0, 'ps'))
def test_div(self):
div = self.jit(div_usecase)
floordiv = self.jit(floordiv_usecase)
def check(a, b, expected):
self.assertPreciseEqual(div(a, b), expected)
self.assertPreciseEqual(floordiv(a, b), expected)
# timedelta64 / non-int64 int
check(TD(-3, 'ps'), np.uint32(2), TD(-1, 'ps'))
# timedelta64 / int
check(TD(3), 2, TD(1))
check(TD(-3, 'ps'), 2, TD(-1, 'ps'))
check(TD('NaT', 'ps'), 2, TD('NaT', 'ps'))
check(TD(3, 'ps'), 0, TD('NaT', 'ps'))
check(TD('NaT', 'ps'), 0, TD('NaT', 'ps'))
# timedelta64 / float
check(TD(7), 0.5, TD(14))
check(TD(-7, 'ps'), 1.5, TD(-4, 'ps'))
check(TD('NaT', 'ps'), 2.5, TD('NaT', 'ps'))
check(TD(3, 'ps'), 0.0, TD('NaT', 'ps'))
check(TD('NaT', 'ps'), 0.0, TD('NaT', 'ps'))
check(TD(3, 'ps'), float('nan'), TD('NaT', 'ps'))
check(TD('NaT', 'ps'), float('nan'), TD('NaT', 'ps'))
def test_homogenous_div(self):
div = self.jit(div_usecase)
def check(a, b, expected):
self.assertPreciseEqual(div(a, b), expected)
# timedelta64 / timedelta64
check(TD(7), TD(3), 7. / 3.)
if not numpy_support.strict_ufunc_typing:
check(TD(7), TD(3, 'ms'), 7. / 3.)
check(TD(7, 'us'), TD(3, 'ms'), 7. / 3000.)
check(TD(7, 'ms'), TD(3, 'us'), 7000. / 3.)
check(TD(7), TD(0), float('+inf'))
check(TD(-7), TD(0), float('-inf'))
check(TD(0), TD(0), float('nan'))
# NaTs
check(TD('nat'), TD(3), float('nan'))
check(TD(3), TD('nat'), float('nan'))
check(TD('nat'), TD(0), float('nan'))
# Cannot div months with days
with self.assertRaises((TypeError, TypingError)):
div(TD(1, 'M'), TD(1, 'D'))
@tag('important')
def test_eq_ne(self):
eq = self.jit(eq_usecase)
ne = self.jit(ne_usecase)
def check(a, b, expected):
self.assertPreciseEqual(eq(a, b), expected)
self.assertPreciseEqual(eq(b, a), expected)
self.assertPreciseEqual(ne(a, b), not expected)
self.assertPreciseEqual(ne(b, a), not expected)
check(TD(1), TD(2), False)
check(TD(1), TD(1), True)
check(TD(1, 's'), TD(2, 's'), False)
check(TD(1, 's'), TD(1, 's'), True)
check(TD(2000, 's'), TD(2, 's'), False)
check(TD(2000, 'ms'), TD(2, 's'), True)
check(TD(1, 'Y'), TD(12, 'M'), True)
# NaTs
check(TD('Nat'), TD('Nat'), True)
check(TD('Nat', 'ms'), TD('Nat', 's'), True)
check(TD('Nat'), TD(1), False)
# Incompatible units => timedeltas compare unequal
check(TD(1, 'Y'), TD(365, 'D'), False)
check(TD(1, 'Y'), TD(366, 'D'), False)
# ... except when both are NaT!
check(TD('NaT', 'W'), TD('NaT', 'D'), True)
def test_lt_ge(self):
lt = self.jit(lt_usecase)
ge = self.jit(ge_usecase)
def check(a, b, expected):
self.assertPreciseEqual(lt(a, b), expected)
self.assertPreciseEqual(ge(a, b), not expected)
check(TD(1), TD(2), True)
check(TD(1), TD(1), False)
check(TD(2), TD(1), False)
check(TD(1, 's'), TD(2, 's'), True)
check(TD(1, 's'), TD(1, 's'), False)
check(TD(2, 's'), TD(1, 's'), False)
check(TD(1, 'm'), TD(61, 's'), True)
check(TD(1, 'm'), TD(60, 's'), False)
# NaTs
check(TD('Nat'), TD('Nat'), False)
check(TD('Nat', 'ms'), TD('Nat', 's'), False)
check(TD('Nat'), TD(-(2**63)+1), True)
# Incompatible units => exception raised
with self.assertRaises((TypeError, TypingError)):
lt(TD(1, 'Y'), TD(365, 'D'))
with self.assertRaises((TypeError, TypingError)):
ge(TD(1, 'Y'), TD(365, 'D'))
# ... even when both are NaT
with self.assertRaises((TypeError, TypingError)):
lt(TD('NaT', 'Y'), TD('NaT', 'D'))
with self.assertRaises((TypeError, TypingError)):
ge(TD('NaT', 'Y'), TD('NaT', 'D'))
def test_le_gt(self):
le = self.jit(le_usecase)
gt = self.jit(gt_usecase)
def check(a, b, expected):
self.assertPreciseEqual(le(a, b), expected)
self.assertPreciseEqual(gt(a, b), not expected)
check(TD(1), TD(2), True)
check(TD(1), TD(1), True)
check(TD(2), TD(1), False)
check(TD(1, 's'), TD(2, 's'), True)
check(TD(1, 's'), TD(1, 's'), True)
check(TD(2, 's'), TD(1, 's'), False)
check(TD(1, 'm'), TD(61, 's'), True)
check(TD(1, 'm'), TD(60, 's'), True)
check(TD(1, 'm'), TD(59, 's'), False)
# NaTs
check(TD('Nat'), TD('Nat'), True)
check(TD('Nat', 'ms'), TD('Nat', 's'), True)
check(TD('Nat'), TD(-(2**63)+1), True)
# Incompatible units => exception raised
with self.assertRaises((TypeError, TypingError)):
le(TD(1, 'Y'), TD(365, 'D'))
with self.assertRaises((TypeError, TypingError)):
gt(TD(1, 'Y'), TD(365, 'D'))
# ... even when both are NaT
with self.assertRaises((TypeError, TypingError)):
le(TD('NaT', 'Y'), TD('NaT', 'D'))
with self.assertRaises((TypeError, TypingError)):
gt(TD('NaT', 'Y'), TD('NaT', 'D'))
def test_pos(self):
pos = self.jit(pos_usecase)
def check(a):
self.assertPreciseEqual(pos(a), +a)
check(TD(3))
check(TD(-4))
check(TD(3, 'ms'))
check(TD(-4, 'ms'))
check(TD('NaT'))
check(TD('NaT', 'ms'))
def test_neg(self):
neg = self.jit(neg_usecase)
def check(a):
self.assertPreciseEqual(neg(a), -a)
check(TD(3))
check(TD(-4))
check(TD(3, 'ms'))
check(TD(-4, 'ms'))
check(TD('NaT'))
check(TD('NaT', 'ms'))
def test_abs(self):
f = self.jit(abs_usecase)
def check(a):
self.assertPreciseEqual(f(a), abs(a))
check(TD(3))
check(TD(-4))
check(TD(3, 'ms'))
check(TD(-4, 'ms'))
check(TD('NaT'))
check(TD('NaT', 'ms'))
class TestTimedeltaArithmeticNoPython(TestTimedeltaArithmetic):
jitargs = dict(nopython=True)
class TestDatetimeArithmetic(TestCase):
jitargs = dict(forceobj=True)
def jit(self, pyfunc):
return jit(**self.jitargs)(pyfunc)
@contextlib.contextmanager
def silence_numpy_warnings(self):
# Numpy can raise warnings when combining e.g. a generic timedelta64
# with a non-generic datetime64.
with warnings.catch_warnings():
warnings.filterwarnings('ignore',
message='Implicitly casting between incompatible kinds',
category=DeprecationWarning)
yield
@tag('important')
def test_add_sub_timedelta(self):
"""
Test `datetime64 + timedelta64` and `datetime64 - timedelta64`.
"""
add = self.jit(add_usecase)
sub = self.jit(sub_usecase)
def check(a, b, expected):
with self.silence_numpy_warnings():
self.assertPreciseEqual(add(a, b), expected, (a, b))
self.assertPreciseEqual(add(b, a), expected, (a, b))
self.assertPreciseEqual(sub(a, -b), expected, (a, b))
# Did we get it right?
self.assertPreciseEqual(a + b, expected)
# Y + ...
if not numpy_support.strict_ufunc_typing:
check(DT('2014'), TD(2), DT('2016'))
check(DT('2014'), TD(2, 'Y'), DT('2016'))
check(DT('2014'), TD(2, 'M'), DT('2014-03'))
check(DT('2014'), TD(3, 'W'), DT('2014-01-16', 'W'))
check(DT('2014'), TD(4, 'D'), DT('2014-01-05'))
check(DT('2000'), TD(365, 'D'), DT('2000-12-31'))
if not numpy_support.strict_ufunc_typing:
check(DT('2001'), TD(61, 'm'), DT('2001-01-01T01:01Z'))
check(DT('2001'), TD(61, 's'), DT('2001-01-01T00:01:01Z'))
# M + ...
if not numpy_support.strict_ufunc_typing:
check(DT('2014-02'), TD(2), DT('2014-04'))
check(DT('2014-02'), TD(2, 'Y'), DT('2016-02'))
check(DT('2014-02'), TD(2, 'M'), DT('2014-04'))
check(DT('2014-02'), TD(2, 'D'), DT('2014-02-03'))
if not numpy_support.strict_ufunc_typing:
check(DT('2014-02'), TD(61, 's'), DT('2014-02-01T00:01:01Z'))
# W + ...
check(DT('2014-01-07', 'W'), TD(2, 'W'), DT('2014-01-16', 'W'))
# D + ...
check(DT('2014-02-02'), TD(27, 'D'), DT('2014-03-01'))
check(DT('2012-02-02'), TD(27, 'D'), DT('2012-02-29'))
check(DT('2012-02-02'), TD(2, 'W'), DT('2012-02-16'))
if not numpy_support.strict_ufunc_typing:
check(DT('2014-02-02'), TD(73, 'h'), DT('2014-02-05T01Z'))
# s + ...
check(DT('2000-01-01T01:02:03Z'), TD(2, 'h'), DT('2000-01-01T03:02:03Z'))
check(DT('2000-01-01T01:02:03Z'), TD(2, 'ms'), DT('2000-01-01T01:02:03.002Z'))
# More thorough checking with leap years and faraway years
for dt_str in ('600', '601', '604', '801',
'1900', '1904', '2200', '2300', '2304',
'2400', '6001'):
for dt_suffix in ('', '-01', '-12'):
dt = DT(dt_str + dt_suffix)
for td in [TD(2, 'D'), TD(2, 'W'),
TD(100, 'D'), TD(10000, 'D'),
TD(-100, 'D'), TD(-10000, 'D'),
TD(100, 'W'), TD(10000, 'W'),
TD(-100, 'W'), TD(-10000, 'W'),
TD(100, 'M'), TD(10000, 'M'),
TD(-100, 'M'), TD(-10000, 'M')]:
self.assertEqual(add(dt, td), dt + td, (dt, td))
self.assertEqual(add(td, dt), dt + td, (dt, td))
self.assertEqual(sub(dt, -td), dt + td, (dt, td))
# NaTs
check(DT('NaT'), TD(2), DT('NaT'))
if not numpy_support.strict_ufunc_typing:
check(DT('NaT', 's'), TD(2), DT('NaT', 's'))
check(DT('NaT', 's'), TD(2, 'h'), DT('NaT', 's'))
check(DT('NaT', 's'), TD(2, 'ms'), DT('NaT', 'ms'))
if not numpy_support.strict_ufunc_typing:
check(DT('2014'), TD('NaT'), DT('NaT', 'Y'))
check(DT('2014'), TD('NaT', 'W'), DT('NaT', 'W'))
check(DT('2014-01-01'), TD('NaT', 'W'), DT('NaT', 'D'))
if not numpy_support.strict_ufunc_typing:
check(DT('NaT', 's'), TD('NaT'), DT('NaT', 's'))
check(DT('NaT', 's'), TD('NaT', 'ms'), DT('NaT', 'ms'))
# Cannot add datetime days and timedelta months or years
for f in (add, sub):
with self.assertRaises((TypeError, TypingError)):
f(DT(1, '2014-01-01'), TD(1, 'Y'))
with self.assertRaises((TypeError, TypingError)):
f(DT(1, '2014-01-01'), TD(1, 'M'))
def datetime_samples(self):
dt_years = ['600', '601', '604', '1968', '1969', '1973',
'2000', '2004', '2005', '2100', '2400', '2401']
dt_suffixes = ['', '-01', '-12', '-02-28', '-12-31',
'-01-05T12:30:56Z', '-01-05T12:30:56.008Z']
dts = [DT(a + b) for (a, b) in itertools.product(dt_years, dt_suffixes)]
dts += [DT(s, 'W') for s in dt_years]
return dts
def test_datetime_difference(self):
"""
Test `datetime64 - datetime64`.
"""
sub = self.jit(sub_usecase)
def check(a, b, expected=None):
with self.silence_numpy_warnings():
self.assertPreciseEqual(sub(a, b), a - b, (a, b))
self.assertPreciseEqual(sub(b, a), b - a, (a, b))
# Did we get it right?
self.assertPreciseEqual(a - b, expected)
check(DT('2014'), DT('2017'), TD(-3, 'Y'))
check(DT('2014-02'), DT('2017-01'), TD(-35, 'M'))
check(DT('2014-02-28'), DT('2015-03-01'), TD(-366, 'D'))
# NaTs
if not numpy_support.strict_ufunc_typing:
check(DT('NaT'), DT('2000'), TD('NaT', 'Y'))
check(DT('NaT', 'M'), DT('2000'), TD('NaT', 'M'))
check(DT('NaT', 'M'), DT('2000-01-01'), TD('NaT', 'D'))
check(DT('NaT'), DT('NaT'), TD('NaT'))
# Test many more values
with self.silence_numpy_warnings():
dts = self.datetime_samples()
for a, b in itertools.product(dts, dts):
if (numpy_support.strict_ufunc_typing
and not npdatetime.same_kind(value_unit(a),
value_unit(b))):
continue
self.assertPreciseEqual(sub(a, b), a - b, (a, b))
@tag('important')
def test_comparisons(self):
# Test all datetime comparisons all at once
eq = self.jit(eq_usecase)
ne = self.jit(ne_usecase)
lt = self.jit(lt_usecase)
le = self.jit(le_usecase)
gt = self.jit(gt_usecase)
ge = self.jit(ge_usecase)
def check_eq(a, b, expected):
with self.silence_numpy_warnings():
self.assertPreciseEqual(eq(a, b), expected, (a, b, expected))
self.assertPreciseEqual(eq(b, a), expected, (a, b, expected))
self.assertPreciseEqual(ne(a, b), not expected, (a, b, expected))
self.assertPreciseEqual(ne(b, a), not expected, (a, b, expected))
if expected:
# If equal, then equal-ordered comparisons are true
self.assertTrue(le(a, b), (a, b))
self.assertTrue(ge(a, b), (a, b))
self.assertTrue(le(b, a), (a, b))
self.assertTrue(ge(b, a), (a, b))
# and strictly ordered comparisons are false
self.assertFalse(lt(a, b), (a, b))
self.assertFalse(gt(a, b), (a, b))
self.assertFalse(lt(b, a), (a, b))
self.assertFalse(gt(b, a), (a, b))
# Did we get it right?
self.assertPreciseEqual(a == b, expected)
def check_lt(a, b, expected):
with self.silence_numpy_warnings():
self.assertPreciseEqual(lt(a, b), expected, (a, b, expected))
self.assertPreciseEqual(gt(b, a), expected, (a, b, expected))
self.assertPreciseEqual(ge(a, b), not expected, (a, b, expected))
self.assertPreciseEqual(le(b, a), not expected, (a, b, expected))
if expected:
# If true, then values are not equal
check_eq(a, b, False)
# Did we get it right?
self.assertPreciseEqual(a < b, expected)
check_eq(DT('2014'), DT('2017'), False)
check_eq(DT('2014'), DT('2014-01'), True)
check_eq(DT('2014'), DT('2014-01-01'), True)
check_eq(DT('2014'), DT('2014-01-01', 'W'), True)
check_eq(DT('2014-01'), DT('2014-01-01', 'W'), True)
# Yes, it's not transitive
check_eq(DT('2014-01-01'), DT('2014-01-01', 'W'), False)
check_eq(DT('2014-01-02'), DT('2014-01-06', 'W'), True)
# with times
check_eq(DT('2014-01-01T00:01:00Z', 's'),
DT('2014-01-01T00:01Z', 'm'), True)
check_eq(DT('2014-01-01T00:01:01Z', 's'),
DT('2014-01-01T00:01Z', 'm'), False)
# NaTs
if not numpy_support.strict_ufunc_typing:
check_lt(DT('NaT'), DT('2017'), True)
check_lt(DT('NaT', 'Y'), DT('2017'), True)
if not numpy_support.strict_ufunc_typing:
check_lt(DT('NaT', 'ms'), DT('2017'), True)
check_eq(DT('NaT'), DT('NaT'), True)
if not numpy_support.strict_ufunc_typing:
check_eq(DT('NaT', 'Y'), DT('NaT'), True)
check_eq(DT('NaT', 'ms'), DT('NaT', 'M'), True)
# Check comparison between various units
dts = self.datetime_samples()
for a in dts:
# Take a number of smaller units
a_unit = a.dtype.str.split('[')[1][:-1]
i = all_units.index(a_unit)
units = all_units[i:i+6]
for unit in units:
# Force conversion
b = a.astype('M8[%s]' % unit)
if (numpy_support.strict_ufunc_typing
and not npdatetime.same_kind(value_unit(a),
value_unit(b))):
continue
check_eq(a, b, True)
check_lt(a, b + np.timedelta64(1, unit), True)
check_lt(b - np.timedelta64(1, unit), a, True)
class TestDatetimeArithmeticNoPython(TestDatetimeArithmetic):
jitargs = dict(nopython=True)
class TestMetadataScalingFactor(TestCase):
"""
Tests than non-1 scaling factors are not supported in datetime64
and timedelta64 dtypes.
"""
def test_datetime(self, **jitargs):
eq = jit(**jitargs)(eq_usecase)
self.assertTrue(eq(DT('2014', '10Y'), DT('2010')))
def test_datetime_npm(self):
with self.assertTypingError():
self.test_datetime(nopython=True)
def test_timedelta(self, **jitargs):
eq = jit(**jitargs)(eq_usecase)
self.assertTrue(eq(TD(2, '10Y'), TD(20, 'Y')))
def test_timedelta_npm(self):
with self.assertTypingError():
self.test_timedelta(nopython=True)
if __name__ == '__main__':
unittest.main()