{
  "instance_id": "django__django-12497",
  "repo": "django/django",
  "created_at": "2020-02-26T18:12:31Z",
  "problem_statement": "Wrong hint about recursive relationship.\nDescription\n\t \n\t\t(last modified by Matheus Cunha Motta)\n\t \nWhen there's more than 2 ForeignKeys in an intermediary model of a m2m field and no through_fields have been set, Django will show an error with the following hint:\nhint=(\n\t'If you want to create a recursive relationship, '\n\t'use ForeignKey(\"%s\", symmetrical=False, through=\"%s\").'\nBut 'symmetrical' and 'through' are m2m keyword arguments, not ForeignKey.\nThis was probably a small mistake where the developer thought ManyToManyField but typed ForeignKey instead. And the symmetrical=False is an outdated requirement to recursive relationships with intermediary model to self, not required since 3.0. I'll provide a PR with a proposed correction shortly after.\nEdit: fixed description.\n",
  "patch": "diff --git a/django/db/models/fields/related.py b/django/db/models/fields/related.py\n--- a/django/db/models/fields/related.py\n+++ b/django/db/models/fields/related.py\n@@ -1309,7 +1309,7 @@ def _check_relationship_model(self, from_model=None, **kwargs):\n                              \"through_fields keyword argument.\") % (self, from_model_name),\n                             hint=(\n                                 'If you want to create a recursive relationship, '\n-                                'use ForeignKey(\"%s\", symmetrical=False, through=\"%s\").'\n+                                'use ManyToManyField(\"%s\", through=\"%s\").'\n                             ) % (\n                                 RECURSIVE_RELATIONSHIP_CONSTANT,\n                                 relationship_model_name,\n@@ -1329,7 +1329,7 @@ def _check_relationship_model(self, from_model=None, **kwargs):\n                             \"through_fields keyword argument.\" % (self, to_model_name),\n                             hint=(\n                                 'If you want to create a recursive relationship, '\n-                                'use ForeignKey(\"%s\", symmetrical=False, through=\"%s\").'\n+                                'use ManyToManyField(\"%s\", through=\"%s\").'\n                             ) % (\n                                 RECURSIVE_RELATIONSHIP_CONSTANT,\n                                 relationship_model_name,\n",
  "similar_bug_items": [
    {
      "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"
        }
      ]
    },
    {
      "pr_number": 12472,
      "pr_title": "Fixed #31286 -- Made database specific fields checks databases aware.",
      "pr_body": "Further tests required(or not?).\r\nCurrent test cases don't cover the case when the user provides multiple databases, e.g.:\r\n```python manage.py check --database dba --database dbb```\r\n",
      "issue_id": 31286,
      "issue_title": "Database specific fields checks should be databases aware.",
      "issue_body": "In an attempt to trigger the check Error mentioned in Tickets:\n#31144\n, I found that the check for database backend doesn't check against the backend specified, e.g. --database mysql, rather, it always checks against the 'default' backend.\nAfter diving into the source code, I found the following method defined in django/db/models/fields/\ninit\n.py\ndef _check_backend_specific_checks(self, **kwargs):\n        app_label = self.model._meta.app_label\n        for db in connections:\n            if router.allow_migrate(db, app_label, model_name=self.model._meta.model_name):\n                return connections[db].validation.check_field(self, **kwargs)\n        return []\nIt checks the first db defined in connections rather those provided by users.\nA proposed change would be:\ndef _check_backend_specific_checks(self, **kwargs):\n        app_label = self.model._meta.app_label\n        errors = []\n        for db in kwargs['databases'] or ['default']:\n            if router.allow_migrate(db, app_label, model_name=self.model._meta.model_name):\n                errors.extend(connections[db].validation.check_field(self, **kwargs))\n        return errors\nIt worked as intended on my local machine.\nI would happily provide a patch for this one.",
      "issue_closed_at": "2020-02-24T10:38:01",
      "base_commit": "94d4bd3a091dd9ac265e14619576b1ee568653b1",
      "changes": [
        {
          "file": "django/db/models/fields/__init__.py",
          "type": "function",
          "name": "_check_null_allowed_for_primary_keys",
          "class_name": "Field",
          "code": "def _check_null_allowed_for_primary_keys(self):\n        if (self.primary_key and self.null and\n                not connection.features.interprets_empty_strings_as_nulls):\n            # We cannot reliably check this for backends like Oracle which\n            # consider NULL and '' to be equal (and thus set up\n            # character-based fields a little differently).\n            return [\n                checks.Error(\n                    'Primary keys must not have null=True.',\n                    hint=('Set null=False on the field, or '\n                          'remove primary_key=True argument.'),\n                    obj=self,\n                    id='fields.E007',\n                )\n            ]\n        else:\n            return []"
        }
      ]
    },
    {
      "pr_number": 9406,
      "pr_title": "Fixed #28871 -- Fixed initialization of autocomplete widgets in \"Add another\" inlines.",
      "pr_body": "https://code.djangoproject.com/ticket/28871",
      "issue_id": 28871,
      "issue_title": "Autocomplete select for new lines in inline model do not open",
      "issue_body": "When a model has an Inline model, autocomplete select does not show for new rows.\nSelect box is is replaced with Select2 style, but clicking it does not show the select list.\nmodel.py\nclass\nItem\n(\nmodels\n.\nModel\n):\nname\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n100\n)\nclass\nCategory\n(\nmodels\n.\nModel\n):\nname\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n100\n)\ndef\n__str__\n(\nself\n):\nreturn\nself\n.\nname\nclass\nLineItem\n(\nmodels\n.\nModel\n):\nitem\n=\nmodels\n.\nForeignKey\n(\n'Item'\n,\non_delete\n=\nmodels\n.\nCASCADE\n)\ncategory\n=\nmodels\n.\nForeignKey\n(\n'Category'\n,\non_delete\n=\nmodels\n.\nCASCADE\n)\nadmin.py\nclass\nLineItemInline\n(\nadmin\n.\nTabularInline\n):\nmodel\n=\nLineItem\nextra\n=\n1\nautocomplete_fields\n=\n(\n'category'\n,)\n@admin\n.\nregister\n(\nItem\n)\nclass\nItemAdmin\n(\nadmin\n.\nModelAdmin\n):\nlist_display\n=\n(\n'name'\n,)\nsearch_fields\n=\n(\n'name'\n,)\ninlines\n=\n[\nLineItemInline\n]\n@admin\n.\nregister\n(\nCategory\n)\nclass\nCategoryAdmin\n(\nadmin\n.\nModelAdmin\n):\nlist_display\n=\n(\n'name'\n,)\nsearch_fields\n=\n(\n'name'\n,)\nGithub repo\nWith demo and instructions on how to reproduce\n​\nhttps://github.com/gotling/bug-django2-autocomplete",
      "issue_closed_at": "2017-12-01T21:42:03",
      "base_commit": "095c1aaa898bed40568009db836aa8434f1b983d",
      "changes": [
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "formfield_for_foreignkey",
          "class_name": "BaseModelAdmin",
          "code": "def formfield_for_foreignkey(self, db_field, request, **kwargs):\n        \"\"\"\n        Get a form Field for a ForeignKey.\n        \"\"\"\n        db = kwargs.get('using')\n\n        if db_field.name in self.get_autocomplete_fields(request):\n            kwargs['widget'] = AutocompleteSelect(db_field.remote_field, using=db)\n        elif db_field.name in self.raw_id_fields:\n            kwargs['widget'] = widgets.ForeignKeyRawIdWidget(db_field.remote_field, self.admin_site, using=db)\n        elif db_field.name in self.radio_fields:\n            kwargs['widget'] = widgets.AdminRadioSelect(attrs={\n                'class': get_ul_class(self.radio_fields[db_field.name]),\n            })\n            kwargs['empty_label'] = _('None') if db_field.blank else None\n\n        if 'queryset' not in kwargs:\n            queryset = self.get_field_queryset(db, db_field, request)\n            if queryset is not None:\n                kwargs['queryset'] = queryset\n\n        return db_field.formfield(**kwargs)"
        },
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "formfield_for_manytomany",
          "class_name": "BaseModelAdmin",
          "code": "def formfield_for_manytomany(self, db_field, request, **kwargs):\n        \"\"\"\n        Get a form Field for a ManyToManyField.\n        \"\"\"\n        # If it uses an intermediary model that isn't auto created, don't show\n        # a field in admin.\n        if not db_field.remote_field.through._meta.auto_created:\n            return None\n        db = kwargs.get('using')\n\n        autocomplete_fields = self.get_autocomplete_fields(request)\n        if db_field.name in autocomplete_fields:\n            kwargs['widget'] = AutocompleteSelectMultiple(db_field.remote_field, using=db)\n        elif db_field.name in self.raw_id_fields:\n            kwargs['widget'] = widgets.ManyToManyRawIdWidget(db_field.remote_field, self.admin_site, using=db)\n        elif db_field.name in list(self.filter_vertical) + list(self.filter_horizontal):\n            kwargs['widget'] = widgets.FilteredSelectMultiple(\n                db_field.verbose_name,\n                db_field.name in self.filter_vertical\n            )\n\n        if 'queryset' not in kwargs:\n            queryset = self.get_field_queryset(db, db_field, request)\n            if queryset is not None:\n                kwargs['queryset'] = queryset\n\n        form_field = db_field.formfield(**kwargs)\n        if (isinstance(form_field.widget, SelectMultiple) and\n                not isinstance(form_field.widget, (CheckboxSelectMultiple, AutocompleteSelectMultiple))):\n            msg = _('Hold down \"Control\", or \"Command\" on a Mac, to select more than one.')\n            help_text = form_field.help_text\n            form_field.help_text = format_lazy('{} {}', help_text, msg) if help_text else msg\n        return form_field"
        },
        {
          "file": "django/contrib/admin/widgets.py",
          "type": "function",
          "name": "__init__",
          "class_name": "AutocompleteMixin",
          "code": "def __init__(self, rel, attrs=None, choices=(), using=None):\n        self.rel = rel\n        self.db = using\n        self.choices = choices\n        if attrs is not None:\n            self.attrs = attrs.copy()\n        else:\n            self.attrs = {}"
        }
      ]
    },
    {
      "pr_number": 6178,
      "pr_title": "Fixed #26186 -- Documented how app relative relationships of abstract models behave.",
      "pr_body": "This partially reverts commit bc7d201bdbaeac14a49f51a9ef292d6312b4c45e.\n\nRefs #25858.\n",
      "issue_id": 26186,
      "issue_title": "When extending from abstract model, ForeignKey points to wrong application",
      "issue_body": "This might be related to\nhttps://code.djangoproject.com/ticket/25858\nI have following code:\napp_A/abstract.py:\nclass AbstractModel1(Model):\n    class Meta:\n        abstract = True\n\n\nclass AbstractModel2(Model):\n    model1 = ForeignKey('Model1')\n\n    class Meta:\n          abstract = True\napp_A/models.py:\nfrom .abstract import AbstractModel1, AbstractModel12\n\nclass Model1(AbstractModel1):\n    pass\n\nclass Model2(AbstractModel12):\n    pass\napp_B/models.py:\nfrom app_A.abstract import AbstractModel1, AbstractModel2\n\nclass Model1(AbstractModel1):\n    pass\n\nclass Model2(AbstractModel12):\n    pass\nin Django 1.8, the\napp_B.Model2.model1\nwould point to\napp_B.Model1\n, but in Django 1.9.2 it points to\napp_A.Model1\nso my code no longer works.\nTest case can be found:\n​\nhttps://github.com/skyjur/django-ticketing/tree/master/ticket_26186",
      "issue_closed_at": "2016-02-29T21:07:53",
      "base_commit": "eac1423f9ebcf432dc5be95d605d124a05ab2686",
      "changes": [
        {
          "file": "django/db/models/fields/related.py",
          "type": "line",
          "name": "line 35",
          "code": "RECURSIVE_RELATIONSHIP_CONSTANT = 'self'\n\n\ndef resolve_relation(scope_model, relation, resolve_recursive_relationship=True):\n    \"\"\"\n    Transform relation into a model or fully-qualified model string of the form\n    \"app_label.ModelName\", relative to scope_model."
        },
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "resolve_relation",
          "class_name": null,
          "code": "def resolve_relation(scope_model, relation, resolve_recursive_relationship=True):\n    \"\"\"\n    Transform relation into a model or fully-qualified model string of the form\n    \"app_label.ModelName\", relative to scope_model.\n\n    The relation argument can be:\n      * RECURSIVE_RELATIONSHIP_CONSTANT, i.e. the string \"self\", in which case\n        the model argument will be returned.\n      * A bare model name without an app_label, in which case scope_model's\n        app_label will be prepended.\n      * An \"app_label.ModelName\" string.\n      * A model class, which will be returned unchanged.\n    \"\"\"\n    # Check for recursive relations\n    if relation == RECURSIVE_RELATIONSHIP_CONSTANT:\n        if resolve_recursive_relationship:\n            relation = scope_model\n    # Look for an \"app.Model\" relation\n    elif isinstance(relation, six.string_types) and '.' not in relation:\n        relation = \"%s.%s\" % (scope_model._meta.app_label, relation)\n\n    return relation"
        },
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "resolve_related_class",
          "class_name": "RelatedField",
          "code": "def resolve_related_class(model, related, field):\n                field.remote_field.model = related\n                field.do_related_class(related, model)"
        },
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "resolve_through_model",
          "class_name": "ManyToManyField",
          "code": "def resolve_through_model(_, model, field):\n                    field.remote_field.through = model"
        }
      ]
    },
    {
      "pr_number": 5618,
      "pr_title": "Fixed #25723 -- Made related field checks lookup against their local apps.",
      "pr_body": "Discovered while working on #5613\n",
      "issue_id": 25723,
      "issue_title": "Related field checks should use their model's apps to lookup related models",
      "issue_body": "Related model fields are using the global apps registry to determine whether or not their related model is registered (\nto\n,\nthrough\n).\nThis prevents the creation of checks tests that define model using an isolated\nApps()\ninstance.\ne.g.\nisolated_apps\n=\nApps\n()\nclass\nFoo\n(\nmodels\n.\nModel\n):\nclass\nMeta\n:\napps\n=\nisolated_apps\nclass\nBar\n(\nmodels\n.\nModel\n):\nfoo\n=\nmodels\n.\nForeignKey\n(\nFoo\n)\nclass\nMeta\n:\napps\n=\nisolated_apps\n# The following assertion will fail since the `to` check\n# will detect a `fields.E300` issue.\nasset\nBar\n.\n_meta\n.\nget_field\n(\n'foo'\n)\n.\ncheck\n()\n==\n[]",
      "issue_closed_at": "2015-11-11T18:34:35",
      "base_commit": "c819780d3f46b6e4e67aa135c840f78dc24468e1",
      "changes": [
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "_check_related_name_is_valid",
          "class_name": "RelatedField",
          "code": "def _check_related_name_is_valid(self):\n        import re\n        import keyword\n        related_name = self.remote_field.related_name\n        if related_name is None:\n            return []\n        is_valid_id = True\n        if keyword.iskeyword(related_name):\n            is_valid_id = False\n        if six.PY3:\n            if not related_name.isidentifier():\n                is_valid_id = False\n        else:\n            if not re.match(r'^[a-zA-Z_][a-zA-Z0-9_]*\\Z', related_name):\n                is_valid_id = False\n        if not (is_valid_id or related_name.endswith('+')):\n            return [\n                checks.Error(\n                    \"The name '%s' is invalid related_name for field %s.%s\" %\n                    (self.remote_field.related_name, self.model._meta.object_name,\n                     self.name),\n                    hint=\"Related name must be a valid Python identifier or end with a '+'\",\n                    obj=self,\n                    id='fields.E306',\n                )\n            ]\n        return []"
        },
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "_check_relation_model_exists",
          "class_name": "RelatedField",
          "code": "def _check_relation_model_exists(self):\n        rel_is_missing = self.remote_field.model not in apps.get_models()\n        rel_is_string = isinstance(self.remote_field.model, six.string_types)\n        model_name = self.remote_field.model if rel_is_string else self.remote_field.model._meta.object_name\n        if rel_is_missing and (rel_is_string or not self.remote_field.model._meta.swapped):\n            return [\n                checks.Error(\n                    (\"Field defines a relation with model '%s', which \"\n                     \"is either not installed, or is abstract.\") % model_name,\n                    hint=None,\n                    obj=self,\n                    id='fields.E300',\n                )\n            ]\n        return []"
        },
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "_check_relationship_model",
          "class_name": "ManyToManyField",
          "code": "def _check_relationship_model(self, from_model=None, **kwargs):\n        if hasattr(self.remote_field.through, '_meta'):\n            qualified_model_name = \"%s.%s\" % (\n                self.remote_field.through._meta.app_label, self.remote_field.through.__name__)\n        else:\n            qualified_model_name = self.remote_field.through\n\n        errors = []\n\n        if self.remote_field.through not in apps.get_models(include_auto_created=True):\n            # The relationship model is not installed.\n            errors.append(\n                checks.Error(\n                    (\"Field specifies a many-to-many relation through model \"\n                     \"'%s', which has not been installed.\") %\n                    qualified_model_name,\n                    hint=None,\n                    obj=self,\n                    id='fields.E331',\n                )\n            )\n\n        else:\n\n            assert from_model is not None, (\n                \"ManyToManyField with intermediate \"\n                \"tables cannot be checked if you don't pass the model \"\n                \"where the field is attached to.\"\n            )\n\n            # Set some useful local variables\n            to_model = resolve_relation(from_model, self.remote_field.model)\n            from_model_name = from_model._meta.object_name\n            if isinstance(to_model, six.string_types):\n                to_model_name = to_model\n            else:\n                to_model_name = to_model._meta.object_name\n            relationship_model_name = self.remote_field.through._meta.object_name\n            self_referential = from_model == to_model\n\n            # Check symmetrical attribute.\n            if (self_referential and self.remote_field.symmetrical and\n                    not self.remote_field.through._meta.auto_created):\n                errors.append(\n                    checks.Error(\n                        'Many-to-many fields with intermediate tables must not be symmetrical.',\n                        hint=None,\n                        obj=self,\n                        id='fields.E332',\n                    )\n                )\n\n            # Count foreign keys in intermediate model\n            if self_referential:\n                seen_self = sum(from_model == getattr(field.remote_field, 'model', None)\n                    for field in self.remote_field.through._meta.fields)\n\n                if seen_self > 2 and not self.remote_field.through_fields:\n                    errors.append(\n                        checks.Error(\n                            (\"The model is used as an intermediate model by \"\n                             \"'%s', but it has more than two foreign keys \"\n                             \"to '%s', which is ambiguous. You must specify \"\n                             \"which two foreign keys Django should use via the \"\n                             \"through_fields keyword argument.\") % (self, from_model_name),\n                            hint=(\"Use through_fields to specify which two \"\n                                  \"foreign keys Django should use.\"),\n                            obj=self.remote_field.through,\n                            id='fields.E333',\n                        )\n                    )\n\n            else:\n                # Count foreign keys in relationship model\n                seen_from = sum(from_model == getattr(field.remote_field, 'model', None)\n                    for field in self.remote_field.through._meta.fields)\n                seen_to = sum(to_model == getattr(field.remote_field, 'model', None)\n                    for field in self.remote_field.through._meta.fields)\n\n                if seen_from > 1 and not self.remote_field.through_fields:\n                    errors.append(\n                        checks.Error(\n                            (\"The model is used as an intermediate model by \"\n                             \"'%s', but it has more than one foreign key \"\n                             \"from '%s', which is ambiguous. You must specify \"\n                             \"which foreign key Django should use via the \"\n                             \"through_fields keyword argument.\") % (self, from_model_name),\n                            hint=('If you want to create a recursive relationship, '\n                                  'use ForeignKey(\"self\", symmetrical=False, '\n                                  'through=\"%s\").') % relationship_model_name,\n                            obj=self,\n                            id='fields.E334',\n                        )\n                    )\n\n                if seen_to > 1 and not self.remote_field.through_fields:\n                    errors.append(\n                        checks.Error(\n                            (\"The model is used as an intermediate model by \"\n                             \"'%s', but it has more than one foreign key \"\n                             \"to '%s', which is ambiguous. You must specify \"\n                             \"which foreign key Django should use via the \"\n                             \"through_fields keyword argument.\") % (self, to_model_name),\n                            hint=('If you want to create a recursive '\n                                  'relationship, use ForeignKey(\"self\", '\n                                  'symmetrical=False, through=\"%s\").') % relationship_model_name,\n                            obj=self,\n                            id='fields.E335',\n                        )\n                    )\n\n                if seen_from == 0 or seen_to == 0:\n                    errors.append(\n                        checks.Error(\n                            (\"The model is used as an intermediate model by \"\n                             \"'%s', but it does not have a foreign key to '%s' or '%s'.\") % (\n                                self, from_model_name, to_model_name\n                            ),\n                            hint=None,\n                            obj=self.remote_field.through,\n                            id='fields.E336',\n                        )\n                    )\n\n        # Validate `through_fields`.\n        if self.remote_field.through_fields is not None:\n            # Validate that we're given an iterable of at least two items\n            # and that none of them is \"falsy\".\n            if not (len(self.remote_field.through_fields) >= 2 and\n                    self.remote_field.through_fields[0] and self.remote_field.through_fields[1]):\n                errors.append(\n                    checks.Error(\n                        (\"Field specifies 'through_fields' but does not \"\n                         \"provide the names of the two link fields that should be \"\n                         \"used for the relation through model \"\n                         \"'%s'.\") % qualified_model_name,\n                        hint=(\"Make sure you specify 'through_fields' as \"\n                              \"through_fields=('field1', 'field2')\"),\n                        obj=self,\n                        id='fields.E337',\n                    )\n                )\n\n            # Validate the given through fields -- they should be actual\n            # fields on the through model, and also be foreign keys to the\n            # expected models.\n            else:\n                assert from_model is not None, (\n                    \"ManyToManyField with intermediate \"\n                    \"tables cannot be checked if you don't pass the model \"\n                    \"where the field is attached to.\"\n                )\n\n                source, through, target = from_model, self.remote_field.through, self.remote_field.model\n                source_field_name, target_field_name = self.remote_field.through_fields[:2]\n\n                for field_name, related_model in ((source_field_name, source),\n                                                  (target_field_name, target)):\n\n                    possible_field_names = []\n                    for f in through._meta.fields:\n                        if hasattr(f, 'remote_field') and getattr(f.remote_field, 'model', None) == related_model:\n                            possible_field_names.append(f.name)\n                    if possible_field_names:\n                        hint = (\"Did you mean one of the following foreign \"\n                                \"keys to '%s': %s?\") % (related_model._meta.object_name,\n                                                        ', '.join(possible_field_names))\n                    else:\n                        hint = None\n\n                    try:\n                        field = through._meta.get_field(field_name)\n                    except exceptions.FieldDoesNotExist:\n                        errors.append(\n                            checks.Error(\n                                (\"The intermediary model '%s' has no field '%s'.\") % (\n                                    qualified_model_name, field_name),\n                                hint=hint,\n                                obj=self,\n                                id='fields.E338',\n                            )\n                        )\n                    else:\n                        if not (hasattr(field, 'remote_field') and\n                                getattr(field.remote_field, 'model', None) == related_model):\n                            errors.append(\n                                checks.Error(\n                                    \"'%s.%s' is not a foreign key to '%s'.\" % (\n                                        through._meta.object_name, field_name,\n                                        related_model._meta.object_name),\n                                    hint=hint,\n                                    obj=self,\n                                    id='fields.E339',\n                                )\n                            )\n\n        return errors"
        }
      ]
    }
  ]
}