{
  "instance_id": "django__django-13315",
  "repo": "django/django",
  "created_at": "2020-08-17T04:24:39Z",
  "problem_statement": "limit_choices_to on a ForeignKey can render duplicate options in formfield\nDescription\n\t\nIf you pass a Q object as limit_choices_to on a ForeignKey field involving a join, you may end up with duplicate options in your form.\nSee regressiontest in patch for a clear view on the problem.\n",
  "patch": "diff --git a/django/forms/models.py b/django/forms/models.py\n--- a/django/forms/models.py\n+++ b/django/forms/models.py\n@@ -97,10 +97,18 @@ def model_to_dict(instance, fields=None, exclude=None):\n \n def apply_limit_choices_to_to_formfield(formfield):\n     \"\"\"Apply limit_choices_to to the formfield's queryset if needed.\"\"\"\n+    from django.db.models import Exists, OuterRef, Q\n     if hasattr(formfield, 'queryset') and hasattr(formfield, 'get_limit_choices_to'):\n         limit_choices_to = formfield.get_limit_choices_to()\n-        if limit_choices_to is not None:\n-            formfield.queryset = formfield.queryset.complex_filter(limit_choices_to)\n+        if limit_choices_to:\n+            complex_filter = limit_choices_to\n+            if not isinstance(complex_filter, Q):\n+                complex_filter = Q(**limit_choices_to)\n+            complex_filter &= Q(pk=OuterRef('pk'))\n+            # Use Exists() to avoid potential duplicates.\n+            formfield.queryset = formfield.queryset.filter(\n+                Exists(formfield.queryset.model._base_manager.filter(complex_filter)),\n+            )\n \n \n def fields_for_model(model, fields=None, exclude=None, widgets=None,\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": 9874,
      "pr_title": "Fixed #29322 -- Made admin check all ModelAdmin fieldsets for duplicates.",
      "pr_body": "Duplicate fields were only detected if they were in the same fieldset.\r\n\r\nhttp://code.djangoproject.com/ticket/29322",
      "issue_id": 29322,
      "issue_title": "admin.E012: Duplicate fields in admin fieldset only recognizes duplicates if they are in the same fieldset",
      "issue_body": "",
      "issue_closed_at": "2018-04-13T10:09:11",
      "base_commit": "0b66c3b442875627fa6daef4ac1e90900d74290b",
      "changes": [
        {
          "file": "django/contrib/admin/checks.py",
          "type": "function",
          "name": "_check_fieldsets",
          "class_name": "BaseModelAdminChecks",
          "code": "def _check_fieldsets(self, obj):\n        \"\"\" Check that fieldsets is properly formatted and doesn't contain\n        duplicates. \"\"\"\n\n        if obj.fieldsets is None:\n            return []\n        elif not isinstance(obj.fieldsets, (list, tuple)):\n            return must_be('a list or tuple', option='fieldsets', obj=obj, id='admin.E007')\n        else:\n            return list(chain.from_iterable(\n                self._check_fieldsets_item(obj, obj.model, fieldset, 'fieldsets[%d]' % index)\n                for index, fieldset in enumerate(obj.fieldsets)\n            ))"
        },
        {
          "file": "django/contrib/admin/checks.py",
          "type": "function",
          "name": "_check_fieldsets_item",
          "class_name": "BaseModelAdminChecks",
          "code": "def _check_fieldsets_item(self, obj, model, fieldset, label):\n        \"\"\" Check an item of `fieldsets`, i.e. check that this is a pair of a\n        set name and a dictionary containing \"fields\" key. \"\"\"\n\n        if not isinstance(fieldset, (list, tuple)):\n            return must_be('a list or tuple', option=label, obj=obj, id='admin.E008')\n        elif len(fieldset) != 2:\n            return must_be('of length 2', option=label, obj=obj, id='admin.E009')\n        elif not isinstance(fieldset[1], dict):\n            return must_be('a dictionary', option='%s[1]' % label, obj=obj, id='admin.E010')\n        elif 'fields' not in fieldset[1]:\n            return [\n                checks.Error(\n                    \"The value of '%s[1]' must contain the key 'fields'.\" % label,\n                    obj=obj.__class__,\n                    id='admin.E011',\n                )\n            ]\n        elif not isinstance(fieldset[1]['fields'], (list, tuple)):\n            return must_be('a list or tuple', option=\"%s[1]['fields']\" % label, obj=obj, id='admin.E008')\n\n        fields = flatten(fieldset[1]['fields'])\n        if len(fields) != len(set(fields)):\n            return [\n                checks.Error(\n                    \"There are duplicate field(s) in '%s[1]'.\" % label,\n                    obj=obj.__class__,\n                    id='admin.E012',\n                )\n            ]\n        return list(chain.from_iterable(\n            self._check_field_spec(obj, model, fieldset_fields, '%s[1][\"fields\"]' % label)\n            for fieldset_fields in fieldset[1]['fields']\n        ))"
        }
      ]
    },
    {
      "pr_number": 5969,
      "pr_title": "[1.9.x] Fixed #25858 -- Bound abstract model application relative relationships.",
      "pr_body": "Thanks to Karl Hobley for the report and Markus, Shai, Aymeric for their input\nand Tim for the review.\n\nBackport of bc7d201bdbaeac14a49f51a9ef292d6312b4c45e from master\n",
      "issue_id": 25858,
      "issue_title": "Deriving from abstract model with foreign key to model in same app broken on Django 1.9",
      "issue_body": "As of Django 1.9, it is no longer possible to derive a new model from abstract model in another app if that abstract model has a local foreign key to a model in the same app.\nSee example project here:\n​\nhttps://github.com/kaedroho/djangobugtest\nFor example, I have two apps: app1 and app2\napp1 contains a concrete model and an abstract model. The abstract model has a foreign key to the concrete one (note the Foreign key is defined as a string and doesn't include the app label):\nclass ConcreteModel(models.Model):\n    pass\n\n\nclass AbstractModel(models.Model):\n    foreign_key = models.ForeignKey('ConcreteModel')\n\n    class Meta:\n        abstract = True\nIn app2, we have another model that derives from the abstract model in app 1\nclass DerivedModel(app1.AbstractModel):\n    pass\nWhen running makemigrations, the following system check error occurs:\napp2.DerivedModel.foreign_key: (fields.E300) Field defines a relation with model 'ConcreteModel', which is either not installed, or is abstract.\nFor apps upgrading to Django 1.9 and already have migrations, the following error occurs when running tests (after the database has been initialised but before the first test runs. It looks like it's happening while serialising the database):\nValueError: Related model u'ConcreteModel' cannot be resolved\nTraceback from an actual site:\n​\nhttps://gist.github.com/kaedroho/01b29332308018abfa29\nThis issue only occurs when the ForeignKey is defined with a string that doesn't contain the app label. So referencing the model class directly or using \"app_label.ModelName\" works.",
      "issue_closed_at": "2016-01-11T12:59:11",
      "base_commit": "76b4015ea9b02fdbc3acebe71ed79bcc902697da",
      "changes": [
        {
          "file": "django/db/models/fields/related.py",
          "type": "line",
          "name": "line 39",
          "code": "RECURSIVE_RELATIONSHIP_CONSTANT = 'self'\n\n\ndef resolve_relation(scope_model, relation):\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):\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        relation = scope_model\n\n    # Look for an \"app.Model\" relation\n    if isinstance(relation, six.string_types):\n        if \".\" 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": 5421,
      "pr_title": "Fixed #24687 -- Added select_related() validation for nested non-relational fields.",
      "pr_body": "The removed test was added in the original select_related() validation\npatch (45d4e43d2d25b902e3821b612209afa951a8bcb8), but there doesn't\nseem to be any reason for it.\n\nhttps://code.djangoproject.com/ticket/24687\n",
      "issue_id": 24687,
      "issue_title": "select_related field validation doesn't work for nested non-relation fields",
      "issue_body": "The validation added in\n#10414\nonly occurs if the field is on the immediate model, or if it's invalid:\n>>>\nPerson\n.\nobjects\n.\nselect_related\n(\n'foo'\n)\ndjango\n.\ncore\n.\nexceptions\n.\nFieldError\n:\nInvalid\nfield\nname\n(\ns\n)\ngiven\nin\nselect_related\n:\n'foo'\n.\nChoices\nare\n:\nhometown\n>>>\nPerson\n.\nobjects\n.\nselect_related\n(\n'hometown__foo'\n)\ndjango\n.\ncore\n.\nexceptions\n.\nFieldError\n:\nInvalid\nfield\nname\n(\ns\n)\ngiven\nin\nselect_related\n:\n'foo'\n.\nChoices\nare\n:\nmayor\n>>>\nPerson\n.\nobjects\n.\nselect_related\n(\n'hometown__name'\n)\n# query = select ... blah\n[\nPerson\nobject\n...\n]\n>>>\nPerson\n.\nobjects\n.\nselect_related\n(\n'name'\n)\ndjango\n.\ncore\n.\nexceptions\n.\nFieldError\n:\nNon\n-\nrelational\nfield\ngiven\nin\nselect_related\n:\n'name'\n.\nChoices\nare\n:\nhometown\nNotice how it will happily accept the third one.  Failing test attached.",
      "issue_closed_at": "2015-10-10T12:38:49",
      "base_commit": "5171f56faeb9ec27153c7e7319c96d3fba774ea5",
      "changes": [
        {
          "file": "django/contrib/gis/db/models/query.py",
          "type": "line",
          "name": "line 10",
          "code": "from django.contrib.gis.geometry.backend import Geometry\nfrom django.contrib.gis.measure import Area, Distance\nfrom django.db import connections\nfrom django.db.models.expressions import RawSQL\nfrom django.db.models.fields import Field\nfrom django.db.models.query import QuerySet"
        },
        {
          "file": "django/contrib/gis/db/models/query.py",
          "type": "function",
          "name": "_geocol_select",
          "class_name": "GeoQuerySet",
          "code": "def _geocol_select(self, geo_field, field_name):\n        \"\"\"\n        Helper routine for constructing the SQL to select the geographic\n        column.  Takes into account if the geographic field is in a\n        ForeignKey relation to the current model.\n        \"\"\"\n        compiler = self.query.get_compiler(self.db)\n        opts = self.model._meta\n        if geo_field not in opts.fields:\n            # Is this operation going to be on a related geographic field?\n            # If so, it'll have to be added to the select related information\n            # (e.g., if 'location__point' was given as the field name).\n            # Note: the operation really is defined as \"must add select related!\"\n            self.query.add_select_related([field_name])\n            # Call pre_sql_setup() so that compiler.select gets populated.\n            compiler.pre_sql_setup()\n            for col, _, _ in compiler.select:\n                if col.output_field == geo_field:\n                    return col.as_sql(compiler, compiler.connection)[0]\n            raise ValueError(\"%r not in compiler's related_select_cols\" % geo_field)\n        elif geo_field not in opts.local_fields:\n            # This geographic field is inherited from another model, so we have to\n            # use the db table for the _parent_ model instead.\n            parent_model = geo_field.model._meta.concrete_model\n            return self._field_column(compiler, geo_field, parent_model._meta.db_table)\n        else:\n            return self._field_column(compiler, geo_field)"
        },
        {
          "file": "django/db/models/sql/compiler.py",
          "type": "function",
          "name": "get_related_klass_infos",
          "class_name": "SQLCompiler",
          "code": "def get_related_klass_infos(klass_info, related_klass_infos):\n            klass_info['related_klass_infos'] = related_klass_infos"
        }
      ]
    },
    {
      "pr_number": 9813,
      "pr_title": "Fixed #29247 -- Allowed blank model field choice to be defined in nested choices.",
      "pr_body": "While sifting through the source code to understand the internals for a package I'm making, I stumbled upon what looked like a bug.\r\n\r\nThese lines right here:\r\n```py\r\nnamed_groups = isinstance(choices[0][1], (list, tuple))\r\nblank_defined = not named_groups and any(choice in ('', None) for choice, __ in choices)\r\n```\r\nAnd indeed a bug it was.\r\n\r\nThe problem was that, had you defined your choices first with a group and then later on a choice that defined the blank value, the current implementation would skip right over it! So you would end up with having both the default blank value and the one you've defined, which doesn't look very pretty.\r\n\r\n![picture](https://user-images.githubusercontent.com/30984274/37772917-39e232e8-2ded-11e8-9767-e3aa5bf98a66.png)\r\n\r\nThis pull request is my solution.\r\n\r\nIt gets rid of the group checking entirely as the documentation quotes that they are \"used for organizational purposes\", so it wouldn't make sense to check them, even if one did get named as either '' or None, which also addresses the issue in #23098 that those lines were put in there for in the first place.",
      "issue_id": 29247,
      "issue_title": "A custom blank value in model field choices doesn't work when using named groups",
      "issue_body": "While sifting through the source code to understand the internals for a package I'm making, I stumbled upon what looked like a bug.\nThese lines right here:\nnamed_groups = isinstance(choices[0][1], (list, tuple))\nblank_defined = not named_groups and any(choice in ('', None) for choice, __ in choices)\nAnd indeed a bug it was.\nThe problem was that, had you defined your choices first with a group and then later on a choice that defined the blank value, the current implementation would skip right over it! So you would end up with having both the default blank value and the one you've defined, which doesn't look very pretty.\nI've created a pull request on github that fixes the issue\n​\nhere.",
      "issue_closed_at": "2018-04-20T10:25:03",
      "base_commit": "e35004966bacbc9ba1fb10dd01edcfd874dad303",
      "changes": [
        {
          "file": "django/db/models/fields/__init__.py",
          "type": "function",
          "name": "get_choices",
          "class_name": "Field",
          "code": "def get_choices(self, include_blank=True, blank_choice=BLANK_CHOICE_DASH, limit_choices_to=None):\n        \"\"\"\n        Return choices with a default blank choices included, for use\n        as <select> choices for this field.\n        \"\"\"\n        if self.choices:\n            choices = list(self.choices)\n            if include_blank:\n                named_groups = isinstance(choices[0][1], (list, tuple))\n                blank_defined = not named_groups and any(choice in ('', None) for choice, __ in choices)\n                if not blank_defined:\n                    choices = blank_choice + choices\n            return choices\n        rel_model = self.remote_field.model\n        limit_choices_to = limit_choices_to or self.get_limit_choices_to()\n        choice_func = operator.attrgetter(\n            self.remote_field.get_related_field().attname\n            if hasattr(self.remote_field, 'get_related_field')\n            else 'pk'\n        )\n        return (blank_choice if include_blank else []) + [\n            (choice_func(x), smart_text(x))\n            for x in rel_model._default_manager.complex_filter(limit_choices_to)\n        ]"
        }
      ]
    }
  ]
}