2010-09-27 23:15:33 +08:00
|
|
|
import datetime
|
2007-09-17 12:50:12 +08:00
|
|
|
|
2020-07-03 14:01:45 +08:00
|
|
|
from django.test import TestCase
|
2013-10-05 03:55:16 +08:00
|
|
|
from django.test.utils import requires_tz_support
|
2019-10-28 18:17:29 +08:00
|
|
|
from django.utils import timezone, translation
|
2015-01-28 20:35:27 +08:00
|
|
|
from django.utils.timesince import timesince, timeuntil
|
2019-10-28 18:17:29 +08:00
|
|
|
from django.utils.translation import npgettext_lazy
|
2013-09-08 16:43:33 +08:00
|
|
|
|
2007-09-17 12:50:12 +08:00
|
|
|
|
2020-07-03 14:01:45 +08:00
|
|
|
class TimesinceTests(TestCase):
|
2007-09-17 12:50:12 +08:00
|
|
|
|
2010-09-27 23:15:33 +08:00
|
|
|
def setUp(self):
|
|
|
|
self.t = datetime.datetime(2007, 8, 14, 13, 46, 0)
|
|
|
|
self.onemicrosecond = datetime.timedelta(microseconds=1)
|
|
|
|
self.onesecond = datetime.timedelta(seconds=1)
|
|
|
|
self.oneminute = datetime.timedelta(minutes=1)
|
|
|
|
self.onehour = datetime.timedelta(hours=1)
|
|
|
|
self.oneday = datetime.timedelta(days=1)
|
|
|
|
self.oneweek = datetime.timedelta(days=7)
|
|
|
|
self.onemonth = datetime.timedelta(days=30)
|
|
|
|
self.oneyear = datetime.timedelta(days=365)
|
2007-09-17 12:50:12 +08:00
|
|
|
|
2010-09-27 23:15:33 +08:00
|
|
|
def test_equal_datetimes(self):
|
|
|
|
""" equal datetimes. """
|
2013-05-18 19:58:45 +08:00
|
|
|
# NOTE: \xa0 avoids wrapping between value and unit
|
|
|
|
self.assertEqual(timesince(self.t, self.t), '0\xa0minutes')
|
2007-09-17 12:50:12 +08:00
|
|
|
|
2010-09-27 23:15:33 +08:00
|
|
|
def test_ignore_microseconds_and_seconds(self):
|
|
|
|
""" Microseconds and seconds are ignored. """
|
2016-04-08 10:04:45 +08:00
|
|
|
self.assertEqual(timesince(self.t, self.t + self.onemicrosecond), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t + self.onesecond), '0\xa0minutes')
|
2007-09-17 12:50:12 +08:00
|
|
|
|
2010-09-27 23:15:33 +08:00
|
|
|
def test_other_units(self):
|
|
|
|
""" Test other units. """
|
2016-04-08 10:04:45 +08:00
|
|
|
self.assertEqual(timesince(self.t, self.t + self.oneminute), '1\xa0minute')
|
2013-11-04 02:08:55 +08:00
|
|
|
self.assertEqual(timesince(self.t, self.t + self.onehour), '1\xa0hour')
|
|
|
|
self.assertEqual(timesince(self.t, self.t + self.oneday), '1\xa0day')
|
|
|
|
self.assertEqual(timesince(self.t, self.t + self.oneweek), '1\xa0week')
|
2016-04-08 10:04:45 +08:00
|
|
|
self.assertEqual(timesince(self.t, self.t + self.onemonth), '1\xa0month')
|
2013-11-04 02:08:55 +08:00
|
|
|
self.assertEqual(timesince(self.t, self.t + self.oneyear), '1\xa0year')
|
2007-09-17 12:50:12 +08:00
|
|
|
|
2010-09-27 23:15:33 +08:00
|
|
|
def test_multiple_units(self):
|
|
|
|
""" Test multiple units. """
|
2016-04-08 10:04:45 +08:00
|
|
|
self.assertEqual(timesince(self.t, self.t + 2 * self.oneday + 6 * self.onehour), '2\xa0days, 6\xa0hours')
|
|
|
|
self.assertEqual(timesince(self.t, self.t + 2 * self.oneweek + 2 * self.oneday), '2\xa0weeks, 2\xa0days')
|
2008-08-26 16:08:55 +08:00
|
|
|
|
2010-09-27 23:15:33 +08:00
|
|
|
def test_display_first_unit(self):
|
|
|
|
"""
|
|
|
|
If the two differing units aren't adjacent, only the first unit is
|
|
|
|
displayed.
|
|
|
|
"""
|
2016-04-08 10:04:45 +08:00
|
|
|
self.assertEqual(
|
|
|
|
timesince(self.t, self.t + 2 * self.oneweek + 3 * self.onehour + 4 * self.oneminute),
|
|
|
|
'2\xa0weeks'
|
|
|
|
)
|
|
|
|
self.assertEqual(timesince(self.t, self.t + 4 * self.oneday + 5 * self.oneminute), '4\xa0days')
|
2010-09-27 23:15:33 +08:00
|
|
|
|
|
|
|
def test_display_second_before_first(self):
|
|
|
|
"""
|
|
|
|
When the second date occurs before the first, we should always
|
|
|
|
get 0 minutes.
|
|
|
|
"""
|
2016-04-08 10:04:45 +08:00
|
|
|
self.assertEqual(timesince(self.t, self.t - self.onemicrosecond), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - self.onesecond), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - self.oneminute), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - self.onehour), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - self.oneday), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - self.oneweek), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - self.onemonth), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - self.oneyear), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - 2 * self.oneday - 6 * self.onehour), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(self.t, self.t - 2 * self.oneweek - 2 * self.oneday), '0\xa0minutes')
|
|
|
|
self.assertEqual(
|
|
|
|
timesince(self.t, self.t - 2 * self.oneweek - 3 * self.onehour - 4 * self.oneminute), '0\xa0minutes'
|
|
|
|
)
|
|
|
|
self.assertEqual(timesince(self.t, self.t - 4 * self.oneday - 5 * self.oneminute), '0\xa0minutes')
|
2010-09-27 23:15:33 +08:00
|
|
|
|
2019-10-28 18:17:29 +08:00
|
|
|
def test_second_before_equal_first_humanize_time_strings(self):
|
|
|
|
time_strings = {
|
2021-01-19 02:03:37 +08:00
|
|
|
'minute': npgettext_lazy(
|
|
|
|
'naturaltime-future',
|
|
|
|
'%(num)d minute',
|
|
|
|
'%(num)d minutes',
|
|
|
|
'num',
|
|
|
|
),
|
2019-10-28 18:17:29 +08:00
|
|
|
}
|
|
|
|
with translation.override('cs'):
|
|
|
|
for now in [self.t, self.t - self.onemicrosecond, self.t - self.oneday]:
|
|
|
|
with self.subTest(now):
|
|
|
|
self.assertEqual(
|
|
|
|
timesince(self.t, now, time_strings=time_strings),
|
|
|
|
'0\xa0minut',
|
|
|
|
)
|
|
|
|
|
2013-10-05 03:55:16 +08:00
|
|
|
@requires_tz_support
|
2010-09-27 23:15:33 +08:00
|
|
|
def test_different_timezones(self):
|
|
|
|
""" When using two different timezones. """
|
|
|
|
now = datetime.datetime.now()
|
2013-09-08 16:43:33 +08:00
|
|
|
now_tz = timezone.make_aware(now, timezone.get_default_timezone())
|
|
|
|
now_tz_i = timezone.localtime(now_tz, timezone.get_fixed_timezone(195))
|
2010-09-27 23:15:33 +08:00
|
|
|
|
2013-05-18 19:58:45 +08:00
|
|
|
self.assertEqual(timesince(now), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(now_tz), '0\xa0minutes')
|
2013-10-05 03:55:16 +08:00
|
|
|
self.assertEqual(timesince(now_tz_i), '0\xa0minutes')
|
|
|
|
self.assertEqual(timesince(now_tz, now_tz_i), '0\xa0minutes')
|
|
|
|
self.assertEqual(timeuntil(now), '0\xa0minutes')
|
|
|
|
self.assertEqual(timeuntil(now_tz), '0\xa0minutes')
|
|
|
|
self.assertEqual(timeuntil(now_tz_i), '0\xa0minutes')
|
2013-05-18 19:58:45 +08:00
|
|
|
self.assertEqual(timeuntil(now_tz, now_tz_i), '0\xa0minutes')
|
2010-09-27 23:15:33 +08:00
|
|
|
|
2012-03-22 15:29:39 +08:00
|
|
|
def test_date_objects(self):
|
|
|
|
""" Both timesince and timeuntil should work on date objects (#17937). """
|
|
|
|
today = datetime.date.today()
|
2013-05-18 19:58:45 +08:00
|
|
|
self.assertEqual(timesince(today + self.oneday), '0\xa0minutes')
|
|
|
|
self.assertEqual(timeuntil(today - self.oneday), '0\xa0minutes')
|
2012-03-22 15:29:39 +08:00
|
|
|
|
2010-09-27 23:15:33 +08:00
|
|
|
def test_both_date_objects(self):
|
|
|
|
""" Timesince should work with both date objects (#9672) """
|
|
|
|
today = datetime.date.today()
|
2013-05-18 19:58:45 +08:00
|
|
|
self.assertEqual(timeuntil(today + self.oneday, today), '1\xa0day')
|
|
|
|
self.assertEqual(timeuntil(today - self.oneday, today), '0\xa0minutes')
|
|
|
|
self.assertEqual(timeuntil(today + self.oneweek, today), '1\xa0week')
|
2011-11-18 21:01:06 +08:00
|
|
|
|
2016-12-27 22:29:11 +08:00
|
|
|
def test_leap_year(self):
|
|
|
|
start_date = datetime.date(2016, 12, 25)
|
|
|
|
self.assertEqual(timeuntil(start_date + self.oneweek, start_date), '1\xa0week')
|
|
|
|
self.assertEqual(timesince(start_date, start_date + self.oneweek), '1\xa0week')
|
|
|
|
|
2017-01-02 21:40:44 +08:00
|
|
|
def test_leap_year_new_years_eve(self):
|
|
|
|
t = datetime.date(2016, 12, 31)
|
|
|
|
now = datetime.datetime(2016, 12, 31, 18, 0, 0)
|
|
|
|
self.assertEqual(timesince(t + self.oneday, now), '0\xa0minutes')
|
|
|
|
self.assertEqual(timeuntil(t - self.oneday, now), '0\xa0minutes')
|
|
|
|
|
2011-11-18 21:01:06 +08:00
|
|
|
def test_naive_datetime_with_tzinfo_attribute(self):
|
|
|
|
class naive(datetime.tzinfo):
|
|
|
|
def utcoffset(self, dt):
|
|
|
|
return None
|
|
|
|
future = datetime.datetime(2080, 1, 1, tzinfo=naive())
|
2013-05-18 19:58:45 +08:00
|
|
|
self.assertEqual(timesince(future), '0\xa0minutes')
|
2011-11-18 21:01:06 +08:00
|
|
|
past = datetime.datetime(1980, 1, 1, tzinfo=naive())
|
2013-05-18 19:58:45 +08:00
|
|
|
self.assertEqual(timeuntil(past), '0\xa0minutes')
|
2015-06-05 06:46:52 +08:00
|
|
|
|
|
|
|
def test_thousand_years_ago(self):
|
|
|
|
t = datetime.datetime(1007, 8, 14, 13, 46, 0)
|
|
|
|
self.assertEqual(timesince(t, self.t), '1000\xa0years')
|
2016-12-27 22:29:11 +08:00
|
|
|
self.assertEqual(timeuntil(self.t, t), '1000\xa0years')
|
2020-07-03 14:01:45 +08:00
|
|
|
|
|
|
|
def test_depth(self):
|
|
|
|
t = self.t + self.oneyear + self.onemonth + self.oneweek + self.oneday + self.onehour
|
|
|
|
tests = [
|
|
|
|
(t, 1, '1\xa0year'),
|
|
|
|
(t, 2, '1\xa0year, 1\xa0month'),
|
|
|
|
(t, 3, '1\xa0year, 1\xa0month, 1\xa0week'),
|
|
|
|
(t, 4, '1\xa0year, 1\xa0month, 1\xa0week, 1\xa0day'),
|
|
|
|
(t, 5, '1\xa0year, 1\xa0month, 1\xa0week, 1\xa0day, 1\xa0hour'),
|
|
|
|
(t, 6, '1\xa0year, 1\xa0month, 1\xa0week, 1\xa0day, 1\xa0hour'),
|
|
|
|
(self.t + self.onehour, 5, '1\xa0hour'),
|
|
|
|
(self.t + (4 * self.oneminute), 3, '4\xa0minutes'),
|
|
|
|
(self.t + self.onehour + self.oneminute, 1, '1\xa0hour'),
|
|
|
|
(self.t + self.oneday + self.onehour, 1, '1\xa0day'),
|
|
|
|
(self.t + self.oneweek + self.oneday, 1, '1\xa0week'),
|
|
|
|
(self.t + self.onemonth + self.oneweek, 1, '1\xa0month'),
|
|
|
|
(self.t + self.oneyear + self.onemonth, 1, '1\xa0year'),
|
|
|
|
(self.t + self.oneyear + self.oneweek + self.oneday, 3, '1\xa0year'),
|
|
|
|
]
|
|
|
|
for value, depth, expected in tests:
|
|
|
|
with self.subTest():
|
|
|
|
self.assertEqual(timesince(self.t, value, depth=depth), expected)
|
|
|
|
self.assertEqual(timeuntil(value, self.t, depth=depth), expected)
|
|
|
|
|
|
|
|
def test_depth_invalid(self):
|
|
|
|
msg = 'depth must be greater than 0.'
|
|
|
|
with self.assertRaisesMessage(ValueError, msg):
|
|
|
|
timesince(self.t, self.t, depth=0)
|