{
  "id": "django__django-14730",
  "question": "Prevent developers from defining a related_name on symmetrical ManyToManyFields\nDescription\n\t\nIn ManyToManyField, if the symmetrical argument is passed, or if it's a self-referential ManyToMany relationship, the related field on the target model is not created. However, if a developer passes in the related_name not understanding this fact, they may be confused until they find the information about symmetrical relationship. Thus, it is proposed to raise an error when the user defines a ManyToManyField in this condition.\n",
  "answer": "{\"task_id\": \"885bb38f-2f6e-46e1-a6d0-4a9cb8644c52\", \"workflow_run_id\": \"955fc908-0f55-42d9-8f86-2986ff93a7ff\", \"data\": {\"id\": \"955fc908-0f55-42d9-8f86-2986ff93a7ff\", \"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/db/models/fields/related.py b/django/db/models/fields/related.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/django/db/models/fields/related.py\\\\n+++ b/django/db/models/fields/related.py\\\\n@@ -52,6 +52,10 @@ class ManyToManyField(RelatedField):\\\\n         symmetrical = kwargs.get('symmetrical', False)\\\\n \\\\n         # Validate that related_name is not set when field is symmetrical\\\\n+        if symmetrical and related_name is not None:\\\\n+            raise ValueError(\\\\n+                'Cannot set related_name on symmetrical ManyToManyField. '\\\\n+                'Either remove the related_name or set symmetrical=False.')\\\\n \\\\n         super().__init__(\\\\n             to,\\\\n@@ -150,0 +154,10 @@ class ManyToManyField(RelatedField):\\\\n+    def _check_related_name_is_not_symmetrical(self):\\\\n+        from django.core.checks import Error\\\\n+        if self.symmetrical and self.remote_field.related_name:\\\\n+            return [\\\\n+                Error(\\\\n+                    'Cannot set related_name on symmetrical ManyToManyField.',\\\\n+                    obj=self,\\\\n+                    id='fields.E001',\\\\n+                )\\\\n+            ]\\\\n+        return []\\\\n@@ -182,6 +192,7 @@ class ManyToManyField(RelatedField):\\\\n         errors = super().check(**kwargs)\\\\n         errors.extend(self._check_relationship_model_exists())\\\\n         errors.extend(self._check_field_name_in_db())\\\\n+        errors.extend(self._check_related_name_is_not_symmetrical())\\\\n         return errors\\\\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 addresses the issue by adding a check in the constructor and a separate method for validation. It raises a clear error message and integrates well with Django's existing error-checking framework. The patch is well-structured and maintains high code quality.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/fields/related.py b/django/db/models/fields/related.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/django/db/models/fields/related.py\\\\n+++ b/django/db/models/fields/related.py\\\\n@@ -52,6 +52,11 @@ class ManyToManyField(RelatedField):\\\\n         symmetrical = kwargs.get('symmetrical', False)\\\\n \\\\n         # Ensure no related_name if the field is symmetrical\\\\n+        if symmetrical and kwargs.get('related_name') is not None:\\\\n+            raise ValueError(\\\\n+                \\\\\\\"'related_name' cannot be set when 'symmetrical' is True. \\\\\\\"\\\\n+                \\\\\\\"Consider removing 'related_name' or set 'symmetrical=False'.\\\\\\\"\\\\n+            )\\\\n \\\\n         super().__init__(\\\\n             to,\\\\n@@ -150,0 +155,11 @@ class ManyToManyField(RelatedField):\\\\n+    def _validate_non_symmetrical(self):\\\\n+        from django.core.checks import Error\\\\n+        errors = []\\\\n+        if self.symmetrical and self.remote_field.related_name is not None:\\\\n+            errors.append(\\\\n+                Error(\\\\n+                    \\\\\\\"Invalid related_name when symmetrical=True.\\\\\\\",\\\\n+                    obj=self,\\\\n+                    id='fields.E003'\\\\n+                )\\\\n+            )\\\\n+        return errors\\\\n@@ -182,6 +193,7 @@ class ManyToManyField(RelatedField):\\\\n         errors = super().check(**kwargs)\\\\n         errors.extend(self._check_relationship_model_exists())\\\\n         errors.extend(self._check_field_name_in_db())\\\\n+        errors.extend(self._validate_non_symmetrical())\\\\n         return errors\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch also addresses the issue effectively by adding a check in the constructor and a validation method. The error message is clear, but slightly less informative than Patch 1. The patch is well-structured and follows best practices, though it introduces a new error ID which might require additional documentation.\\\",\\n      \\\"recommendation\\\": \\\"Recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/fields/related.py b/django/db/models/fields/related.py\\\\nindex 1234567..89abcde 100644\\\\n--- a/django/db/models/fields/related.py\\\\n+++ b/django/db/models/fields/related.py\\\\n@@ -52,6 +52,12 @@ class ManyToManyField(RelatedField):\\\\n         symmetrical = kwargs.get('symmetrical', False)\\\\n \\\\n         # Check that related_name is not defined if symmetrical is True\\\\n+        self._check_related_name_on_symmetrical(related_name, symmetrical)\\\\n \\\\n         super().__init__(\\\\n             to,\\\\n@@ -150,0 +156,10 @@ class ManyToManyField(RelatedField):\\\\n+    def _check_related_name_on_symmetrical(self, related_name, symmetrical):\\\\n+        if symmetrical and related_name is not None:\\\\n+            raise ValueError(\\\\n+                'Cannot set related_name on a symmetrical ManyToManyField.',\\\\n+            )\\\\n+\\\\n+    def _check_for_system_behaviors(self):\\\\n+        from django.core.checks import Error\\\\n+        if self.symmetrical and self.remote_field.related_name:\\\\n+            return [\\\\n+                Error(\\\\n+                    'related_name must be None if symmetrical is True.',\\\\n+                    obj=self,\\\\n+                    id='fields.E002',\\\\n+                )\\\\n+            ]\\\\n+        return []\\\\n@@ -182,6 +192,7 @@ class ManyToManyField(RelatedField):\\\\n         errors = super().check(**kwargs)\\\\n         errors.extend(self._check_relationship_model_exists())\\\\n         errors.extend(self._check_field_name_in_db())\\\\n+        errors.extend(self._check_for_system_behaviors())\\\\n         return errors\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.75\\n      },\\n      \\\"reasoning\\\": \\\"This patch introduces a check in the constructor and a validation method, similar to the others. However, it uses a slightly more complex approach with two separate methods for checking, which could be simplified. The error message is less informative, and the patch introduces a new error ID, which might require additional documentation.\\\",\\n      \\\"recommendation\\\": \\\"Acceptable\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"All patches address the issue of preventing the use of related_name with symmetrical ManyToManyFields by adding checks in the constructor and validation methods. Patch 1 is the most straightforward and maintains high code quality, making it the best choice. Patch 3 is also effective but slightly less informative. Patch 2 is the least preferred due to its complexity and less informative error message.\\\"\\n}\\n```\", \"generated_tests\": \"```json\\n{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_issue_with_symmetrical_field\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_issue_with_symmetrical_field():\\\\n    symmetrical = True\\\\n    related_name = 'test_related_name'\\\\n    try:\\\\n        field = ManyToManyField(OtherModel, symmetrical=symmetrical, related_name=related_name)\\\\n        assert False, 'No exception raised'\\\\n    except ValueError as e:\\\\n        assert str(e) == 'Cannot set related_name on symmetrical ManyToManyField. Either remove the related_name or set symmetrical=False.', 'Incorrect error message'\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by defining a ManyToManyField with symmetrical as True and providing a related_name\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise a ValueError with the correct error message\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_issue_with_self_referential_field\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_issue_with_self_referential_field():\\\\n    symmetrical = False\\\\n    related_name = 'test_related_name'\\\\n    try:\\\\n        field = ManyToManyField('self', symmetrical=symmetrical, related_name=related_name)\\\\n        assert False, 'No exception raised'\\\\n    except ValueError as e:\\\\n        assert str(e) == 'Cannot set related_name on symmetrical ManyToManyField. Either remove the related_name or set symmetrical=False.', 'Incorrect error message'\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue by defining a self-referential ManyToManyField with a related_name\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise a ValueError with the correct error message\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch_1\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch_1():\\\\n    symmetrical = True\\\\n    related_name = 'test_related_name'\\\\n    try:\\\\n        field = ManyToManyField(OtherModel, symmetrical=symmetrical, related_name=related_name)\\\\n        assert False, 'No exception raised'\\\\n    except ValueError as e:\\\\n        assert str(e) == 'Cannot set related_name on symmetrical ManyToManyField. Either remove the related_name or set symmetrical=False.', 'Incorrect error message'\\\",\\n      \\\"description\\\": \\\"This test validates Patch 1 by defining a ManyToManyField with symmetrical as True and providing a related_name\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise a ValueError with the correct error message as specified in Patch 1\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch_2\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch_2():\\\\n    symmetrical = True\\\\n    related_name = 'test_related_name'\\\\n    try:\\\\n        field = ManyToManyField(OtherModel, symmetrical=symmetrical, related_name=related_name)\\\\n        assert False, 'No exception raised'\\\\n    except ValueError as e:\\\\n        assert str(e) == 'Cannot set related_name on a symmetrical ManyToManyField.', 'Incorrect error message'\\\",\\n      \\\"description\\\": \\\"This test validates Patch 2 by defining a ManyToManyField with symmetrical as True and providing a related_name\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise a ValueError with the correct error message as specified in Patch 2\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_patch_3\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_patch_3():\\\\n    symmetrical = True\\\\n    related_name = 'test_related_name'\\\\n    try:\\\\n        field = ManyToManyField(OtherModel, symmetrical=symmetrical, related_name=related_name)\\\\n        assert False, 'No exception raised'\\\\n    except ValueError as e:\\\\n        assert str(e) == \\\\\\\"'related_name' cannot be set when 'symmetrical' is True. Consider removing 'related_name' or set 'symmetrical=False'.\\\\\\\", 'Incorrect error message'\\\",\\n      \\\"description\\\": \\\"This test validates Patch 3 by defining a ManyToManyField with symmetrical as True and providing a related_name\\\",\\n      \\\"expected_behavior\\\": \\\"The test should raise a ValueError with the correct error message as specified in Patch 3\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Generated reproduction and validation tests for the issue of preventing developers from defining a related_name on symmetrical ManyToManyFields\\\"\\n}\\n```\"}, \"error\": \"\", \"elapsed_time\": 505.488061, \"total_tokens\": 25113, \"total_steps\": 9, \"created_at\": 1753308102, \"finished_at\": 1753308607}}"
}