{
  "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": "The popular Web Faction hosting service uses a shared database server. Users can create databases using the web UI or XML RPC calls, but not using the SQL CREATE syntax.\nRunning tests throws a ProgrammingError, with the message 'permission denied to create database', even if the test database has been previously created manually.\nThe error code for this error is '42501', which appears to correspond to errorcodes.INSUFFICIENT_PRIVILEGE.\ndjango/db/backends/postgresql/creation.py only handles the error errorcodes.DUPLICATE_DATABASE in _execute_create_test_db(), line 35. Because the error code does not match the program exits with a log message. But it would be fine to proceed with the error code '42501' also, making use of the --keepdb mechanism.\nThis appears to be a regression, as I did not experience this issue either using postgresql_psycopg2 driver or using Django 1.11",
      "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": "The documentation at\n​\nhttps://docs.djangoproject.com/en/dev/ref/class-based-views/mixins-single-object/#django.views.generic.detail.SingleObjectMixin\nsays that 'object' will be in the context, as well as potentially context_object_name if given. However, actually only context_object_name is set in the context by this mixin; only if object is passed to the mixin (which BaseDetailView does) will it be included.\nThe MultipleObjectMixin does include 'object_list' in its get_context_data (though assumes it must be passed in rather than on the object, I'll file that as another ticket), so I assume this is an issue with SingleObjectMixin, and that it should set 'object' on context whether it is passed in as a kwarg or not.\nI have a patch on a branch of my github fork:\n​\nhttps://github.com/dracos/django/compare/singleobjectfixin",
      "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": "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"
        }
      ]
    },
    {
      "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": "When using a namespaced\n_meta.db_table\n(e.g. Oracle's\n'schema\".\"table'\n) it's possible that\n_create_index_name\nreturns an index name truncating the namespace resulting in an invalid identifier or one that isn't namespaced anymore and thus created in the user's namespace.\nFor example, given the following model:\nclass\nFoo\n(\nmodels\n.\nModel\n):\nfield\n=\nmodels\n.\nIntegerField\n(\nindex\n=\nTrue\n)\nclass\nMeta\n:\ndb_table\n=\n'long_name\".\"table_name'\nThe resulting index name will be\n'long_name\"_field_d21c9e0a'\nwhich is invalid SQL even when quoted to\n'\"long_name\"_field_d21c9e0a\"'\n.\nMarking as a release blocker because this is a regression which I believe was introduced by\n#27458\nand wasn't addressed by\n#27843\n. I confirm that this uses to work on Django 1.10 but was broken on 1.11 as I stumbled upon the issue when upgrading a Django 1.8 LTS codebase to 1.11 LTS on the 1.10 -> 1.11 step.\nThe tests added by\n#27458\njust happened to work because the index name truncation cut the string the in a way that both double quotes are stripped. It should be possible to tweak the table names to trigger the errors but I felt like directly testing\n_create_index_name\nwas more appropriate.",
      "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": "My app has a Group model with an m2m field in it which used to look like:\nthings = models.ManyToManyField(Thing, blank=True, null=True)\nDjango 1.8a1 on my test environment warned me that the\nblank\nand\nnull\narguments were unnecessary, so I removed them:\nthings = models.ManyToManyField(Thing)\nThen I ran\nmanage.py makemigrations\nand applied the resulting migration with\nmanage.py migrate\n. To my horror, the data in the relevant \"linking\" table was all deleted.\nThe section from the migration file looks like this:\noperations = [\n        migrations.RemoveField(\n            model_name='group',\n            name='things',\n        ),\n        migrations.AddField(\n            model_name='group',\n            name='things',\n            field=models.ManyToManyField(to='sa.Thing'),\n        ),\n    ]\nThis doesn't look like the right behaviour to me, and cost me a lot of time recovering data from backup. Have I done something wrong (other than trusting the migration file without inspecting it) or is this a genuine bug?",
      "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)"
        }
      ]
    }
  ]
}