{
  "id": "django__django-14608",
  "question": "Add `nonform` CSS class for non form errors in FormSets\nDescription\n\t \n\t\t(last modified by Ties Jan Hefting)\n\t \nForms add the nonfield CSS class for non field errors in ErrorList instances. This is documented in a section on ​rendering form error messages. Similarly, in FormSets I'd expect to see the nonform CSS class added for non form errors. This would allow a custom ErrorList to make a distinction in form field errors, non field errors (forms) and non form errors (FormSets) when rendering error messages. Therefore I'd suggest to add this nonform CSS class and document it for developers to use.\n",
  "answer": "{\"task_id\": \"21dcda86-adb2-41fb-910c-68d72ffb1e23\", \"workflow_run_id\": \"5d11872e-57b9-45c6-a690-ce438c3d3583\", \"data\": {\"id\": \"5d11872e-57b9-45c6-a690-ce438c3d3583\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/forms/formsets.py b/django/forms/formsets.py\\\\nindex abc1234..def5678 100644\\\\n--- a/django/forms/formsets.py\\\\n+++ b/django/forms/formsets.py\\\\n@@ -300,7 +300,13 @@ class BaseFormSet:\\\\n         Return an ErrorList of errors that aren't associated with a particular form.\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n-        return self.error_class(self.errors)\\\\n+        return self.error_class(self.errors, error_class='nonform')\\\\n\\\\ndiff --git a/django/forms/utils.py b/django/forms/utils.py\\\\nindex 0123456..6543210 100644\\\\n--- a/django/forms/utils.py\\\\n+++ b/django/forms/utils.py\\\\n@@ -50,8 +50,10 @@ class ErrorList(UserList):\\\\n             return ''\\\\n         error_class = 'errorlist'\\\\n+        if self.error_class:\\\\n+            error_class += ' ' + self.error_class\\\\n         return '<ul class=\\\\\\\"%s\\\\\\\">%s</ul>' % (\\\\n-            ' '.join(['errorlist'] + self._css_classes), ''.join('<li>%s</li>' % e for e in self)\\\\n+            ' '.join([error_class] + self._css_classes), ''.join('<li>%s</li>' % e for e in self)\\\\n         )\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.05,\\n        \\\"best_practices\\\": 0.95,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch modifies the existing ErrorList class to accept an additional error_class parameter, which is a clean and maintainable solution. It minimizes changes to the existing codebase and leverages existing structures, reducing the risk of introducing new bugs.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/forms/formsets.py b/django/forms/formsets.py\\\\nindex abc1234..def5678 100644\\\\n--- a/django/forms/formsets.py\\\\n+++ b/django/forms/formsets.py\\\\n@@ -300,7 +300,13 @@ class BaseFormSet:\\\\n         Return an ErrorList of errors that aren't associated with a particular form.\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n-        return self.error_class(self.errors)\\\\n+        return NonFormErrorList(self.errors)\\\\n+\\\\n+\\\\n+class NonFormErrorList(ErrorList):\\\\n+    def __init__(self, *args, **kwargs):\\\\n+        super().__init__(*args, **kwargs)\\\\n+\\\\n+    def as_ul(self):\\\\n+        if not self:\\\\n+            return ''\\\\n+        return '<ul class=\\\\\\\"{}\\\\\\\">{}</ul>'.format('nonform', ''.join('<li>%s</li>' % e for e in self))\\\\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 introduces a new NonFormErrorList class, which is a straightforward approach but adds additional complexity by creating a new class. It is less maintainable compared to modifying existing structures.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/forms/formsets.py b/django/forms/formsets.py\\\\nindex abc1234..def5678 100644\\\\n--- a/django/forms/formsets.py\\\\n+++ b/django/forms/formsets.py\\\\n@@ -300,7 +300,17 @@ class BaseFormSet:\\\\n         Return an ErrorList of errors that aren't associated with a particular form.\\\\n         \\\\\\\"\\\\\\\"\\\\\\\"\\\\n-        return self.error_class(self.errors)\\\\n+        return NonFormErrorList(self.errors)\\\\n \\\\n+\\\\n+class NonFormErrorList(ErrorList):\\\\n+    def _get_error_class(self):\\\\n+        return 'nonform'\\\\n+\\\\n+    def as_ul(self):\\\\n+        if not self:\\\\n+            return ''\\\\n+        return '<ul class=\\\\\\\"%s\\\\\\\">%s</ul>' % (\\\\n+            self._get_error_class(), ''.join('<li>%s</li>' % e for e in self)\\\\n+        )\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.75,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.75,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.7\\n      },\\n      \\\"reasoning\\\": \\\"This patch also introduces a new NonFormErrorList class but adds an unnecessary method _get_error_class, which increases complexity without significant benefit. It is less aligned with best practices compared to the other patches.\\\",\\n      \\\"recommendation\\\": \\\"Not recommended\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 2 is the best solution as it modifies the existing ErrorList class to handle non-form errors with minimal changes and risk. Patch 1 is a viable alternative but introduces a new class, adding complexity. Patch 3 is the least preferred due to unnecessary complexity and deviation from best practices.\\\"\\n}\\n```\", \"generated_tests\": \"```json\\n{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue():\\\\n    # Simulate a FormSet with a non-form error\\\\n    formset_errors = NonFormErrorList(['Error 1', 'Error 2'])\\\\n    # Validate if the 'nonform' CSS class is added when rendering as ul\\\\n    assert formset_errors.as_ul() == '<ul class=\\\\\\\"nonform\\\\\\\"><li>Error 1</li><li>Error 2</li></ul>', 'Non-form errors rendering with incorrect CSS class'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_edge_cases\\\",\\n      \\\"test_code\\\": \\\"def test_edge_cases():\\\\n    # Test an empty NonFormErrorList\\\\n    empty_errors = NonFormErrorList()\\\\n    assert empty_errors.as_ul() == '', 'Empty NonFormErrorList not rendered correctly'\\\\n    # Test NonFormErrorList with single error message\\\\n    single_error = NonFormErrorList(['Single Error'])\\\\n    assert single_error.as_ul() == '<ul class=\\\\\\\"nonform\\\\\\\"><li>Single Error</li></ul>', 'NonFormErrorList with single error not rendered correctly'\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_method_1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_method_1():\\\\n    # Validate patch method 1: Using NonFormErrorList class\\\\n    formset_errors = NonFormErrorList(['Validation Error'])\\\\n    assert formset_errors.as_ul() == '<ul class=\\\\\\\"nonform\\\\\\\"><li>Validation Error</li></ul>', 'Method 1 patch validation failed'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_method_2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_method_2():\\\\n    # Validate patch method 2: Modifying error_class in BaseFormSet\\\\n    formset_errors = ErrorList(['Validation Error'], error_class='nonform')\\\\n    assert formset_errors.as_ul() == '<ul class=\\\\\\\"nonform\\\\\\\"><li>Validation Error</li></ul>', 'Method 2 patch validation failed'\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_method_3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_method_3():\\\\n    # Validate patch method 3: Using _get_error_class in NonFormErrorList\\\\n    formset_errors = NonFormErrorList(['Validation Error'])\\\\n    assert formset_errors.as_ul() == '<ul class=\\\\\\\"nonform\\\\\\\"><li>Validation Error</li></ul>', 'Method 3 patch validation failed'\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Comprehensive test cases generated to reproduce the original issue, test edge cases, and validate each patch method for distinguishing non-form errors in FormSets.\\\"\\n}\\n```\"}, \"error\": \"\", \"elapsed_time\": 502.305189, \"total_tokens\": 24528, \"total_steps\": 9, \"created_at\": 1753306972, \"finished_at\": 1753307474}}"
}