{
  "Selected_candidate": {
    "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"
      }
    ]
  },
  "Justification": "Candidate E is the most relevant as it directly pertains to the sitemap generation in Django, which is the focus of the current bug report regarding sitemaps with no items. Both issues involve handling attributes within sitemaps, and Candidate E addresses a similar type of ValueError related to missing information ('alternates') in the sitemap rendering process. The similarities in components and error types suggest that the insight gained from the fix in Candidate E could significantly aid in resolving the CURRENT bug involving the lastmod callable when the sitemap is empty.",
  "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"
}