{
  "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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"
        }
      ]
    }
  ]
}