{
  "Selected_candidate": {
    "pr_number": 10199,
    "pr_title": "Fixed #26352 -- Made system check allow ManyToManyField to target the same model if through_fields differs.",
    "pr_body": "See https://code.djangoproject.com/ticket/26352",
    "issue_id": 26352,
    "issue_title": "models.E003 check incorrectly prevents duplicate ManyToMany through-self that differ by through_fields",
    "issue_body": "Let's say we're building a Twitter-style friendship model, where a user can follow other users, and can have users that follow them.\nA sensible way to model that might look like this:\nclass\nUser\n(\nmodels\n.\nModel\n):\nfriends\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'user'\n,\n'target'\n),\nrelated_name\n=\n'+'\n)\nfollowers\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'target'\n,\n'user'\n),\nrelated_name\n=\n'+'\n)\nclass\nFollowship\n(\nmodels\n.\nModel\n):\nuser\n=\nmodels\n.\nForeignKey\n(\nUser\n,\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'+'\n)\ntarget\n=\nmodels\n.\nForeignKey\n(\nUser\n,\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'+'\n)\nHere we have an intermediary table called \"Followship\", which relates a user to the target user that they are following.\nWe also have two helpful convenience ManyToMany fields defined on the user:\nuser.friends.all()\nreturns all other users that the user is following, while\nuser.followers.all()\nreturns the users that are following our current user.\nThe above models should work fine... but they don't, because Django's model checking framework throws the following error:\nusers.User: (models.E003) The model has two many-to-many relations through the intermediate model 'users.Followship'.\nI don't think this warning is warranted in this case, because the\nthrough_fields\narguments provided to the friends/followers ManyToManyFields mean that the relationships defined here make sense and the models should work correctly.\nI've tested this theory by disabling the warning entirely using an over-ride class method on user, like this:\nclass\nUser\n(\nmodels\n.\nModel\n):\nfriends\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'user'\n,\n'target'\n),\nrelated_name\n=\n'+'\n)\nfollowers\n=\nmodels\n.\nManyToManyField\n(\n'self'\n,\nthrough\n=\n'Followship'\n,\nsymmetrical\n=\nFalse\n,\nthrough_fields\n=\n(\n'target'\n,\n'user'\n),\nrelated_name\n=\n'+'\n)\n@classmethod\ndef\n_check_m2m_through_same_relationship\n(\ncls\n):\n# Disable models.E003 check for this model\nreturn\n[]\nThis does the trick: the check is suppressed, and the models work as expected.\nI think the model check framework is being overly strict here. I think it should be modified to enable this pattern, provided the through_fields= parameters on the duplicate ManyToMany fields are present and differentiate the fields correctly.",
    "issue_closed_at": "2018-08-22T11:28:35",
    "base_commit": "f2d5dafec93e6b3100f004c559ebe21e2b783ae7",
    "changes": [
      {
        "file": "django/db/models/base.py",
        "type": "function",
        "name": "_check_m2m_through_same_relationship",
        "class_name": "Model",
        "code": "def _check_m2m_through_same_relationship(cls):\n        \"\"\" Check if no relationship model is used by more than one m2m field.\n        \"\"\"\n\n        errors = []\n        seen_intermediary_signatures = []\n\n        fields = cls._meta.local_many_to_many\n\n        # Skip when the target model wasn't found.\n        fields = (f for f in fields if isinstance(f.remote_field.model, ModelBase))\n\n        # Skip when the relationship model wasn't found.\n        fields = (f for f in fields if isinstance(f.remote_field.through, ModelBase))\n\n        for f in fields:\n            signature = (f.remote_field.model, cls, f.remote_field.through)\n            if signature in seen_intermediary_signatures:\n                errors.append(\n                    checks.Error(\n                        \"The model has two many-to-many relations through \"\n                        \"the intermediate model '%s'.\" % f.remote_field.through._meta.label,\n                        obj=cls,\n                        id='models.E003',\n                    )\n                )\n            else:\n                seen_intermediary_signatures.append(signature)\n        return errors"
      }
    ]
  },
  "Justification": "Candidate A is the most helpful because it directly relates to Django's model validation system, similar to the current issue with ValidationErrors. Both involve improvements in data validation mechanisms, and the need for nuanced equality checks in fields implies shared underlying concerns about how Django handles model relationships and errors. The fact that both reports deal with the equality of complex structures (models in Candidate A and ValidationErrors in the CURRENT bug report) suggests that insights from the patch in Candidate A could inform the implementation of the __eq__ method for ValidationErrors.",
  "instance_id": "django__django-13220",
  "repo": "django/django",
  "created_at": "2020-07-21T19:54:16Z",
  "problem_statement": "Allow ValidationErrors to equal each other when created identically\nDescription\n\t \n\t\t(last modified by kamni)\n\t \nCurrently ValidationErrors (django.core.exceptions.ValidationError) that have identical messages don't equal each other, which is counter-intuitive, and can make certain kinds of testing more complicated. Please add an __eq__ method that allows two ValidationErrors to be compared. \nIdeally, this would be more than just a simple self.messages == other.messages. It would be most helpful if the comparison were independent of the order in which errors were raised in a field or in non_field_errors.\n",
  "patch": "diff --git a/django/core/exceptions.py b/django/core/exceptions.py\n--- a/django/core/exceptions.py\n+++ b/django/core/exceptions.py\n@@ -1,6 +1,9 @@\n \"\"\"\n Global Django exception and warning classes.\n \"\"\"\n+import operator\n+\n+from django.utils.hashable import make_hashable\n \n \n class FieldDoesNotExist(Exception):\n@@ -182,6 +185,23 @@ def __str__(self):\n     def __repr__(self):\n         return 'ValidationError(%s)' % self\n \n+    def __eq__(self, other):\n+        if not isinstance(other, ValidationError):\n+            return NotImplemented\n+        return hash(self) == hash(other)\n+\n+    def __hash__(self):\n+        # Ignore params and messages ordering.\n+        if hasattr(self, 'message'):\n+            return hash((\n+                self.message,\n+                self.code,\n+                tuple(sorted(make_hashable(self.params))) if self.params else None,\n+            ))\n+        if hasattr(self, 'error_dict'):\n+            return hash(tuple(sorted(make_hashable(self.error_dict))))\n+        return hash(tuple(sorted(self.error_list, key=operator.attrgetter('message'))))\n+\n \n class EmptyResultSet(Exception):\n     \"\"\"A database query predicate is impossible.\"\"\"\n"
}