Fixed #30803 -- Allowed comma separators for milliseconds in django.utils.dateparse functions.
Co-Authored-By: Ben Wilber <benwilber@gmail.com>
This commit is contained in:
parent
42b23d1e79
commit
1f817daa20
|
@ -16,13 +16,13 @@ date_re = _lazy_re_compile(
|
||||||
|
|
||||||
time_re = _lazy_re_compile(
|
time_re = _lazy_re_compile(
|
||||||
r'(?P<hour>\d{1,2}):(?P<minute>\d{1,2})'
|
r'(?P<hour>\d{1,2}):(?P<minute>\d{1,2})'
|
||||||
r'(?::(?P<second>\d{1,2})(?:\.(?P<microsecond>\d{1,6})\d{0,6})?)?'
|
r'(?::(?P<second>\d{1,2})(?:[\.,](?P<microsecond>\d{1,6})\d{0,6})?)?'
|
||||||
)
|
)
|
||||||
|
|
||||||
datetime_re = _lazy_re_compile(
|
datetime_re = _lazy_re_compile(
|
||||||
r'(?P<year>\d{4})-(?P<month>\d{1,2})-(?P<day>\d{1,2})'
|
r'(?P<year>\d{4})-(?P<month>\d{1,2})-(?P<day>\d{1,2})'
|
||||||
r'[T ](?P<hour>\d{1,2}):(?P<minute>\d{1,2})'
|
r'[T ](?P<hour>\d{1,2}):(?P<minute>\d{1,2})'
|
||||||
r'(?::(?P<second>\d{1,2})(?:\.(?P<microsecond>\d{1,6})\d{0,6})?)?'
|
r'(?::(?P<second>\d{1,2})(?:[\.,](?P<microsecond>\d{1,6})\d{0,6})?)?'
|
||||||
r'(?P<tzinfo>Z|[+-]\d{2}(?::?\d{2})?)?$'
|
r'(?P<tzinfo>Z|[+-]\d{2}(?::?\d{2})?)?$'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -33,7 +33,7 @@ standard_duration_re = _lazy_re_compile(
|
||||||
r'((?:(?P<hours>\d+):)(?=\d+:\d+))?'
|
r'((?:(?P<hours>\d+):)(?=\d+:\d+))?'
|
||||||
r'(?:(?P<minutes>\d+):)?'
|
r'(?:(?P<minutes>\d+):)?'
|
||||||
r'(?P<seconds>\d+)'
|
r'(?P<seconds>\d+)'
|
||||||
r'(?:\.(?P<microseconds>\d{1,6})\d{0,6})?'
|
r'(?:[\.,](?P<microseconds>\d{1,6})\d{0,6})?'
|
||||||
r'$'
|
r'$'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
|
@ -139,6 +139,10 @@ The functions defined in this module share the following properties:
|
||||||
UTC offsets aren't supported; if ``value`` describes one, the result is
|
UTC offsets aren't supported; if ``value`` describes one, the result is
|
||||||
``None``.
|
``None``.
|
||||||
|
|
||||||
|
.. versionchanged:: 3.1
|
||||||
|
|
||||||
|
Support for comma separators for milliseconds was added.
|
||||||
|
|
||||||
.. function:: parse_datetime(value)
|
.. function:: parse_datetime(value)
|
||||||
|
|
||||||
Parses a string and returns a :class:`datetime.datetime`.
|
Parses a string and returns a :class:`datetime.datetime`.
|
||||||
|
@ -146,18 +150,23 @@ The functions defined in this module share the following properties:
|
||||||
UTC offsets are supported; if ``value`` describes one, the result's
|
UTC offsets are supported; if ``value`` describes one, the result's
|
||||||
``tzinfo`` attribute is a :class:`datetime.timezone` instance.
|
``tzinfo`` attribute is a :class:`datetime.timezone` instance.
|
||||||
|
|
||||||
|
.. versionchanged:: 3.1
|
||||||
|
|
||||||
|
Support for comma separators for milliseconds was added.
|
||||||
|
|
||||||
.. function:: parse_duration(value)
|
.. function:: parse_duration(value)
|
||||||
|
|
||||||
Parses a string and returns a :class:`datetime.timedelta`.
|
Parses a string and returns a :class:`datetime.timedelta`.
|
||||||
|
|
||||||
Expects data in the format ``"DD HH:MM:SS.uuuuuu"`` or as specified by ISO
|
Expects data in the format ``"DD HH:MM:SS.uuuuuu"``,
|
||||||
8601 (e.g. ``P4DT1H15M20S`` which is equivalent to ``4 1:15:20``) or
|
``"DD HH:MM:SS,uuuuuu"``, or as specified by ISO 8601 (e.g.
|
||||||
PostgreSQL's day-time interval format (e.g. ``3 days 04:05:06``).
|
``P4DT1H15M20S`` which is equivalent to ``4 1:15:20``) or PostgreSQL's
|
||||||
|
day-time interval format (e.g. ``3 days 04:05:06``).
|
||||||
|
|
||||||
.. versionchanged:: 3.1
|
.. versionchanged:: 3.1
|
||||||
|
|
||||||
Support for comma separators for decimal fractions in the ISO 8601
|
Support for comma separators for decimal fractions in the ISO 8601
|
||||||
format was added.
|
format and for the format ``"DD HH:MM:SS,uuuuuu"`` was added.
|
||||||
|
|
||||||
``django.utils.decorators``
|
``django.utils.decorators``
|
||||||
===========================
|
===========================
|
||||||
|
|
|
@ -259,6 +259,11 @@ Utilities
|
||||||
* :func:`~django.utils.dateparse.parse_duration` now supports comma separators
|
* :func:`~django.utils.dateparse.parse_duration` now supports comma separators
|
||||||
for decimal fractions in the ISO 8601 format.
|
for decimal fractions in the ISO 8601 format.
|
||||||
|
|
||||||
|
* :func:`~django.utils.dateparse.parse_datetime`,
|
||||||
|
:func:`~django.utils.dateparse.parse_duration`, and
|
||||||
|
:func:`~django.utils.dateparse.parse_time` now support comma separators for
|
||||||
|
milliseconds.
|
||||||
|
|
||||||
Validators
|
Validators
|
||||||
~~~~~~~~~~
|
~~~~~~~~~~
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,7 @@ class DateParseTests(unittest.TestCase):
|
||||||
self.assertEqual(parse_time('09:15:00'), time(9, 15))
|
self.assertEqual(parse_time('09:15:00'), time(9, 15))
|
||||||
self.assertEqual(parse_time('10:10'), time(10, 10))
|
self.assertEqual(parse_time('10:10'), time(10, 10))
|
||||||
self.assertEqual(parse_time('10:20:30.400'), time(10, 20, 30, 400000))
|
self.assertEqual(parse_time('10:20:30.400'), time(10, 20, 30, 400000))
|
||||||
|
self.assertEqual(parse_time('10:20:30,400'), time(10, 20, 30, 400000))
|
||||||
self.assertEqual(parse_time('4:8:16'), time(4, 8, 16))
|
self.assertEqual(parse_time('4:8:16'), time(4, 8, 16))
|
||||||
# Invalid inputs
|
# Invalid inputs
|
||||||
self.assertIsNone(parse_time('091500'))
|
self.assertIsNone(parse_time('091500'))
|
||||||
|
@ -38,6 +39,7 @@ class DateParseTests(unittest.TestCase):
|
||||||
('2012-04-23T10:20:30.400+02:30', datetime(2012, 4, 23, 10, 20, 30, 400000, get_fixed_timezone(150))),
|
('2012-04-23T10:20:30.400+02:30', datetime(2012, 4, 23, 10, 20, 30, 400000, get_fixed_timezone(150))),
|
||||||
('2012-04-23T10:20:30.400+02', datetime(2012, 4, 23, 10, 20, 30, 400000, get_fixed_timezone(120))),
|
('2012-04-23T10:20:30.400+02', datetime(2012, 4, 23, 10, 20, 30, 400000, get_fixed_timezone(120))),
|
||||||
('2012-04-23T10:20:30.400-02', datetime(2012, 4, 23, 10, 20, 30, 400000, get_fixed_timezone(-120))),
|
('2012-04-23T10:20:30.400-02', datetime(2012, 4, 23, 10, 20, 30, 400000, get_fixed_timezone(-120))),
|
||||||
|
('2012-04-23T10:20:30,400-02', datetime(2012, 4, 23, 10, 20, 30, 400000, get_fixed_timezone(-120))),
|
||||||
)
|
)
|
||||||
for source, expected in valid_inputs:
|
for source, expected in valid_inputs:
|
||||||
with self.subTest(source=source):
|
with self.subTest(source=source):
|
||||||
|
@ -104,6 +106,7 @@ class DurationParseTests(unittest.TestCase):
|
||||||
('15:30.0001', timedelta(minutes=15, seconds=30, microseconds=100)),
|
('15:30.0001', timedelta(minutes=15, seconds=30, microseconds=100)),
|
||||||
('15:30.00001', timedelta(minutes=15, seconds=30, microseconds=10)),
|
('15:30.00001', timedelta(minutes=15, seconds=30, microseconds=10)),
|
||||||
('15:30.000001', timedelta(minutes=15, seconds=30, microseconds=1)),
|
('15:30.000001', timedelta(minutes=15, seconds=30, microseconds=1)),
|
||||||
|
('15:30,000001', timedelta(minutes=15, seconds=30, microseconds=1)),
|
||||||
)
|
)
|
||||||
for source, expected in test_values:
|
for source, expected in test_values:
|
||||||
with self.subTest(source=source):
|
with self.subTest(source=source):
|
||||||
|
@ -116,6 +119,7 @@ class DurationParseTests(unittest.TestCase):
|
||||||
('-15:30', timedelta(minutes=-15, seconds=-30)),
|
('-15:30', timedelta(minutes=-15, seconds=-30)),
|
||||||
('-1:15:30', timedelta(hours=-1, minutes=-15, seconds=-30)),
|
('-1:15:30', timedelta(hours=-1, minutes=-15, seconds=-30)),
|
||||||
('-30.1', timedelta(seconds=-30, milliseconds=-100)),
|
('-30.1', timedelta(seconds=-30, milliseconds=-100)),
|
||||||
|
('-30,1', timedelta(seconds=-30, milliseconds=-100)),
|
||||||
('-00:01:01', timedelta(minutes=-1, seconds=-1)),
|
('-00:01:01', timedelta(minutes=-1, seconds=-1)),
|
||||||
('-01:01', timedelta(seconds=-61)),
|
('-01:01', timedelta(seconds=-61)),
|
||||||
('-01:-01', None),
|
('-01:-01', None),
|
||||||
|
|
Loading…
Reference in New Issue