{
  "instance_id": "django__django-16816",
  "repo": "django/django",
  "created_at": "2023-04-30T15:37:43Z",
  "problem_statement": "Error E108 does not cover some cases\nDescription\n\t \n\t\t(last modified by Baha Sdtbekov)\n\t \nI have two models, Question and Choice. And if I write list_display = [\"choice\"] in QuestionAdmin, I get no errors.\nBut when I visit /admin/polls/question/, the following trace is returned:\nInternal Server Error: /admin/polls/question/\nTraceback (most recent call last):\n File \"/some/path/django/contrib/admin/utils.py\", line 334, in label_for_field\n\tfield = _get_non_gfk_field(model._meta, name)\n File \"/some/path/django/contrib/admin/utils.py\", line 310, in _get_non_gfk_field\n\traise FieldDoesNotExist()\ndjango.core.exceptions.FieldDoesNotExist\nDuring handling of the above exception, another exception occurred:\nTraceback (most recent call last):\n File \"/some/path/django/core/handlers/exception.py\", line 55, in inner\n\tresponse = get_response(request)\n File \"/some/path/django/core/handlers/base.py\", line 220, in _get_response\n\tresponse = response.render()\n File \"/some/path/django/template/response.py\", line 111, in render\n\tself.content = self.rendered_content\n File \"/some/path/django/template/response.py\", line 89, in rendered_content\n\treturn template.render(context, self._request)\n File \"/some/path/django/template/backends/django.py\", line 61, in render\n\treturn self.template.render(context)\n File \"/some/path/django/template/base.py\", line 175, in render\n\treturn self._render(context)\n File \"/some/path/django/template/base.py\", line 167, in _render\n\treturn self.nodelist.render(context)\n File \"/some/path/django/template/base.py\", line 1005, in render\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 1005, in <listcomp>\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 966, in render_annotated\n\treturn self.render(context)\n File \"/some/path/django/template/loader_tags.py\", line 157, in render\n\treturn compiled_parent._render(context)\n File \"/some/path/django/template/base.py\", line 167, in _render\n\treturn self.nodelist.render(context)\n File \"/some/path/django/template/base.py\", line 1005, in render\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 1005, in <listcomp>\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 966, in render_annotated\n\treturn self.render(context)\n File \"/some/path/django/template/loader_tags.py\", line 157, in render\n\treturn compiled_parent._render(context)\n File \"/some/path/django/template/base.py\", line 167, in _render\n\treturn self.nodelist.render(context)\n File \"/some/path/django/template/base.py\", line 1005, in render\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 1005, in <listcomp>\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 966, in render_annotated\n\treturn self.render(context)\n File \"/some/path/django/template/loader_tags.py\", line 63, in render\n\tresult = block.nodelist.render(context)\n File \"/some/path/django/template/base.py\", line 1005, in render\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 1005, in <listcomp>\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 966, in render_annotated\n\treturn self.render(context)\n File \"/some/path/django/template/loader_tags.py\", line 63, in render\n\tresult = block.nodelist.render(context)\n File \"/some/path/django/template/base.py\", line 1005, in render\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 1005, in <listcomp>\n\treturn SafeString(\"\".join([node.render_annotated(context) for node in self]))\n File \"/some/path/django/template/base.py\", line 966, in render_annotated\n\treturn self.render(context)\n File \"/some/path/django/contrib/admin/templatetags/base.py\", line 45, in render\n\treturn super().render(context)\n File \"/some/path/django/template/library.py\", line 258, in render\n\t_dict = self.func(*resolved_args, **resolved_kwargs)\n File \"/some/path/django/contrib/admin/templatetags/admin_list.py\", line 326, in result_list\n\theaders = list(result_headers(cl))\n File \"/some/path/django/contrib/admin/templatetags/admin_list.py\", line 90, in result_headers\n\ttext, attr = label_for_field(\n File \"/some/path/django/contrib/admin/utils.py\", line 362, in label_for_field\n\traise AttributeError(message)\nAttributeError: Unable to lookup 'choice' on Question or QuestionAdmin\n[24/Apr/2023 15:43:32] \"GET /admin/polls/question/ HTTP/1.1\" 500 349913\nI suggest that error E108 be updated to cover this case as well\nFor reproduce see \u200bgithub\n",
  "patch": "diff --git a/django/contrib/admin/checks.py b/django/contrib/admin/checks.py\n--- a/django/contrib/admin/checks.py\n+++ b/django/contrib/admin/checks.py\n@@ -916,9 +916,10 @@ def _check_list_display_item(self, obj, item, label):\n                         id=\"admin.E108\",\n                     )\n                 ]\n-        if isinstance(field, models.ManyToManyField) or (\n-            getattr(field, \"rel\", None) and field.rel.field.many_to_one\n-        ):\n+        if (\n+            getattr(field, \"is_relation\", False)\n+            and (field.many_to_many or field.one_to_many)\n+        ) or (getattr(field, \"rel\", None) and field.rel.field.many_to_one):\n             return [\n                 checks.Error(\n                     f\"The value of '{label}' must not be a many-to-many field or a \"\n",
  "similar_bug_items": [
    {
      "pr_number": 16250,
      "pr_title": "Fixed #33701 -- Added fine-grained error locations to the technical 500 debug page.",
      "pr_body": "Solves Ticket [33701](https://code.djangoproject.com/ticket/33701)\r\n\r\nWith Python 3.11 the technical_500.html debug page now shows, where in the line the error occurred.\r\n\r\n![traceback](https://user-images.githubusercontent.com/39874595/199703281-3f755068-fd41-45bc-961c-957a78d51df5.PNG)\r\n",
      "issue_id": 33701,
      "issue_title": "Highlight error location in the technical 500 debug page on Python 3.11+.",
      "issue_body": "",
      "issue_closed_at": "2022-11-29T02:25:55",
      "base_commit": "9d726c7902979d4ad53945ed8f1037266a88010d",
      "changes": [
        {
          "file": "django/views/debug.py",
          "type": "line",
          "name": "line 1",
          "code": "import functools\nimport re\nimport sys\nimport types"
        },
        {
          "file": "django/views/debug.py",
          "type": "line",
          "name": "line 15",
          "code": "from django.utils.encoding import force_str\nfrom django.utils.module_loading import import_string\nfrom django.utils.regex_helper import _lazy_re_compile\nfrom django.utils.version import get_docs_version\n\n# Minimal Django templates engine to render the error templates\n# regardless of the project's TEMPLATES setting. Templates are"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "get_exception_traceback_frames",
          "class_name": "ExceptionReporter",
          "code": "def get_exception_traceback_frames(self, exc_value, tb):\n        exc_cause = self._get_explicit_or_implicit_cause(exc_value)\n        exc_cause_explicit = getattr(exc_value, \"__cause__\", True)\n        if tb is None:\n            yield {\n                \"exc_cause\": exc_cause,\n                \"exc_cause_explicit\": exc_cause_explicit,\n                \"tb\": None,\n                \"type\": \"user\",\n            }\n        while tb is not None:\n            # Support for __traceback_hide__ which is used by a few libraries\n            # to hide internal frames.\n            if tb.tb_frame.f_locals.get(\"__traceback_hide__\"):\n                tb = tb.tb_next\n                continue\n            filename = tb.tb_frame.f_code.co_filename\n            function = tb.tb_frame.f_code.co_name\n            lineno = tb.tb_lineno - 1\n            loader = tb.tb_frame.f_globals.get(\"__loader__\")\n            module_name = tb.tb_frame.f_globals.get(\"__name__\") or \"\"\n            (\n                pre_context_lineno,\n                pre_context,\n                context_line,\n                post_context,\n            ) = self._get_lines_from_file(\n                filename,\n                lineno,\n                7,\n                loader,\n                module_name,\n            )\n            if pre_context_lineno is None:\n                pre_context_lineno = lineno\n                pre_context = []\n                context_line = \"<source code not available>\"\n                post_context = []\n            yield {\n                \"exc_cause\": exc_cause,\n                \"exc_cause_explicit\": exc_cause_explicit,\n                \"tb\": tb,\n                \"type\": \"django\" if module_name.startswith(\"django.\") else \"user\",\n                \"filename\": filename,\n                \"function\": function,\n                \"lineno\": lineno + 1,\n                \"vars\": self.filter.get_traceback_frame_variables(\n                    self.request, tb.tb_frame\n                ),\n                \"id\": id(tb),\n                \"pre_context\": pre_context,\n                \"context_line\": context_line,\n                \"post_context\": post_context,\n                \"pre_context_lineno\": pre_context_lineno + 1,\n            }\n            tb = tb.tb_next"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "get_exception_traceback_frames",
          "class_name": "ExceptionReporter",
          "code": "def get_exception_traceback_frames(self, exc_value, tb):\n        exc_cause = self._get_explicit_or_implicit_cause(exc_value)\n        exc_cause_explicit = getattr(exc_value, \"__cause__\", True)\n        if tb is None:\n            yield {\n                \"exc_cause\": exc_cause,\n                \"exc_cause_explicit\": exc_cause_explicit,\n                \"tb\": None,\n                \"type\": \"user\",\n            }\n        while tb is not None:\n            # Support for __traceback_hide__ which is used by a few libraries\n            # to hide internal frames.\n            if tb.tb_frame.f_locals.get(\"__traceback_hide__\"):\n                tb = tb.tb_next\n                continue\n            filename = tb.tb_frame.f_code.co_filename\n            function = tb.tb_frame.f_code.co_name\n            lineno = tb.tb_lineno - 1\n            loader = tb.tb_frame.f_globals.get(\"__loader__\")\n            module_name = tb.tb_frame.f_globals.get(\"__name__\") or \"\"\n            (\n                pre_context_lineno,\n                pre_context,\n                context_line,\n                post_context,\n            ) = self._get_lines_from_file(\n                filename,\n                lineno,\n                7,\n                loader,\n                module_name,\n            )\n            if pre_context_lineno is None:\n                pre_context_lineno = lineno\n                pre_context = []\n                context_line = \"<source code not available>\"\n                post_context = []\n            yield {\n                \"exc_cause\": exc_cause,\n                \"exc_cause_explicit\": exc_cause_explicit,\n                \"tb\": tb,\n                \"type\": \"django\" if module_name.startswith(\"django.\") else \"user\",\n                \"filename\": filename,\n                \"function\": function,\n                \"lineno\": lineno + 1,\n                \"vars\": self.filter.get_traceback_frame_variables(\n                    self.request, tb.tb_frame\n                ),\n                \"id\": id(tb),\n                \"pre_context\": pre_context,\n                \"context_line\": context_line,\n                \"post_context\": post_context,\n                \"pre_context_lineno\": pre_context_lineno + 1,\n            }\n            tb = tb.tb_next"
        }
      ]
    },
    {
      "pr_number": 14367,
      "pr_title": "Fixed #28935 -- Fixed display of errors in extended blocks.",
      "pr_body": "Fixed #27694 -- Exception debug info on correct template.",
      "issue_id": 28935,
      "issue_title": "Template error raised in an {% extends %} child template shows incorrect source location on debug page",
      "issue_body": "",
      "issue_closed_at": "2021-07-02T05:20:43",
      "base_commit": "9f3cce172f6913c5ac74272fa5fc07f847b4e112",
      "changes": [
        {
          "file": "django/template/base.py",
          "type": "function",
          "name": "render_annotated",
          "class_name": "TextNode",
          "code": "def render_annotated(self, context):\n        \"\"\"\n        Return the given value.\n\n        The default implementation of this method handles exceptions raised\n        during rendering, which is not necessary for text nodes.\n        \"\"\"\n        return self.s"
        }
      ]
    },
    {
      "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": 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": 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"
        }
      ]
    }
  ]
}