{
  "instance_id": "django__django-16408",
  "repo": "django/django",
  "created_at": "2022-12-29T02:08:29Z",
  "problem_statement": "Multi-level FilteredRelation with select_related() may set wrong related object.\nDescription\n\t\ntest case:\n# add to known_related_objects.tests.ExistingRelatedInstancesTests\n\tdef test_wrong_select_related(self):\n\t\twith self.assertNumQueries(3):\n\t\t\tp = list(PoolStyle.objects.annotate(\n\t\t\t\ttournament_pool=FilteredRelation('pool__tournament__pool'),\n\t\t\t\t).select_related('tournament_pool'))\n\t\t\tself.assertEqual(p[0].pool.tournament, p[0].tournament_pool.tournament)\nresult:\n======================================================================\nFAIL: test_wrong_select_related (known_related_objects.tests.ExistingRelatedInstancesTests.test_wrong_select_related)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n File \"D:\\Work\\django\\tests\\known_related_objects\\tests.py\", line 171, in test_wrong_select_related\n\tself.assertEqual(p[0].pool.tournament, p[0].tournament_pool.tournament)\nAssertionError: <Tournament: Tournament object (1)> != <PoolStyle: PoolStyle object (1)>\n----------------------------------------------------------------------\n",
  "patch": "diff --git a/django/db/models/sql/compiler.py b/django/db/models/sql/compiler.py\n--- a/django/db/models/sql/compiler.py\n+++ b/django/db/models/sql/compiler.py\n@@ -1274,6 +1274,9 @@ def local_setter(final_field, obj, from_obj):\n                 if from_obj:\n                     final_field.remote_field.set_cached_value(from_obj, obj)\n \n+            def local_setter_noop(obj, from_obj):\n+                pass\n+\n             def remote_setter(name, obj, from_obj):\n                 setattr(from_obj, name, obj)\n \n@@ -1295,7 +1298,11 @@ def remote_setter(name, obj, from_obj):\n                         \"model\": model,\n                         \"field\": final_field,\n                         \"reverse\": True,\n-                        \"local_setter\": partial(local_setter, final_field),\n+                        \"local_setter\": (\n+                            partial(local_setter, final_field)\n+                            if len(joins) <= 2\n+                            else local_setter_noop\n+                        ),\n                         \"remote_setter\": partial(remote_setter, name),\n                         \"from_parent\": from_parent,\n                     }\n",
  "similar_bug_items": [
    {
      "pr_number": 13982,
      "pr_title": "Fixed #32425 -- Fixed adding nullable field with default on MySQL.",
      "pr_body": "ticket link: https://code.djangoproject.com/ticket/32425#ticket\r\n\r\nHi, team. I wanna fix some picky issue. \r\nI suggest below code for solving this problem. (same code creates different schemas)\r\n\r\nbut It has duplicate code with base/schema.py _alter_column_default_sql.\r\n\r\nWe can do like this. but it need to one more SQL. \r\nFor mysql users, performance is not good, but better than schema changes.\r\n\r\nmysql/schema.py\r\n\r\n```\r\ndef _alter_column_default_sql(self, model, old_field, new_field, drop=False):\r\n      super()._alter_column_default_sql(model, old_field, new_field, drop)\r\n\r\n      if drop and new_field.null:\r\n          new_db_params = new_field.db_parameters(connection=self.connection)\r\n          new_default = self.effective_default(new_field)\r\n          default = self._column_default_sql(new_field)\r\n          params = [new_default]\r\n          sql = self.sql_alter_column_default_null\r\n          return (\r\n              sql % {\r\n                  'column': self.quote_name(new_field.column),\r\n                  'type': new_db_params['type'],\r\n                  'default': default,\r\n              },\r\n              params,\r\n          )\r\n\r\n```",
      "issue_id": 32425,
      "issue_title": "MySQL Schema is different about the same class definitions. (depends on create table vs alter table)",
      "issue_body": "Hi, My name is Jordan. When I delete the column, I found some picky points in Django.\nMySQL Schema is different about the same class definitions.\nMySQL Schemas are different about the nullable column default value.\nFor example,\n1) when it was made by 'create table'\nclass PhoneBook(models.Model):\n    name = models.CharField(max_length=32, null=True, blank=True, default='jordan')\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\nThe above code creates the migrations file as shown below.\noperations = [\n        migrations.CreateModel(\n            name='PhoneBook',\n            fields=[\n                ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),\n                ('name', models.CharField(blank=True, default='jordan', max_length=32, null=True)),\n                ('phone_number', models.CharField(blank=True, max_length=32, null=True)),\n            ],\n        ),\n]\nWhen Migrate, SQL is executed as shown below.\nCREATE TABLE `main_phonebook` (\n\t`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY,\n\t`name` varchar(32) NULL,\n\t`phone_number` varchar(32) NULL\n)ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\n2) when it was made by 'alter table'\nFor the situation where the ‘alter table’ is applied, let's define the model class first as below and then add the name column.\nclass PhoneBook2(models.Model):\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\nand I added 'name' column.\nclass PhoneBook2(models.Model):\n    name = models.CharField(max_length=32, null=True, blank=True, default='jordan')\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\noperations = [\n        migrations.AddField(\n            model_name='phonebook2',\n            name='name',\n            field=models.CharField(blank=True, default='jordan', max_length=32, null=True),\n        ),\n    ]\nthis operations make this SQL.\nALTER TABLE `main_phonebook2` ADD COLUMN `name` varchar(32) DEFAULT %s NULL ['jordan']\nALTER TABLE `main_phonebook2` ALTER COLUMN `name` DROP DEFAULT []\nand table schema is like below.\nCREATE TABLE `main_phonebook2` (\n  `id` int(11) NOT NULL AUTO_INCREMENT,\n  `phone_number` varchar(32) DEFAULT NULL,\n  `name` varchar(32),\n  PRIMARY KEY (`id`)\n) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;\nWe can see that the same code creates different schemas.\n+ when i try to remove the column\nclass PhoneBook(models.Model):\n    # name = models.CharField(max_length=32, null=True, blank=True, default='jordan')\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\n\n\nclass PhoneBook2(models.Model):\n    # name = models.CharField(max_length=32, null=True, blank=True, default='jordan')\n    phone_number = models.CharField(max_length=32, null=True, blank=True)\nhappened like this.\nIn [2]: p=PhoneBook(phone_number='010-1234-1234')                         \n\nIn [3]: p.save()                                                          \nINSERT INTO `main_phonebook` (`phone_number`) VALUES (%s) ['010-1234-1234']\n\nIn [4]: p2=PhoneBook2(phone_number='010-1234-1234')                       \n\nIn [5]: p2.save()                                                         \nINSERT INTO `main_phonebook2` (`phone_number`) VALUES (%s) ['010-1234-1234']\n\n!!!!!!!!error!!!!!!!!!!!!!!!\nIntegrityError: (1364, \"Field 'name' doesn't have a default value\")\nI made PR for fixing this.\n​\nhttps://github.com/django/django/pull/13982",
      "issue_closed_at": "2021-02-09T01:25:39",
      "base_commit": "b99d6c9cbc8eecf480892599201eef0d14b20d71",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "_alter_column_default_sql",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def _alter_column_default_sql(self, model, old_field, new_field, drop=False):\n        \"\"\"\n        Hook to specialize column default alteration.\n\n        Return a (sql, params) fragment to add or drop (depending on the drop\n        argument) a default to new_field's column.\n        \"\"\"\n        new_default = self.effective_default(new_field)\n        default = self._column_default_sql(new_field)\n        params = [new_default]\n\n        if drop:\n            params = []\n        elif self.connection.features.requires_literal_defaults:\n            # Some databases (Oracle) can't take defaults as a parameter\n            # If this is the case, the SchemaEditor for that database should\n            # implement prepare_default().\n            default = self.prepare_default(new_default)\n            params = []\n\n        new_db_params = new_field.db_parameters(connection=self.connection)\n        sql = self.sql_alter_column_no_default if drop else self.sql_alter_column_default\n        return (\n            sql % {\n                'column': self.quote_name(new_field.column),\n                'type': new_db_params['type'],\n                'default': default,\n            },\n            params,\n        )"
        },
        {
          "file": "django/db/backends/mysql/schema.py",
          "type": "class",
          "name": "DatabaseSchemaEditor",
          "code": "class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):\n\n    sql_rename_table = \"RENAME TABLE %(old_table)s TO %(new_table)s\"\n\n    sql_alter_column_null = \"MODIFY %(column)s %(type)s NULL\"\n    sql_alter_column_not_null = \"MODIFY %(column)s %(type)s NOT NULL\"\n    sql_alter_column_type = \"MODIFY %(column)s %(type)s\"\n    sql_alter_column_collate = \"MODIFY %(column)s %(type)s%(collation)s\"\n\n    # No 'CASCADE' which works as a no-op in MySQL but is undocumented\n    sql_delete_column = \"ALTER TABLE %(table)s DROP COLUMN %(column)s\"\n\n    sql_delete_unique = \"ALTER TABLE %(table)s DROP INDEX %(name)s\"\n    sql_create_column_inline_fk = (\n        ', ADD CONSTRAINT %(name)s FOREIGN KEY (%(column)s) '\n        'REFERENCES %(to_table)s(%(to_column)s)'\n    )\n    sql_delete_fk = \"ALTER TABLE %(table)s DROP FOREIGN KEY %(name)s\"\n\n    sql_delete_index = \"DROP INDEX %(name)s ON %(table)s\"\n\n    sql_create_pk = \"ALTER TABLE %(table)s ADD CONSTRAINT %(name)s PRIMARY KEY (%(columns)s)\"\n    sql_delete_pk = \"ALTER TABLE %(table)s DROP PRIMARY KEY\"\n\n    sql_create_index = 'CREATE INDEX %(name)s ON %(table)s (%(columns)s)%(extra)s'\n\n    @property\n    def sql_delete_check(self):\n        if self.connection.mysql_is_mariadb:\n            # The name of the column check constraint is the same as the field\n            # name on MariaDB. Adding IF EXISTS clause prevents migrations\n            # crash. Constraint is removed during a \"MODIFY\" column statement.\n            return 'ALTER TABLE %(table)s DROP CONSTRAINT IF EXISTS %(name)s'\n        return 'ALTER TABLE %(table)s DROP CHECK %(name)s'\n\n    @property\n    def sql_rename_column(self):\n        # MariaDB >= 10.5.2 and MySQL >= 8.0.4 support an\n        # \"ALTER TABLE ... RENAME COLUMN\" statement.\n        if self.connection.mysql_is_mariadb:\n            if self.connection.mysql_version >= (10, 5, 2):\n                return super().sql_rename_column\n        elif self.connection.mysql_version >= (8, 0, 4):\n            return super().sql_rename_column\n        return 'ALTER TABLE %(table)s CHANGE %(old_column)s %(new_column)s %(type)s'\n\n    def quote_value(self, value):\n        self.connection.ensure_connection()\n        if isinstance(value, str):\n            value = value.replace('%', '%%')\n        # MySQLdb escapes to string, PyMySQL to bytes.\n        quoted = self.connection.connection.escape(value, self.connection.connection.encoders)\n        if isinstance(value, str) and isinstance(quoted, bytes):\n            quoted = quoted.decode()\n        return quoted\n\n    def _is_limited_data_type(self, field):\n        db_type = field.db_type(self.connection)\n        return db_type is not None and db_type.lower() in self.connection._limited_data_types\n\n    def skip_default(self, field):\n        if not self._supports_limited_data_type_defaults:\n            return self._is_limited_data_type(field)\n        return False\n\n    @property\n    def _supports_limited_data_type_defaults(self):\n        # MariaDB >= 10.2.1 and MySQL >= 8.0.13 supports defaults for BLOB\n        # and TEXT.\n        if self.connection.mysql_is_mariadb:\n            return self.connection.mysql_version >= (10, 2, 1)\n        return self.connection.mysql_version >= (8, 0, 13)\n\n    def _column_default_sql(self, field):\n        if (\n            not self.connection.mysql_is_mariadb and\n            self._supports_limited_data_type_defaults and\n            self._is_limited_data_type(field)\n        ):\n            # MySQL supports defaults for BLOB and TEXT columns only if the\n            # default value is written as an expression i.e. in parentheses.\n            return '(%s)'\n        return super()._column_default_sql(field)\n\n    def add_field(self, model, field):\n        super().add_field(model, field)\n\n        # Simulate the effect of a one-off default.\n        # field.default may be unhashable, so a set isn't used for \"in\" check.\n        if self.skip_default(field) and field.default not in (None, NOT_PROVIDED):\n            effective_default = self.effective_default(field)\n            self.execute('UPDATE %(table)s SET %(column)s = %%s' % {\n                'table': self.quote_name(model._meta.db_table),\n                'column': self.quote_name(field.column),\n            }, [effective_default])\n\n    def _field_should_be_indexed(self, model, field):\n        create_index = super()._field_should_be_indexed(model, field)\n        storage = self.connection.introspection.get_storage_engine(\n            self.connection.cursor(), model._meta.db_table\n        )\n        # No need to create an index for ForeignKey fields except if\n        # db_constraint=False because the index from that constraint won't be\n        # created.\n        if (storage == \"InnoDB\" and\n                create_index and\n                field.get_internal_type() == 'ForeignKey' and\n                field.db_constraint):\n            return False\n        return not self._is_limited_data_type(field) and create_index\n\n    def _delete_composed_index(self, model, fields, *args):\n        \"\"\"\n        MySQL can remove an implicit FK index on a field when that field is\n        covered by another index like a unique_together. \"covered\" here means\n        that the more complex index starts like the simpler one.\n        http://bugs.mysql.com/bug.php?id=37910 / Django ticket #24757\n        We check here before removing the [unique|index]_together if we have to\n        recreate a FK index.\n        \"\"\"\n        first_field = model._meta.get_field(fields[0])\n        if first_field.get_internal_type() == 'ForeignKey':\n            constraint_names = self._constraint_names(model, [first_field.column], index=True)\n            if not constraint_names:\n                self.execute(\n                    self._create_index_sql(model, fields=[first_field], suffix='')\n                )\n        return super()._delete_composed_index(model, fields, *args)\n\n    def _set_field_new_type_null_status(self, field, new_type):\n        \"\"\"\n        Keep the null property of the old field. If it has changed, it will be\n        handled separately.\n        \"\"\"\n        if field.null:\n            new_type += \" NULL\"\n        else:\n            new_type += \" NOT NULL\"\n        return new_type\n\n    def _alter_column_type_sql(self, model, old_field, new_field, new_type):\n        new_type = self._set_field_new_type_null_status(old_field, new_type)\n        return super()._alter_column_type_sql(model, old_field, new_field, new_type)\n\n    def _rename_field_sql(self, table, old_field, new_field, new_type):\n        new_type = self._set_field_new_type_null_status(old_field, new_type)\n        return super()._rename_field_sql(table, old_field, new_field, new_type)"
        },
        {
          "file": "django/db/backends/oracle/schema.py",
          "type": "class",
          "name": "DatabaseSchemaEditor",
          "code": "class DatabaseSchemaEditor(BaseDatabaseSchemaEditor):\n\n    sql_create_column = \"ALTER TABLE %(table)s ADD %(column)s %(definition)s\"\n    sql_alter_column_type = \"MODIFY %(column)s %(type)s\"\n    sql_alter_column_null = \"MODIFY %(column)s NULL\"\n    sql_alter_column_not_null = \"MODIFY %(column)s NOT NULL\"\n    sql_alter_column_default = \"MODIFY %(column)s DEFAULT %(default)s\"\n    sql_alter_column_no_default = \"MODIFY %(column)s DEFAULT NULL\"\n    sql_alter_column_collate = \"MODIFY %(column)s %(type)s%(collation)s\"\n\n    sql_delete_column = \"ALTER TABLE %(table)s DROP COLUMN %(column)s\"\n    sql_create_column_inline_fk = 'CONSTRAINT %(name)s REFERENCES %(to_table)s(%(to_column)s)%(deferrable)s'\n    sql_delete_table = \"DROP TABLE %(table)s CASCADE CONSTRAINTS\"\n    sql_create_index = \"CREATE INDEX %(name)s ON %(table)s (%(columns)s)%(extra)s\"\n\n    def quote_value(self, value):\n        if isinstance(value, (datetime.date, datetime.time, datetime.datetime)):\n            return \"'%s'\" % value\n        elif isinstance(value, str):\n            return \"'%s'\" % value.replace(\"\\'\", \"\\'\\'\").replace('%', '%%')\n        elif isinstance(value, (bytes, bytearray, memoryview)):\n            return \"'%s'\" % value.hex()\n        elif isinstance(value, bool):\n            return \"1\" if value else \"0\"\n        else:\n            return str(value)\n\n    def remove_field(self, model, field):\n        # If the column is an identity column, drop the identity before\n        # removing the field.\n        if self._is_identity_column(model._meta.db_table, field.column):\n            self._drop_identity(model._meta.db_table, field.column)\n        super().remove_field(model, field)\n\n    def delete_model(self, model):\n        # Run superclass action\n        super().delete_model(model)\n        # Clean up manually created sequence.\n        self.execute(\"\"\"\n            DECLARE\n                i INTEGER;\n            BEGIN\n                SELECT COUNT(1) INTO i FROM USER_SEQUENCES\n                    WHERE SEQUENCE_NAME = '%(sq_name)s';\n                IF i = 1 THEN\n                    EXECUTE IMMEDIATE 'DROP SEQUENCE \"%(sq_name)s\"';\n                END IF;\n            END;\n        /\"\"\" % {'sq_name': self.connection.ops._get_no_autofield_sequence_name(model._meta.db_table)})\n\n    def alter_field(self, model, old_field, new_field, strict=False):\n        try:\n            super().alter_field(model, old_field, new_field, strict)\n        except DatabaseError as e:\n            description = str(e)\n            # If we're changing type to an unsupported type we need a\n            # SQLite-ish workaround\n            if 'ORA-22858' in description or 'ORA-22859' in description:\n                self._alter_field_type_workaround(model, old_field, new_field)\n            # If an identity column is changing to a non-numeric type, drop the\n            # identity first.\n            elif 'ORA-30675' in description:\n                self._drop_identity(model._meta.db_table, old_field.column)\n                self.alter_field(model, old_field, new_field, strict)\n            # If a primary key column is changing to an identity column, drop\n            # the primary key first.\n            elif 'ORA-30673' in description and old_field.primary_key:\n                self._delete_primary_key(model, strict=True)\n                self._alter_field_type_workaround(model, old_field, new_field)\n            else:\n                raise\n\n    def _alter_field_type_workaround(self, model, old_field, new_field):\n        \"\"\"\n        Oracle refuses to change from some type to other type.\n        What we need to do instead is:\n        - Add a nullable version of the desired field with a temporary name. If\n          the new column is an auto field, then the temporary column can't be\n          nullable.\n        - Update the table to transfer values from old to new\n        - Drop old column\n        - Rename the new column and possibly drop the nullable property\n        \"\"\"\n        # Make a new field that's like the new one but with a temporary\n        # column name.\n        new_temp_field = copy.deepcopy(new_field)\n        new_temp_field.null = (new_field.get_internal_type() not in ('AutoField', 'BigAutoField', 'SmallAutoField'))\n        new_temp_field.column = self._generate_temp_name(new_field.column)\n        # Add it\n        self.add_field(model, new_temp_field)\n        # Explicit data type conversion\n        # https://docs.oracle.com/en/database/oracle/oracle-database/18/sqlrf\n        # /Data-Type-Comparison-Rules.html#GUID-D0C5A47E-6F93-4C2D-9E49-4F2B86B359DD\n        new_value = self.quote_name(old_field.column)\n        old_type = old_field.db_type(self.connection)\n        if re.match('^N?CLOB', old_type):\n            new_value = \"TO_CHAR(%s)\" % new_value\n            old_type = 'VARCHAR2'\n        if re.match('^N?VARCHAR2', old_type):\n            new_internal_type = new_field.get_internal_type()\n            if new_internal_type == 'DateField':\n                new_value = \"TO_DATE(%s, 'YYYY-MM-DD')\" % new_value\n            elif new_internal_type == 'DateTimeField':\n                new_value = \"TO_TIMESTAMP(%s, 'YYYY-MM-DD HH24:MI:SS.FF')\" % new_value\n            elif new_internal_type == 'TimeField':\n                # TimeField are stored as TIMESTAMP with a 1900-01-01 date part.\n                new_value = \"TO_TIMESTAMP(CONCAT('1900-01-01 ', %s), 'YYYY-MM-DD HH24:MI:SS.FF')\" % new_value\n        # Transfer values across\n        self.execute(\"UPDATE %s set %s=%s\" % (\n            self.quote_name(model._meta.db_table),\n            self.quote_name(new_temp_field.column),\n            new_value,\n        ))\n        # Drop the old field\n        self.remove_field(model, old_field)\n        # Rename and possibly make the new field NOT NULL\n        super().alter_field(model, new_temp_field, new_field)\n\n    def _alter_column_type_sql(self, model, old_field, new_field, new_type):\n        auto_field_types = {'AutoField', 'BigAutoField', 'SmallAutoField'}\n        # Drop the identity if migrating away from AutoField.\n        if (\n            old_field.get_internal_type() in auto_field_types and\n            new_field.get_internal_type() not in auto_field_types and\n            self._is_identity_column(model._meta.db_table, new_field.column)\n        ):\n            self._drop_identity(model._meta.db_table, new_field.column)\n        return super()._alter_column_type_sql(model, old_field, new_field, new_type)\n\n    def normalize_name(self, name):\n        \"\"\"\n        Get the properly shortened and uppercased identifier as returned by\n        quote_name() but without the quotes.\n        \"\"\"\n        nn = self.quote_name(name)\n        if nn[0] == '\"' and nn[-1] == '\"':\n            nn = nn[1:-1]\n        return nn\n\n    def _generate_temp_name(self, for_name):\n        \"\"\"Generate temporary names for workarounds that need temp columns.\"\"\"\n        suffix = hex(hash(for_name)).upper()[1:]\n        return self.normalize_name(for_name + \"_\" + suffix)\n\n    def prepare_default(self, value):\n        return self.quote_value(value)\n\n    def _field_should_be_indexed(self, model, field):\n        create_index = super()._field_should_be_indexed(model, field)\n        db_type = field.db_type(self.connection)\n        if db_type is not None and db_type.lower() in self.connection._limited_data_types:\n            return False\n        return create_index\n\n    def _unique_should_be_added(self, old_field, new_field):\n        return (\n            super()._unique_should_be_added(old_field, new_field) and\n            not self._field_became_primary_key(old_field, new_field)\n        )\n\n    def _is_identity_column(self, table_name, column_name):\n        with self.connection.cursor() as cursor:\n            cursor.execute(\"\"\"\n                SELECT\n                    CASE WHEN identity_column = 'YES' THEN 1 ELSE 0 END\n                FROM user_tab_cols\n                WHERE table_name = %s AND\n                      column_name = %s\n            \"\"\", [self.normalize_name(table_name), self.normalize_name(column_name)])\n            row = cursor.fetchone()\n            return row[0] if row else False\n\n    def _drop_identity(self, table_name, column_name):\n        self.execute('ALTER TABLE %(table)s MODIFY %(column)s DROP IDENTITY' % {\n            'table': self.quote_name(table_name),\n            'column': self.quote_name(column_name),\n        })\n\n    def _get_default_collation(self, table_name):\n        with self.connection.cursor() as cursor:\n            cursor.execute(\"\"\"\n                SELECT default_collation FROM user_tables WHERE table_name = %s\n            \"\"\", [self.normalize_name(table_name)])\n            return cursor.fetchone()[0]\n\n    def _alter_column_collation_sql(self, model, new_field, new_type, new_collation):\n        if new_collation is None:\n            new_collation = self._get_default_collation(model._meta.db_table)\n        return super()._alter_column_collation_sql(model, new_field, new_type, new_collation)"
        }
      ]
    },
    {
      "pr_number": 8396,
      "pr_title": "Fixed #28116 -- Used error code filtering in PostgreSQL test database creation.",
      "pr_body": "Ticket [28116](https://code.djangoproject.com/ticket/28116).",
      "issue_id": 28116,
      "issue_title": "Filtering PostgreSQL exception based on message is too brittle",
      "issue_body": "In commit\n64264c9a19b7dae6cd1d5e112177cdbcafafc93c\n, the behavior of\n_execute_create_test_db\ndepends on the databae error message\n''database %s already exists'\n.\nOn my system for example, the error message is localized so I'm finally getting a message\nGot an error creating the test database: la base de données « test_testgis1 » existe déjà\neven when I provide\n--keepdb\n.",
      "issue_closed_at": "2017-04-24T23:01:44",
      "base_commit": "8ef35468b660e1c25af67a8299202b8bc108679f",
      "changes": [
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "line",
          "name": "line 1",
          "code": "import sys\n\nfrom django.db.backends.base.creation import BaseDatabaseCreation\n\n"
        },
        {
          "file": "django/db/backends/postgresql/creation.py",
          "type": "function",
          "name": "_execute_create_test_db",
          "class_name": "DatabaseCreation",
          "code": "def _execute_create_test_db(self, cursor, parameters, keepdb=False):\n        try:\n            super()._execute_create_test_db(cursor, parameters, keepdb)\n        except Exception as e:\n            exc_msg = 'database %s already exists' % parameters['dbname']\n            if exc_msg not in str(e):\n                # All errors except \"database already exists\" cancel tests\n                sys.stderr.write('Got an error creating the test database: %s\\n' % e)\n                sys.exit(2)\n            elif not keepdb:\n                # If the database should be kept, ignore \"database already\n                # exists\".\n                raise e"
        }
      ]
    },
    {
      "pr_number": 4655,
      "pr_title": "Fixed #24757 -- Recreated MySQL index when needed during unique_together removal",
      "pr_body": "",
      "issue_id": 24757,
      "issue_title": "Removing unique_together constraint make Django 1.8 migration fail with MySQL",
      "issue_body": "Migrations containing\nmigrations.AlterUniqueTogether(name='xxx', unique_together=set([]),)\nmay fail on Django 1.8 but not on Django 1.7 if using the backend mysql.\nGiven the database engine is mysql and the following\nmodels.py\n:\nfrom django.db import models\n\n\nclass MyModelA(models.Model):\n    s = models.CharField(max_length=120)\n\n\nclass MyModelB(models.Model):\n    a = models.ForeignKey(MyModelA)\n    b = models.IntegerField()\nAnd the 2 following migrations:\n# -*- coding: utf-8 -*-\nfrom __future__ import unicode_literals\n\nfrom django.db import models, migrations\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n    ]\n\n    operations = [\n        migrations.CreateModel(\n            name='MyModelA',\n            fields=[\n                ('id', models.AutoField(serialize=False, auto_created=True, verbose_name='ID', primary_key=True)),\n                ('s', models.CharField(max_length=120)),\n            ],\n        ),\n        migrations.CreateModel(\n            name='MyModelB',\n            fields=[\n                ('id', models.AutoField(serialize=False, auto_created=True, verbose_name='ID', primary_key=True)),\n                ('b', models.IntegerField()),\n                ('a', models.ForeignKey(to='mysite.MyModelA')),\n            ],\n        ),\n        migrations.AlterUniqueTogether(\n            name='mymodelb',\n            unique_together=set([('a', 'b')]),\n        ),\n    ]\n# -*- coding: utf-8 -*-\nfrom __future__ import unicode_literals\n\nfrom django.db import models, migrations\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n        ('mysite', '0001_initial'),\n    ]\n\n    operations = [\n        migrations.AlterUniqueTogether(\n            name='mymodelb',\n            unique_together=set([]),\n        ),\n    ]\nRunning\n./manage.py migrate\nworks fine on Django 1.7, but on Django 1.8 I get the following error:\n$ ./manage.py migrate                \nOperations to perform:\n  Synchronize unmigrated apps: staticfiles, messages\n  Apply all migrations: contenttypes, auth, mysite, sessions, admin\nSynchronizing apps without migrations:\n  Creating tables...\n    Running deferred SQL...\n  Installing custom SQL...\nRunning migrations:\n  Rendering model states... DONE\n  [..]\n  Applying mysite.0001_initial... OK\n  Applying mysite.0002_auto...Traceback (most recent call last):\n  [..]\ndjango.db.utils.OperationalError: (1553, \"Cannot drop index 'mysite_mymodelb_a_id_b53c781c93aab9a_uniq': needed in a foreign key constraint\")\nThe same migrations work on psql, I did not have the chance to test it against sqlite.\nInspecting the generated sql I found Django 1.7 add an index in addition to the unique constraint and Django 1.8 does not.\nOn Django 1.7\nBEGIN;\nCREATE TABLE `mysite_mymodela` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `s` varchar(120) NOT NULL);\nCREATE TABLE `mysite_mymodelb` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `b` integer NOT NULL, `a_id` integer NOT NULL);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_48275010b87402ed_uniq` UNIQUE (`a_id`, `b`);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_4981d3920f0fef1e_fk_mysite_mymodela_id` FOREIGN KEY (`a_id`) REFERENCES `mysite_mymodela` (`id`);\nCREATE INDEX `mysite_mymodelb_e2b453f4` ON `mysite_mymodelb` (`a_id`);\n\nCOMMIT;\nNote the\nCREATE INDEX `mysite_mymodelb_e2b453f4` ON `mysite_mymodelb` (`a_id`);\n.\nAnd on Django 1.8\nBEGIN;\nCREATE TABLE `mysite_mymodela` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `s` varchar(120) NOT NULL);\nCREATE TABLE `mysite_mymodelb` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `b` integer NOT NULL, `a_id` integer NOT NULL);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_784bb266d0e363ab_uniq` UNIQUE (`a_id`, `b`);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_50892c78e36678b8_fk_mysite_mymodela_id` FOREIGN KEY (`a_id`) REFERENCES `mysite_mymodela` (`id`);\n\nCOMMIT;\nI joined the full traceback.",
      "issue_closed_at": "2015-05-15T10:06:22",
      "base_commit": "0eaef8e527af556c0c517a64035929404cf94a39",
      "changes": [
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "alter_unique_together",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def alter_unique_together(self, model, old_unique_together, new_unique_together):\n        \"\"\"\n        Deals with a model changing its unique_together.\n        Note: The input unique_togethers must be doubly-nested, not the single-\n        nested [\"foo\", \"bar\"] format.\n        \"\"\"\n        olds = set(tuple(fields) for fields in old_unique_together)\n        news = set(tuple(fields) for fields in new_unique_together)\n        # Deleted uniques\n        for fields in olds.difference(news):\n            columns = [model._meta.get_field(field).column for field in fields]\n            constraint_names = self._constraint_names(model, columns, unique=True)\n            if len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of constraints for %s(%s)\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    \", \".join(columns),\n                ))\n            self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_names[0]))\n        # Created uniques\n        for fields in news.difference(olds):\n            columns = [model._meta.get_field(field).column for field in fields]\n            self.execute(self._create_unique_sql(model, columns))"
        },
        {
          "file": "django/db/backends/base/schema.py",
          "type": "function",
          "name": "alter_index_together",
          "class_name": "BaseDatabaseSchemaEditor",
          "code": "def alter_index_together(self, model, old_index_together, new_index_together):\n        \"\"\"\n        Deals with a model changing its index_together.\n        Note: The input index_togethers must be doubly-nested, not the single-\n        nested [\"foo\", \"bar\"] format.\n        \"\"\"\n        olds = set(tuple(fields) for fields in old_index_together)\n        news = set(tuple(fields) for fields in new_index_together)\n        # Deleted indexes\n        for fields in olds.difference(news):\n            columns = [model._meta.get_field(field).column for field in fields]\n            constraint_names = self._constraint_names(model, list(columns), index=True)\n            if len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of constraints for %s(%s)\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    \", \".join(columns),\n                ))\n            self.execute(self._delete_constraint_sql(self.sql_delete_index, model, constraint_names[0]))\n        # Created indexes\n        for field_names in news.difference(olds):\n            fields = [model._meta.get_field(field) for field in field_names]\n            self.execute(self._create_index_sql(model, fields, suffix=\"_idx\"))"
        },
        {
          "file": "django/db/backends/mysql/schema.py",
          "type": "function",
          "name": "_model_indexes_sql",
          "class_name": "DatabaseSchemaEditor",
          "code": "def _model_indexes_sql(self, model):\n        storage = self.connection.introspection.get_storage_engine(\n            self.connection.cursor(), model._meta.db_table\n        )\n        if storage == \"InnoDB\":\n            for field in model._meta.local_fields:\n                if field.db_index and not field.unique and field.get_internal_type() == \"ForeignKey\":\n                    # Temporary setting db_index to False (in memory) to disable\n                    # index creation for FKs (index automatically created by MySQL)\n                    field.db_index = False\n        return super(DatabaseSchemaEditor, self)._model_indexes_sql(model)"
        }
      ]
    },
    {
      "pr_number": 8905,
      "pr_title": "Fixed #28375 -- Fixed KeyError raised on reverse prefetch of a model with OneToOne primary key to non-pk field",
      "pr_body": "https://code.djangoproject.com/ticket/28375",
      "issue_id": 28375,
      "issue_title": "QuerySet.prefetch_related() crashes with KeyError if model uses to_field and string primary key",
      "issue_body": "The issue:\nprefetch_related failed if prefetching by char primary key.\nDjango version 1.11.3\nPython 2.7\nreproducible steps:\n1) django-admin startproject pk_string\n2) cd pk_string\n3) django-admin startapp users\n4) update  users.models\n# -*- coding: utf-8 -*-\nfrom\n__future__\nimport\nunicode_literals\nfrom\ndjango.db\nimport\nmodels\n# Create your models here.\nclass\nUser\n(\nmodels\n.\nModel\n):\nemail\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n255\n,\nunique\n=\nTrue\n)\nclass\nUserData\n(\nmodels\n.\nModel\n):\nemail\n=\nmodels\n.\nOneToOneField\n(\nUser\n,\nto_field\n=\n'email'\n,\nprimary_key\n=\nTrue\n)\nnote\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n255\n);\n5) install app pk_string.settings\n...\nINSTALLED_APPS\n=\n[\n'django.contrib.admin'\n,\n'django.contrib.auth'\n,\n'django.contrib.contenttypes'\n,\n'django.contrib.sessions'\n,\n'django.contrib.messages'\n,\n'django.contrib.staticfiles'\n,\n'users'\n,\n]\n...\n6) ./manage.py makemigrations\n7) ./manage.py migrate\n8) ./manage.py shell\nfrom\nusers.models\nimport\nUser\n,\nUserData\nUser\n.\nobjects\n.\ncreate\n(\nemail\n=\n'111111'\n)\nUser\n.\nobjects\n.\ncreate\n(\nemail\n=\n'222222'\n)\nUser\n.\nobjects\n.\ncreate\n(\nemail\n=\n'333333'\n)\nUser\n.\nobjects\n.\ncreate\n(\nemail\n=\n'444444'\n)\nusers\n=\nUser\n.\nobjects\n.\nall\n()\n.\nprefetch_related\n(\n'userdata'\n)\nusers\n[\n0\n]\n>>>\n\"<User: User object>\"\nUserData\n.\nobjects\n.\ncreate\n(\nemail\n=\nusers\n[\n0\n]\n,\nnote\n=\n'111'\n)\nUserData\n.\nobjects\n.\ncreate\n(\nemail\n=\nusers\n[\n2\n]\n,\nnote\n=\n'222'\n)\nUserData\n.\nobjects\n.\ncreate\n(\nemail\n=\nusers\n[\n3\n]\n,\nnote\n=\n'333'\n)\nusers\n=\nUser\n.\nobjects\n.\nall\n()\n.\nprefetch_related\n(\n'userdata'\n)\n>>>\nusers\n[\n0\n]\nTraceback\n(\nmost\nrecent\ncall\nlast\n):\nFile\n\"<console>\"\n,\nline\n1\n,\nin\n<\nmodule\n>\nFile\n\"/usr/local/lib/python2.7/site-packages/django/db/models/query.py\"\n,\nline\n289\n,\nin\n__getitem__\nreturn\nlist\n(\nqs\n)[\n0\n]\nFile\n\"/usr/local/lib/python2.7/site-packages/django/db/models/query.py\"\n,\nline\n250\n,\nin\n__iter__\nself\n.\n_fetch_all\n()\nFile\n\"/usr/local/lib/python2.7/site-packages/django/db/models/query.py\"\n,\nline\n1120\n,\nin\n_fetch_all\nself\n.\n_prefetch_related_objects\n()\nFile\n\"/usr/local/lib/python2.7/site-packages/django/db/models/query.py\"\n,\nline\n675\n,\nin\n_prefetch_related_objects\nprefetch_related_objects\n(\nself\n.\n_result_cache\n,\n*\nself\n.\n_prefetch_related_lookups\n)\nFile\n\"/usr/local/lib/python2.7/site-packages/django/db/models/query.py\"\n,\nline\n1469\n,\nin\nprefetch_related_objects\nobj_list\n,\nadditional_lookups\n=\nprefetch_one_level\n(\nobj_list\n,\nprefetcher\n,\nlookup\n,\nlevel\n)\nFile\n\"/usr/local/lib/python2.7/site-packages/django/db/models/query.py\"\n,\nline\n1582\n,\nin\nprefetch_one_level\nprefetcher\n.\nget_prefetch_queryset\n(\ninstances\n,\nlookup\n.\nget_current_queryset\n(\nlevel\n)))\nFile\n\"/usr/local/lib/python2.7/site-packages/django/db/models/fields/related_descriptors.py\"\n,\nline\n362\n,\nin\nget_prefetch_queryset\ninstance\n=\ninstances_dict\n[\nrel_obj_attr\n(\nrel_obj\n)]\nKeyError\n:\nu\n'111111'",
      "issue_closed_at": "2017-08-21T15:47:49",
      "base_commit": "b5ad5c628a0327c2208d76e5cacb3cb6f48750b5",
      "changes": [
        {
          "file": "django/db/models/fields/related_descriptors.py",
          "type": "function",
          "name": "get_prefetch_queryset",
          "class_name": "ManyRelatedManager",
          "code": "def get_prefetch_queryset(self, instances, queryset=None):\n            if queryset is None:\n                queryset = super().get_queryset()\n\n            queryset._add_hints(instance=instances[0])\n            queryset = queryset.using(queryset._db or self._db)\n\n            query = {'%s__in' % self.query_field_name: instances}\n            queryset = queryset._next_is_sticky().filter(**query)\n\n            # M2M: need to annotate the query in order to get the primary model\n            # that the secondary model was actually related to. We know that\n            # there will already be a join on the join table, so we can just add\n            # the select.\n\n            # For non-autocreated 'through' models, can't assume we are\n            # dealing with PK values.\n            fk = self.through._meta.get_field(self.source_field_name)\n            join_table = fk.model._meta.db_table\n            connection = connections[queryset.db]\n            qn = connection.ops.quote_name\n            queryset = queryset.extra(select={\n                '_prefetch_related_val_%s' % f.attname:\n                '%s.%s' % (qn(join_table), qn(f.column)) for f in fk.local_related_fields})\n            return (\n                queryset,\n                lambda result: tuple(\n                    getattr(result, '_prefetch_related_val_%s' % f.attname)\n                    for f in fk.local_related_fields\n                ),\n                lambda inst: tuple(\n                    f.get_db_prep_value(getattr(inst, f.attname), connection)\n                    for f in fk.foreign_related_fields\n                ),\n                False,\n                self.prefetch_cache_name,\n                False,\n            )"
        }
      ]
    },
    {
      "pr_number": 16250,
      "pr_title": "Fixed #33701 -- Added fine-grained error locations to the technical 500 debug page.",
      "pr_body": "Solves Ticket [33701](https://code.djangoproject.com/ticket/33701)\r\n\r\nWith Python 3.11 the technical_500.html debug page now shows, where in the line the error occurred.\r\n\r\n![traceback](https://user-images.githubusercontent.com/39874595/199703281-3f755068-fd41-45bc-961c-957a78d51df5.PNG)\r\n",
      "issue_id": 33701,
      "issue_title": "Highlight error location in the technical 500 debug page on Python 3.11+.",
      "issue_body": "Similar to these tickets for extending the debug page for Python 3.11 exception changes:\n#33747\n,\n#33752\n.\nPython 3.11 adds \"where in the line\" highlighting to tracebacks:\n$ python3.11 t.py\nTraceback (most recent call last):\n  File \"/.../example.py\", line 5, in <module>\n    formula(1, 0)\n    ^^^^^^^^^^^^^\n  File \"/.../example.py\", line 2, in formula\n    return a / b + b / a\n           ~~^~~\nZeroDivisionError: division by zero\nRelease note:\n​\nhttps://docs.python.org/3.11/whatsnew/3.11.html#enhanced-error-locations-in-tracebacks\nIt would be good if we could use this on Django's error pages as well.\nThe new code column information API may be required:\n​\nhttps://docs.python.org/3.11/whatsnew/3.11.html#column-information-for-code-objects\n. Note it can be disabled.",
      "issue_closed_at": "2022-11-29T02:25:55",
      "base_commit": "9d726c7902979d4ad53945ed8f1037266a88010d",
      "changes": [
        {
          "file": "django/views/debug.py",
          "type": "line",
          "name": "line 1",
          "code": "import functools\nimport re\nimport sys\nimport types"
        },
        {
          "file": "django/views/debug.py",
          "type": "line",
          "name": "line 15",
          "code": "from django.utils.encoding import force_str\nfrom django.utils.module_loading import import_string\nfrom django.utils.regex_helper import _lazy_re_compile\nfrom django.utils.version import get_docs_version\n\n# Minimal Django templates engine to render the error templates\n# regardless of the project's TEMPLATES setting. Templates are"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "get_exception_traceback_frames",
          "class_name": "ExceptionReporter",
          "code": "def get_exception_traceback_frames(self, exc_value, tb):\n        exc_cause = self._get_explicit_or_implicit_cause(exc_value)\n        exc_cause_explicit = getattr(exc_value, \"__cause__\", True)\n        if tb is None:\n            yield {\n                \"exc_cause\": exc_cause,\n                \"exc_cause_explicit\": exc_cause_explicit,\n                \"tb\": None,\n                \"type\": \"user\",\n            }\n        while tb is not None:\n            # Support for __traceback_hide__ which is used by a few libraries\n            # to hide internal frames.\n            if tb.tb_frame.f_locals.get(\"__traceback_hide__\"):\n                tb = tb.tb_next\n                continue\n            filename = tb.tb_frame.f_code.co_filename\n            function = tb.tb_frame.f_code.co_name\n            lineno = tb.tb_lineno - 1\n            loader = tb.tb_frame.f_globals.get(\"__loader__\")\n            module_name = tb.tb_frame.f_globals.get(\"__name__\") or \"\"\n            (\n                pre_context_lineno,\n                pre_context,\n                context_line,\n                post_context,\n            ) = self._get_lines_from_file(\n                filename,\n                lineno,\n                7,\n                loader,\n                module_name,\n            )\n            if pre_context_lineno is None:\n                pre_context_lineno = lineno\n                pre_context = []\n                context_line = \"<source code not available>\"\n                post_context = []\n            yield {\n                \"exc_cause\": exc_cause,\n                \"exc_cause_explicit\": exc_cause_explicit,\n                \"tb\": tb,\n                \"type\": \"django\" if module_name.startswith(\"django.\") else \"user\",\n                \"filename\": filename,\n                \"function\": function,\n                \"lineno\": lineno + 1,\n                \"vars\": self.filter.get_traceback_frame_variables(\n                    self.request, tb.tb_frame\n                ),\n                \"id\": id(tb),\n                \"pre_context\": pre_context,\n                \"context_line\": context_line,\n                \"post_context\": post_context,\n                \"pre_context_lineno\": pre_context_lineno + 1,\n            }\n            tb = tb.tb_next"
        },
        {
          "file": "django/views/debug.py",
          "type": "function",
          "name": "get_exception_traceback_frames",
          "class_name": "ExceptionReporter",
          "code": "def get_exception_traceback_frames(self, exc_value, tb):\n        exc_cause = self._get_explicit_or_implicit_cause(exc_value)\n        exc_cause_explicit = getattr(exc_value, \"__cause__\", True)\n        if tb is None:\n            yield {\n                \"exc_cause\": exc_cause,\n                \"exc_cause_explicit\": exc_cause_explicit,\n                \"tb\": None,\n                \"type\": \"user\",\n            }\n        while tb is not None:\n            # Support for __traceback_hide__ which is used by a few libraries\n            # to hide internal frames.\n            if tb.tb_frame.f_locals.get(\"__traceback_hide__\"):\n                tb = tb.tb_next\n                continue\n            filename = tb.tb_frame.f_code.co_filename\n            function = tb.tb_frame.f_code.co_name\n            lineno = tb.tb_lineno - 1\n            loader = tb.tb_frame.f_globals.get(\"__loader__\")\n            module_name = tb.tb_frame.f_globals.get(\"__name__\") or \"\"\n            (\n                pre_context_lineno,\n                pre_context,\n                context_line,\n                post_context,\n            ) = self._get_lines_from_file(\n                filename,\n                lineno,\n                7,\n                loader,\n                module_name,\n            )\n            if pre_context_lineno is None:\n                pre_context_lineno = lineno\n                pre_context = []\n                context_line = \"<source code not available>\"\n                post_context = []\n            yield {\n                \"exc_cause\": exc_cause,\n                \"exc_cause_explicit\": exc_cause_explicit,\n                \"tb\": tb,\n                \"type\": \"django\" if module_name.startswith(\"django.\") else \"user\",\n                \"filename\": filename,\n                \"function\": function,\n                \"lineno\": lineno + 1,\n                \"vars\": self.filter.get_traceback_frame_variables(\n                    self.request, tb.tb_frame\n                ),\n                \"id\": id(tb),\n                \"pre_context\": pre_context,\n                \"context_line\": context_line,\n                \"post_context\": post_context,\n                \"pre_context_lineno\": pre_context_lineno + 1,\n            }\n            tb = tb.tb_next"
        }
      ]
    }
  ]
}