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