{
  "instance_id": "django__django-15790",
  "repo": "django/django",
  "created_at": "2022-06-23T11:02:06Z",
  "problem_statement": "check_for_template_tags_with_the_same_name with libraries in TEMPLATES\nDescription\n\t\nI didn't explore this thoroughly, but I think there might be an issue with the check_for_template_tags_with_the_same_name when you add a template tag library into TEMPLATES['OPTIONS']['librairies'].\nI'm getting an error like: \n(templates.E003) 'my_tags' is used for multiple template tag modules: 'someapp.templatetags.my_tags', 'someapp.templatetags.my_tags'\n",
  "patch": "diff --git a/django/core/checks/templates.py b/django/core/checks/templates.py\n--- a/django/core/checks/templates.py\n+++ b/django/core/checks/templates.py\n@@ -50,15 +50,15 @@ def check_string_if_invalid_is_string(app_configs, **kwargs):\n @register(Tags.templates)\n def check_for_template_tags_with_the_same_name(app_configs, **kwargs):\n     errors = []\n-    libraries = defaultdict(list)\n+    libraries = defaultdict(set)\n \n     for conf in settings.TEMPLATES:\n         custom_libraries = conf.get(\"OPTIONS\", {}).get(\"libraries\", {})\n         for module_name, module_path in custom_libraries.items():\n-            libraries[module_name].append(module_path)\n+            libraries[module_name].add(module_path)\n \n     for module_name, module_path in get_template_tag_modules():\n-        libraries[module_name].append(module_path)\n+        libraries[module_name].add(module_path)\n \n     for library_name, items in libraries.items():\n         if len(items) > 1:\n@@ -66,7 +66,7 @@ def check_for_template_tags_with_the_same_name(app_configs, **kwargs):\n                 Error(\n                     E003.msg.format(\n                         repr(library_name),\n-                        \", \".join(repr(item) for item in items),\n+                        \", \".join(repr(item) for item in sorted(items)),\n                     ),\n                     id=E003.id,\n                 )\n",
  "similar_bug_items": [
    {
      "pr_number": 9255,
      "pr_title": "Fixed #28719 -- Added a helpful exception if MultipleObjectTemplateResponseMixin doesn't generate any template names.",
      "pr_body": "https://code.djangoproject.com/ticket/28719",
      "issue_id": 28719,
      "issue_title": "Add a helpful exception message when ListView.get_queryset() returns None",
      "issue_body": "One of my\nListViews\nsuddenly started raising\nTemplateDoesNotExist\nwith a rather cryptic (to me) message:\nTemplate-loader postmortem\n \nDjango tried loading these templates, in this order:\n \nUsing engine :\nThis engine did not provide a list of tried templates.\nIt took me a while to realise this was because my get_queryset wasn't returning anything. It did some filtering based on user settings, and I didn't have a fallback for when none of the filtering steps applied.\nThought it might be helpful to have a better message is no template names are found and\nobject_list\nis\nNone\n. Suggesting: \"Expected a queryset, but found None. Please check that <cls>.get_queryset() returns a queryset.\" Pull request coming up.",
      "issue_closed_at": "2017-11-07T18:04:43",
      "base_commit": "a2851f204c6431330042d0343ee99f33449f78e0",
      "changes": [
        {
          "file": "django/views/generic/list.py",
          "type": "function",
          "name": "get_template_names",
          "class_name": "MultipleObjectTemplateResponseMixin",
          "code": "def get_template_names(self):\n        \"\"\"\n        Return a list of template names to be used for the request. Must return\n        a list. May not be called if render_to_response is overridden.\n        \"\"\"\n        try:\n            names = super().get_template_names()\n        except ImproperlyConfigured:\n            # If template_name isn't specified, it's not a problem --\n            # we just start with an empty list.\n            names = []\n\n        # If the list is a queryset, we'll invent a template name based on the\n        # app and model name. This name gets put at the end of the template\n        # name list so that user-supplied names override the automatically-\n        # generated ones.\n        if hasattr(self.object_list, 'model'):\n            opts = self.object_list.model._meta\n            names.append(\"%s/%s%s.html\" % (opts.app_label, opts.model_name, self.template_name_suffix))\n\n        return names"
        }
      ]
    },
    {
      "pr_number": 4607,
      "pr_title": "Fixed #24685 -- Fixed check for template name unicity.",
      "pr_body": "Thanks Preston Timmons for the report.\n",
      "issue_id": 24685,
      "issue_title": "EngineHandler doesn't raise exception when duplicate aliases are registered",
      "issue_body": "The\ndjango.template.utils.EngineHandler\nhas a check to raise\nImproperlyConfigured\nif multiple engines are specified with the same alias:\n​\nhttps://github.com/django/django/blob/master/django/template/utils.py#L72\nThis won't ever raise an error, though, because configuration is stored in an\nOrderedDict\n. Duplicate aliases simply overwrite previous ones.\nCurtis ran into this problem when using settings like this without the\nNAME\noption:\nTEMPLATES = [\n    {\n        'BACKEND': 'knights.django.KnightsTemplater',\n        ...\n    },\n    {\n        'BACKEND': 'django.template.backends.django.DjangoTemplates',\n        ...\n    },\n]",
      "issue_closed_at": "2015-05-04T00:37:45",
      "base_commit": "db0a0c4b8ae490d0596f2c210d3edba3d700374a",
      "changes": [
        {
          "file": "django/template/utils.py",
          "type": "function",
          "name": "templates",
          "class_name": "EngineHandler",
          "code": "def templates(self):\n        if self._templates is None:\n            self._templates = settings.TEMPLATES\n\n        if not self._templates:\n            warnings.warn(\n                \"You haven't defined a TEMPLATES setting. You must do so \"\n                \"before upgrading to Django 2.0. Otherwise Django will be \"\n                \"unable to load templates.\", RemovedInDjango20Warning)\n            self._templates = [\n                {\n                    'BACKEND': 'django.template.backends.django.DjangoTemplates',\n                    'DIRS': settings.TEMPLATE_DIRS,\n                    'OPTIONS': {\n                        'allowed_include_roots': settings.ALLOWED_INCLUDE_ROOTS,\n                        'context_processors': settings.TEMPLATE_CONTEXT_PROCESSORS,\n                        'debug': settings.TEMPLATE_DEBUG,\n                        'loaders': settings.TEMPLATE_LOADERS,\n                        'string_if_invalid': settings.TEMPLATE_STRING_IF_INVALID,\n                    },\n                },\n            ]\n\n        templates = OrderedDict()\n        for tpl in self._templates:\n            tpl = tpl.copy()\n            try:\n                # This will raise an exception if 'BACKEND' doesn't exist or\n                # isn't a string containing at least one dot.\n                default_name = tpl['BACKEND'].rsplit('.', 2)[-2]\n            except Exception:\n                invalid_backend = tpl.get('BACKEND', '<not defined>')\n                raise ImproperlyConfigured(\n                    \"Invalid BACKEND for a template engine: {}. Check \"\n                    \"your TEMPLATES setting.\".format(invalid_backend))\n\n            tpl.setdefault('NAME', default_name)\n            tpl.setdefault('DIRS', [])\n            tpl.setdefault('APP_DIRS', False)\n            tpl.setdefault('OPTIONS', {})\n\n            templates[tpl['NAME']] = tpl\n\n        counts = Counter(list(templates))\n        duplicates = [alias for alias, count in counts.most_common() if count > 1]\n        if duplicates:\n            raise ImproperlyConfigured(\n                \"Template engine aliases aren't unique, duplicates: {}. \"\n                \"Set a unique NAME for each engine in settings.TEMPLATES.\"\n                .format(\", \".join(duplicates)))\n\n        return templates"
        },
        {
          "file": "django/template/utils.py",
          "type": "function",
          "name": "templates",
          "class_name": "EngineHandler",
          "code": "def templates(self):\n        if self._templates is None:\n            self._templates = settings.TEMPLATES\n\n        if not self._templates:\n            warnings.warn(\n                \"You haven't defined a TEMPLATES setting. You must do so \"\n                \"before upgrading to Django 2.0. Otherwise Django will be \"\n                \"unable to load templates.\", RemovedInDjango20Warning)\n            self._templates = [\n                {\n                    'BACKEND': 'django.template.backends.django.DjangoTemplates',\n                    'DIRS': settings.TEMPLATE_DIRS,\n                    'OPTIONS': {\n                        'allowed_include_roots': settings.ALLOWED_INCLUDE_ROOTS,\n                        'context_processors': settings.TEMPLATE_CONTEXT_PROCESSORS,\n                        'debug': settings.TEMPLATE_DEBUG,\n                        'loaders': settings.TEMPLATE_LOADERS,\n                        'string_if_invalid': settings.TEMPLATE_STRING_IF_INVALID,\n                    },\n                },\n            ]\n\n        templates = OrderedDict()\n        for tpl in self._templates:\n            tpl = tpl.copy()\n            try:\n                # This will raise an exception if 'BACKEND' doesn't exist or\n                # isn't a string containing at least one dot.\n                default_name = tpl['BACKEND'].rsplit('.', 2)[-2]\n            except Exception:\n                invalid_backend = tpl.get('BACKEND', '<not defined>')\n                raise ImproperlyConfigured(\n                    \"Invalid BACKEND for a template engine: {}. Check \"\n                    \"your TEMPLATES setting.\".format(invalid_backend))\n\n            tpl.setdefault('NAME', default_name)\n            tpl.setdefault('DIRS', [])\n            tpl.setdefault('APP_DIRS', False)\n            tpl.setdefault('OPTIONS', {})\n\n            templates[tpl['NAME']] = tpl\n\n        counts = Counter(list(templates))\n        duplicates = [alias for alias, count in counts.most_common() if count > 1]\n        if duplicates:\n            raise ImproperlyConfigured(\n                \"Template engine aliases aren't unique, duplicates: {}. \"\n                \"Set a unique NAME for each engine in settings.TEMPLATES.\"\n                .format(\", \".join(duplicates)))\n\n        return templates"
        }
      ]
    },
    {
      "pr_number": 12263,
      "pr_title": "Fixed #31166 -- Used \"raise from\" when raising ImproperlyConfigured exceptions in django.urls.resolvers.",
      "pr_body": "This will make it have a text of \"this exception is the direct result of\" instead of \"During handling of the above exception, another exception occurred\" \r\n\r\nThis is more accurate for the case of this exception.",
      "issue_id": 31166,
      "issue_title": "Provide context for ImproperlyConfigured exceptions in URL resolver.",
      "issue_body": "The patch is ready here:\n​\nhttps://github.com/django/django/pull/12263\nThis will make it have a text of \"this exception is the direct result of\" instead of \"During handling of the above exception, another exception occurred\"\nThis is more accurate for the case of this exception.\nIf this change will be merged, there are a couple more place in this module where it can be added.\nWhen it happened to me a few weeks ago, it was due to a circular import, which I then fixed. Now of course, when I tried to reproduce the circular import to show you, I couldn't. But I could achieve the same thing by renaming\nurlpatterns\nto\nxurlpatterns\n, so it wouldn't be available. This is the error I get:\nException in thread django-main-thread:\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 590, in url_patterns\n    iter(patterns)\nTypeError: 'module' object is not iterable\n\nDuring handling of the above exception, another exception occurred:\n\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 932, in _bootstrap_inner\n    self.run()\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 870, in run\n    self._target(*self._args, **self._kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\autoreload.py\", line 53, in wrapper\n    fn(*args, **kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\commands\\runserver.py\", line 117, in inner_run\n    self.check(display_num_errors=True)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 392, in check\n    all_issues = self._run_checks(\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 382, in _run_checks\n    return checks.run_checks(**kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\registry.py\", line 72, in run_checks\n    new_errors = check(app_configs=app_configs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 13, in check_url_config\n    return check_resolver(resolver)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 408, in check\n    messages.extend(check_resolver(pattern))\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 407, in check\n    for pattern in self.url_patterns:\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\functional.py\", line 48, in __get__\n    res = instance.__dict__[self.name] = self.func(instance)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 597, in url_patterns\n    raise ImproperlyConfigured(msg.format(name=self.urlconf_name))\ndjango.core.exceptions.ImproperlyConfigured: The included URLconf '<module 'barb.urls' from 'C:\\\\Users\\\\Administrator\\\\Desktop\\\\foof\\\\barb\\\\urls.py'>' does not appear to have any patterns in it. If you see valid patterns in the file then the issue is probably caused by a circular import.\nThe bad part is\nDuring handling of the above exception, another exception occurred\n.\nAfter my fix, it looks like this:\nException in thread django-main-thread:\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 590, in url_patterns\n    iter(patterns)\nTypeError: 'module' object is not iterable\n\nThe above exception was the direct cause of the following exception:\n\nTraceback (most recent call last):\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 932, in _bootstrap_inner\n    self.run()\n  File \"C:\\Program Files\\Python38\\lib\\threading.py\", line 870, in run\n    self._target(*self._args, **self._kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\autoreload.py\", line 53, in wrapper\n    fn(*args, **kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\commands\\runserver.py\", line 117, in inner_run\n    self.check(display_num_errors=True)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 392, in check\n    all_issues = self._run_checks(\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\management\\base.py\", line 382, in _run_checks\n    return checks.run_checks(**kwargs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\registry.py\", line 72, in run_checks\n    new_errors = check(app_configs=app_configs)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 13, in check_url_config\n    return check_resolver(resolver)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 408, in check\n    messages.extend(check_resolver(pattern))\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\core\\checks\\urls.py\", line 23, in check_resolver\n    return check_method()\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 407, in check\n    for pattern in self.url_patterns:\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\utils\\functional.py\", line 48, in __get__\n    res = instance.__dict__[self.name] = self.func(instance)\n  File \"C:\\Program Files\\Python38\\lib\\site-packages\\django\\urls\\resolvers.py\", line 597, in url_patterns\n    raise ImproperlyConfigured(msg.format(name=self.urlconf_name)) from type_error # Ram hack, remove\ndjango.core.exceptions.ImproperlyConfigured: The included URLconf '<module 'barb.urls' from 'C:\\\\Users\\\\Administrator\\\\Desktop\\\\foof\\\\barb\\\\urls.py'>' does not appear to have any patterns in it. If you see valid patterns in the file then the issue is probably caused by a circular import.\nNow the text is corrected to\nThe above exception was the direct cause of the following exception",
      "issue_closed_at": "2020-01-17T05:21:04",
      "base_commit": "73563183c2ea92e9ef1d3a1f790a503acc14ade2",
      "changes": [
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "_compile",
          "class_name": "RoutePattern",
          "code": "def _compile(self, route):\n        return re.compile(_route_to_regex(route, self._is_endpoint)[0])"
        },
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "_route_to_regex",
          "class_name": null,
          "code": "def _route_to_regex(route, is_endpoint=False):\n    \"\"\"\n    Convert a path pattern into a regular expression. Return the regular\n    expression and a dictionary mapping the capture names to the converters.\n    For example, 'foo/<int:pk>' returns '^foo\\\\/(?P<pk>[0-9]+)'\n    and {'pk': <django.urls.converters.IntConverter>}.\n    \"\"\"\n    if not set(route).isdisjoint(string.whitespace):\n        raise ImproperlyConfigured(\"URL route '%s' cannot contain whitespace.\" % route)\n    original_route = route\n    parts = ['^']\n    converters = {}\n    while True:\n        match = _PATH_PARAMETER_COMPONENT_RE.search(route)\n        if not match:\n            parts.append(re.escape(route))\n            break\n        parts.append(re.escape(route[:match.start()]))\n        route = route[match.end():]\n        parameter = match.group('parameter')\n        if not parameter.isidentifier():\n            raise ImproperlyConfigured(\n                \"URL route '%s' uses parameter name %r which isn't a valid \"\n                \"Python identifier.\" % (original_route, parameter)\n            )\n        raw_converter = match.group('converter')\n        if raw_converter is None:\n            # If a converter isn't specified, the default is `str`.\n            raw_converter = 'str'\n        try:\n            converter = get_converter(raw_converter)\n        except KeyError as e:\n            raise ImproperlyConfigured(\n                \"URL route '%s' uses invalid converter %s.\" % (original_route, e)\n            )\n        converters[parameter] = converter\n        parts.append('(?P<' + parameter + '>' + converter.regex + ')')\n    if is_endpoint:\n        parts.append('$')\n    return ''.join(parts), converters"
        },
        {
          "file": "django/urls/resolvers.py",
          "type": "function",
          "name": "url_patterns",
          "class_name": "URLResolver",
          "code": "def url_patterns(self):\n        # urlconf_module might be a valid set of patterns, so we default to it\n        patterns = getattr(self.urlconf_module, \"urlpatterns\", self.urlconf_module)\n        try:\n            iter(patterns)\n        except TypeError:\n            msg = (\n                \"The included URLconf '{name}' does not appear to have any \"\n                \"patterns in it. If you see valid patterns in the file then \"\n                \"the issue is probably caused by a circular import.\"\n            )\n            raise ImproperlyConfigured(msg.format(name=self.urlconf_name))\n        return patterns"
        }
      ]
    },
    {
      "pr_number": 9861,
      "pr_title": "Fixed #29296 -- Fix handling of callable object views in admindocs",
      "pr_body": "See https://code.djangoproject.com/ticket/29296\r\n\r\nI'm not sure how you normally handle backports, but a backported version of this commit can be found here:\r\nhttps://github.com/PaulSD/django/commit/b2128a2afa14c64b39a5fae1f9a63b2959c4f752",
      "issue_id": 29296,
      "issue_title": "admindocs ViewIndexView crashes if a syndication Feed (or something without __qualname__) is configured",
      "issue_body": "If a Feed (\n​\nhttps://docs.djangoproject.com/en/2.0/ref/contrib/syndication/\n) is configured anywhere in the project, then /admin/doc/views/ (\n​\nhttps://docs.djangoproject.com/en/2.0/ref/contrib/admin/admindocs/\n) crashes with AttributeError:\n'GlobalFeed' object has no attribute '__qualname__'\nThe problem is that Feed is a callable object (\n​\nhttps://github.com/django/django/blob/master/django/contrib/syndication/views.py#L34\n) and the documentation uses that class directly as a View (\npath('latest/feed/', LatestEntriesFeed()),\n), but admindocs assumes that all views are functions and it does not work properly with a callable object (\n​\nhttps://github.com/django/django/blob/master/django/contrib/admindocs/views.py#L130\n).\nIn Django 1.11 and earlier, admindocs supported callable objects on Python 2 because it fell back to\nview.__class__.__name__\n, but it appears that the Python 3 code in admindocs has not supported this since it was added in\n​\nhttps://github.com/django/django/commit/ae0f55eb491255217d6df31296ec8102007224a6\n(\nhttps://code.djangoproject.com/ticket/27018\n).\nAs a work-around,\n__qualname__\ncan be manually defined on Feed objects:\nclass MyFeed(Feed):\n    def __init__(self):\n        super().__init__()\n        self.__qualname__ = self.__class__.__name__\nWithout knowing Django's opinion of the use of callable objects as views, I don't know the appropriate way to fix this: Should admindocs support callable objects as views, or should Feed not use a callable object as a view?",
      "issue_closed_at": "2018-04-12T12:34:31",
      "base_commit": "ee17bb8a67a9e7e688da6e6f4b3be1b3a69c09b0",
      "changes": [
        {
          "file": "django/contrib/admindocs/middleware.py",
          "type": "line",
          "name": "line 2",
          "code": "from django.http import HttpResponse\nfrom django.utils.deprecation import MiddlewareMixin\n\n\nclass XViewMiddleware(MiddlewareMixin):\n    \"\"\""
        },
        {
          "file": "django/contrib/admindocs/middleware.py",
          "type": "function",
          "name": "process_view",
          "class_name": "XViewMiddleware",
          "code": "def process_view(self, request, view_func, view_args, view_kwargs):\n        \"\"\"\n        If the request method is HEAD and either the IP is internal or the\n        user is a logged-in staff member, return a responsewith an x-view\n        header indicating the view function. This is used to lookup the view\n        function for an arbitrary page.\n        \"\"\"\n        assert hasattr(request, 'user'), (\n            \"The XView middleware requires authentication middleware to be \"\n            \"installed. Edit your MIDDLEWARE%s setting to insert \"\n            \"'django.contrib.auth.middleware.AuthenticationMiddleware'.\" % (\n                \"_CLASSES\" if settings.MIDDLEWARE is None else \"\"\n            )\n        )\n        if request.method == 'HEAD' and (request.META.get('REMOTE_ADDR') in settings.INTERNAL_IPS or\n                                         (request.user.is_active and request.user.is_staff)):\n            response = HttpResponse()\n            response['X-View'] = \"%s.%s\" % (view_func.__module__, view_func.__name__)\n            return response"
        },
        {
          "file": "django/contrib/admindocs/utils.py",
          "type": "line",
          "name": "line 18",
          "code": "    docutils_is_available = True\n\n\ndef trim_docstring(docstring):\n    \"\"\"\n    Uniformly trim leading/trailing whitespace from docstrings."
        },
        {
          "file": "django/contrib/admindocs/views.py",
          "type": "line",
          "name": "line 23",
          "code": "from django.utils.translation import gettext as _\nfrom django.views.generic import TemplateView\n\n# Exclude methods starting with these strings from documentation\nMODEL_METHODS_EXCLUDE = ('_', 'add_', 'delete', 'save', 'set_')\n"
        },
        {
          "file": "django/contrib/admindocs/views.py",
          "type": "function",
          "name": "get_context_data",
          "class_name": "TemplateDetailView",
          "code": "def get_context_data(self, **kwargs):\n        template = self.kwargs['template']\n        templates = []\n        try:\n            default_engine = Engine.get_default()\n        except ImproperlyConfigured:\n            # Non-trivial TEMPLATES settings aren't supported (#24125).\n            pass\n        else:\n            # This doesn't account for template loaders (#24128).\n            for index, directory in enumerate(default_engine.dirs):\n                template_file = os.path.join(directory, template)\n                if os.path.exists(template_file):\n                    with open(template_file) as f:\n                        template_contents = f.read()\n                else:\n                    template_contents = ''\n                templates.append({\n                    'file': template_file,\n                    'exists': os.path.exists(template_file),\n                    'contents': template_contents,\n                    'order': index,\n                })\n        return super().get_context_data(**{\n            **kwargs,\n            'name': template,\n            'templates': templates,\n        })"
        }
      ]
    },
    {
      "pr_number": 5437,
      "pr_title": "Fixed #25560 -- Made empty string related name invalid.",
      "pr_body": "Thanks to Ali Lotfi for the initial report and patch.\n",
      "issue_id": 25560,
      "issue_title": "Empty string related name should be invalid",
      "issue_body": "The actual model field check is\n​\nskipped\nif the\nrelated_name\nis an empty string when a warning should be raised since it's an invalid Python identifier.",
      "issue_closed_at": "2015-10-16T13:18:21",
      "base_commit": "4dcc2a195595f8d7ddad45bc4baf98ffdeec7f41",
      "changes": [
        {
          "file": "django/db/models/fields/related.py",
          "type": "function",
          "name": "_check_related_name_is_valid",
          "class_name": "RelatedField",
          "code": "def _check_related_name_is_valid(self):\n        import re\n        import keyword\n        related_name = self.remote_field.related_name\n        if not related_name:\n            return []\n        is_valid_id = True\n        if keyword.iskeyword(related_name):\n            is_valid_id = False\n        if six.PY3:\n            if not related_name.isidentifier():\n                is_valid_id = False\n        else:\n            if not re.match(r'^[a-zA-Z_][a-zA-Z0-9_]*\\Z', related_name):\n                is_valid_id = False\n        if not (is_valid_id or related_name.endswith('+')):\n            return [\n                checks.Error(\n                    \"The name '%s' is invalid related_name for field %s.%s\" %\n                    (self.remote_field.related_name, self.model._meta.object_name,\n                     self.name),\n                    hint=\"Related name must be a valid Python identifier or end with a '+'\",\n                    obj=self,\n                    id='fields.E306',\n                )\n            ]\n        return []"
        },
        {
          "file": "django/db/models/fields/reverse_related.py",
          "type": "function",
          "name": "get_db_prep_lookup",
          "class_name": "ForeignObjectRel",
          "code": "def get_db_prep_lookup(self, lookup_type, value, connection, prepared=False):\n        # Defer to the actual field definition for db prep\n        return self.field.get_db_prep_lookup(lookup_type, value, connection=connection, prepared=prepared)"
        }
      ]
    }
  ]
}