{
  "instance_id": "django__django-15814",
  "repo": "django/django",
  "created_at": "2022-07-03T19:10:56Z",
  "problem_statement": "QuerySet.only() after select_related() crash on proxy models.\nDescription\n\t\nWhen I optimize a query using select_related() and only() methods from the proxy model I encounter an error:\nWindows 10; Python 3.10; Django 4.0.5\nTraceback (most recent call last):\n File \"D:\\study\\django_college\\manage.py\", line 22, in <module>\n\tmain()\n File \"D:\\study\\django_college\\manage.py\", line 18, in main\n\texecute_from_command_line(sys.argv)\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\core\\management\\__init__.py\", line 446, in execute_from_command_line\n\tutility.execute()\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\core\\management\\__init__.py\", line 440, in execute\n\tself.fetch_command(subcommand).run_from_argv(self.argv)\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\core\\management\\base.py\", line 414, in run_from_argv\n\tself.execute(*args, **cmd_options)\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\core\\management\\base.py\", line 460, in execute\n\toutput = self.handle(*args, **options)\n File \"D:\\study\\django_college\\project\\users\\management\\commands\\test_proxy.py\", line 9, in handle\n\tobjs = list(AnotherModel.objects.select_related(\"custom\").only(\"custom__name\").all())\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\db\\models\\query.py\", line 302, in __len__\n\tself._fetch_all()\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\db\\models\\query.py\", line 1507, in _fetch_all\n\tself._result_cache = list(self._iterable_class(self))\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\db\\models\\query.py\", line 71, in __iter__\n\trelated_populators = get_related_populators(klass_info, select, db)\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\db\\models\\query.py\", line 2268, in get_related_populators\n\trel_cls = RelatedPopulator(rel_klass_info, select, db)\n File \"D:\\Anaconda3\\envs\\django\\lib\\site-packages\\django\\db\\models\\query.py\", line 2243, in __init__\n\tself.pk_idx = self.init_list.index(self.model_cls._meta.pk.attname)\nValueError: 'id' is not in list\nModels:\nclass CustomModel(models.Model):\n\tname = models.CharField(max_length=16)\nclass ProxyCustomModel(CustomModel):\n\tclass Meta:\n\t\tproxy = True\nclass AnotherModel(models.Model):\n\tcustom = models.ForeignKey(\n\t\tProxyCustomModel,\n\t\ton_delete=models.SET_NULL,\n\t\tnull=True,\n\t\tblank=True,\n\t)\nCommand:\nclass Command(BaseCommand):\n\tdef handle(self, *args, **options):\n\t\tlist(AnotherModel.objects.select_related(\"custom\").only(\"custom__name\").all())\nAt django/db/models/sql/query.py in 745 line there is snippet:\nopts = cur_model._meta\nIf I replace it by \nopts = cur_model._meta.concrete_model._meta\nall works as expected.\n",
  "patch": "diff --git a/django/db/models/sql/query.py b/django/db/models/sql/query.py\n--- a/django/db/models/sql/query.py\n+++ b/django/db/models/sql/query.py\n@@ -748,6 +748,7 @@ def deferred_to_data(self, target):\n                     cur_model = source.related_model\n                 else:\n                     cur_model = source.remote_field.model\n+                cur_model = cur_model._meta.concrete_model\n                 opts = cur_model._meta\n                 # Even if we're \"just passing through\" this model, we must add\n                 # both the current model's pk and the related reference field\n",
  "similar_bug_items": [
    {
      "pr_number": 15034,
      "pr_title": "Fixed #33234 -- Fixed autodetector crash for proxy models inheriting from non-model class.",
      "pr_body": "ticket-33234\r\n\r\nRegression in aa4acc164d1247c0de515c959f7b09648b57dc42.\r\n\r\nThanks Kevin Marsh for the report.",
      "issue_id": 33234,
      "issue_title": "Migrations autodetector breaking with proxy models inheriting from mixin",
      "issue_body": "",
      "issue_closed_at": "2021-11-02T14:10:58",
      "base_commit": "073b7b5915fdfb89a144e81173176ee13ff92a25",
      "changes": [
        {
          "file": "django/db/migrations/state.py",
          "type": "function",
          "name": "_get_concrete_models_mapping_and_proxy_models",
          "class_name": "ProjectState",
          "code": "def _get_concrete_models_mapping_and_proxy_models(self):\n        concrete_models_mapping = {}\n        proxy_models = {}\n        # Split models to proxy and concrete models.\n        for model_key, model_state in self.models.items():\n            if model_state.options.get('proxy'):\n                proxy_models[model_key] = model_state\n                # Find a concrete model for the proxy.\n                concrete_models_mapping[model_key] = self._find_concrete_model_from_proxy(\n                    proxy_models, model_state,\n                )\n            else:\n                concrete_models_mapping[model_key] = model_key\n        return concrete_models_mapping, proxy_models"
        }
      ]
    },
    {
      "pr_number": 9112,
      "pr_title": "Fixed #27846 -- clear all cached reverse relationships on refresh_from_db()",
      "pr_body": "https://code.djangoproject.com/ticket/27846",
      "issue_id": 27846,
      "issue_title": "refresh_from_db() doesn't clear reverse OneToOneFields",
      "issue_body": "",
      "issue_closed_at": "2017-10-12T16:25:22",
      "base_commit": "df0aebc893973c78d7d2cda712ba4133dbe29b6e",
      "changes": [
        {
          "file": "django/db/models/base.py",
          "type": "function",
          "name": "refresh_from_db",
          "class_name": "Model",
          "code": "def refresh_from_db(self, using=None, fields=None):\n        \"\"\"\n        Reload field values from the database.\n\n        By default, the reloading happens from the database this instance was\n        loaded from, or by the read router if this instance wasn't loaded from\n        any database. The using parameter will override the default.\n\n        Fields can be used to specify which fields to reload. The fields\n        should be an iterable of field attnames. If fields is None, then\n        all non-deferred fields are reloaded.\n\n        When accessing deferred fields of an instance, the deferred loading\n        of the field will call this method.\n        \"\"\"\n        if fields is not None:\n            if len(fields) == 0:\n                return\n            if any(LOOKUP_SEP in f for f in fields):\n                raise ValueError(\n                    'Found \"%s\" in fields argument. Relations and transforms '\n                    'are not allowed in fields.' % LOOKUP_SEP)\n\n        db = using if using is not None else self._state.db\n        db_instance_qs = self.__class__._default_manager.using(db).filter(pk=self.pk)\n\n        # Use provided fields, if not set then reload all non-deferred fields.\n        deferred_fields = self.get_deferred_fields()\n        if fields is not None:\n            fields = list(fields)\n            db_instance_qs = db_instance_qs.only(*fields)\n        elif deferred_fields:\n            fields = [f.attname for f in self._meta.concrete_fields\n                      if f.attname not in deferred_fields]\n            db_instance_qs = db_instance_qs.only(*fields)\n\n        db_instance = db_instance_qs.get()\n        non_loaded_fields = db_instance.get_deferred_fields()\n        for field in self._meta.concrete_fields:\n            if field.attname in non_loaded_fields:\n                # This field wasn't refreshed - skip ahead.\n                continue\n            setattr(self, field.attname, getattr(db_instance, field.attname))\n            # Throw away stale foreign key references.\n            if field.is_relation and field.is_cached(self):\n                rel_instance = field.get_cached_value(self)\n                local_val = getattr(db_instance, field.attname)\n                related_val = None if rel_instance is None else getattr(rel_instance, field.target_field.attname)\n                if local_val != related_val or (local_val is None and related_val is None):\n                    field.delete_cached_value(self)\n        self._state.db = db_instance._state.db"
        }
      ]
    },
    {
      "pr_number": 14578,
      "pr_title": "Fixed #32144 -- Made makemessages remove temporary files when locale path doesn't exist.",
      "pr_body": "Alternative to  #13609 ticket-32144",
      "issue_id": 32144,
      "issue_title": "makemessages leaves temporary files when locale directory doesn't exist.",
      "issue_body": "",
      "issue_closed_at": "2021-07-01T03:11:23",
      "base_commit": "62988afbea7c7ea6ea7eb76382b3a87a5ccf310c",
      "changes": [
        {
          "file": "django/core/management/commands/makemessages.py",
          "type": "function",
          "name": "process_locale_dir",
          "class_name": "Command",
          "code": "def process_locale_dir(self, locale_dir, files):\n        \"\"\"\n        Extract translatable literals from the specified files, creating or\n        updating the POT file for a given locale directory.\n\n        Use the xgettext GNU gettext utility.\n        \"\"\"\n        build_files = []\n        for translatable in files:\n            if self.verbosity > 1:\n                self.stdout.write('processing file %s in %s' % (\n                    translatable.file, translatable.dirpath\n                ))\n            if self.domain not in ('djangojs', 'django'):\n                continue\n            build_file = self.build_file_class(self, self.domain, translatable)\n            try:\n                build_file.preprocess()\n            except UnicodeDecodeError as e:\n                self.stdout.write(\n                    'UnicodeDecodeError: skipped file %s in %s (reason: %s)' % (\n                        translatable.file, translatable.dirpath, e,\n                    )\n                )\n                continue\n            build_files.append(build_file)\n\n        if self.domain == 'djangojs':\n            is_templatized = build_file.is_templatized\n            args = [\n                'xgettext',\n                '-d', self.domain,\n                '--language=%s' % ('C' if is_templatized else 'JavaScript',),\n                '--keyword=gettext_noop',\n                '--keyword=gettext_lazy',\n                '--keyword=ngettext_lazy:1,2',\n                '--keyword=pgettext:1c,2',\n                '--keyword=npgettext:1c,2,3',\n                '--output=-',\n            ]\n        elif self.domain == 'django':\n            args = [\n                'xgettext',\n                '-d', self.domain,\n                '--language=Python',\n                '--keyword=gettext_noop',\n                '--keyword=gettext_lazy',\n                '--keyword=ngettext_lazy:1,2',\n                '--keyword=pgettext:1c,2',\n                '--keyword=npgettext:1c,2,3',\n                '--keyword=pgettext_lazy:1c,2',\n                '--keyword=npgettext_lazy:1c,2,3',\n                '--output=-',\n            ]\n        else:\n            return\n\n        input_files = [bf.work_path for bf in build_files]\n        with NamedTemporaryFile(mode='w+') as input_files_list:\n            input_files_list.write('\\n'.join(input_files))\n            input_files_list.flush()\n            args.extend(['--files-from', input_files_list.name])\n            args.extend(self.xgettext_options)\n            msgs, errors, status = popen_wrapper(args)\n\n        if errors:\n            if status != STATUS_OK:\n                for build_file in build_files:\n                    build_file.cleanup()\n                raise CommandError(\n                    'errors happened while running xgettext on %s\\n%s' %\n                    ('\\n'.join(input_files), errors)\n                )\n            elif self.verbosity > 0:\n                # Print warnings\n                self.stdout.write(errors)\n\n        if msgs:\n            if locale_dir is NO_LOCALE_DIR:\n                file_path = os.path.normpath(build_files[0].path)\n                raise CommandError(\n                    \"Unable to find a locale path to store translations for \"\n                    \"file %s. Make sure the 'locale' directory exists in an \"\n                    \"app or LOCALE_PATHS setting is set.\" % file_path\n                )\n            for build_file in build_files:\n                msgs = build_file.postprocess_messages(msgs)\n            potfile = os.path.join(locale_dir, '%s.pot' % self.domain)\n            write_pot_file(potfile, msgs)\n\n        for build_file in build_files:\n            build_file.cleanup()"
        },
        {
          "file": "django/core/management/commands/makemessages.py",
          "type": "function",
          "name": "process_locale_dir",
          "class_name": "Command",
          "code": "def process_locale_dir(self, locale_dir, files):\n        \"\"\"\n        Extract translatable literals from the specified files, creating or\n        updating the POT file for a given locale directory.\n\n        Use the xgettext GNU gettext utility.\n        \"\"\"\n        build_files = []\n        for translatable in files:\n            if self.verbosity > 1:\n                self.stdout.write('processing file %s in %s' % (\n                    translatable.file, translatable.dirpath\n                ))\n            if self.domain not in ('djangojs', 'django'):\n                continue\n            build_file = self.build_file_class(self, self.domain, translatable)\n            try:\n                build_file.preprocess()\n            except UnicodeDecodeError as e:\n                self.stdout.write(\n                    'UnicodeDecodeError: skipped file %s in %s (reason: %s)' % (\n                        translatable.file, translatable.dirpath, e,\n                    )\n                )\n                continue\n            build_files.append(build_file)\n\n        if self.domain == 'djangojs':\n            is_templatized = build_file.is_templatized\n            args = [\n                'xgettext',\n                '-d', self.domain,\n                '--language=%s' % ('C' if is_templatized else 'JavaScript',),\n                '--keyword=gettext_noop',\n                '--keyword=gettext_lazy',\n                '--keyword=ngettext_lazy:1,2',\n                '--keyword=pgettext:1c,2',\n                '--keyword=npgettext:1c,2,3',\n                '--output=-',\n            ]\n        elif self.domain == 'django':\n            args = [\n                'xgettext',\n                '-d', self.domain,\n                '--language=Python',\n                '--keyword=gettext_noop',\n                '--keyword=gettext_lazy',\n                '--keyword=ngettext_lazy:1,2',\n                '--keyword=pgettext:1c,2',\n                '--keyword=npgettext:1c,2,3',\n                '--keyword=pgettext_lazy:1c,2',\n                '--keyword=npgettext_lazy:1c,2,3',\n                '--output=-',\n            ]\n        else:\n            return\n\n        input_files = [bf.work_path for bf in build_files]\n        with NamedTemporaryFile(mode='w+') as input_files_list:\n            input_files_list.write('\\n'.join(input_files))\n            input_files_list.flush()\n            args.extend(['--files-from', input_files_list.name])\n            args.extend(self.xgettext_options)\n            msgs, errors, status = popen_wrapper(args)\n\n        if errors:\n            if status != STATUS_OK:\n                for build_file in build_files:\n                    build_file.cleanup()\n                raise CommandError(\n                    'errors happened while running xgettext on %s\\n%s' %\n                    ('\\n'.join(input_files), errors)\n                )\n            elif self.verbosity > 0:\n                # Print warnings\n                self.stdout.write(errors)\n\n        if msgs:\n            if locale_dir is NO_LOCALE_DIR:\n                file_path = os.path.normpath(build_files[0].path)\n                raise CommandError(\n                    \"Unable to find a locale path to store translations for \"\n                    \"file %s. Make sure the 'locale' directory exists in an \"\n                    \"app or LOCALE_PATHS setting is set.\" % file_path\n                )\n            for build_file in build_files:\n                msgs = build_file.postprocess_messages(msgs)\n            potfile = os.path.join(locale_dir, '%s.pot' % self.domain)\n            write_pot_file(potfile, msgs)\n\n        for build_file in build_files:\n            build_file.cleanup()"
        }
      ]
    },
    {
      "pr_number": 11886,
      "pr_title": "Fixed #30405 -- Fixed source code mismatch crash in ExceptionReporter. ",
      "pr_body": "[ticket 30405](https://code.djangoproject.com/ticket/30405)",
      "issue_id": 30405,
      "issue_title": "IndexError in _get_lines_from_file when module does not match file contents (via loader)",
      "issue_body": "",
      "issue_closed_at": "2019-11-12T04:53:04",
      "base_commit": "6e2f05b2e33a6c80c7a411ce76af7b5a08acb835",
      "changes": [
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "get_traceback_text",
          "class_name": "ExceptionReporter",
          "code": "def get_traceback_text(self):\n        \"\"\"Return plain text version of debug 500 HTTP error page.\"\"\"\n        with Path(CURRENT_DIR, 'templates', 'technical_500.txt').open(encoding='utf-8') as fh:\n            t = DEBUG_ENGINE.from_string(fh.read())\n        c = Context(self.get_traceback_data(), autoescape=False, use_l10n=False)\n        return t.render(c)"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "_get_lines_from_file",
          "class_name": "ExceptionReporter",
          "code": "def _get_lines_from_file(self, filename, lineno, context_lines, loader=None, module_name=None):\n        \"\"\"\n        Return context_lines before and after lineno from file.\n        Return (pre_context_lineno, pre_context, context_line, post_context).\n        \"\"\"\n        source = None\n        if hasattr(loader, 'get_source'):\n            try:\n                source = loader.get_source(module_name)\n            except ImportError:\n                pass\n            if source is not None:\n                source = source.splitlines()\n        if source is None:\n            try:\n                with open(filename, 'rb') as fp:\n                    source = fp.read().splitlines()\n            except OSError:\n                pass\n        if source is None:\n            return None, [], None, []\n\n        # If we just read the source from a file, or if the loader did not\n        # apply tokenize.detect_encoding to decode the source into a\n        # string, then we should do that ourselves.\n        if isinstance(source[0], bytes):\n            encoding = 'ascii'\n            for line in source[:2]:\n                # File coding may be specified. Match pattern from PEP-263\n                # (https://www.python.org/dev/peps/pep-0263/)\n                match = re.search(br'coding[:=]\\s*([-\\w.]+)', line)\n                if match:\n                    encoding = match.group(1).decode('ascii')\n                    break\n            source = [str(sline, encoding, 'replace') for sline in source]\n\n        lower_bound = max(0, lineno - context_lines)\n        upper_bound = lineno + context_lines\n\n        pre_context = source[lower_bound:lineno]\n        context_line = source[lineno]\n        post_context = source[lineno + 1:upper_bound]\n\n        return lower_bound, pre_context, context_line, post_context"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "_get_lines_from_file",
          "class_name": "ExceptionReporter",
          "code": "def _get_lines_from_file(self, filename, lineno, context_lines, loader=None, module_name=None):\n        \"\"\"\n        Return context_lines before and after lineno from file.\n        Return (pre_context_lineno, pre_context, context_line, post_context).\n        \"\"\"\n        source = None\n        if hasattr(loader, 'get_source'):\n            try:\n                source = loader.get_source(module_name)\n            except ImportError:\n                pass\n            if source is not None:\n                source = source.splitlines()\n        if source is None:\n            try:\n                with open(filename, 'rb') as fp:\n                    source = fp.read().splitlines()\n            except OSError:\n                pass\n        if source is None:\n            return None, [], None, []\n\n        # If we just read the source from a file, or if the loader did not\n        # apply tokenize.detect_encoding to decode the source into a\n        # string, then we should do that ourselves.\n        if isinstance(source[0], bytes):\n            encoding = 'ascii'\n            for line in source[:2]:\n                # File coding may be specified. Match pattern from PEP-263\n                # (https://www.python.org/dev/peps/pep-0263/)\n                match = re.search(br'coding[:=]\\s*([-\\w.]+)', line)\n                if match:\n                    encoding = match.group(1).decode('ascii')\n                    break\n            source = [str(sline, encoding, 'replace') for sline in source]\n\n        lower_bound = max(0, lineno - context_lines)\n        upper_bound = lineno + context_lines\n\n        pre_context = source[lower_bound:lineno]\n        context_line = source[lineno]\n        post_context = source[lineno + 1:upper_bound]\n\n        return lower_bound, pre_context, context_line, post_context"
        }
      ]
    },
    {
      "pr_number": 8228,
      "pr_title": "Fixed #27966 -- Bumped required psycopg2 version to 2.5.4.",
      "pr_body": "",
      "issue_id": 27966,
      "issue_title": "Bump required version of pyscopg2 to 2.5.4",
      "issue_body": "",
      "issue_closed_at": "2017-03-21T11:23:31",
      "base_commit": "7063a85579f40585f2601ba6e6887b0982e7ce43",
      "changes": [
        {
          "file": "django/db/backends/postgresql/base.py",
          "type": "function",
          "name": "psycopg2_version",
          "class_name": "DatabaseWrapper",
          "code": "def psycopg2_version(self):\n        return PSYCOPG2_VERSION"
        }
      ]
    }
  ]
}