{
  "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",
  "similar_bug_items": [
    {
      "pr_number": 10260,
      "pr_title": "Fixed #29613 -- Fixed --keepdb on PostgreSQL if the database exists and the user can't create databases.",
      "pr_body": "Ticket [29613](https://code.djangoproject.com/ticket/29613).",
      "issue_id": 29613,
      "issue_title": "Allow --keepdb to work on PostgreSQL if the database exists but the user can't create databases",
      "issue_body": "",
      "issue_closed_at": "2018-08-03T03:32:30",
      "base_commit": "d8e2be459f97f1773c7edf7d37de180139146176",
      "changes": [
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "line",
          "name": "line 3",
          "code": "from psycopg2 import errorcodes\n\nfrom django.db.backends.base.creation import BaseDatabaseCreation\n\n\nclass DatabaseCreation(BaseDatabaseCreation):"
        },
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "function",
          "name": "sql_table_creation_suffix",
          "class_name": "DatabaseCreation",
          "code": "def sql_table_creation_suffix(self):\n        test_settings = self.connection.settings_dict['TEST']\n        assert test_settings['COLLATION'] is None, (\n            \"PostgreSQL does not support collation setting at database creation time.\"\n        )\n        return self._get_database_create_suffix(\n            encoding=test_settings['CHARSET'],\n            template=test_settings.get('TEMPLATE'),\n        )"
        }
      ]
    },
    {
      "pr_number": 1116,
      "pr_title": "Ticket 20234 20236",
      "pr_body": "Resolved: https://code.djangoproject.com/ticket/20234 and https://code.djangoproject.com/ticket/20236\n",
      "issue_id": 20234,
      "issue_title": "SingleObjectMixin does not add 'object' key to context",
      "issue_body": "",
      "issue_closed_at": "2013-05-19T03:32:50",
      "base_commit": "0038296135502331c302935106d7aa568f715200",
      "changes": [
        {
          "file": "django/views/generic/detail.py",
          "type": "function",
          "name": "get_context_data",
          "class_name": "SingleObjectMixin",
          "code": "def get_context_data(self, **kwargs):\n        \"\"\"\n        Insert the single object into the context dict.\n        \"\"\"\n        context = {}\n        context_object_name = self.get_context_object_name(self.object)\n        if context_object_name:\n            context[context_object_name] = self.object\n        context.update(kwargs)\n        return super(SingleObjectMixin, self).get_context_data(**context)"
        },
        {
          "file": "django/views/generic/detail.py",
          "type": "function",
          "name": "get_template_names",
          "class_name": "SingleObjectTemplateResponseMixin",
          "code": "def get_template_names(self):\n        \"\"\"\n        Return a list of template names to be used for the request. May not be\n        called if render_to_response is overridden. Returns the following list:\n\n        * the value of ``template_name`` on the view (if provided)\n        * the contents of the ``template_name_field`` field on the\n          object instance that the view is operating upon (if available)\n        * ``<app_label>/<object_name><template_name_suffix>.html``        \n        \"\"\"\n        try:\n            names = super(SingleObjectTemplateResponseMixin, self).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 self.template_name_field is set, grab the value of the field\n        # of that name from the object; this is the most specific template\n        # name, if given.\n        if self.object and self.template_name_field:\n            name = getattr(self.object, self.template_name_field, None)\n            if name:\n                names.insert(0, name)\n\n        # The least-specific option is the default <app>/<model>_detail.html;\n        # only use this if the object in question is a model.\n        if isinstance(self.object, models.Model):\n            names.append(\"%s/%s%s.html\" % (\n                self.object._meta.app_label,\n                self.object._meta.model_name,\n                self.template_name_suffix\n            ))\n        elif hasattr(self, 'model') and self.model is not None and issubclass(self.model, models.Model):\n            names.append(\"%s/%s%s.html\" % (\n                self.model._meta.app_label,\n                self.model._meta.model_name,\n                self.template_name_suffix\n            ))\n        return names"
        },
        {
          "file": "django/views/generic/edit.py",
          "type": "function",
          "name": "form_valid",
          "class_name": "ModelFormMixin",
          "code": "def form_valid(self, form):\n        \"\"\"\n        If the form is valid, save the associated model.\n        \"\"\"\n        self.object = form.save()\n        return super(ModelFormMixin, self).form_valid(form)"
        }
      ]
    },
    {
      "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": "",
      "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"
        }
      ]
    },
    {
      "pr_number": 9345,
      "pr_title": "Fixed #28792 -- Fixed index name truncation of namespaced tables.",
      "pr_body": "Refs #27458, #27843.",
      "issue_id": 28792,
      "issue_title": "Index names can be incorrectly truncated when using a namespaced table name",
      "issue_body": "",
      "issue_closed_at": "2017-11-14T20:53:46",
      "base_commit": "532a4f22ad94db320cb0fd66f4c7ee57d17ac65a",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "line",
          "name": "line 5",
          "code": "from django.db.backends.ddl_references import (\n    Columns, ForeignKeyName, IndexName, Statement, Table,\n)\nfrom django.db.backends.utils import strip_quotes\nfrom django.db.models import Index\nfrom django.db.transaction import TransactionManagementError, atomic\nfrom django.utils import timezone"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_create_index_name",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _create_index_name(self, table_name, column_names, suffix=\"\"):\n        \"\"\"\n        Generate a unique name for an index/unique constraint.\n\n        The name is divided into 3 parts: the table name, the column names,\n        and a unique digest and suffix.\n        \"\"\"\n        table_name = strip_quotes(table_name)\n        hash_data = [table_name] + list(column_names)\n        hash_suffix_part = '%s%s' % (self._digest(*hash_data), suffix)\n        max_length = self.connection.ops.max_name_length() or 200\n        # If everything fits into max_length, use that name.\n        index_name = '%s_%s_%s' % (table_name, '_'.join(column_names), hash_suffix_part)\n        if len(index_name) <= max_length:\n            return index_name\n        # Shorten a long suffix.\n        if len(hash_suffix_part) > max_length / 3:\n            hash_suffix_part = hash_suffix_part[:max_length // 3]\n        other_length = (max_length - len(hash_suffix_part)) // 2 - 1\n        index_name = '%s_%s_%s' % (\n            table_name[:other_length],\n            '_'.join(column_names)[:other_length],\n            hash_suffix_part,\n        )\n        # Prepend D if needed to prevent the name from starting with an\n        # underscore or a number (not permitted on Oracle).\n        if index_name[0] == \"_\" or index_name[0].isdigit():\n            index_name = \"D%s\" % index_name[:-1]\n        return index_name"
        },
        {
          "file": "django/db/backends/utils.py",
          "type": "line",
          "name": "line 3",
          "code": "import functools\nimport hashlib\nimport logging\nimport re\nfrom time import time\n\nfrom django.conf import settings"
        },
        {
          "file": "django/db/backends/utils.py",
          "type": "function",
          "name": "rev_typecast_decimal",
          "class_name": null,
          "code": "def rev_typecast_decimal(d):\n    if d is None:\n        return None\n    return str(d)"
        }
      ]
    },
    {
      "pr_number": 4233,
      "pr_title": "Fixed #24435 -- Prevented m2m field removal and addition in migrations when changing blank",
      "pr_body": "https://code.djangoproject.com/ticket/24435\n",
      "issue_id": 24435,
      "issue_title": "Removing blank=True, null=True from ManyToMany field causes data deletion in migration",
      "issue_body": "",
      "issue_closed_at": "2015-03-04T08:41:51",
      "base_commit": "70123cf084e3af7dfc61bb7bd2090ff802c3cda4",
      "changes": [
        {
          "file": "django/db/migrations/autodetector.py",
          "type": "function",
          "name": "generate_altered_fields",
          "class_name": "MigrationAutodetector",
          "code": "def generate_altered_fields(self):\n        \"\"\"\n        Fields that have been altered.\n        \"\"\"\n        for app_label, model_name, field_name in sorted(self.old_field_keys.intersection(self.new_field_keys)):\n            # Did the field change?\n            old_model_name = self.renamed_models.get((app_label, model_name), model_name)\n            old_field_name = self.renamed_fields.get((app_label, model_name, field_name), field_name)\n            old_field = self.old_apps.get_model(app_label, old_model_name)._meta.get_field(old_field_name)\n            new_field = self.new_apps.get_model(app_label, model_name)._meta.get_field(field_name)\n            # Implement any model renames on relations; these are handled by RenameModel\n            # so we need to exclude them from the comparison\n            if hasattr(new_field, \"rel\") and getattr(new_field.rel, \"to\", None):\n                rename_key = (\n                    new_field.rel.to._meta.app_label,\n                    new_field.rel.to._meta.model_name,\n                )\n                if rename_key in self.renamed_models:\n                    new_field.rel.to = old_field.rel.to\n            old_field_dec = self.deep_deconstruct(old_field)\n            new_field_dec = self.deep_deconstruct(new_field)\n            if old_field_dec != new_field_dec:\n                if (not isinstance(old_field, models.ManyToManyField) and\n                        not isinstance(new_field, models.ManyToManyField)):\n                    preserve_default = True\n                    if (old_field.null and not new_field.null and not new_field.has_default() and\n                            not isinstance(new_field, models.ManyToManyField)):\n                        field = new_field.clone()\n                        new_default = self.questioner.ask_not_null_alteration(field_name, model_name)\n                        if new_default is not models.NOT_PROVIDED:\n                            field.default = new_default\n                            preserve_default = False\n                    else:\n                        field = new_field\n                    self.add_operation(\n                        app_label,\n                        operations.AlterField(\n                            model_name=model_name,\n                            name=field_name,\n                            field=field,\n                            preserve_default=preserve_default,\n                        )\n                    )\n                else:\n                    self._generate_removed_field(app_label, model_name, field_name)\n                    self._generate_added_field(app_label, model_name, field_name)"
        },
        {
          "file": "django/db/migrations/autodetector.py",
          "type": "function",
          "name": "generate_altered_fields",
          "class_name": "MigrationAutodetector",
          "code": "def generate_altered_fields(self):\n        \"\"\"\n        Fields that have been altered.\n        \"\"\"\n        for app_label, model_name, field_name in sorted(self.old_field_keys.intersection(self.new_field_keys)):\n            # Did the field change?\n            old_model_name = self.renamed_models.get((app_label, model_name), model_name)\n            old_field_name = self.renamed_fields.get((app_label, model_name, field_name), field_name)\n            old_field = self.old_apps.get_model(app_label, old_model_name)._meta.get_field(old_field_name)\n            new_field = self.new_apps.get_model(app_label, model_name)._meta.get_field(field_name)\n            # Implement any model renames on relations; these are handled by RenameModel\n            # so we need to exclude them from the comparison\n            if hasattr(new_field, \"rel\") and getattr(new_field.rel, \"to\", None):\n                rename_key = (\n                    new_field.rel.to._meta.app_label,\n                    new_field.rel.to._meta.model_name,\n                )\n                if rename_key in self.renamed_models:\n                    new_field.rel.to = old_field.rel.to\n            old_field_dec = self.deep_deconstruct(old_field)\n            new_field_dec = self.deep_deconstruct(new_field)\n            if old_field_dec != new_field_dec:\n                if (not isinstance(old_field, models.ManyToManyField) and\n                        not isinstance(new_field, models.ManyToManyField)):\n                    preserve_default = True\n                    if (old_field.null and not new_field.null and not new_field.has_default() and\n                            not isinstance(new_field, models.ManyToManyField)):\n                        field = new_field.clone()\n                        new_default = self.questioner.ask_not_null_alteration(field_name, model_name)\n                        if new_default is not models.NOT_PROVIDED:\n                            field.default = new_default\n                            preserve_default = False\n                    else:\n                        field = new_field\n                    self.add_operation(\n                        app_label,\n                        operations.AlterField(\n                            model_name=model_name,\n                            name=field_name,\n                            field=field,\n                            preserve_default=preserve_default,\n                        )\n                    )\n                else:\n                    self._generate_removed_field(app_label, model_name, field_name)\n                    self._generate_added_field(app_label, model_name, field_name)"
        }
      ]
    }
  ]
}