{
  "instance_id": "django__django-14017",
  "problem": "Add a helpful exception message when ListView.get_queryset() returns None\nOne of my\nListViews\nsuddenly started raising\nTemplateDoesNotExist\nwith a rather cryptic (to me) message:\nTemplate-loader postmortem\n \nDjango tried loading these templates, in this order:\n \nUsing engine :\nThis engine did not provide a list of tried templates.\nIt took me a while to realise this was because my get_queryset wasn't returning anything. It did some filtering based on user settings, and I didn't have a fallback for when none of the filtering steps applied.\nThought it might be helpful to have a better message is no template names are found and\nobject_list\nis\nNone\n. Suggesting: \"Expected a queryset, but found None. Please check that <cls>.get_queryset() returns a queryset.\" Pull request coming up.",
  "patch": "diff --git a/django/views/generic/list.py b/django/views/generic/list.py\nindex 906e2a60163d..6753c1ab225a 100644\n--- a/django/views/generic/list.py\n+++ b/django/views/generic/list.py\n@@ -181,7 +181,13 @@ def get_template_names(self):\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+        elif not names:\n+            raise ImproperlyConfigured(\n+                \"%(cls)s requires either a 'template_name' attribute \"\n+                \"or a get_queryset() method that returns a QuerySet.\" % {\n+                    'cls': self.__class__.__name__,\n+                }\n+            )\n         return names\n \n \n",
  "file_think": "Step 1: Extract key terms from the issue: ListView, get_queryset(), object_list is None, no template names found, TemplateDoesNotExist, and suggestion to add a helpful exception message.\nStep 2: Map these terms to Django components: ListView is part of Django’s generic class-based views; object_list and template resolution for list views are handled by MultipleObjectMixin/MultipleObjectTemplateResponseMixin, which live with ListView.\nStep 3: Identify likely file locations: In Django, generic views reside under django/views/generic/. ListView and related mixins are implemented in django/views/generic/list.py. Template loading errors are surfaced by the template engine, but the root cause here is the view not providing a queryset.\nStep 4: Narrow candidates: While django/template/* or django/views/generic/base.py could be involved, the fix should occur where queryset/object_list is obtained and template names are derived for list views—i.e., in ListView/MultipleObjectTemplateResponseMixin within django/views/generic/list.py.\nStep 5: Conclude the best modification point: To provide a helpful message when get_queryset() returns None, the check and error message should be added in the ListView code path (e.g., in get_queryset(), get_context_data(), or get_template_names within list.py). Therefore, django/views/generic/list.py is the most appropriate file.",
  "func_think": "Step 1: From the bug report, the failure occurs in ListView when get_queryset() returns None, leading to TemplateDoesNotExist with an unhelpful message due to no template names being found.\nStep 2: In Django’s generic views architecture, ListView relies on MultipleObjectMixin for queryset handling and MultipleObjectTemplateResponseMixin for template name resolution; both are located in django/views/generic/list.py.\nStep 3: When get_queryset() returns None, ListView.get() sets self.object_list = None. The template resolution logic then attempts to infer template names from the model or object_list.\nStep 4: The method responsible for computing template names in list-based views is get_template_names in MultipleObjectTemplateResponseMixin (in django/views/generic/list.py). If object_list is None and no template_name is defined, it returns an empty list, which later triggers TemplateDoesNotExist without helpful context.\nStep 5: Therefore, the most direct and appropriate place to add a helpful error when object_list is None and no template names are determined is get_template_names. This is where we can detect the None queryset and raise a clearer ImproperlyConfigured message guiding developers to ensure get_queryset() returns a queryset.\nStep 6: Other candidates like ListView.get() or MultipleObjectMixin.get_queryset could be inspected, but the specific user-facing error about missing templates is constructed during template name resolution, making get_template_names the correct single modification point.\nStep 7: Thus, the key location to fix the issue matches the ground truth: django/views/generic/list.py, function get_template_names.",
  "edit_think": "Step 1: Identify the root cause: when ListView.get_queryset() returns None, get_template_names() can’t infer any template names, returning an empty list which triggers a cryptic TemplateDoesNotExist error.\nStep 2: Locate the logic that builds template name candidates for list views: MultipleObjectTemplateResponseMixin.get_template_names() in django/views/generic/list.py.\nStep 3: Determine the fix point: after attempting to infer template names from model/object_list, detect the case where object_list is None and no names were found.\nStep 4: Add an explicit, helpful ImproperlyConfigured error message indicating that get_queryset() returned None and must return a QuerySet.\nStep 5: Confirm that ImproperlyConfigured is already available in this file (it is typically used/handled here), so no additional imports should be necessary.",
  "patch_think": "Step 1:\nUnderstand the issue: When a ListView's get_queryset() returns None, Django proceeds and later fails to find any template names, resulting in a cryptic TemplateDoesNotExist error without useful context.\n\nStep 2:\nDesired behavior: Fail fast with a clear ImproperlyConfigured error message when get_queryset() returns None, pointing the developer to ensure get_queryset() returns a QuerySet.\n\nStep 3:\nLocate where to add the check: In BaseListView.get(), immediately after calling self.get_queryset(), before any further logic depends on object_list.\n\nStep 4:\nImplement a None-check for self.object_list and raise ImproperlyConfigured with a helpful message, including the class name.\n\nStep 5:\nVerify imports: ImproperlyConfigured is already used within this module (MultipleObjectMixin.get_queryset), so the import should already exist. No additional import changes needed.\n\nStep 6:\nPrepare the SEARCH/REPLACE patch to insert the check between retrieving object_list and evaluating allow_empty.\n\n```python\n### django/views/generic/list.py\n<<<<<<< SEARCH\n        self.object_list = self.get_queryset()\n        allow_empty = self.get_allow_empty()\n=======\n        self.object_list = self.get_queryset()\n        if self.object_list is None:\n            raise ImproperlyConfigured(\n                \"Expected a QuerySet, but None returned from %s.get_queryset(). \"\n                \"Ensure that %s.get_queryset() returns a QuerySet.\"\n                % (self.__class__.__name__, self.__class__.__name__)\n            )\n        allow_empty = self.get_allow_empty()\n>>>>>>> REPLACE\n```"
}