{
    "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 addresses an error in model relationships that involves Django's model checking framework, bringing a similar context to the issue with table name clashes among models. This candidate provides insights into how Django's version change might affect model definitions similarly, which could help in debugging the table name conflict caused by different apps in the current issue."
}