{
  "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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"
        }
      ]
    }
  ]
}