{
  "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": "So I've run into an issue with the new automigrations feature in Django 4.0b1 (and 4.0a1)  relating to proxy models inheriting from a non-Model class.\nFor examples given these three models:\nfrom\ndjango.db\nimport\nmodels\nclass\nMyMixin\n:\n...\nclass\nMyBaseModel\n(\nmodels\n.\nModel\n):\n...\nclass\nMyProxyModel\n(\nMyMixin\n,\nMyBaseModel\n):\nclass\nMeta\n:\nproxy\n=\nTrue\nRunning\nmigrate\n/\nmakemigrations\nwill result in\nMyProxyModel\ncrashing when it tries running\n_find_concrete_model_from_proxy\n:\nFile \"/some/annonymized/path/django/core/management/commands/makemigrations.py\", line 172, in handle\n    changes = autodetector.changes(\n  File \"/some/annonymized/path/django/db/migrations/autodetector.py\", line 43, in changes\n    changes = self._detect_changes(convert_apps, graph)\n  File \"/some/annonymized/path/django/db/migrations/autodetector.py\", line 156, in _detect_changes\n    self.to_state.resolve_fields_and_relations()\n  File \"/some/annonymized/path/django/db/migrations/state.py\", line 449, in resolve_fields_and_relations\n    concretes, proxies = self._get_concrete_models_mapping_and_proxy_models()\n  File \"/some/annonymized/path/django/db/migrations/state.py\", line 470, in _get_concrete_models_mapping_and_proxy_models\n    concrete_models_mapping[model_key] = self._find_concrete_model_from_proxy(\n  File \"/some/annonymized/path/django/db/migrations/state.py\", line 479, in _find_concrete_model_from_proxy\n    base_key = make_model_tuple(base)\n  File \"/some/annonymized/path/django/db/models/utils.py\", line 18, in make_model_tuple\n    model_tuple = model._meta.app_label, model._meta.model_name\nAttributeError: type object 'MyMixin' has no attribute '_meta'\nChanging\nMyMixin\nto inherit from\nmodels.Model\nprevents this crash but might not always be possible depending on how the user is using that mixin. Or if we've decided that all inheritance for proxy models must be from\nmodels.Model\nthen that should be flagged up more explicitly to developers (eg. checking for it in\nModelBase.__new__\nor something).",
      "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": "Sorry for the poor summary, it is difficult to explain in words. I have a project to demo this bug attached to this ticket, but I will try to explain the bug anyway in steps and the setup.\nSetup:\n2 models (A and B)\nB has a OneToOne to A\nBoth A and B have a field (ie TextField)\nSetup either a signal or override save() for A to update B's TextField to equal that of A's on save() or post_save for signals\nSteps:\nCreate instance of A\nGet another copy of the instance of A via A.objects.get()\nCreate instance of B using the copy of the instance of A\nDo refresh_from_db() on original instance of A\nTry to access B from A\nThe project I have provided is a slim version of this problem that demonstrates it with signals, overriden save(), and basic set and save inside the test. The basic set and save works, but the other two fail when using the above steps. Run the test suite to see.",
      "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": "If you run makemessages and you end up with the error message \"Unable to find a locale path to store translations for file [...]\", you get a lot of temporary files left over that you need to clean up.\nThe fix is pretty trivial so a PR is on the way.",
      "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": "self = <django.views.debug.ExceptionReporter object at 0x7f2a7908ac18>\nfilename = '…/project/.venv/lib/python3.7/site-packages/pdb.py'\nlineno = 230\ncontext_lines = 7\nloader = <_frozen_importlib_external.SourceFileLoader object at 0x7f2a73609278>\nmodule_name = 'pdb'\n\n[23]   …/Vcs/django/django/core/handlers/exception.py(90)response_for_exception()\n-> response = handle_uncaught_exception(request, get_resolver(get_urlconf()), sys.exc_info())\n[24]   …/Vcs/django/django/core/handlers/exception.py(125)handle_uncaught_exception()\n-> return debug.technical_500_response(request, *exc_info)\n[25]   …/Vcs/django/django/views/debug.py(94)technical_500_response()\n-> html = reporter.get_traceback_html()\n[26]   …/Vcs/django/django/views/debug.py(333)get_traceback_html()\n-> c = Context(self.get_traceback_data(), use_l10n=False)\n[27]   …/Vcs/django/django/views/debug.py(264)get_traceback_data()\n-> frames = self.get_traceback_frames()\n[28]   …/Vcs/django/django/views/debug.py(427)get_traceback_frames()\n-> filename, lineno, 7, loader, module_name,\n\n 385             try:\n 386                 context_line = source[lineno]\n 387             except:\n 388                 __import__('pdb').set_trace()\n 389  ->         post_context = source[lineno + 1:upper_bound]\n 390\n 391             return lower_bound, pre_context, context_line, post_context\n(Pdb++) source\n['# this file is needed to hijack pdb without eggs', 'import os.path', \"pdb_path = os.path.join(os.path.dirname(os.path.dirname(__file__)), 'pdb.py')\", 'with open(pdb_path) as f:', \"    exec(compile(f.read(), pdb_path, 'exec'))\"]\nIt uses the loader (\n​\nhttps://github.com/django/django/blob/47885278c669dd7a13a4c3ff7e58e1cbe88af385/django/views/debug.py#L351\n), which picks up the\npth\n, and then the contents does not match the expected line number.\nI think it should maybe always use the given filename?!",
      "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": "​\nthis commit\nuses the cursor as context manager (line in question is marked), which were added in psycopg2 2.5 (\n​\nrelease notes\n) (see third item)\nbut\n​\nhere\ndjango checks only for 2.4.5.\n​\nthis commit here\nmade 2.4.5 a requirement and documented that in a few places.",
      "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"
        }
      ]
    }
  ]
}