{
  "id": "django__django-10999",
  "question": "Fix parse_duration() for some negative durations\nDescription\n\t\nThe ​https://docs.djangoproject.com/en/2.1/_modules/django/utils/dateparse/ defines:\nstandard_duration_re = re.compile(\n\tr'^'\n\tr'(?:(?P<days>-?\\d+) (days?, )?)?'\n\tr'((?:(?P<hours>-?\\d+):)(?=\\d+:\\d+))?'\n\tr'(?:(?P<minutes>-?\\d+):)?'\n\tr'(?P<seconds>-?\\d+)'\n\tr'(?:\\.(?P<microseconds>\\d{1,6})\\d{0,6})?'\n\tr'$'\n)\nthat doesn't match to negative durations, because of the <hours> definition final (lookahead) part does not have '-?' in it. The following will work:\n\tr'((?:(?P<hours>-?\\d+):)(?=-?\\d+:-?\\d+))?'\n(Thanks to Konstantin Senichev for finding the fix.)\n",
  "answer": "{\"task_id\": \"8c1741e6-75cb-4ba3-9ff6-44b291d1a897\", \"workflow_run_id\": \"a5712a66-c208-4514-86a6-eae1163a1c91\", \"data\": {\"id\": \"a5712a66-c208-4514-86a6-eae1163a1c91\", \"workflow_id\": \"9a0a6efa-4d15-4876-b13f-4fb6a999caa2\", \"status\": \"succeeded\", \"outputs\": {\"final_patch\": [\"--- a/django/utils/dateparse.py\\n+++ b/django/utils/dateparse.py\\n@@ -15,7 +15,7 @@ standard_duration_re = re.compile(\\n     r'^'\\n     r'(?:(?P<days>-?\\\\d+) (days?, )?)?'\\n-    r'((?:(?P<hours>-?\\\\d+):)(?=\\\\d+:\\\\d+))?'\\n+    r'((?:(?P<hours>-?\\\\d+):)(?=-?\\\\d+:-?\\\\d+))?'\\n     r'(?:(?P<minutes>-?\\\\d+):)?'\\n     r'(?P<seconds>-?\\\\d+)'\\n     r'(?:\\\\.(?P<microseconds>\\\\d{1,6})\\\\d{0,6})?'\\n\", \"--- a/django/utils/dateparse.py\\n+++ b/django/utils/dateparse.py\\n@@ -15,7 +15,7 @@ standard_duration_re = re.compile(\\n     r'^'\\n     r'(?:(?P<days>-?\\\\d+) (days?, )?)?'\\n-    r'((?:(?P<hours>-?\\\\d+):)(?=\\\\d+:\\\\d+))?'\\n+    r'((?:(?P<hours>-?\\\\d+):)(?=-?\\\\d+:-?\\\\d+))?'\\n     r'(?:(?P<minutes>-?\\\\d+):)?'\\n     r'(?P<seconds>-?\\\\d+)'\\n     r'(?:\\\\.(?P<microseconds>\\\\d{1,6})\\\\d{0,6})?'\\n--- a/tests/utils_tests/test_dateparse.py\\n+++ b/tests/utils_tests/test_dateparse.py\\n@@ -45,6 +45,12 @@ class DateParseTests(unittest.TestCase):\\n         self.assertEqual(parse_duration('1 02:03:04.000005'), expected)\\n         self.assertEqual(parse_duration('1 day, 02:03:04.000005'), expected)\\n \\n+        # Test negative durations\\n+        expected = datetime.timedelta(days=-1, hours=-2, minutes=-3, seconds=-4, microseconds=-5)\\n+        self.assertEqual(parse_duration('-1 -02:-03:-04.000005'), expected)\\n+        expected = datetime.timedelta(hours=-2, minutes=-3, seconds=-4)\\n+        self.assertEqual(parse_duration('-02:-03:-04'), expected)\\n+\\n     def test_parse_duration_invalid(self):\\n         with self.assertRaises(ValueError):\\n             parse_duration('invalid')\\n\", \"--- a/django/utils/dateparse.py\\n+++ b/django/utils/dateparse.py\\n@@ -15,7 +15,7 @@ standard_duration_re = re.compile(\\n     r'^'\\n     r'(?:(?P<days>-?\\\\d+) (days?, )?)?'\\n-    r'((?:(?P<hours>-?\\\\d+):)(?=\\\\d+:\\\\d+))?'\\n+    r'((?:(?P<hours>-?\\\\d+):)(?=[-]?\\\\d+:[-]?\\\\d+))?'\\n     r'(?:(?P<minutes>-?\\\\d+):)?'\\n     r'(?P<seconds>-?\\\\d+)'\\n     r'(?:\\\\.(?P<microseconds>\\\\d{1,6})\\\\d{0,6})?'\\n\", \"--- a/django/utils/dateparse.py\\n+++ b/django/utils/dateparse.py\\n@@ -15,7 +15,7 @@ standard_duration_re = re.compile(\\n     r'^'\\n     r'(?:(?P<days>-?\\\\d+) (days?, )?)?'\\n-    r'((?:(?P<hours>-?\\\\d+):)(?=\\\\d+:\\\\d+))?'\\n+    r'((?:(?P<hours>-?\\\\d+):)(?=-?\\\\d+:-?\\\\d+))?'\\n     r'(?:(?P<minutes>-?\\\\d+):)?'\\n     r'(?P<seconds>-?\\\\d+)'\\n     r'(?:\\\\.(?P<microseconds>\\\\d{1,6})\\\\d{0,6})?'\\n--- a/tests/utils_tests/test_dateparse.py\\n+++ b/tests/utils_tests/test_dateparse.py\\n@@ -45,6 +45,18 @@ class DateParseTests(unittest.TestCase):\\n         self.assertEqual(parse_duration('1 02:03:04.000005'), expected)\\n         self.assertEqual(parse_duration('1 day, 02:03:04.000005'), expected)\\n \\n+        # Test negative durations\\n+        expected = datetime.timedelta(days=-1, hours=-2, minutes=-3, seconds=-4, microseconds=-5)\\n+        self.assertEqual(parse_duration('-1 -02:-03:-04.000005'), expected)\\n+        \\n+        expected = datetime.timedelta(hours=-2, minutes=-3, seconds=-4)\\n+        self.assertEqual(parse_duration('-02:-03:-04'), expected)\\n+        \\n+        expected = datetime.timedelta(minutes=-30, seconds=-45)\\n+        self.assertEqual(parse_duration('-30:-45'), expected)\\n+        \\n+        expected = datetime.timedelta(seconds=-123)\\n+        self.assertEqual(parse_duration('-123'), expected)\\n+\\n     def test_parse_duration_invalid(self):\\n         with self.assertRaises(ValueError):\\n             parse_duration('invalid')\\n\", \"--- a/django/utils/dateparse.py\\n+++ b/django/utils/dateparse.py\\n@@ -18,7 +18,7 @@ standard_duration_re = re.compile(\\n     r'(?:(?P<days>-?\\\\d+) (days?, )?)?'\\n-    r'((?:(?P<hours>-?\\\\d+):)(?=\\\\d+:\\\\d+))?'\\n+    r'((?:(?P<hours>-?\\\\d+):)(?=-?\\\\d+:-?\\\\d+))?'\\n     r'(?:(?P<minutes>-?\\\\d+):)?'\\n     r'(?P<seconds>-?\\\\d+)'\\n     r'(?:\\\\.(?P<microseconds>\\\\d{1,6})\\\\d{0,6})?'\\n\", \"diff\\n--- a/django/utils/dateparse.py\\n+++ b/django/utils/dateparse.py\\n@@ -15,7 +15,7 @@ standard_duration_re = re.compile(\\n     r'^'\\n     r'(?:(?P<days>-?\\\\d+) (days?, )?)?'\\n-    r'((?:(?P<hours>-?\\\\d+):)(?=\\\\d+:\\\\d+))?'\\n+    r'((?:(?P<hours>-?\\\\d+):)(?=-?\\\\d+:-?\\\\d+))?'\\n     r'(?:(?P<minutes>-?\\\\d+):)?'\\n     r'(?P<seconds>-?\\\\d+)'\\n     r'(?:\\\\.(?P<microseconds>\\\\d{1,6})\\\\d{0,6})?'\\n--- a/tests/utils_tests/test_dateparse.py\\n+++ b/tests/utils_tests/test_dateparse.py\\n@@ -45,6 +45,18 @@ class DateParseTests(unittest.TestCase):\\n         self.assertEqual(parse_duration('1 02:03:04.000005'), expected)\\n         self.assertEqual(parse_duration('1 day, 02:03:04.000005'), expected)\\n \\n+        # Test negative durations\\n+        expected = datetime.timedelta(days=-1, hours=-2, minutes=-3, seconds=-4, microseconds=-5)\\n+        self.assertEqual(parse_duration('-1 -02:-03:-04.000005'), expected)\\n+        \\n+        expected = datetime.timedelta(hours=-2, minutes=-3, seconds=-4)\\n+        self.assertEqual(parse_duration('-02:-03:-04'), expected)\\n+        \\n+        expected = datetime.timedelta(minutes=-30, seconds=-45)\\n+        self.assertEqual(parse_duration('-30:-45'), expected)\\n+        \\n+        expected = datetime.timedelta(seconds=-123)\\n+        self.assertEqual(parse_duration('-123'), expected)\\n+\\n     def test_parse_duration_invalid(self):\\n         with self.assertRaises(ValueError):\\n             parse_duration('invalid')\\n\", \"diff\\n--- a/django/utils/dateparse.py\\n+++ b/django/utils/dateparse.py\\n@@ -15,7 +15,7 @@ standard_duration_re = re.compile(\\n     r'^'\\n     r'(?:(?P<days>-?\\\\d+) (days?, )?)?'\\n-    r'((?:(?P<hours>-?\\\\d+):)(?=\\\\d+:\\\\d+))?'\\n+    r'((?:(?P<hours>-?\\\\d+):)(?=-?\\\\d+:-?\\\\d+))?'\\n     r'(?:(?P<minutes>-?\\\\d+):)?'\\n     r'(?P<seconds>-?\\\\d+)'\\n     r'(?:\\\\.(?P<microseconds>\\\\d{1,6})\\\\d{0,6})?'\\n--- a/tests/utils_tests/test_dateparse.py\\n+++ b/tests/utils_tests/test_dateparse.py\\n@@ -45,6 +45,18 @@ class DateParseTests(unittest.TestCase):\\n         self.assertEqual(parse_duration('1 02:03:04.000005'), expected)\\n         self.assertEqual(parse_duration('1 day, 02:03:04.000005'), expected)\\n \\n+        # Test negative durations\\n+        expected = datetime.timedelta(days=-1, hours=-2, minutes=-3, seconds=-4, microseconds=-5)\\n+        self.assertEqual(parse_duration('-1 -02:-03:-04.000005'), expected)\\n+        \\n+        expected = datetime.timedelta(hours=-2, minutes=-3, seconds=-4)\\n+        self.assertEqual(parse_duration('-02:-03:-04'), expected)\\n+        \\n+        expected = datetime.timedelta(minutes=-30, seconds=-45)\\n+        self.assertEqual(parse_duration('-30:-45'), expected)\\n+        \\n+        expected = datetime.timedelta(seconds=-123)\\n+        self.assertEqual(parse_duration('-123'), expected)\\n+\\n     def test_parse_duration_invalid(self):\\n         with self.assertRaises(ValueError):\\n             parse_duration('invalid')\\n\", \"diff\\n--- a/django/utils/dateparse.py\\n+++ b/django/utils/dateparse.py\\n@@ -15,7 +15,7 @@ standard_duration_re = re.compile(\\n     r'^'\\n     r'(?:(?P<days>-?\\\\d+) (days?, )?)?'\\n-    r'((?:(?P<hours>-?\\\\d+):)(?=\\\\d+:\\\\d+))?'\\n+    r'((?:(?P<hours>-?\\\\d+):)(?=-?\\\\d+:-?\\\\d+))?'\\n     r'(?:(?P<minutes>-?\\\\d+):)?'\\n     r'(?P<seconds>-?\\\\d+)'\\n     r'(?:\\\\.(?P<microseconds>\\\\d{1,6})\\\\d{0,6})?'\\n--- a/tests/utils_tests/test_dateparse.py\\n+++ b/tests/utils_tests/test_dateparse.py\\n@@ -45,6 +45,18 @@ class DateParseTests(unittest.TestCase):\\n         self.assertEqual(parse_duration('1 02:03:04.000005'), expected)\\n         self.assertEqual(parse_duration('1 day, 02:03:04.000005'), expected)\\n \\n+        # Test negative durations\\n+        expected = datetime.timedelta(days=-1, hours=-2, minutes=-3, seconds=-4, microseconds=-5)\\n+        self.assertEqual(parse_duration('-1 -02:-03:-04.000005'), expected)\\n+        \\n+        expected = datetime.timedelta(hours=-2, minutes=-3, seconds=-4)\\n+        self.assertEqual(parse_duration('-02:-03:-04'), expected)\\n+        \\n+        expected = datetime.timedelta(minutes=-30, seconds=-45)\\n+        self.assertEqual(parse_duration('-30:-45'), expected)\\n+        \\n+        expected = datetime.timedelta(seconds=-123)\\n+        self.assertEqual(parse_duration('-123'), expected)\\n+\\n     def test_parse_duration_invalid(self):\\n         with self.assertRaises(ValueError):\\n             parse_duration('invalid')\\n\"]}, \"error\": \"\", \"elapsed_time\": 91.25867, \"total_tokens\": 40849, \"total_steps\": 30, \"created_at\": 1754647009, \"finished_at\": 1754647100}}"
}