{
  "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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)"
        }
      ]
    }
  ]
}