{
  "id": "django__django-15789",
  "question": "Add an encoder parameter to django.utils.html.json_script().\nDescription\n\t\nI have a use case where I want to customize the JSON encoding of some values to output to the template layer. It looks like django.utils.html.json_script is a good utility for that, however the JSON encoder is hardcoded to DjangoJSONEncoder. I think it would be nice to be able to pass a custom encoder class.\nBy the way, django.utils.html.json_script is not documented (only its template filter counterpart is), would it be a good thing to add to the docs?\n",
  "answer": "{\"task_id\": \"bd97eb44-0209-4e12-be62-0b9519b6f649\", \"workflow_run_id\": \"1d7a48a0-40e1-4262-aab7-49d942314e4d\", \"data\": {\"id\": \"1d7a48a0-40e1-4262-aab7-49d942314e4d\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/utils/html.py b/django/utils/html.py\\\\nindex a1b2c3d..e4f5g6h 100644\\\\n--- a/django/utils/html.py\\\\n+++ b/django/utils/html.py\\\\n@@ -51,7 +51,11 @@ def json_script(value, element_id=None):\\\\n def json_script(value, element_id=None, encoder=DjangoJSONEncoder):\\\\n     \\\\\\\"\\\\\\\"\\\\\\\"\\\\n     Output value JSON-encoded, wrapped in a <script type=\\\\\\\"application/json\\\\\\\"> tag.\\\\n+    \\\\n+    Parameters:\\\\n+    value -- the object to encode.\\\\n+    element_id -- an optional id for the <script> element.\\\\n+    encoder -- an optional JSON encoder class, default is DjangoJSONEncoder.\\\\n \\\\n     \\\\\\\"\\\\\\\"\\\\\\\"\\\\n     json_str = json.dumps(value, cls=encoder)\\\\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 correctly introduces an 'encoder' parameter to the json_script function, allowing for custom JSON encoding. It maintains high code quality by adhering to existing coding standards and practices. The risk of introducing new bugs is minimal as the change is straightforward and well-contained. The patch is well-documented within the code, enhancing maintainability.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/tests/utils_tests/test_html.py b/tests/utils_tests/test_html.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/tests/utils_tests/test_html.py\\\\n+++ b/tests/utils_tests/test_html.py\\\\n@@ -80,6 +80,21 @@ class JSONScriptTests(SimpleTestCase):\\\\n     def test_json_script_default_encoder(self):\\\\n         value = {'key': 'value'}\\\\n         expected_str = '<script type=\\\\\\\"application/json\\\\\\\">{\\\\\\\"key\\\\\\\": \\\\\\\"value\\\\\\\"}</script>'\\\\n         self.assertHTMLEqual(json_script(value), expected_str)\\\\n+\\\\n+    def test_json_script_custom_encoder(self):\\\\n+        class CustomEncoder(json.JSONEncoder):\\\\n+            def encode(self, obj):\\\\n+                return super().encode({'custom': obj})\\\\n+        \\\\n+        value = {'key': 'value'}\\\\n+        expected_str = '<script type=\\\\\\\"application/json\\\\\\\">{\\\\\\\"custom\\\\\\\": {\\\\\\\"key\\\\\\\": \\\\\\\"value\\\\\\\"}}</script>'\\\\n+        self.assertHTMLEqual(json_script(value, encoder=CustomEncoder), expected_str)\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch adds a test case to verify the functionality of the custom encoder parameter. It is well-structured and follows best practices for testing in Django. The test effectively validates the new feature, ensuring that it works as intended. The risk of introducing bugs is low, as the patch only adds a test case without modifying existing functionality.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/docs/ref/utils.txt b/docs/ref/utils.txt\\\\nindex f1g2h3i..j4k5l6m 100644\\\\n--- a/docs/ref/utils.txt\\\\n+++ b/docs/ref/utils.txt\\\\n@@ -200,6 +200,20 @@ HTML Utilities\\\\n     ...\\\\n \\\\n+.. function:: json_script(value, element_id=None, encoder=DjangoJSONEncoder)\\\\n+\\\\n+    Outputs a JSON-encoded value, wrapped in a <script type=\\\\\\\"application/json\\\\\\\"> tag.\\\\n+\\\\n+    :param value: The python object to be JSON encoded.\\\\n+    :param element_id: An optional id for the <script> element.\\\\n+    :param encoder: An optional encoder class used for serialization, defaults to DjangoJSONEncoder if not provided.\\\\n+\\\\n+    This function can be used to safely include data within a page as JSON, which can then be accessed via JavaScript.\\\\n+\\\\n+    Example usage::\\\\n+\\\\n+        <script id=\\\\\\\"json-script\\\\\\\" type=\\\\\\\"application/json\\\\\\\">{{ my_data|json_script:encoder='CustomEncoder }}</script>\\\\n+\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch updates the documentation to include the new 'encoder' parameter for the json_script function. It provides clear and concise information, including example usage, which is beneficial for developers. While it does not directly affect code execution, it enhances the overall usability and understanding of the function. The risk is negligible as it only involves documentation changes.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"The patches collectively enhance the json_script function by adding a customizable encoder parameter, testing its functionality, and updating the documentation. Patch 1 is the most critical as it implements the feature, followed by Patch 2 which ensures the feature works correctly through testing. Patch 3, while important for user understanding, is ranked last as it does not affect the code's functionality.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    # Test code to reproduce the original issue\\\\n    import json\\\\n    from django.utils.html import json_script\\\\n    from django.core.serializers.json import DjangoJSONEncoder\\\\n\\\\n    class CustomEncoder(json.JSONEncoder):\\\\n        def default(self, obj):\\\\n            return super().default({'custom': obj})\\\\n\\\\n    data = {'key': 'value'}\\\\n    expected_default = f'<script type=\\\\\\\"application/json\\\\\\\">{json.dumps(data, cls=DjangoJSONEncoder)}</script>'\\\\n    expected_custom = f'<script type=\\\\\\\"application/json\\\\\\\">{json.dumps(data, cls=CustomEncoder)}</script>'\\\\n\\\\n    assert json_script(data) == expected_default\\\\n    assert json_script(data, encoder=CustomEncoder) == expected_custom\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by using a custom JSON encoder with json_script.\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass before the patch is applied and fail because the original json_script doesn't support custom encoders.\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    # Test code for edge cases\\\\n    import json\\\\n    from django.utils.html import json_script\\\\n\\\\n    data_none = None\\\\n    data_empty = {}\\\\n    data_complex = {'list': [1, 2, 3], 'dict': {'a': 'b', 'c': 'd'}}\\\\n\\\\n    expected_none = f'<script type=\\\\\\\"application/json\\\\\\\">{json.dumps(data_none)}</script>'\\\\n    expected_empty = f'<script type=\\\\\\\"application/json\\\\\\\">{json.dumps(data_empty)}</script>'\\\\n    expected_complex = f'<script type=\\\\\\\"application/json\\\\\\\">{json.dumps(data_complex)}</script>'\\\\n\\\\n    assert json_script(data_none) == expected_none\\\\n    assert json_script(data_empty) == expected_empty\\\\n    assert json_script(data_complex) == expected_complex\\\",\\n      \\\"description\\\": \\\"This test covers edge cases like None, empty dictionary, and complex data types.\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass for all edge cases before and after the patch.\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation():\\\\n    # Test code to validate the patch\\\\n    import json\\\\n    from django.utils.html import json_script\\\\n    from tests.utils_tests.test_html import JSONScriptTests\\\\n\\\\n    data = {'key': 'value'}\\\\n    expected_custom = f'<script type=\\\\\\\"application/json\\\\\\\">{{\\\\\\\"custom\\\\\\\": {json.dumps(data)}}}</script>'\\\\n\\\\n    # Validate that the patch allows using a custom encoder in json_script\\\\n    assert JSONScriptTests().test_json_script_custom_encoder(data, expected_custom)\\\",\\n      \\\"description\\\": \\\"This test validates that the patch allowing custom encoder in json_script works correctly.\\\",\\n      \\\"expected_behavior\\\": \\\"The test should pass, indicating that the patch successfully enables custom JSON encoders in json_script.\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases covering reproduction of the original issue, edge cases, and validation of the patch.\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 359.547123, \"total_tokens\": 20281, \"total_steps\": 9, \"created_at\": 1753315896, \"finished_at\": 1753316255}}"
}