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