2012-06-07 18:08:47 +02:00
|
|
|
from __future__ import unicode_literals
|
2015-01-28 07:35:27 -05:00
|
|
|
|
2011-11-18 13:01:06 +00:00
|
|
|
import datetime
|
2013-02-23 13:19:21 -03:00
|
|
|
from decimal import Decimal
|
2013-07-01 14:22:27 +02:00
|
|
|
from unittest import skipIf
|
2010-10-11 12:55:17 +00:00
|
|
|
|
2012-07-19 23:02:22 +02:00
|
|
|
from django.contrib.humanize.templatetags import humanize
|
2015-01-28 07:35:27 -05:00
|
|
|
from django.template import Context, Template, defaultfilters
|
2015-02-10 08:44:52 -05:00
|
|
|
from django.test import TestCase, modify_settings, override_settings
|
2012-07-19 23:02:22 +02:00
|
|
|
from django.utils import translation
|
2015-01-28 07:35:27 -05:00
|
|
|
from django.utils.html import escape
|
|
|
|
from django.utils.timezone import get_fixed_timezone, utc
|
2012-07-19 23:02:22 +02:00
|
|
|
from django.utils.translation import ugettext as _
|
|
|
|
|
2015-01-28 07:35:27 -05:00
|
|
|
try:
|
|
|
|
import pytz
|
|
|
|
except ImportError:
|
|
|
|
pytz = None
|
|
|
|
|
2012-07-19 23:02:22 +02:00
|
|
|
|
|
|
|
# Mock out datetime in some tests so they don't fail occasionally when they
|
|
|
|
# run too slow. Use a fixed datetime for datetime.now(). DST change in
|
|
|
|
# America/Chicago (the default time zone) happened on March 11th in 2012.
|
|
|
|
|
|
|
|
now = datetime.datetime(2012, 3, 9, 22, 30)
|
|
|
|
|
2013-09-06 12:59:44 -05:00
|
|
|
|
2012-07-19 23:02:22 +02:00
|
|
|
class MockDateTime(datetime.datetime):
|
|
|
|
@classmethod
|
2013-12-07 20:13:53 -05:00
|
|
|
def now(cls, tz=None):
|
2012-07-19 23:02:22 +02:00
|
|
|
if tz is None or tz.utcoffset(now) is None:
|
|
|
|
return now
|
|
|
|
else:
|
|
|
|
# equals now.replace(tzinfo=utc)
|
|
|
|
return now.replace(tzinfo=tz) + tz.utcoffset(now)
|
2007-02-25 16:09:30 +00:00
|
|
|
|
2011-04-22 12:02:55 +00:00
|
|
|
|
2015-02-10 08:44:52 -05:00
|
|
|
@modify_settings(INSTALLED_APPS={'append': 'django.contrib.humanize'})
|
2014-01-26 14:29:04 +01:00
|
|
|
class HumanizeTests(TestCase):
|
2007-02-25 16:09:30 +00:00
|
|
|
|
2013-09-06 12:59:44 -05:00
|
|
|
def humanize_tester(self, test_list, result_list, method, normalize_result_func=escape):
|
2011-05-17 10:15:58 +00:00
|
|
|
for test_content, result in zip(test_list, result_list):
|
2011-10-24 21:30:55 +00:00
|
|
|
t = Template('{%% load humanize %%}{{ test_content|%s }}' % method)
|
2007-02-25 16:09:30 +00:00
|
|
|
rendered = t.render(Context(locals())).strip()
|
2013-09-06 12:59:44 -05:00
|
|
|
self.assertEqual(rendered, normalize_result_func(result),
|
2011-05-17 10:15:58 +00:00
|
|
|
msg="%s test failed, produced '%s', should've produced '%s'" % (method, rendered, result))
|
Merged Unicode branch into trunk (r4952:5608). This should be fully
backwards compatible for all practical purposes.
Fixed #2391, #2489, #2996, #3322, #3344, #3370, #3406, #3432, #3454, #3492, #3582, #3690, #3878, #3891, #3937, #4039, #4141, #4227, #4286, #4291, #4300, #4452, #4702
git-svn-id: http://code.djangoproject.com/svn/django/trunk@5609 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2007-07-04 12:11:04 +00:00
|
|
|
|
2007-02-25 16:09:30 +00:00
|
|
|
def test_ordinal(self):
|
2011-09-23 16:45:40 +00:00
|
|
|
test_list = ('1', '2', '3', '4', '11', '12',
|
|
|
|
'13', '101', '102', '103', '111',
|
2010-01-10 21:37:20 +00:00
|
|
|
'something else', None)
|
2007-02-25 16:09:30 +00:00
|
|
|
result_list = ('1st', '2nd', '3rd', '4th', '11th',
|
|
|
|
'12th', '13th', '101st', '102nd', '103rd',
|
2010-01-10 21:37:20 +00:00
|
|
|
'111th', 'something else', None)
|
2007-02-25 16:09:30 +00:00
|
|
|
|
2013-03-17 18:17:35 -03:00
|
|
|
with translation.override('en'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'ordinal')
|
2007-02-25 16:09:30 +00:00
|
|
|
|
2013-09-06 12:59:44 -05:00
|
|
|
def test_i18n_html_ordinal(self):
|
|
|
|
"""Allow html in output on i18n strings"""
|
|
|
|
test_list = ('1', '2', '3', '4', '11', '12',
|
|
|
|
'13', '101', '102', '103', '111',
|
|
|
|
'something else', None)
|
|
|
|
result_list = ('1<sup>er</sup>', '2<sup>e</sup>', '3<sup>e</sup>', '4<sup>e</sup>',
|
|
|
|
'11<sup>e</sup>', '12<sup>e</sup>', '13<sup>e</sup>', '101<sup>er</sup>',
|
|
|
|
'102<sup>e</sup>', '103<sup>e</sup>', '111<sup>e</sup>', 'something else',
|
|
|
|
'None')
|
|
|
|
|
|
|
|
with translation.override('fr-fr'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'ordinal', lambda x: x)
|
|
|
|
|
2007-02-25 16:09:30 +00:00
|
|
|
def test_intcomma(self):
|
2007-03-31 11:45:24 +00:00
|
|
|
test_list = (100, 1000, 10123, 10311, 1000000, 1234567.25,
|
2013-02-23 13:19:21 -03:00
|
|
|
'100', '1000', '10123', '10311', '1000000', '1234567.1234567', Decimal('1234567.1234567'),
|
2010-12-21 15:07:43 +00:00
|
|
|
None)
|
2007-04-01 05:23:08 +00:00
|
|
|
result_list = ('100', '1,000', '10,123', '10,311', '1,000,000', '1,234,567.25',
|
2013-02-23 13:19:21 -03:00
|
|
|
'100', '1,000', '10,123', '10,311', '1,000,000', '1,234,567.1234567', '1,234,567.1234567',
|
2010-12-21 15:07:43 +00:00
|
|
|
None)
|
2007-02-25 16:09:30 +00:00
|
|
|
|
2013-03-17 18:17:35 -03:00
|
|
|
with translation.override('en'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'intcomma')
|
2007-02-25 16:09:30 +00:00
|
|
|
|
2011-09-09 17:57:45 +00:00
|
|
|
def test_l10n_intcomma(self):
|
|
|
|
test_list = (100, 1000, 10123, 10311, 1000000, 1234567.25,
|
2013-02-23 13:19:21 -03:00
|
|
|
'100', '1000', '10123', '10311', '1000000', '1234567.1234567', Decimal('1234567.1234567'),
|
2011-09-09 17:57:45 +00:00
|
|
|
None)
|
|
|
|
result_list = ('100', '1,000', '10,123', '10,311', '1,000,000', '1,234,567.25',
|
2013-02-23 13:19:21 -03:00
|
|
|
'100', '1,000', '10,123', '10,311', '1,000,000', '1,234,567.1234567', '1,234,567.1234567',
|
2011-09-09 17:57:45 +00:00
|
|
|
None)
|
|
|
|
|
2013-09-22 13:49:46 +02:00
|
|
|
with self.settings(USE_L10N=True, USE_THOUSAND_SEPARATOR=False):
|
|
|
|
with translation.override('en'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'intcomma')
|
2011-09-09 17:57:45 +00:00
|
|
|
|
2011-12-24 11:01:57 +00:00
|
|
|
def test_intcomma_without_number_grouping(self):
|
|
|
|
# Regression for #17414
|
2013-08-16 20:12:10 +02:00
|
|
|
with translation.override('ja'), self.settings(USE_L10N=True):
|
|
|
|
self.humanize_tester([100], ['100'], 'intcomma')
|
2011-12-24 11:01:57 +00:00
|
|
|
|
2007-02-25 16:09:30 +00:00
|
|
|
def test_intword(self):
|
|
|
|
test_list = ('100', '1000000', '1200000', '1290000',
|
2011-05-06 13:29:58 +00:00
|
|
|
'1000000000', '2000000000', '6000000000000',
|
2011-09-23 16:45:40 +00:00
|
|
|
'1300000000000000', '3500000000000000000000',
|
|
|
|
'8100000000000000000000000000000000', None)
|
2007-02-25 16:09:30 +00:00
|
|
|
result_list = ('100', '1.0 million', '1.2 million', '1.3 million',
|
2010-12-21 15:07:43 +00:00
|
|
|
'1.0 billion', '2.0 billion', '6.0 trillion',
|
2011-09-23 16:45:40 +00:00
|
|
|
'1.3 quadrillion', '3.5 sextillion',
|
|
|
|
'8.1 decillion', None)
|
2013-03-17 18:17:35 -03:00
|
|
|
with translation.override('en'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'intword')
|
2007-02-25 16:09:30 +00:00
|
|
|
|
2011-05-06 13:29:58 +00:00
|
|
|
def test_i18n_intcomma(self):
|
|
|
|
test_list = (100, 1000, 10123, 10311, 1000000, 1234567.25,
|
|
|
|
'100', '1000', '10123', '10311', '1000000', None)
|
|
|
|
result_list = ('100', '1.000', '10.123', '10.311', '1.000.000', '1.234.567,25',
|
|
|
|
'100', '1.000', '10.123', '10.311', '1.000.000', None)
|
2013-09-22 13:49:46 +02:00
|
|
|
with self.settings(USE_L10N=True, USE_THOUSAND_SEPARATOR=True):
|
|
|
|
with translation.override('de'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'intcomma')
|
2011-05-06 13:29:58 +00:00
|
|
|
|
|
|
|
def test_i18n_intword(self):
|
|
|
|
test_list = ('100', '1000000', '1200000', '1290000',
|
2011-09-23 16:45:40 +00:00
|
|
|
'1000000000', '2000000000', '6000000000000')
|
2011-05-06 13:29:58 +00:00
|
|
|
result_list = ('100', '1,0 Million', '1,2 Millionen', '1,3 Millionen',
|
|
|
|
'1,0 Milliarde', '2,0 Milliarden', '6,0 Billionen')
|
2013-09-22 13:49:46 +02:00
|
|
|
with self.settings(USE_L10N=True, USE_THOUSAND_SEPARATOR=True):
|
|
|
|
with translation.override('de'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'intword')
|
2011-05-06 13:29:58 +00:00
|
|
|
|
2007-02-25 16:09:30 +00:00
|
|
|
def test_apnumber(self):
|
Merged Unicode branch into trunk (r4952:5608). This should be fully
backwards compatible for all practical purposes.
Fixed #2391, #2489, #2996, #3322, #3344, #3370, #3406, #3432, #3454, #3492, #3582, #3690, #3878, #3891, #3937, #4039, #4141, #4227, #4286, #4291, #4300, #4452, #4702
git-svn-id: http://code.djangoproject.com/svn/django/trunk@5609 bcc190cf-cafb-0310-a4f2-bffc1f526a37
2007-07-04 12:11:04 +00:00
|
|
|
test_list = [str(x) for x in range(1, 11)]
|
2010-12-21 15:07:43 +00:00
|
|
|
test_list.append(None)
|
2012-06-07 18:08:47 +02:00
|
|
|
result_list = ('one', 'two', 'three', 'four', 'five', 'six',
|
|
|
|
'seven', 'eight', 'nine', '10', None)
|
2013-03-17 18:17:35 -03:00
|
|
|
with translation.override('en'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'apnumber')
|
2007-02-25 16:09:30 +00:00
|
|
|
|
2007-08-20 08:50:08 +00:00
|
|
|
def test_naturalday(self):
|
2011-11-18 13:01:06 +00:00
|
|
|
today = datetime.date.today()
|
|
|
|
yesterday = today - datetime.timedelta(days=1)
|
|
|
|
tomorrow = today + datetime.timedelta(days=1)
|
|
|
|
someday = today - datetime.timedelta(days=10)
|
2012-06-07 18:08:47 +02:00
|
|
|
notdate = "I'm not a date value"
|
2007-08-20 08:50:08 +00:00
|
|
|
|
2010-12-21 15:07:43 +00:00
|
|
|
test_list = (today, yesterday, tomorrow, someday, notdate, None)
|
2007-08-20 08:50:08 +00:00
|
|
|
someday_result = defaultfilters.date(someday)
|
2012-06-07 18:08:47 +02:00
|
|
|
result_list = (_('today'), _('yesterday'), _('tomorrow'),
|
|
|
|
someday_result, "I'm not a date value", None)
|
2007-08-20 08:50:08 +00:00
|
|
|
self.humanize_tester(test_list, result_list, 'naturalday')
|
|
|
|
|
2011-04-22 12:02:55 +00:00
|
|
|
def test_naturalday_tz(self):
|
2011-11-18 13:01:06 +00:00
|
|
|
today = datetime.date.today()
|
2013-09-08 10:43:33 +02:00
|
|
|
tz_one = get_fixed_timezone(-720)
|
|
|
|
tz_two = get_fixed_timezone(720)
|
2011-04-22 12:02:55 +00:00
|
|
|
|
|
|
|
# Can be today or yesterday
|
2011-11-18 13:01:06 +00:00
|
|
|
date_one = datetime.datetime(today.year, today.month, today.day, tzinfo=tz_one)
|
2012-07-19 23:02:22 +02:00
|
|
|
naturalday_one = humanize.naturalday(date_one)
|
2011-04-22 12:02:55 +00:00
|
|
|
# Can be today or tomorrow
|
2011-11-18 13:01:06 +00:00
|
|
|
date_two = datetime.datetime(today.year, today.month, today.day, tzinfo=tz_two)
|
2012-07-19 23:02:22 +02:00
|
|
|
naturalday_two = humanize.naturalday(date_two)
|
2011-04-22 12:02:55 +00:00
|
|
|
|
|
|
|
# As 24h of difference they will never be the same
|
|
|
|
self.assertNotEqual(naturalday_one, naturalday_two)
|
2011-05-17 10:16:12 +00:00
|
|
|
|
2014-02-09 16:36:08 +00:00
|
|
|
@skipIf(pytz is None, "this test requires pytz")
|
2012-07-19 23:02:22 +02:00
|
|
|
def test_naturalday_uses_localtime(self):
|
|
|
|
# Regression for #18504
|
2012-10-28 22:35:01 +01:00
|
|
|
# This is 2012-03-08HT19:30:00-06:00 in America/Chicago
|
2012-07-19 23:02:22 +02:00
|
|
|
dt = datetime.datetime(2012, 3, 9, 1, 30, tzinfo=utc)
|
|
|
|
|
|
|
|
orig_humanize_datetime, humanize.datetime = humanize.datetime, MockDateTime
|
|
|
|
try:
|
2013-09-22 13:49:46 +02:00
|
|
|
with override_settings(TIME_ZONE="America/Chicago", USE_TZ=True):
|
|
|
|
with translation.override('en'):
|
|
|
|
self.humanize_tester([dt], ['yesterday'], 'naturalday')
|
2012-07-19 23:02:22 +02:00
|
|
|
finally:
|
|
|
|
humanize.datetime = orig_humanize_datetime
|
|
|
|
|
2011-05-17 10:16:12 +00:00
|
|
|
def test_naturaltime(self):
|
2011-11-18 13:01:06 +00:00
|
|
|
class naive(datetime.tzinfo):
|
|
|
|
def utcoffset(self, dt):
|
|
|
|
return None
|
2011-05-17 10:16:12 +00:00
|
|
|
test_list = [
|
|
|
|
now,
|
2011-11-18 13:01:06 +00:00
|
|
|
now - datetime.timedelta(seconds=1),
|
|
|
|
now - datetime.timedelta(seconds=30),
|
|
|
|
now - datetime.timedelta(minutes=1, seconds=30),
|
|
|
|
now - datetime.timedelta(minutes=2),
|
|
|
|
now - datetime.timedelta(hours=1, minutes=30, seconds=30),
|
|
|
|
now - datetime.timedelta(hours=23, minutes=50, seconds=50),
|
|
|
|
now - datetime.timedelta(days=1),
|
|
|
|
now - datetime.timedelta(days=500),
|
|
|
|
now + datetime.timedelta(seconds=1),
|
|
|
|
now + datetime.timedelta(seconds=30),
|
|
|
|
now + datetime.timedelta(minutes=1, seconds=30),
|
|
|
|
now + datetime.timedelta(minutes=2),
|
|
|
|
now + datetime.timedelta(hours=1, minutes=30, seconds=30),
|
|
|
|
now + datetime.timedelta(hours=23, minutes=50, seconds=50),
|
|
|
|
now + datetime.timedelta(days=1),
|
2012-07-19 23:02:22 +02:00
|
|
|
now + datetime.timedelta(days=2, hours=6),
|
2011-11-18 13:01:06 +00:00
|
|
|
now + datetime.timedelta(days=500),
|
|
|
|
now.replace(tzinfo=naive()),
|
|
|
|
now.replace(tzinfo=utc),
|
2011-05-17 10:16:12 +00:00
|
|
|
]
|
|
|
|
result_list = [
|
|
|
|
'now',
|
|
|
|
'a second ago',
|
2013-05-18 13:58:45 +02:00
|
|
|
'30\xa0seconds ago',
|
2011-05-17 10:16:12 +00:00
|
|
|
'a minute ago',
|
2013-05-18 13:58:45 +02:00
|
|
|
'2\xa0minutes ago',
|
2011-05-17 10:16:12 +00:00
|
|
|
'an hour ago',
|
2013-05-18 13:58:45 +02:00
|
|
|
'23\xa0hours ago',
|
|
|
|
'1\xa0day ago',
|
|
|
|
'1\xa0year, 4\xa0months ago',
|
2011-05-17 10:16:12 +00:00
|
|
|
'a second from now',
|
2013-05-18 13:58:45 +02:00
|
|
|
'30\xa0seconds from now',
|
2011-05-17 10:16:12 +00:00
|
|
|
'a minute from now',
|
2013-05-18 13:58:45 +02:00
|
|
|
'2\xa0minutes from now',
|
2011-05-17 10:16:12 +00:00
|
|
|
'an hour from now',
|
2013-05-18 13:58:45 +02:00
|
|
|
'23\xa0hours from now',
|
|
|
|
'1\xa0day from now',
|
|
|
|
'2\xa0days, 6\xa0hours from now',
|
|
|
|
'1\xa0year, 4\xa0months from now',
|
2011-11-18 13:01:06 +00:00
|
|
|
'now',
|
|
|
|
'now',
|
2011-05-17 10:16:12 +00:00
|
|
|
]
|
2012-07-19 23:02:22 +02:00
|
|
|
# Because of the DST change, 2 days and 6 hours after the chosen
|
|
|
|
# date in naive arithmetic is only 2 days and 5 hours after in
|
|
|
|
# aware arithmetic.
|
|
|
|
result_list_with_tz_support = result_list[:]
|
2013-05-18 13:58:45 +02:00
|
|
|
assert result_list_with_tz_support[-4] == '2\xa0days, 6\xa0hours from now'
|
|
|
|
result_list_with_tz_support[-4] == '2\xa0days, 5\xa0hours from now'
|
2012-07-19 23:02:22 +02:00
|
|
|
|
|
|
|
orig_humanize_datetime, humanize.datetime = humanize.datetime, MockDateTime
|
2011-10-22 06:06:44 +00:00
|
|
|
try:
|
2013-03-17 18:17:35 -03:00
|
|
|
with translation.override('en'):
|
|
|
|
self.humanize_tester(test_list, result_list, 'naturaltime')
|
|
|
|
with override_settings(USE_TZ=True):
|
|
|
|
self.humanize_tester(
|
|
|
|
test_list, result_list_with_tz_support, 'naturaltime')
|
2011-10-22 06:06:44 +00:00
|
|
|
finally:
|
2011-10-26 15:17:43 +00:00
|
|
|
humanize.datetime = orig_humanize_datetime
|
2014-09-05 16:55:24 -07:00
|
|
|
|
|
|
|
def test_naturaltime_as_documented(self):
|
|
|
|
"""
|
|
|
|
#23340 -- Verify the documented behavior of humanize.naturaltime.
|
|
|
|
"""
|
|
|
|
time_format = '%d %b %Y %H:%M:%S'
|
|
|
|
documented_now = datetime.datetime.strptime('17 Feb 2007 16:30:00', time_format)
|
|
|
|
|
|
|
|
test_data = (
|
|
|
|
('17 Feb 2007 16:30:00', 'now'),
|
|
|
|
('17 Feb 2007 16:29:31', '29 seconds ago'),
|
|
|
|
('17 Feb 2007 16:29:00', 'a minute ago'),
|
|
|
|
('17 Feb 2007 16:25:35', '4 minutes ago'),
|
|
|
|
('17 Feb 2007 15:30:29', '59 minutes ago'),
|
|
|
|
('17 Feb 2007 15:30:01', '59 minutes ago'),
|
|
|
|
('17 Feb 2007 15:30:00', 'an hour ago'),
|
|
|
|
('17 Feb 2007 13:31:29', '2 hours ago'),
|
|
|
|
('16 Feb 2007 13:31:29', '1 day, 2 hours ago'),
|
|
|
|
('16 Feb 2007 13:30:01', '1 day, 2 hours ago'),
|
|
|
|
('16 Feb 2007 13:30:00', '1 day, 3 hours ago'),
|
|
|
|
('17 Feb 2007 16:30:30', '30 seconds from now'),
|
|
|
|
('17 Feb 2007 16:30:29', '29 seconds from now'),
|
|
|
|
('17 Feb 2007 16:31:00', 'a minute from now'),
|
|
|
|
('17 Feb 2007 16:34:35', '4 minutes from now'),
|
|
|
|
('17 Feb 2007 17:30:29', 'an hour from now'),
|
|
|
|
('17 Feb 2007 18:31:29', '2 hours from now'),
|
|
|
|
('18 Feb 2007 16:31:29', '1 day from now'),
|
|
|
|
('26 Feb 2007 18:31:29', '1 week, 2 days from now'),
|
|
|
|
)
|
|
|
|
|
|
|
|
class DocumentedMockDateTime(datetime.datetime):
|
|
|
|
@classmethod
|
|
|
|
def now(cls, tz=None):
|
|
|
|
if tz is None or tz.utcoffset(documented_now) is None:
|
|
|
|
return documented_now
|
|
|
|
else:
|
|
|
|
return documented_now.replace(tzinfo=tz) + tz.utcoffset(now)
|
|
|
|
|
2014-09-06 07:59:27 -04:00
|
|
|
orig_humanize_datetime = humanize.datetime
|
2014-09-05 16:55:24 -07:00
|
|
|
humanize.datetime = DocumentedMockDateTime
|
2014-09-06 07:59:27 -04:00
|
|
|
try:
|
|
|
|
for test_time_string, expected_natural_time in test_data:
|
|
|
|
test_time = datetime.datetime.strptime(test_time_string, time_format)
|
|
|
|
natural_time = humanize.naturaltime(test_time).replace('\xa0', ' ')
|
2014-09-10 12:26:15 -04:00
|
|
|
self.assertEqual(expected_natural_time, natural_time)
|
2014-09-06 07:59:27 -04:00
|
|
|
finally:
|
|
|
|
humanize.datetime = orig_humanize_datetime
|