{
  "instance_id": "django__django-16255",
  "repo": "django/django",
  "created_at": "2022-11-04T13:49:40Z",
  "problem_statement": "Sitemaps without items raise ValueError on callable lastmod.\nDescription\n\t\nWhen sitemap contains not items, but supports returning lastmod for an item, it fails with a ValueError:\nTraceback (most recent call last):\n File \"/usr/local/lib/python3.10/site-packages/django/core/handlers/exception.py\", line 55, in inner\n\tresponse = get_response(request)\n File \"/usr/local/lib/python3.10/site-packages/django/core/handlers/base.py\", line 197, in _get_response\n\tresponse = wrapped_callback(request, *callback_args, **callback_kwargs)\n File \"/usr/local/lib/python3.10/site-packages/django/utils/decorators.py\", line 133, in _wrapped_view\n\tresponse = view_func(request, *args, **kwargs)\n File \"/usr/local/lib/python3.10/site-packages/django/contrib/sitemaps/views.py\", line 34, in inner\n\tresponse = func(request, *args, **kwargs)\n File \"/usr/local/lib/python3.10/site-packages/django/contrib/sitemaps/views.py\", line 76, in index\n\tsite_lastmod = site.get_latest_lastmod()\n File \"/usr/local/lib/python3.10/site-packages/django/contrib/sitemaps/__init__.py\", line 170, in get_latest_lastmod\n\treturn max([self.lastmod(item) for item in self.items()])\nException Type: ValueError at /sitemap.xml\nException Value: max() arg is an empty sequence\nSomething like this might be a solution:\n\t def get_latest_lastmod(self):\n\t\t if not hasattr(self, \"lastmod\"):\n\t\t\t return None\n\t\t if callable(self.lastmod):\n\t\t\t try:\n\t\t\t\t return max([self.lastmod(item) for item in self.items()])\n-\t\t\texcept TypeError:\n+\t\t\texcept (TypeError, ValueError):\n\t\t\t\t return None\n\t\t else:\n\t\t\t return self.lastmod\n",
  "patch": "diff --git a/django/contrib/sitemaps/__init__.py b/django/contrib/sitemaps/__init__.py\n--- a/django/contrib/sitemaps/__init__.py\n+++ b/django/contrib/sitemaps/__init__.py\n@@ -167,7 +167,7 @@ def get_latest_lastmod(self):\n             return None\n         if callable(self.lastmod):\n             try:\n-                return max([self.lastmod(item) for item in self.items()])\n+                return max([self.lastmod(item) for item in self.items()], default=None)\n             except TypeError:\n                 return None\n         else:\n",
  "similar_bug_items": [
    {
      "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": 6540,
      "pr_title": "Fixed #26341 (again) -- Addressed multiple occurrences per line use case",
      "pr_body": "",
      "issue_id": 26341,
      "issue_title": "Weird comments in PO files (.html.py filenames)",
      "issue_body": "I just upgraded Django to 1.9.4 from 1.8.10, and sometimes, the filenames in PO files comments contain \".html.py\" extensions.\nThis is visible on django's main repository :\n​\nhttps://github.com/django/django/blob/ae4d932b1ac12651a7c57d89742c25483ee8c9f9/django/contrib/admin/locale/en/LC_MESSAGES/django.po#L282\n​\nhttps://github.com/django/django/blob/4323676ea5ab6994feb1385522665069d84f397b/django/contrib/admin/locale/en/LC_MESSAGES/django.po#L302\nIn this example, the \"contrib/admin/templates/admin/base_site.html\" file is now named \"contrib/admin/templates/admin/base_site.html.py\" (with a trailing \".py\") in the po file.\nThis seems to appear only on lines with a python file before the template html file.\nclaudep found that this could be the faulty commit :\n​\nhttps://github.com/django/django/commit/e75882332c",
      "issue_closed_at": "2016-04-30T05:07:43",
      "base_commit": "4e2ee8662753ca6a2619039b903f11c60709f398",
      "changes": [
        {
          "file": "django/core/management/commands/makemessages.py",
          "type": "function",
          "name": "postprocess_messages",
          "class_name": "BuildFile",
          "code": "def postprocess_messages(self, msgs):\n        \"\"\"\n        Postprocess messages generated by xgettext GNU gettext utility.\n\n        Transform paths as if these messages were generated from original\n        translatable files rather than from preprocessed versions.\n        \"\"\"\n        if not self.is_templatized:\n            return msgs\n\n        # Remove '.py' suffix\n        if os.name == 'nt':\n            # Preserve '.\\' prefix on Windows to respect gettext behavior\n            old_path = self.work_path\n            new_path = self.path\n        else:\n            old_path = self.work_path[2:]\n            new_path = self.path[2:]\n\n        return re.sub(\n            r'^(#: .*)(' + re.escape(old_path) + r')',\n            r'\\1' + new_path,\n            msgs,\n            flags=re.MULTILINE\n        )"
        }
      ]
    },
    {
      "pr_number": 11551,
      "pr_title": "Fixed #30543 -- Fixed checks of ModelAdmin.list_display for fields accessible only via instance.",
      "pr_body": "Continue [ticke 30543](https://code.djangoproject.com/ticket/30543).\r\nTest has been added.",
      "issue_id": 30543,
      "issue_title": "admin.E108 is raised on fields accessible only via instance.",
      "issue_body": "As part of startup django validates the ModelAdmin's list_display list/tuple for correctness (django.admin.contrib.checks._check_list_display). Having upgraded django from 2.07 to 2.2.1 I found that a ModelAdmin with a list display that used to pass the checks and work fine in admin now fails validation, preventing django from starting. A PositionField from the django-positions library triggers this bug, explanation why follows.\nfrom django.db import models\nfrom position.Fields import PositionField\n\nclass Thing(models.Model)\n  number = models.IntegerField(default=0)\n  order = PositionField()\nfrom django.contrib import admin\nfrom .models import Thing\n\n@admin.register(Thing)\nclass ThingAdmin(admin.ModelAdmin)\n  list_display = ['number', 'order']\nUnder 2.2.1 this raises an incorrect admin.E108 message saying \"The value of list_display\n[1]\nrefers to 'order' which is not a callable...\".\nUnder 2.0.7 django starts up successfully.\nIf you change 'number' to 'no_number' or 'order' to 'no_order' then the validation correctly complains about those.\nThe reason for this bug is commit\n​\nhttps://github.com/django/django/commit/47016adbf54b54143d4cf052eeb29fc72d27e6b1\nwhich was proposed and accepted as a fix for bug\nhttps://code.djangoproject.com/ticket/28490\n. The problem is while it fixed that bug it broke the functionality of _check_list_display_item in other cases. The rationale for that change was that after field=getattr(model, item) field could be None if item was a descriptor returning None, but subsequent code incorrectly interpreted field being None as meaning getattr raised an AttributeError. As this was done\nafter\ntrying field = model._meta.get_field(item) and that failing that meant the validation error should be returned. However, after the above change if hasattr(model, item) is false then we no longer even try field = model._meta.get_field(item) before returning an error. The reason hasattr(model, item) is false in the case of a PositionField is its\nget\nmethod throws an exception if called on an instance of the PositionField class on the Thing model class, rather than a Thing instance.\nFor clarity, here are the various logical tests that _check_list_display_item needs to deal with and the behaviour before the above change, after it, and the correct behaviour (which my suggested patch exhibits). Note this is assuming the first 2 tests callable(item) and hasattr(obj, item) are both false (corresponding to item is an actual function/lambda rather than string or an attribute of ThingAdmin).\nhasattr(model, item) returns True or  False (which is the same as seeing if getattr(model, item) raises AttributeError)\nmodel._meta.get_field(item) returns a field or raises FieldDoesNotExist\nGet a field from somewhere, could either be from getattr(model,item) if hasattr was True or from get_field.\nIs that field an instance of ManyToManyField?\nIs that field None? (True in case of bug 28490)\nhasattr\nget_field\nfield is None?\nfield ManyToMany?\n2.0 returns\n2.2 returns\nCorrect behaviour\nComments\nTrue\nok\nFalse\nFalse\n[]\n[]\n[]\n-\nTrue\nok\nFalse\nTrue\nE109\nE109\nE109\n-\nTrue\nok\nTrue\nFalse\nE108\n[]\n[]\ngood bit of 28490 fix, 2.0 was wrong\nTrue\nraises\nFalse\nFalse\n[]\n[]\n[]\n-\nTrue\nraises\nFalse\nTrue\nE109\n[]\nE109\nAnother bug introduced by 28490 fix, fails to check if ManyToMany in get_field raise case\nTrue\nraises\nTrue\nFalse\nE108\n[]\n[]\ngood bit of 28490 fix, 2.0 was wrong\nFalse\nok\nFalse\nFalse\n[]\nE108\n[]\nbad bit of 28490 fix, bug hit with PositionField\nFalse\nok\nFalse\nTrue\n[]\nE108\nE109\nboth 2.0 and 2.2 wrong\nFalse\nok\nTrue\nFalse\n[]\nE108\n[]\nbad 28490 fix\nFalse\nraises\nFalse\nFalse\nE108\nE108\nE108\n-\nFalse\nraises\nFalse\nTrue\nE108\nE108\nE108\nimpossible condition, we got no field assigned to be a ManyToMany\nFalse\nraises\nTrue\nFalse\nE108\nE108\nE108\nimpossible condition, we got no field assigned to be None\nThe following code exhibits the correct behaviour in all cases. The key changes are there is no longer a check for hasattr(model, item), as that being false should not prevent us form attempting to get the field via get_field, and only return an E108 in the case both of them fail. If either of those means or procuring it are successful then we need to check if it's a ManyToMany. Whether or not the field is None is irrelevant, and behaviour is contained within the exception catching blocks that should cause it instead of signalled through a variable being set to None which is a source of conflation of different cases.\ndef _check_list_display_item(self, obj, item, label):\n    if callable(item):\n        return []\n    elif hasattr(obj, item):\n        return []\n    else:\n        try:\n            field = obj.model._meta.get_field(item)\n        except FieldDoesNotExist:\n            try:\n                field = getattr(obj.model, item)\n            except AttributeError:\n                return [\n                    checks.Error(\n                        \"The value of '%s' refers to '%s', which is not a callable, \"\n                        \"an attribute of '%s', or an attribute or method on '%s.%s'.\" % (\n                            label, item, obj.__class__.__name__,\n                            obj.model._meta.app_label, obj.model._meta.object_name,\n                        ),\n                        obj=obj.__class__,\n                        id='admin.E108',\n                    )\n                ]\n\n        if isinstance(field, models.ManyToManyField):\n            return [\n                checks.Error(\n                    \"The value of '%s' must not be a ManyToManyField.\" % label,\n                    obj=obj.__class__,\n                    id='admin.E109',\n                )\n            ]\n        return []",
      "issue_closed_at": "2019-07-10T03:59:12",
      "base_commit": "7991111af12056ec9a856f35935d273526338c1f",
      "changes": [
        {
          "file": "django/contrib/admin/checks.py",
          "type": "function",
          "name": "_check_list_display_item",
          "class_name": "ModelAdminChecks",
          "code": "def _check_list_display_item(self, obj, item, label):\n        if callable(item):\n            return []\n        elif hasattr(obj, item):\n            return []\n        elif hasattr(obj.model, item):\n            try:\n                field = obj.model._meta.get_field(item)\n            except FieldDoesNotExist:\n                return []\n            else:\n                if isinstance(field, models.ManyToManyField):\n                    return [\n                        checks.Error(\n                            \"The value of '%s' must not be a ManyToManyField.\" % label,\n                            obj=obj.__class__,\n                            id='admin.E109',\n                        )\n                    ]\n                return []\n        else:\n            return [\n                checks.Error(\n                    \"The value of '%s' refers to '%s', which is not a callable, \"\n                    \"an attribute of '%s', or an attribute or method on '%s.%s'.\" % (\n                        label, item, obj.__class__.__name__,\n                        obj.model._meta.app_label, obj.model._meta.object_name,\n                    ),\n                    obj=obj.__class__,\n                    id='admin.E108',\n                )\n            ]"
        }
      ]
    },
    {
      "pr_number": 11634,
      "pr_title": "Fixed #30687 -- Fixed using of OuterRef() expressions in distance lookups.",
      "pr_body": "https://code.djangoproject.com/ticket/30687\r\n",
      "issue_id": 30687,
      "issue_title": "GIS distance lookups fail within subqueries using OuterRef",
      "issue_body": "I discovered this when trying to make a query of this form:\nfrom\ndjango.db\nimport\nmodels\nfrom\ndjango.contrib.gis.db.models.fields\nimport\nPointField\n,\nRasterField\nclass\nModelA\n(\nmodels\n.\nModel\n):\npointA\n=\nPointField\n()\nclass\nModelB\n(\nmodels\n.\nModel\n):\npointB\n=\nPointField\n()\ndef\nquery1\n():\nreturn\nModelA\n.\nobjects\n.\nannotate\n(\nhas_value\n=\nExists\n(\nModelB\n.\nobjects\n.\nfilter\n(\npointB__dwithin\n=\n(\nOuterRef\n(\n\"pointA\"\n),\n10\n)\n))\n)\n.\nfilter\n(\nhas_value\n=\nTrue\n)\nThis fails with \"ValueError: This queryset contains a reference to an outer query and may only be used in a subquery\"\nI dug into things, and while I may not fully understand how queries are processed, I think I've identified the problem in\nQuery.resolve_lookup_value()\n:\ndef\nresolve_lookup_value\n(\nself\n,\nvalue\n,\ncan_reuse\n,\nallow_joins\n,\nsimple_col\n):\nif\nhasattr\n(\nvalue\n,\n'resolve_expression'\n):\nkwargs\n=\n{\n'reuse'\n:\ncan_reuse\n,\n'allow_joins'\n:\nallow_joins\n}\nif\nisinstance\n(\nvalue\n,\nF\n):\nkwargs\n[\n'simple_col'\n]\n=\nsimple_col\nvalue\n=\nvalue\n.\nresolve_expression\n(\nself\n,\n**\nkwargs\n)\nelif\nisinstance\n(\nvalue\n,\n(\nlist\n,\ntuple\n)):\n# The items of the iterable may be expressions and therefore need\n# to be resolved independently.\nfor\nsub_value\nin\nvalue\n:\nif\nhasattr\n(\nsub_value\n,\n'resolve_expression'\n):\nif\nisinstance\n(\nsub_value\n,\nF\n):\nsub_value\n.\nresolve_expression\n(\nself\n,\nreuse\n=\ncan_reuse\n,\nallow_joins\n=\nallow_joins\n,\nsimple_col\n=\nsimple_col\n,\n)\nelse\n:\nsub_value\n.\nresolve_expression\n(\nself\n,\nreuse\n=\ncan_reuse\n,\nallow_joins\n=\nallow_joins\n)\nreturn\nvalue\nThis resolves the value passed in as the rhs of a filter. For single objects, it calls\nresolve_expression()\nand then returns the result. But for multiple objects in a list or tuple, it calls\nresolve_expression()\non each but doesn't return the resolved objects. Rather it returns the original list.\nAs a consequence, during the call to filter, the passed in\nOuterRef\nobject doesn't get resolved to a\nResolvedOuterRef\nobject, since the\n__dwithin\nlookup takes a tuple of\n(value, distance)\n, triggering that second code path above. Later during the call to annotate, the\nOuterRef\ndoes resolve to a\nResolvedOuterRef\nbut when the query is compiled,\nResolvedOuterRef.as_sql()\nis called which raises the error.\nI modified\nresolve_lookup_value()\nto return the list or tuple of resolved values, and wrote a test for the query above, which I will submit in a PR shortly.",
      "issue_closed_at": "2019-08-16T02:03:57",
      "base_commit": "8289fc55fff879df273cb95fdd1b039447f85783",
      "changes": [
        {
          "file": "django/db/models/sql/query.py",
          "type": "function",
          "name": "resolve_lookup_value",
          "class_name": "Query",
          "code": "def resolve_lookup_value(self, value, can_reuse, allow_joins, simple_col):\n        if hasattr(value, 'resolve_expression'):\n            kwargs = {'reuse': can_reuse, 'allow_joins': allow_joins}\n            if isinstance(value, F):\n                kwargs['simple_col'] = simple_col\n            value = value.resolve_expression(self, **kwargs)\n        elif isinstance(value, (list, tuple)):\n            # The items of the iterable may be expressions and therefore need\n            # to be resolved independently.\n            for sub_value in value:\n                if hasattr(sub_value, 'resolve_expression'):\n                    if isinstance(sub_value, F):\n                        sub_value.resolve_expression(\n                            self, reuse=can_reuse, allow_joins=allow_joins,\n                            simple_col=simple_col,\n                        )\n                    else:\n                        sub_value.resolve_expression(self, reuse=can_reuse, allow_joins=allow_joins)\n        return value"
        }
      ]
    },
    {
      "pr_number": 14259,
      "pr_title": "Fixed #32648 -- Fixed VariableDoesNotExist rendering sitemaps template.",
      "pr_body": "See https://code.djangoproject.com/ticket/32648#ticket",
      "issue_id": 32648,
      "issue_title": "New sitemap 'alternates' generation feature is bugged using default values.",
      "issue_body": "(First time reporting a bug in Django, please be kind)\nHi,\nWhen closing ticket\n#27395\nwith the following commit\n​\nhttps://github.com/django/django/commit/16218c20606d8cd89c5393970c83da04598a3e04\n# a bug was added in Django.\nIf alternates is set to False, or if i18n is disabled on the sitemap, this line\n​\nhttps://github.com/django/django/commit/16218c20606d8cd89c5393970c83da04598a3e04#diff-d0316d5baddb3fd017c4a17ac10d784a4668a05ae39bf8a0485ec80da1409c51R189\nwill not get executed, meaning\nurl_info\nwill not have\nalternates\nset.\nLater, when rendering in the template, the inner loop (\n​\nhttps://github.com/django/django/commit/16218c20606d8cd89c5393970c83da04598a3e04#diff-a2c649c9d199c72cb1df4204ce54d92a480f4f077e7b423db91ee1ab421895d8R10\n) will try to access alternates anyway, causing the following stacktrace to be printed\n[\nDEBUG\n]\n(\nbase\n.\n_resolve_lookup\n)\nException\nwhile\nresolving\nvariable\n'alternates'\nin\ntemplate\n'sitemap.xml'\n.\nTraceback\n(\nmost\nrecent\ncall\nlast\n):\nFile\n\"/usr/local/lib/python3.9/site-packages/django/template/base.py\"\n,\nline\n829\n,\nin\n_resolve_lookup\ncurrent\n=\ncurrent\n[\nbit\n]\nKeyError\n:\n'alternates'\nDuring\nhandling\nof\nthe\nabove\nexception\n,\nanother\nexception\noccurred\n:\nTraceback\n(\nmost\nrecent\ncall\nlast\n):\nFile\n\"/usr/local/lib/python3.9/site-packages/django/template/base.py\"\n,\nline\n837\n,\nin\n_resolve_lookup\ncurrent\n=\ngetattr\n(\ncurrent\n,\nbit\n)\nAttributeError\n:\n'dict'\nobject\nhas\nno\nattribute\n'alternates'\nDuring\nhandling\nof\nthe\nabove\nexception\n,\nanother\nexception\noccurred\n:\nTraceback\n(\nmost\nrecent\ncall\nlast\n):\nFile\n\"/usr/local/lib/python3.9/site-packages/django/template/base.py\"\n,\nline\n843\n,\nin\n_resolve_lookup\ncurrent\n=\ncurrent\n[\nint\n(\nbit\n)]\nValueError\n:\ninvalid\nliteral\nfor\nint\n()\nwith\nbase\n10\n:\n'alternates'\nDuring\nhandling\nof\nthe\nabove\nexception\n,\nanother\nexception\noccurred\n:\nTraceback\n(\nmost\nrecent\ncall\nlast\n):\nFile\n\"/usr/local/lib/python3.9/site-packages/django/template/base.py\"\n,\nline\n848\n,\nin\n_resolve_lookup\nraise\nVariableDoesNotExist\n(\n\"Failed lookup for key \"\ndjango\n.\ntemplate\n.\nbase\n.\nVariableDoesNotExist\n:\nFailed\nlookup\nfor\nkey\n[\nalternates\n]\nin\n{\n'item'\n:\n<\nItem\n__str__\n>\n,\n'location'\n:\n'https://myurl'\n,\n'lastmod'\n:\nNone\n,\n'changefreq'\n:\n'always'\n,\n'priority'\n:\n''\n}\nA simple fix is the meantime is to redefine the _urls() method in the Sitemap to include the alternates attribute to an empty list :\nclass\nSitemap\n(\nsitemaps\n.\nSitemap\n):\n\"\"\"\nFixes the Exception while resolving variable 'alternates' in template 'sitemap.xml'\n\"\"\"\ndef\n_urls\n(\nself\n,\n*\nargs\n,\n**\nkwargs\n):\nurls\n=\nsuper\n()\n.\n_urls\n(\n*\nargs\n,\n**\nkwargs\n)\nfor\nurl_info\nin\nurls\n:\nurl_info\n[\n'alternates'\n]\n=\n[]\nreturn\nurls\nThat said, the patch is probably a 1-liner, to check if url.alternates exist before using it.",
      "issue_closed_at": "2021-04-14T12:40:44",
      "base_commit": "23fa29f6a6659e0f600d216de6bcb79e7f6818c9",
      "changes": [
        {
          "file": "django/contrib/sitemaps/__init__.py",
          "type": "function",
          "name": "_urls",
          "class_name": "Sitemap",
          "code": "def _urls(self, page, protocol, domain):\n        urls = []\n        latest_lastmod = None\n        all_items_lastmod = True  # track if all items have a lastmod\n\n        paginator_page = self.paginator.page(page)\n        for item in paginator_page.object_list:\n            loc = f'{protocol}://{domain}{self._location(item)}'\n            priority = self._get('priority', item)\n            lastmod = self._get('lastmod', item)\n\n            if all_items_lastmod:\n                all_items_lastmod = lastmod is not None\n                if (all_items_lastmod and\n                        (latest_lastmod is None or lastmod > latest_lastmod)):\n                    latest_lastmod = lastmod\n\n            url_info = {\n                'item': item,\n                'location': loc,\n                'lastmod': lastmod,\n                'changefreq': self._get('changefreq', item),\n                'priority': str(priority if priority is not None else ''),\n            }\n\n            if self.i18n and self.alternates:\n                alternates = []\n                for lang_code in self._languages():\n                    loc = f'{protocol}://{domain}{self._location(item, lang_code)}'\n                    alternates.append({\n                        'location': loc,\n                        'lang_code': lang_code,\n                    })\n                if self.x_default:\n                    lang_code = settings.LANGUAGE_CODE\n                    loc = f'{protocol}://{domain}{self._location(item, lang_code)}'\n                    loc = loc.replace(f'/{lang_code}/', '/', 1)\n                    alternates.append({\n                        'location': loc,\n                        'lang_code': 'x-default',\n                    })\n                url_info['alternates'] = alternates\n\n            urls.append(url_info)\n\n        if all_items_lastmod and latest_lastmod:\n            self.latest_lastmod = latest_lastmod\n\n        return urls"
        },
        {
          "file": "django/contrib/sitemaps/__init__.py",
          "type": "function",
          "name": "_urls",
          "class_name": "Sitemap",
          "code": "def _urls(self, page, protocol, domain):\n        urls = []\n        latest_lastmod = None\n        all_items_lastmod = True  # track if all items have a lastmod\n\n        paginator_page = self.paginator.page(page)\n        for item in paginator_page.object_list:\n            loc = f'{protocol}://{domain}{self._location(item)}'\n            priority = self._get('priority', item)\n            lastmod = self._get('lastmod', item)\n\n            if all_items_lastmod:\n                all_items_lastmod = lastmod is not None\n                if (all_items_lastmod and\n                        (latest_lastmod is None or lastmod > latest_lastmod)):\n                    latest_lastmod = lastmod\n\n            url_info = {\n                'item': item,\n                'location': loc,\n                'lastmod': lastmod,\n                'changefreq': self._get('changefreq', item),\n                'priority': str(priority if priority is not None else ''),\n            }\n\n            if self.i18n and self.alternates:\n                alternates = []\n                for lang_code in self._languages():\n                    loc = f'{protocol}://{domain}{self._location(item, lang_code)}'\n                    alternates.append({\n                        'location': loc,\n                        'lang_code': lang_code,\n                    })\n                if self.x_default:\n                    lang_code = settings.LANGUAGE_CODE\n                    loc = f'{protocol}://{domain}{self._location(item, lang_code)}'\n                    loc = loc.replace(f'/{lang_code}/', '/', 1)\n                    alternates.append({\n                        'location': loc,\n                        'lang_code': 'x-default',\n                    })\n                url_info['alternates'] = alternates\n\n            urls.append(url_info)\n\n        if all_items_lastmod and latest_lastmod:\n            self.latest_lastmod = latest_lastmod\n\n        return urls"
        }
      ]
    }
  ]
}