{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate B is the most helpful because it directly addresses a configuration issue with duplicate aliases in the template settings, which is closely related to the current bug regarding template tag libraries. Both bugs deal with template management and configurations, and the error message generated in Candidate B highlights potential misconfigurations similar to the issue with multiple uses of the same template tag, as seen in the CURRENT bug report. Furthermore, resolving similar issues with the handling of templates provides insight into error reporting and configuration checks, making this report particularly useful in identifying and fixing the current problem.",
  "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"
}