{
  "instance_id": "django__django-14730",
  "repo": "django/django",
  "created_at": "2021-08-03T04:27:52Z",
  "problem_statement": "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",
  "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@@ -1258,6 +1258,16 @@ def _check_ignored_options(self, **kwargs):\n                 )\n             )\n \n+        if self.remote_field.symmetrical and self._related_name:\n+            warnings.append(\n+                checks.Warning(\n+                    'related_name has no effect on ManyToManyField '\n+                    'with a symmetrical relationship, e.g. to \"self\".',\n+                    obj=self,\n+                    id='fields.W345',\n+                )\n+            )\n+\n         return warnings\n \n     def _check_relationship_model(self, from_model=None, **kwargs):\n",
  "similar_bug_items": [
    {
      "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": 5437,
      "pr_title": "Fixed #25560 -- Made empty string related name invalid.",
      "pr_body": "Thanks to Ali Lotfi for the initial report and patch.\n",
      "issue_id": 25560,
      "issue_title": "Empty string related name should be invalid",
      "issue_body": "The actual model field check is\n​\nskipped\nif the\nrelated_name\nis an empty string when a warning should be raised since it's an invalid Python identifier.",
      "issue_closed_at": "2015-10-16T13:18:21",
      "base_commit": "4dcc2a195595f8d7ddad45bc4baf98ffdeec7f41",
      "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 not related_name:\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/reverse_related.py",
          "type": "function",
          "name": "get_db_prep_lookup",
          "class_name": "ForeignObjectRel",
          "code": "def get_db_prep_lookup(self, lookup_type, value, connection, prepared=False):\n        # Defer to the actual field definition for db prep\n        return self.field.get_db_prep_lookup(lookup_type, value, connection=connection, prepared=prepared)"
        }
      ]
    },
    {
      "pr_number": 5657,
      "pr_title": "Ticket #25745 1.9 backport",
      "pr_body": "Will merge if CI pass.\n",
      "issue_id": 25745,
      "issue_title": "Fix RuntimeWarnings about model reloading in test suite",
      "issue_body": "Similar warnings to those reported in\n#24812\nare back in several apps:\norder_with_respect_to\n,\nforeign_object\n,\nschema\n.\nTo see the\norder_with_respect_to\nwarnings, you must also include\ncontenttypes_tests\n(these apps share models):\n$ python -Wall runtests.py order_with_respect_to contenttypes_tests\nExample:\n/home/tim/code/django/django/db/models/base.py:283: RuntimeWarning: Model 'order_with_respect_to.bar' was already registered. Reloading models is not advised as it can lead to inconsistencies, most notably with related models.\nWe should probably promote\nRuntimeWarning\nto \"error\" in\ntests/runtests.py\nto prevent future regressions.",
      "issue_closed_at": "2015-11-14T10:34:07",
      "base_commit": "84006fda55ffcaf272ca4fcd4addf7874302e884",
      "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"
        }
      ]
    },
    {
      "pr_number": 7097,
      "pr_title": "Fixed #27068 -- Unified form field initial data retrieval.",
      "pr_body": "https://code.djangoproject.com/ticket/27068\n",
      "issue_id": 27068,
      "issue_title": "Acquire form's initial data more consistently",
      "issue_body": "In\nBoundField\n, sometimes initial data\n​\nhandles callables\n. While other times,\n​\nit doesn't\n. This can lead to a theoretical bug when the initial data is a callable, but the field is disabled.\nOther times, the initial callable is handled, but microseconds are\n​\nnot chopped\nleading to theoretical false positives that data has changed.\nInitial data should be handled more consistently across all cases.\nMarking as a bug due to the theoretical edge cases that can produce unexpected results. Noticed these inconsistencies while working on\n#27037\n.",
      "issue_closed_at": "2016-08-18T19:51:32",
      "base_commit": "13857b45ca54a519a58361238442af84262c0d23",
      "changes": [
        {
          "file": "django/forms/boundfield.py",
          "type": "function",
          "name": "value",
          "class_name": "BoundField",
          "code": "def value(self):\n        \"\"\"\n        Returns the value for this BoundField, using the initial value if\n        the form is not bound or the data otherwise.\n        \"\"\"\n        if not self.form.is_bound:\n            data = self.initial\n        else:\n            data = self.field.bound_data(\n                self.data, self.form.initial.get(self.name, self.field.initial)\n            )\n        return self.field.prepare_value(data)"
        },
        {
          "file": "django/forms/boundfield.py",
          "type": "function",
          "name": "id_for_label",
          "class_name": "BoundField",
          "code": "def id_for_label(self):\n        \"\"\"\n        Wrapper around the field widget's `id_for_label` method.\n        Useful, for example, for focusing on this field regardless of whether\n        it has a single widget or a MultiWidget.\n        \"\"\"\n        widget = self.field.widget\n        id_ = widget.attrs.get('id') or self.auto_id\n        return widget.id_for_label(id_)"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "_clean_fields",
          "class_name": "BaseForm",
          "code": "def _clean_fields(self):\n        for name, field in self.fields.items():\n            # value_from_datadict() gets the data from the data dictionaries.\n            # Each widget type knows how to retrieve its own data, because some\n            # widgets split data over several HTML fields.\n            if field.disabled:\n                value = self.initial.get(name, field.initial)\n            else:\n                value = field.widget.value_from_datadict(self.data, self.files, self.add_prefix(name))\n            try:\n                if isinstance(field, FileField):\n                    initial = self.initial.get(name, field.initial)\n                    value = field.clean(value, initial)\n                else:\n                    value = field.clean(value)\n                self.cleaned_data[name] = value\n                if hasattr(self, 'clean_%s' % name):\n                    value = getattr(self, 'clean_%s' % name)()\n                    self.cleaned_data[name] = value\n            except ValidationError as e:\n                self.add_error(name, e)"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "changed_data",
          "class_name": "BaseForm",
          "code": "def changed_data(self):\n        data = []\n        for name, field in self.fields.items():\n            prefixed_name = self.add_prefix(name)\n            data_value = field.widget.value_from_datadict(self.data, self.files, prefixed_name)\n            if not field.show_hidden_initial:\n                initial_value = self.initial.get(name, field.initial)\n                if callable(initial_value):\n                    initial_value = initial_value()\n            else:\n                initial_prefixed_name = self.add_initial_prefix(name)\n                hidden_widget = field.hidden_widget()\n                try:\n                    initial_value = field.to_python(hidden_widget.value_from_datadict(\n                        self.data, self.files, initial_prefixed_name))\n                except ValidationError:\n                    # Always assume data has changed if validation fails.\n                    data.append(name)\n                    continue\n            if field.has_changed(initial_value, data_value):\n                data.append(name)\n        return data"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "visible_fields",
          "class_name": "BaseForm",
          "code": "def visible_fields(self):\n        \"\"\"\n        Returns a list of BoundField objects that aren't hidden fields.\n        The opposite of the hidden_fields() method.\n        \"\"\"\n        return [field for field in self if not field.is_hidden]"
        }
      ]
    },
    {
      "pr_number": 13109,
      "pr_title": "Fixed #31596 -- Changed ForeignKey.validate() to use the base manager.",
      "pr_body": "Follow up from #12923 for which ~~GH seems~~ I seem to have _had a moment_...\r\n\r\n\r\ncc @jdufresne ",
      "issue_id": 31596,
      "issue_title": "ForeignKey.validate() should validate using the base manager.",
      "issue_body": "ForeignKey.validate()\nshould validate using the base manager instead of the default manager.\nConsider the models:\nclass ArticleManager(models.Manage):\n    def get_queryset(self):\n        qs = super().get_queryset()\n        return qs.filter(archived=False)\n\nclass Article(models.Model):\n    title = models.CharField(max_length=100)\n    archived = models.BooleanField(default=False)\n\n    # Don't include archived articles by default.\n    objects = ArticleManager()\n\nclass FavoriteAricles(models.Model):\n    article = models.ForeignKey(Article, on_delete=models.CASCADE)\nIn the example, now consider a form that allows users to pick a favorite article including archived articles.\nclass FavoriteAriclesForm(forms.ModelForm):\n    class Meta:\n        model = FavoriteArticle\n        fields = '__all__'\n\n    def __init__(self, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        # Use the base manager instead of the default manager to allow archived articles.\n        self.fields['article'].queryset = Article._base_manager.all()\nThe above form will never validate as\nTrue\nwhen a user selects an archived article. This is because the ForeignKey validation always uses\n_default_manager\ninstead of\n_base_manager\n. The user facing error message is \"article instance with id 123 does not exist.\" (quite confusing to typical users). The code for this validation is here:\n​\nhttps://github.com/django/django/blob/94f63b926fd32d7a7b6e2591ef72aa8f040f25cc/django/db/models/fields/related.py#L917-L919\nThe\nFavoriteAriclesForm\nis specifically designed to use a different manager, but the ForeignKey validation makes this difficult.\nIn this example scenario, it is not acceptable to change the model's default manager as the default should avoid archived articles in other typical scenarios.\nSuggested solution: the ForeignKey validation should use the\n_base_manager\ninstead which does not include the default filters.",
      "issue_closed_at": "2020-06-25T04:36:37",
      "base_commit": "fbe82f82555bc25dccb476c749ca062f0b522be3",
      "changes": [
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "validate",
          "class_name": "ForeignKey",
          "code": "def validate(self, value, model_instance):\n        if self.remote_field.parent_link:\n            return\n        super().validate(value, model_instance)\n        if value is None:\n            return\n\n        using = router.db_for_read(self.remote_field.model, instance=model_instance)\n        qs = self.remote_field.model._default_manager.using(using).filter(\n            **{self.remote_field.field_name: value}\n        )\n        qs = qs.complex_filter(self.get_limit_choices_to())\n        if not qs.exists():\n            raise exceptions.ValidationError(\n                self.error_messages['invalid'],\n                code='invalid',\n                params={\n                    'model': self.remote_field.model._meta.verbose_name, 'pk': value,\n                    'field': self.remote_field.field_name, 'value': value,\n                },  # 'pk' is included for backwards compatibility\n            )"
        }
      ]
    }
  ]
}