{
  "instance_id": "django__django-13757",
  "repo": "django/django",
  "created_at": "2020-12-09T14:48:53Z",
  "problem_statement": "Using __isnull=True on a KeyTransform should not match JSON null on SQLite and Oracle\nDescription\n\t\nThe KeyTransformIsNull lookup borrows the logic from HasKey for isnull=False, which is correct. If isnull=True, the query should only match objects that do not have the key. The query is correct for MariaDB, MySQL, and PostgreSQL. However, on SQLite and Oracle, the query also matches objects that have the key with the value null, which is incorrect.\nTo confirm, edit tests.model_fields.test_jsonfield.TestQuerying.test_isnull_key. For the first assertion, change\n\t\tself.assertSequenceEqual(\n\t\t\tNullableJSONModel.objects.filter(value__a__isnull=True),\n\t\t\tself.objs[:3] + self.objs[5:],\n\t\t)\nto\n\t\tself.assertSequenceEqual(\n\t\t\tNullableJSONModel.objects.filter(value__j__isnull=True),\n\t\t\tself.objs[:4] + self.objs[5:],\n\t\t)\nThe test previously only checks with value__a which could not catch this behavior because the value is not JSON null.\n",
  "patch": "diff --git a/django/db/models/fields/json.py b/django/db/models/fields/json.py\n--- a/django/db/models/fields/json.py\n+++ b/django/db/models/fields/json.py\n@@ -366,14 +366,25 @@ def process_rhs(self, compiler, connection):\n class KeyTransformIsNull(lookups.IsNull):\n     # key__isnull=False is the same as has_key='key'\n     def as_oracle(self, compiler, connection):\n+        sql, params = HasKey(\n+            self.lhs.lhs,\n+            self.lhs.key_name,\n+        ).as_oracle(compiler, connection)\n         if not self.rhs:\n-            return HasKey(self.lhs.lhs, self.lhs.key_name).as_oracle(compiler, connection)\n-        return super().as_sql(compiler, connection)\n+            return sql, params\n+        # Column doesn't have a key or IS NULL.\n+        lhs, lhs_params, _ = self.lhs.preprocess_lhs(compiler, connection)\n+        return '(NOT %s OR %s IS NULL)' % (sql, lhs), tuple(params) + tuple(lhs_params)\n \n     def as_sqlite(self, compiler, connection):\n+        template = 'JSON_TYPE(%s, %%s) IS NULL'\n         if not self.rhs:\n-            return HasKey(self.lhs.lhs, self.lhs.key_name).as_sqlite(compiler, connection)\n-        return super().as_sql(compiler, connection)\n+            template = 'JSON_TYPE(%s, %%s) IS NOT NULL'\n+        return HasKey(self.lhs.lhs, self.lhs.key_name).as_sql(\n+            compiler,\n+            connection,\n+            template=template,\n+        )\n \n \n class KeyTransformIn(lookups.In):\n",
  "similar_bug_items": [
    {
      "pr_number": 12472,
      "pr_title": "Fixed #31286 -- Made database specific fields checks databases aware.",
      "pr_body": "Further tests required(or not?).\r\nCurrent test cases don't cover the case when the user provides multiple databases, e.g.:\r\n```python manage.py check --database dba --database dbb```\r\n",
      "issue_id": 31286,
      "issue_title": "Database specific fields checks should be databases aware.",
      "issue_body": "In an attempt to trigger the check Error mentioned in Tickets:\n#31144\n, I found that the check for database backend doesn't check against the backend specified, e.g. --database mysql, rather, it always checks against the 'default' backend.\nAfter diving into the source code, I found the following method defined in django/db/models/fields/\ninit\n.py\ndef _check_backend_specific_checks(self, **kwargs):\n        app_label = self.model._meta.app_label\n        for db in connections:\n            if router.allow_migrate(db, app_label, model_name=self.model._meta.model_name):\n                return connections[db].validation.check_field(self, **kwargs)\n        return []\nIt checks the first db defined in connections rather those provided by users.\nA proposed change would be:\ndef _check_backend_specific_checks(self, **kwargs):\n        app_label = self.model._meta.app_label\n        errors = []\n        for db in kwargs['databases'] or ['default']:\n            if router.allow_migrate(db, app_label, model_name=self.model._meta.model_name):\n                errors.extend(connections[db].validation.check_field(self, **kwargs))\n        return errors\nIt worked as intended on my local machine.\nI would happily provide a patch for this one.",
      "issue_closed_at": "2020-02-24T10:38:01",
      "base_commit": "94d4bd3a091dd9ac265e14619576b1ee568653b1",
      "changes": [
        {
          "file": "django/db/models/fields/__init__.py",
          "type": "function",
          "name": "_check_null_allowed_for_primary_keys",
          "class_name": "Field",
          "code": "def _check_null_allowed_for_primary_keys(self):\n        if (self.primary_key and self.null and\n                not connection.features.interprets_empty_strings_as_nulls):\n            # We cannot reliably check this for backends like Oracle which\n            # consider NULL and '' to be equal (and thus set up\n            # character-based fields a little differently).\n            return [\n                checks.Error(\n                    'Primary keys must not have null=True.',\n                    hint=('Set null=False on the field, or '\n                          'remove primary_key=True argument.'),\n                    obj=self,\n                    id='fields.E007',\n                )\n            ]\n        else:\n            return []"
        }
      ]
    },
    {
      "pr_number": 8754,
      "pr_title": "Fixed #28391 -- Fixed Cast() with CharField and max_length on MySQL.",
      "pr_body": "https://code.djangoproject.com/ticket/28391",
      "issue_id": 28391,
      "issue_title": "Cast doesn't take into account CharField's max_length on MySQL.",
      "issue_body": "Correct behavior, e.g. (based on\ndb_functions/test_cast.py\n):\n>>> Author.objects.create(name='Bob', age=1111)\n>>> numbers = Author.objects.annotate(cast_string=Cast('age', models.CharField(max_length=3)))\n>>> numbers.get().cast_string\n111\non MySQL it returns\n1111\n(related with\n#28371\n).",
      "issue_closed_at": "2017-07-17T14:12:39",
      "base_commit": "feeafdad02e2874e2e2f879a825d3527f6b193ad",
      "changes": [
        {
          "file": "django/db/backends/sqlite3/features.py",
          "type": "class",
          "name": "DatabaseFeatures",
          "code": "class DatabaseFeatures(BaseDatabaseFeatures):\n    # SQLite cannot handle us only partially reading from a cursor's result set\n    # and then writing the same rows to the database in another cursor. This\n    # setting ensures we always read result sets fully into memory all in one\n    # go.\n    can_use_chunked_reads = False\n    test_db_allows_multiple_connections = False\n    supports_unspecified_pk = True\n    supports_timezones = False\n    max_query_params = 999\n    supports_mixed_date_datetime_comparisons = False\n    has_bulk_insert = True\n    supports_column_check_constraints = False\n    autocommits_when_autocommit_is_off = True\n    can_introspect_decimal_field = False\n    can_introspect_positive_integer_field = True\n    can_introspect_small_integer_field = True\n    supports_transactions = True\n    atomic_transactions = False\n    can_rollback_ddl = True\n    supports_paramstyle_pyformat = False\n    supports_sequence_reset = False\n    can_clone_databases = True\n    supports_temporal_subtraction = True\n    ignores_table_name_case = True\n\n    @cached_property\n    def uses_savepoints(self):\n        return Database.sqlite_version_info >= (3, 6, 8)\n\n    @cached_property\n    def supports_index_column_ordering(self):\n        return Database.sqlite_version_info >= (3, 3, 0)\n\n    @cached_property\n    def can_release_savepoints(self):\n        return self.uses_savepoints\n\n    @cached_property\n    def can_share_in_memory_db(self):\n        return (\n            Database.__name__ == 'sqlite3.dbapi2' and\n            Database.sqlite_version_info >= (3, 7, 13)\n        )\n\n    @cached_property\n    def supports_stddev(self):\n        \"\"\"\n        Confirm support for STDDEV and related stats functions.\n\n        SQLite supports STDDEV as an extension package; so\n        connection.ops.check_expression_support() can't unilaterally\n        rule out support for STDDEV. Manually check whether the call works.\n        \"\"\"\n        with self.connection.cursor() as cursor:\n            cursor.execute('CREATE TABLE STDDEV_TEST (X INT)')\n            try:\n                cursor.execute('SELECT STDDEV(*) FROM STDDEV_TEST')\n                has_support = True\n            except utils.DatabaseError:\n                has_support = False\n            cursor.execute('DROP TABLE STDDEV_TEST')\n        return has_support"
        },
        {
          "file": "django/db/models/fields/__init__.py",
          "type": "function",
          "name": "clean",
          "class_name": "Field",
          "code": "def clean(self, value, model_instance):\n        \"\"\"\n        Convert the value's type and run validation. Validation errors\n        from to_python() and validate() are propagated. Return the correct\n        value if no error is raised.\n        \"\"\"\n        value = self.to_python(value)\n        self.validate(value, model_instance)\n        self.run_validators(value)\n        return value"
        },
        {
          "file": "django/db/models/fields/__init__.py",
          "type": "function",
          "name": "db_type",
          "class_name": "Field",
          "code": "def db_type(self, connection):\n        \"\"\"\n        Return the database column data type for this field, for the provided\n        connection.\n        \"\"\"\n        # The default implementation of this method looks at the\n        # backend-specific data_types dictionary, looking up the field by its\n        # \"internal type\".\n        #\n        # A Field class can implement the get_internal_type() method to specify\n        # which *preexisting* Django Field class it's most similar to -- i.e.,\n        # a custom field might be represented by a TEXT column type, which is\n        # the same as the TextField Django field type, which means the custom\n        # field's get_internal_type() returns 'TextField'.\n        #\n        # But the limitation of the get_internal_type() / data_types approach\n        # is that it cannot handle database column types that aren't already\n        # mapped to one of the built-in Django field types. In this case, you\n        # can implement db_type() instead of get_internal_type() to specify\n        # exactly which wacky database column type you want to use.\n        data = DictWrapper(self.__dict__, connection.ops.quote_name, \"qn_\")\n        try:\n            return connection.data_types[self.get_internal_type()] % data\n        except KeyError:\n            return None"
        },
        {
          "file": "django/db/models/functions/base.py",
          "type": "class",
          "name": "Cast",
          "code": "class Cast(Func):\n    \"\"\"Coerce an expression to a new field type.\"\"\"\n    function = 'CAST'\n    template = '%(function)s(%(expressions)s AS %(db_type)s)'\n\n    mysql_types = {\n        fields.CharField: 'char',\n        fields.IntegerField: 'signed integer',\n        fields.BigIntegerField: 'signed integer',\n        fields.SmallIntegerField: 'signed integer',\n        fields.FloatField: 'signed',\n        fields.PositiveIntegerField: 'unsigned integer',\n        fields.PositiveSmallIntegerField: 'unsigned integer',\n    }\n\n    def __init__(self, expression, output_field):\n        super().__init__(expression, output_field=output_field)\n\n    def as_sql(self, compiler, connection, **extra_context):\n        if 'db_type' not in extra_context:\n            extra_context['db_type'] = self.output_field.db_type(connection)\n        return super().as_sql(compiler, connection, **extra_context)\n\n    def as_mysql(self, compiler, connection):\n        extra_context = {}\n        output_field_class = type(self.output_field)\n        if output_field_class in self.mysql_types:\n            extra_context['db_type'] = self.mysql_types[output_field_class]\n        return self.as_sql(compiler, connection, **extra_context)\n\n    def as_postgresql(self, compiler, connection):\n        # CAST would be valid too, but the :: shortcut syntax is more readable.\n        return self.as_sql(compiler, connection, template='%(expressions)s::%(db_type)s')"
        },
        {
          "file": "django/db/models/functions/base.py",
          "type": "function",
          "name": "as_mysql",
          "class_name": "Length",
          "code": "def as_mysql(self, compiler, connection):\n        return super().as_sql(compiler, connection, function='CHAR_LENGTH')"
        }
      ]
    },
    {
      "pr_number": 8629,
      "pr_title": "Fixed #28293 -- Fixed union(), intersection(), and difference() when combining with an EmptyQuerySet.",
      "pr_body": "https://code.djangoproject.com/ticket/28293",
      "issue_id": 28293,
      "issue_title": "QuerySet.union(QuerySet.none()) results in an empty queryset, should be the original queryset",
      "issue_body": "Tested on Django 1.11.2.\nAs\nQuerySet.union()\nuses the SQL\nUNION\noperator, I would expect \"SET1 UNION <EMPTY SET>\" to result in SET1. Currently it results in the empty set.\n>>> from django.contrib.auth.models import User\n>>> User.objects.count()\n100\n>>> list(User.objects.all().union(User.objects.none()))\n[]\nFrom\n​\nhttps://www.postgresql.org/docs/current/static/queries-union.html\nUNION effectively appends the result of query2 to the result of query1 ...\nQuerySet.difference()\nlooks to suffer from the same issue.",
      "issue_closed_at": "2017-06-13T01:16:29",
      "base_commit": "9dc83c356d363c090f3351c908cad6f823aeb7bf",
      "changes": [
        {
          "file": "django/db/models/query.py",
          "type": "function",
          "name": "_combinator_query",
          "class_name": "QuerySet",
          "code": "def _combinator_query(self, combinator, *other_qs, all=False):\n        # Clone the query to inherit the select list and everything\n        clone = self._clone()\n        # Clear limits and ordering so they can be reapplied\n        clone.query.clear_ordering(True)\n        clone.query.clear_limits()\n        clone.query.combined_queries = (self.query,) + tuple(qs.query for qs in other_qs)\n        clone.query.combinator = combinator\n        clone.query.combinator_all = all\n        return clone"
        },
        {
          "file": "django/db/models/sql/compiler.py",
          "type": "function",
          "name": "get_combinator_sql",
          "class_name": "SQLCompiler",
          "code": "def get_combinator_sql(self, combinator, all):\n        features = self.connection.features\n        compilers = [\n            query.get_compiler(self.using, self.connection)\n            for query in self.query.combined_queries\n        ]\n        if not features.supports_slicing_ordering_in_compound:\n            for query, compiler in zip(self.query.combined_queries, compilers):\n                if query.low_mark or query.high_mark:\n                    raise DatabaseError('LIMIT/OFFSET not allowed in subqueries of compound statements.')\n                if compiler.get_order_by():\n                    raise DatabaseError('ORDER BY not allowed in subqueries of compound statements.')\n        parts = (compiler.as_sql() for compiler in compilers)\n        combinator_sql = self.connection.ops.set_operators[combinator]\n        if all and combinator == 'union':\n            combinator_sql += ' ALL'\n        braces = '({})' if features.supports_slicing_ordering_in_compound else '{}'\n        sql_parts, args_parts = zip(*((braces.format(sql), args) for sql, args in parts))\n        result = [' {} '.format(combinator_sql).join(sql_parts)]\n        params = []\n        for part in args_parts:\n            params.extend(part)\n        return result, params"
        }
      ]
    },
    {
      "pr_number": 8116,
      "pr_title": "Fixed #25406 -- Removed exception hiding in PostgreSQL test database …",
      "pr_body": "…creation / cloning during `--keepdb`. Ticket [25406](https://code.djangoproject.com/ticket/25406).\r\n\r\nI'm going to prepare separate PR with similar modification in the MySQL backend.",
      "issue_id": 25406,
      "issue_title": "_create_test_db hides errors like 'source database \"template1\" is being accessed by other users' with --keepdb",
      "issue_body": "The _create_test_db method will hide errors like 'source database \"template1\" is being accessed by other users', and will assume that the test database exists already.\n> …/pyenv/project/lib/python3.4/site-packages/psycopg2/__init__.py(165)connect()\n    164     import ipdb; ipdb.set_trace()\n--> 165     conn = _connect(dsn, connection_factory=connection_factory, async=async)\n    166     if cursor_factory is not None:\n\nipdb> c\nsource database \"template1\" is being accessed by other users\nDETAIL:  There are 3 other sessions using the database.\n\n> …/pyenv/project/lib/python3.4/site-packages/django/db/backends/base/creation.py(458)_create_test_db()\n    457                 # just return and skip it all.\n--> 458                 if keepdb:\n    459                     return test_database_name\nSource reference:\n​\nhttps://github.com/blueyed/django/blob/9e530b08d5858d7063d081b60ec86d24173e4df5/django/db/backends/base/creation.py#L146-L165\nThis will then result in an error when trying to connect to it, because it has not been created:\npsycopg2.OperationalError: FATAL:  database \"test_project\" does not exist\nBut instead the initial error should be displayed:\nsource database \"template1\" is being accessed by other users\nDETAIL:  There are 3 other sessions using the database.\nTo reproduce this:\nconnect to the \"template1\" database\nrun Django tests\nI think the SQL could use\nCREATE DATABASE IF NOT EXISTS\n(in case\nIF NOT EXISTS\n) is supported by all backends (maybe that needs to be subclassed then), and then would not assume that an Exception can be ignored with\nkeepdb\n.\nAn even better way would be to check if it exists, instead of trying to create it.\nWith pytest-django we're using the following code:\ndef test_database_exists_from_previous_run(connection):\n    # Try to open a cursor to the test database\n    test_db_name = connection.creation._get_test_db_name()\n\n    # When using a real SQLite backend (via TEST_NAME), check if the file\n    # exists, because it gets created automatically.\n    if connection.settings_dict['ENGINE'] == 'django.db.backends.sqlite3':\n        if not os.path.exists(test_db_name):\n            return False\n\n    orig_db_name = connection.settings_dict['NAME']\n    connection.settings_dict['NAME'] = test_db_name\n\n    # With SQLite memory databases the db never exists.\n    if connection.settings_dict['NAME'] == ':memory:':\n        return False\n\n    try:\n        connection.cursor()\n        return True\n    except Exception:  # TODO: Be more discerning but still DB agnostic.\n        return False\n    finally:\n        connection.close()\n        connection.settings_dict['NAME'] = orig_db_name\n(Source:\n​\nhttps://github.com/blueyed/pytest_django/blob/93fca47feea39016dd93e657a9328450e9b6e891/pytest_django/db_reuse.py#L11-L35\n)",
      "issue_closed_at": "2017-04-10T12:04:07",
      "base_commit": "5d3b322dce452dd75e8602ced9f0d02f9d6a5837",
      "changes": [
        {
          "file": "django/db/backends/base/creation.py",
          "type": "function",
          "name": "_get_test_db_name",
          "class_name": "BaseDatabaseCreation",
          "code": "def _get_test_db_name(self):\n        \"\"\"\n        Internal implementation - return the name of the test DB that will be\n        created. Only useful when called from create_test_db() and\n        _create_test_db() and when no external munging is done with the 'NAME'\n        settings.\n        \"\"\"\n        if self.connection.settings_dict['TEST']['NAME']:\n            return self.connection.settings_dict['TEST']['NAME']\n        return TEST_DATABASE_PREFIX + self.connection.settings_dict['NAME']"
        },
        {
          "file": "django/db/backends/base/creation.py",
          "type": "function",
          "name": "_create_test_db",
          "class_name": "BaseDatabaseCreation",
          "code": "def _create_test_db(self, verbosity, autoclobber, keepdb=False):\n        \"\"\"\n        Internal implementation - create the test db tables.\n        \"\"\"\n        suffix = self.sql_table_creation_suffix()\n\n        test_database_name = self._get_test_db_name()\n\n        qn = self.connection.ops.quote_name\n\n        # Create the test database and connect to it.\n        with self._nodb_connection.cursor() as cursor:\n            try:\n                cursor.execute(\n                    \"CREATE DATABASE %s %s\" % (qn(test_database_name), suffix))\n            except Exception as e:\n                # if we want to keep the db, then no need to do any of the below,\n                # just return and skip it all.\n                if keepdb:\n                    return test_database_name\n\n                sys.stderr.write(\n                    \"Got an error creating the test database: %s\\n\" % e)\n                if not autoclobber:\n                    confirm = input(\n                        \"Type 'yes' if you would like to try deleting the test \"\n                        \"database '%s', or 'no' to cancel: \" % test_database_name)\n                if autoclobber or confirm == 'yes':\n                    try:\n                        if verbosity >= 1:\n                            print(\"Destroying old test database for alias %s...\" % (\n                                self._get_database_display_str(verbosity, test_database_name),\n                            ))\n                        cursor.execute(\n                            \"DROP DATABASE %s\" % qn(test_database_name))\n                        cursor.execute(\n                            \"CREATE DATABASE %s %s\" % (qn(test_database_name),\n                                                       suffix))\n                    except Exception as e:\n                        sys.stderr.write(\n                            \"Got an error recreating the test database: %s\\n\" % e)\n                        sys.exit(2)\n                else:\n                    print(\"Tests cancelled.\")\n                    sys.exit(1)\n\n        return test_database_name"
        },
        {
          "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        )"
        },
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "function",
          "name": "_clone_test_db",
          "class_name": "DatabaseCreation",
          "code": "def _clone_test_db(self, number, verbosity, keepdb=False):\n        # CREATE DATABASE ... WITH TEMPLATE ... requires closing connections\n        # to the template database.\n        self.connection.close()\n\n        source_database_name = self.connection.settings_dict['NAME']\n        target_database_name = self.get_test_db_clone_settings(number)['NAME']\n        suffix = self._get_database_create_suffix(template=source_database_name)\n        creation_sql = \"CREATE DATABASE {} {}\".format(self._quote_name(target_database_name), suffix)\n\n        with self._nodb_connection.cursor() as cursor:\n            try:\n                cursor.execute(creation_sql)\n            except Exception:\n                if keepdb:\n                    return\n                try:\n                    if verbosity >= 1:\n                        print(\"Destroying old test database for alias %s...\" % (\n                            self._get_database_display_str(verbosity, target_database_name),\n                        ))\n                    cursor.execute(\"DROP DATABASE %s\" % self._quote_name(target_database_name))\n                    cursor.execute(creation_sql)\n                except Exception as e:\n                    sys.stderr.write(\"Got an error cloning the test database: %s\\n\" % e)\n                    sys.exit(2)"
        }
      ]
    },
    {
      "pr_number": 5139,
      "pr_title": "Fixed #19263 -- Fixed crash when filtering using __in and an empty QuerySet.",
      "pr_body": "https://code.djangoproject.com/ticket/19263\n",
      "issue_id": 19263,
      "issue_title": "Filtering __in a sliced queryset with a 0 limit raises an error",
      "issue_body": "I've noticed that after upgrading to Django 1.4,\n__in\nqueries really don't like empty sets. Simple queries still work, like\nUser.objects.filter(groups__in=[])\n, but most failures I've seen are with Paginators. I think this is the minimum set to cause a DatabaseError, create any app, add a models.py with:\nfrom\ndjango.db\nimport\nmodels\nclass\nAuthor\n(\nmodels\n.\nModel\n):\npass\nclass\nBook\n(\nmodels\n.\nModel\n):\nauthor\n=\nmodels\n.\nForeignKey\n(\nAuthor\n)\ndef\ncrash\n():\nfrom\ndjango.core.paginator\nimport\nPaginator\npages\n=\nPaginator\n(\nAuthor\n.\nobjects\n.\nall\n(),\n25\n)\npage\n=\npages\n.\npage\n(\n1\n)\nbooks\n=\nBook\n.\nobjects\n.\nfilter\n(\nauthor__in\n=\npage\n.\nobject_list\n)\nprint\nbooks\ncalling crash() will cause this stack trace:\nC:\\Workspace\\someproject\\src\\someproject\\test.py in <module>()\n      6\n      7 books = Book.objects.filter(author__in=page.object_list)\n----> 8 print books\n      9\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\query.pyc in __repr__(self)\n     70\n     71     def __repr__(self):\n---> 72         data = list(self[:REPR_OUTPUT_SIZE + 1])\n     73         if len(data) > REPR_OUTPUT_SIZE:\n     74             data[-1] = \"...(remaining elements truncated)...\"\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\query.pyc in __len__(self)\n     85                 self._result_cache = list(self.iterator())\n     86         elif self._iter:\n---> 87             self._result_cache.extend(self._iter)\n     88         if self._prefetch_related_lookups and not self._prefetch_done:\n     89             self._prefetch_related_objects()\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\query.pyc in iterator(self)\n    289             klass_info = get_klass_info(model, max_depth=max_depth,\n    290                                         requested=requested, only_load=only_load)\n--> 291         for row in compiler.results_iter():\n    292             if fill_cache:\n    293                 obj, _ = get_cached_row(row, index_start, db, klass_info,\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\sql\\compiler.pyc in results_iter(self)\n    761         if self.query.select_for_update and transaction.is_managed(self.using):\n    762             transaction.set_dirty(self.using)\n--> 763         for rows in self.execute_sql(MULTI):\n    764             for row in rows:\n    765                 if resolve_columns:\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\models\\sql\\compiler.pyc in execute_sql(self, result_type)\n    816\n    817         cursor = self.connection.cursor()\n--> 818         cursor.execute(sql, params)\n    819\n    820         if not result_type:\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\backends\\util.pyc in execute(self, sql, params)\n     38         start = time()\n     39         try:\n---> 40             return self.cursor.execute(sql, params)\n     41         finally:\n     42             stop = time()\n\nC:\\Dev\\Python27\\lib\\site-packages\\django\\db\\backends\\postgresql_psycopg2\\base.pyc in execute(self, query, args)\n     50     def execute(self, query, args=None):\n     51         try:\n---> 52             return self.cursor.execute(query, args)\n     53         except Database.IntegrityError, e:\n     54             raise utils.IntegrityError, utils.IntegrityError(*tuple(e)), sys.exc_info()[2]\n\nDatabaseError: syntax error at or near \")\"\nLINE 1: ...ugtest_book\" WHERE \"bugtest_book\".\"author_id\" IN () LIMIT 21\nThe SQL statement created is:\nSELECT \"bugtest_book\".\"id\", \"bugtest_book\".\"author_id\" FROM \"bugtest_book\" WHERE \"bugtest_book\".\"author_id\" IN () LIMIT 21",
      "issue_closed_at": "2015-09-04T07:00:51",
      "base_commit": "7c0850028f25eebaa9b521b5d02afac084ff2c6f",
      "changes": [
        {
          "file": "django/db/models/sql/compiler.py",
          "type": "function",
          "name": "as_nested_sql",
          "class_name": "SQLCompiler",
          "code": "def as_nested_sql(self):\n        \"\"\"\n        Perform the same functionality as the as_sql() method, returning an\n        SQL string and parameters. However, the alias prefixes are bumped\n        beforehand (in a copy -- the current query isn't changed), and any\n        ordering is removed if the query is unsliced.\n\n        Used when nesting this query inside another.\n        \"\"\"\n        obj = self.query.clone()\n        if obj.low_mark == 0 and obj.high_mark is None and not self.query.distinct_fields:\n            # If there is no slicing in use, then we can safely drop all ordering\n            obj.clear_ordering(True)\n        return obj.get_compiler(connection=self.connection).as_sql(subquery=True)"
        }
      ]
    }
  ]
}