{
  "id": "marshmallow-code__marshmallow-1252",
  "question": "ISO8601 DateTimes ending with Z considered not valid in 2.19.4\nProbably related to #1247 and #1234 - in marshmallow `2.19.4`, with `python-dateutil` _not_ installed, it seems that loading a datetime in ISO8601 that ends in `Z` (UTC time) results in an error:\r\n\r\n```python\r\nclass Foo(Schema):\r\n    date = DateTime(required=True)\r\n\r\n\r\nfoo_schema = Foo(strict=True)\r\n\r\na_date_with_z = '2019-06-17T00:57:41.000Z'\r\nfoo_schema.load({'date': a_date_with_z})\r\n```\r\n\r\n```\r\nmarshmallow.exceptions.ValidationError: {'date': ['Not a valid datetime.']}\r\n```\r\n\r\nDigging a bit deeper, it seems [`from_iso_datetime`](https://github.com/marshmallow-code/marshmallow/blob/dev/src/marshmallow/utils.py#L213-L215) is failing with a `unconverted data remains: Z` - my understanding of the spec is rather limited, but it seems that they are indeed valid ISO8601 dates (and in `marshmallow==2.19.3` and earlier, the previous snippet seems to work without raising validation errors).\r\n\n",
  "answer": "{\"task_id\": \"14f8a3ec-eb7c-4f37-b1db-8b2258d85f67\", \"workflow_run_id\": \"c9b65104-1a64-44cb-af05-71bd130fdc72\", \"data\": {\"id\": \"c9b65104-1a64-44cb-af05-71bd130fdc72\", \"workflow_id\": \"4faf996e-a613-49ba-90e4-a0af9c740cdf\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/src/marshmallow/utils.py b/src/marshmallow/utils.py\\\\nindex cdef123..adef456 100644\\\\n--- a/src/marshmallow/utils.py\\\\n+++ b/src/marshmallow/utils.py\\\\n@@ -213,6 +213,8 @@ def from_iso_datetime(value):\\\\n     :returns: A :class:`datetime.datetime` object\\\\n     :rtype: :class:`datetime.datetime`\\\\n     \\\"\\\"\\\"\\\\n+    # Handle UTC 'Z' suffix by replacing it with '+00:00'\\\\n+    if value.endswith('Z'):\\\\n+        value = value[:-1] + '+00:00'\\\\n     try:\\\\n         return datetime.datetime.strptime(value, '%Y-%m-%dT%H:%M:%S.%f%z')\\\\n     except ValueError:\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch directly addresses the issue by modifying the `from_iso_datetime` function to handle the 'Z' suffix, which is a common representation for UTC in ISO8601. It is a minimal change with low risk of introducing new bugs and aligns well with best practices.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/src/marshmallow/fields.py b/src/marshmallow/fields.py\\\\nindex abcdefg..bcdefgh 100644\\\\n--- a/src/marshmallow/fields.py\\\\n+++ b/src/marshmallow/fields.py\\\\n@@ -610,7 +610,11 @@ class DateTime(Field):\\\\n         try:\\\\n             if not self.dateformat:\\\\n                 return utils.from_iso_datetime(value)\\\\n+            elif value.endswith('Z'):\\\\n+                # Handle datetime strings ending with 'Z' suffix as UTC\\\\n+                return datetime.datetime.strptime(value[:-1] + '+0000', '%Y-%m-%dT%H:%M:%S.%f%z')\\\\n             return datetime.datetime.strptime(value, self.dateformat)\\\\n+        except ValueError as ve:\\\\n+            raise ValidationError('Not a valid datetime: %s' % ve)\\\\n         except (TypeError, ValueError):\\\\n             self.fail('invalid')\\\\n \\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch modifies the `DateTime` field's `_deserialize` method to handle 'Z' as UTC. While it addresses the issue, it introduces additional complexity by handling the 'Z' suffix in multiple places, which could lead to maintenance challenges.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/tests/test_utils.py b/tests/test_utils.py\\\\nindex 123ab45..678cd90 100644\\\\n--- a/tests/test_utils.py\\\\n+++ b/tests/test_utils.py\\\\n@@ -50,6 +50,11 @@ class TestFromISO:\\\\n     def test_from_iso_datetime(self):\\\\n         assert utils.from_iso_datetime('2010-11-12T13:14:15.000+00:00') == datetime(2010, 11, 12, 13, 14, 15, 0, tzinfo=utc)\\\\n \\\\n+    def test_from_iso_datetime_utc_z(self):\\\\n+        # Test handling datetime string ending with 'Z'\\\\n+        dt_str = '2019-06-17T00:57:41.000Z'\\\\n+        expected_dt = datetime(2019, 6, 17, 0, 57, 41, 0, tzinfo=utc)\\\\n+        assert utils.from_iso_datetime(dt_str) == expected_dt\\\\n\\\\n     def test_from_iso_datetime_invalid(self):\\\\n         with pytest.raises(TypeError):\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.8,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch adds a test case to ensure that the `from_iso_datetime` function correctly handles datetime strings ending with 'Z'. It is a valuable addition for validation but does not directly fix the issue.\\\",\\n      \\\"recommendation\\\": \\\"Useful for validation\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 4,\\n      \\\"patch\\\": \\\"diff --git a/tests/test_fields.py b/tests/test_fields.py\\\\nindex a1b2c3d..d4e5f6g 100644\\\\n--- a/tests/test_fields.py\\\\n+++ b/tests/test_fields.py\\\\n@@ -120,6 +120,12 @@ class TestDateTimeField:\\\\n             ('2014-12-23T09:12:12'),\\\\n         ]\\\\n \\\\n+    def test_datetime_field_with_z_suffix(self):\\\\n+        # Test DateTime field with 'Z' suffix\\\\n+        field = fields.DateTime()\\\\n+        input_data = '2019-06-17T00:57:41.000Z'\\\\n+        expected = datetime(2019, 6, 17, 0, 57, 41, tzinfo=utc)\\\\n+        assert field.deserialize(input_data) == expected\\\\n\\\\n     def test_datetime_field_custom(self):\\\\n         field = fields.DateTime(format='%m-%d-%Y')\\\\n\\\",\\n      \\\"rank\\\": 4,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.75,\\n        \\\"code_quality\\\": 0.75,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch adds a test case for the `DateTime` field to ensure it handles 'Z' suffix correctly. It is important for ensuring the robustness of the solution but does not directly address the issue.\\\",\\n      \\\"recommendation\\\": \\\"Useful for validation\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 1 is the most effective solution as it directly addresses the issue with minimal changes and low risk. Patch 2 also fixes the issue but introduces complexity by handling the 'Z' suffix in multiple places. Patches 3 and 4 provide valuable test coverage to ensure the robustness of the solution but do not directly fix the issue.\\\"\\n}\\n```\", \"generated_tests\": \"```json\\n{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_issue():\\\\n    class Foo(Schema):\\\\n        date = DateTime(required=True)\\\\n\\\\n    foo_schema = Foo(strict=True)\\\\n\\\\n    a_date_with_z = '2019-06-17T00:57:41.000Z'\\\\n    with pytest.raises(ValidationError):\\\\n        foo_schema.load({'date': a_date_with_z})\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by attempting to load a datetime string ending with 'Z' using marshmallow Schema\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise a validation error before applying the patch\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    class Foo(Schema):\\\\n        date = DateTime(required=True)\\\\n\\\\n    foo_schema = Foo(strict=True)\\\\n\\\\n    a_date_with_z = '2019-06-17T00:57:41.000Z'\\\\n    loaded_data = foo_schema.load({'date': a_date_with_z})\\\\n    assert loaded_data['date'].isoformat() == '2019-06-17T00:57:41+00:00'\\\\n\\\\n    another_date_with_z = '2022-01-01T00:00:00.000Z'\\\\n    loaded_data = foo_schema.load({'date': another_date_with_z})\\\\n    assert loaded_data['date'].isoformat() == '2022-01-01T00:00:00+00:00'\\\",\\n      \\\"description\\\": \\\"This test covers edge cases by loading datetime strings ending with 'Z' and verifying the correct parsing and handling after applying the patch\\\",\\n      \\\"expected_behavior\\\": \\\"The loaded datetime values should match the expected UTC time format\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    class Foo(Schema):\\\\n        date = DateTime(required=True)\\\\n\\\\n    foo_schema = Foo(strict=True)\\\\n\\\\n    a_date_with_z = '2019-06-17T00:57:41.000Z'\\\\n    loaded_data = foo_schema.load({'date': a_date_with_z})\\\\n    assert loaded_data['date'].isoformat() == '2019-06-17T00:57:41+00:00'\\\",\\n      \\\"description\\\": \\\"This test validates that the patch correctly handles datetime strings ending with 'Z' by loading and verifying the parsed datetime object\\\",\\n      \\\"expected_behavior\\\": \\\"The loaded datetime value should match the expected UTC time format after applying the patch\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases generated to reproduce the original issue, cover edge cases, and validate the patches for handling ISO8601 datetime strings ending with 'Z'.\\\"\\n}\\n```\"}, \"error\": \"\", \"elapsed_time\": 230.8503, \"total_tokens\": 17016, \"total_steps\": 9, \"created_at\": 1753365493, \"finished_at\": 1753365724}}"
}