Skip to content

Commit 2f59e94

Browse files
committed
Fixed #18728 -- Made colon optional in tzinfo
Made two-digit hours and minutes mandatory in tzinfo (the code used to crash if a one-digit representation was provided). Added standalone tests for django.utils.dateparse.
1 parent a43ecc0 commit 2f59e94

File tree

3 files changed

+52
-7
lines changed

3 files changed

+52
-7
lines changed

django/utils/dateparse.py

Lines changed: 7 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -15,16 +15,16 @@
1515
r'(?P<year>\d{4})-(?P<month>\d{1,2})-(?P<day>\d{1,2})$'
1616
)
1717

18-
datetime_re = re.compile(
19-
r'(?P<year>\d{4})-(?P<month>\d{1,2})-(?P<day>\d{1,2})'
20-
r'[T ](?P<hour>\d{1,2}):(?P<minute>\d{1,2})'
18+
time_re = re.compile(
19+
r'(?P<hour>\d{1,2}):(?P<minute>\d{1,2})'
2120
r'(?::(?P<second>\d{1,2})(?:\.(?P<microsecond>\d{1,6})\d{0,6})?)?'
22-
r'(?P<tzinfo>Z|[+-]\d{1,2}:\d{1,2})?$'
2321
)
2422

25-
time_re = re.compile(
26-
r'(?P<hour>\d{1,2}):(?P<minute>\d{1,2})'
23+
datetime_re = re.compile(
24+
r'(?P<year>\d{4})-(?P<month>\d{1,2})-(?P<day>\d{1,2})'
25+
r'[T ](?P<hour>\d{1,2}):(?P<minute>\d{1,2})'
2726
r'(?::(?P<second>\d{1,2})(?:\.(?P<microsecond>\d{1,6})\d{0,6})?)?'
27+
r'(?P<tzinfo>Z|[+-]\d{2}:?\d{2})?$'
2828
)
2929

3030
def parse_date(value):
@@ -73,7 +73,7 @@ def parse_datetime(value):
7373
if tzinfo == 'Z':
7474
tzinfo = utc
7575
elif tzinfo is not None:
76-
offset = 60 * int(tzinfo[1:3]) + int(tzinfo[4:6])
76+
offset = 60 * int(tzinfo[1:3]) + int(tzinfo[-2:])
7777
if tzinfo[0] == '-':
7878
offset = -offset
7979
tzinfo = FixedOffset(offset)
Lines changed: 44 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
from __future__ import unicode_literals
2+
3+
from datetime import date, time, datetime
4+
5+
from django.utils.dateparse import parse_date, parse_time, parse_datetime
6+
from django.utils import unittest
7+
from django.utils.tzinfo import FixedOffset
8+
9+
10+
class DateParseTests(unittest.TestCase):
11+
12+
def test_parse_date(self):
13+
# Valid inputs
14+
self.assertEqual(parse_date('2012-04-23'), date(2012, 4, 23))
15+
self.assertEqual(parse_date('2012-4-9'), date(2012, 4, 9))
16+
# Invalid inputs
17+
self.assertEqual(parse_date('20120423'), None)
18+
self.assertRaises(ValueError, parse_date, '2012-04-56')
19+
20+
def test_parse_time(self):
21+
# Valid inputs
22+
self.assertEqual(parse_time('09:15:00'), time(9, 15))
23+
self.assertEqual(parse_time('10:10'), time(10, 10))
24+
self.assertEqual(parse_time('10:20:30.400'), time(10, 20, 30, 400000))
25+
self.assertEqual(parse_time('4:8:16'), time(4, 8, 16))
26+
# Invalid inputs
27+
self.assertEqual(parse_time('091500'), None)
28+
self.assertRaises(ValueError, parse_time, '09:15:90')
29+
30+
def test_parse_datetime(self):
31+
# Valid inputs
32+
self.assertEqual(parse_datetime('2012-04-23T09:15:00'),
33+
datetime(2012, 4, 23, 9, 15))
34+
self.assertEqual(parse_datetime('2012-4-9 4:8:16'),
35+
datetime(2012, 4, 9, 4, 8, 16))
36+
self.assertEqual(parse_datetime('2012-04-23T09:15:00Z'),
37+
datetime(2012, 4, 23, 9, 15, 0, 0, FixedOffset(0)))
38+
self.assertEqual(parse_datetime('2012-4-9 4:8:16-0320'),
39+
datetime(2012, 4, 9, 4, 8, 16, 0, FixedOffset(-200)))
40+
self.assertEqual(parse_datetime('2012-04-23T10:20:30.400+02:30'),
41+
datetime(2012, 4, 23, 10, 20, 30, 400000, FixedOffset(150)))
42+
# Invalid inputs
43+
self.assertEqual(parse_datetime('20120423091500'), None)
44+
self.assertRaises(ValueError, parse_datetime, '2012-04-56T09:15:90')

tests/regressiontests/utils/tests.py

Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -26,3 +26,4 @@
2626
from .crypto import TestUtilsCryptoPBKDF2
2727
from .archive import TestZip, TestTar, TestGzipTar, TestBzip2Tar
2828
from .regex_helper import NormalizeTests
29+
from .dateparse import DateParseTests

0 commit comments

Comments
 (0)