{
  "Selected_candidate": {
    "pr_number": 9255,
    "pr_title": "Fixed #28719 -- Added a helpful exception if MultipleObjectTemplateResponseMixin doesn't generate any template names.",
    "pr_body": "https://code.djangoproject.com/ticket/28719",
    "issue_id": 28719,
    "issue_title": "Add a helpful exception message when ListView.get_queryset() returns None",
    "issue_body": "One 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.",
    "issue_closed_at": "2017-11-07T18:04:43",
    "base_commit": "a2851f204c6431330042d0343ee99f33449f78e0",
    "changes": [
      {
        "file": "django/views/generic/list.py",
        "type": "function",
        "name": "get_template_names",
        "class_name": "MultipleObjectTemplateResponseMixin",
        "code": "def get_template_names(self):\n        \"\"\"\n        Return a list of template names to be used for the request. Must return\n        a list. May not be called if render_to_response is overridden.\n        \"\"\"\n        try:\n            names = super().get_template_names()\n        except ImproperlyConfigured:\n            # If template_name isn't specified, it's not a problem --\n            # we just start with an empty list.\n            names = []\n\n        # If the list is a queryset, we'll invent a template name based on the\n        # app and model name. This name gets put at the end of the template\n        # name list so that user-supplied names override the automatically-\n        # generated ones.\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        return names"
      }
    ]
  },
  "Justification": "Candidate C is the most helpful because it addresses issues related to the `get_queryset()` method potentially returning None during template rendering, similar to the behavior observed in the CURRENT bug report where the combination of `Q` and `Exists` raises a TypeError. The behavior of the Q-Exists combination suggests that the operations should handle edge cases correctly, akin to how a better exception handling was proposed in Candidate C. This shared emphasis on ensuring proper conditions before proceeding with operations makes Candidate C particularly relevant for debugging the CURRENT bug.",
  "instance_id": "django__django-14017",
  "repo": "django/django",
  "created_at": "2021-02-18T13:05:27Z",
  "problem_statement": "Q(...) & Exists(...) raises a TypeError\nDescription\n\t\nExists(...) & Q(...) works, but Q(...) & Exists(...) raise a TypeError\nHere's a minimal example:\nIn [3]: Exists(Product.objects.all()) & Q()\nOut[3]: <Q: (AND: <django.db.models.expressions.Exists object at 0x7fc18dd0ed90>, (AND: ))>\nIn [4]: Q() & Exists(Product.objects.all())\n---------------------------------------------------------------------------\nTypeError\t\t\t\t\t\t\t\t Traceback (most recent call last)\n<ipython-input-4-21d3dea0fcb9> in <module>\n----> 1 Q() & Exists(Product.objects.all())\n~/Code/venv/ecom/lib/python3.8/site-packages/django/db/models/query_utils.py in __and__(self, other)\n\t 90 \n\t 91\t def __and__(self, other):\n---> 92\t\t return self._combine(other, self.AND)\n\t 93 \n\t 94\t def __invert__(self):\n~/Code/venv/ecom/lib/python3.8/site-packages/django/db/models/query_utils.py in _combine(self, other, conn)\n\t 71\t def _combine(self, other, conn):\n\t 72\t\t if not isinstance(other, Q):\n---> 73\t\t\t raise TypeError(other)\n\t 74 \n\t 75\t\t # If the other Q() is empty, ignore it and just use `self`.\nTypeError: <django.db.models.expressions.Exists object at 0x7fc18dd21400>\nThe & (and |) operators should be commutative on Q-Exists pairs, but it's not\nI think there's a missing definition of __rand__ somewhere.\n",
  "patch": "diff --git a/django/db/models/query_utils.py b/django/db/models/query_utils.py\n--- a/django/db/models/query_utils.py\n+++ b/django/db/models/query_utils.py\n@@ -40,7 +40,7 @@ def __init__(self, *args, _connector=None, _negated=False, **kwargs):\n         super().__init__(children=[*args, *sorted(kwargs.items())], connector=_connector, negated=_negated)\n \n     def _combine(self, other, conn):\n-        if not isinstance(other, Q):\n+        if not(isinstance(other, Q) or getattr(other, 'conditional', False) is True):\n             raise TypeError(other)\n \n         # If the other Q() is empty, ignore it and just use `self`.\n"
}