{
  "Selected_candidate": {
    "pr_number": 10058,
    "pr_title": "Fixed #29496 -- Fixed crash on Oracle when converting a non-unique field to primary key.",
    "pr_body": "Ticket [29496](https://code.djangoproject.com/ticket/29496).",
    "issue_id": 29496,
    "issue_title": "Altering non-unique field to primary key fails on Oracle",
    "issue_body": "I am trying to migrate a table from an auto-incrementing Integer PK to a UUID PK.\nI have custom-written migrations that add the new UUID field, add new foreign key fields in dependent tables, populate them, remove the old FKs, etc.\nOne of the final steps of the process is to alter the new uuid field in the table to be a primary_key:\nmigrations\n.\nAlterField\n(\nmodel_name\n=\n'examplelayout'\n,\nname\n=\n'uuid'\n,\nfield\n=\nmodels\n.\nUUIDField\n(\ndefault\n=\nuuid\n.\nuuid4\n,\nprimary_key\n=\nTrue\n,\neditable\n=\nFalse\n),\n),\nThe whole process works great on Postgres, but this specific step ends up in an error on Oracle 12, despite the fact that before this step there are no indexes in the DB on this field.\nquery = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT \"VARIO_EXA_UUID_8498C9C8_P\" PRIMARY KEY (\"UUID\")'\n...\ndjango.db.utils.DatabaseError: ORA-02261: such unique or primary key already exists in the table\nThe issue seems to be that Django tries to first add a UNIQUE constraint for the field, then a PRIMARY KEY on the same field. Here are all the SQL queries executed by this specific migration step:\n==================== STEP =========================  4  =====================\n\n\n[2018-06-15 08:51:11.141 | DEBUG | django.db.backends.schema:124] ALTER TABLE \"VARIO_EXAMPLELAYOUT\" MODIFY \"UUID\" DEFAULT '9cf3fe16f11d4fa09b62e337c50ae5f7'; (params [])\n[2018-06-15 08:51:11.212 | DEBUG | django.db.backends:111] (0.071) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" MODIFY \"UUID\" DEFAULT \\'9cf3fe16f11d4fa09b62e337c50ae5f7\\'' - PARAMS = (); args=[]\n[2018-06-15 08:51:11.212 | DEBUG | django.db.backends.schema:124] UPDATE \"VARIO_EXAMPLELAYOUT\" SET \"UUID\" = %s WHERE \"UUID\" IS NULL; (params ['347ac08714bd4c208ffd2f79e7932bb3'])\n[2018-06-15 08:51:11.269 | DEBUG | django.db.backends:111] (0.056) QUERY = 'UPDATE \"VARIO_EXAMPLELAYOUT\" SET \"UUID\" = :arg0 WHERE \"UUID\" IS NULL' - PARAMS = ('347ac08714bd4c208ffd2f79e7932bb3',); args=['347ac08714bd4c208ffd2f79e7932bb3']\n[2018-06-15 08:51:11.269 | DEBUG | django.db.backends.schema:124] ALTER TABLE \"VARIO_EXAMPLELAYOUT\" MODIFY \"UUID\" NOT NULL; (params [])\n[2018-06-15 08:51:11.343 | DEBUG | django.db.backends:111] (0.073) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" MODIFY \"UUID\" NOT NULL' - PARAMS = (); args=[]\n[2018-06-15 08:51:11.343 | DEBUG | django.db.backends.schema:124] ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT vario_exa_uuid_8498c9c8_u UNIQUE (\"UUID\"); (params ())\n[2018-06-15 08:51:11.437 | DEBUG | django.db.backends:111] (0.094) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT vario_exa_uuid_8498c9c8_u UNIQUE (\"UUID\")' - PARAMS = (); args=()\n[2018-06-15 08:51:11.792 | DEBUG | django.db.backends:111] (0.354) QUERY = \"\\n            SELECT\\n                user_constraints.constraint_name,\\n                LISTAGG(LOWER(cols.column_name), ',') WITHIN GROUP (ORDER BY cols.position),\\n                CASE user_constraints.constraint_type\\n                    WHEN 'P' THEN 1\\n                    ELSE 0\\n                END AS is_primary_key,\\n                CASE\\n                    WHEN user_constraints.constraint_type IN ('P', 'U') THEN 1\\n                    ELSE 0\\n                END AS is_unique,\\n                CASE user_constraints.constraint_type\\n                    WHEN 'C' THEN 1\\n                    ELSE 0\\n                END AS is_check_constraint\\n            FROM\\n                user_constraints\\n            LEFT OUTER JOIN\\n                user_cons_columns cols ON user_constraints.constraint_name = cols.constraint_name\\n            WHERE\\n                user_constraints.constraint_type = ANY('P', 'U', 'C')\\n                AND user_constraints.table_name = UPPER(:arg0)\\n            GROUP BY user_constraints.constraint_name, user_constraints.constraint_type\\n        \" - PARAMS = ('vario_examplelayout',); args=['vario_examplelayout']\n[2018-06-15 08:51:12.480 | DEBUG | django.db.backends:111] (0.688) QUERY = \"\\n            SELECT\\n                cons.constraint_name,\\n                LISTAGG(LOWER(cols.column_name), ',') WITHIN GROUP (ORDER BY cols.position),\\n                LOWER(rcols.table_name),\\n                LOWER(rcols.column_name)\\n            FROM\\n                user_constraints cons\\n            INNER JOIN\\n                user_cons_columns rcols ON rcols.constraint_name = cons.r_constraint_name AND rcols.position = 1\\n            LEFT OUTER JOIN\\n                user_cons_columns cols ON cons.constraint_name = cols.constraint_name\\n            WHERE\\n                cons.constraint_type = 'R' AND\\n                cons.table_name = UPPER(:arg0)\\n            GROUP BY cons.constraint_name, rcols.table_name, rcols.column_name\\n        \" - PARAMS = ('vario_examplelayout',); args=['vario_examplelayout']\n[2018-06-15 08:51:13.695 | DEBUG | django.db.backends:111] (1.214) QUERY = \"\\n            SELECT\\n                ind.index_name,\\n                LOWER(ind.index_type),\\n                LISTAGG(LOWER(cols.column_name), ',') WITHIN GROUP (ORDER BY cols.column_position),\\n                LISTAGG(cols.descend, ',') WITHIN GROUP (ORDER BY cols.column_position)\\n            FROM\\n                user_ind_columns cols, user_indexes ind\\n            WHERE\\n                cols.table_name = UPPER(:arg0) AND\\n                NOT EXISTS (\\n                    SELECT 1\\n                    FROM user_constraints cons\\n                    WHERE ind.index_name = cons.index_name\\n                ) AND cols.index_name = ind.index_name\\n            GROUP BY ind.index_name, ind.index_type\\n        \" - PARAMS = ('vario_examplelayout',); args=['vario_examplelayout']\n[2018-06-15 08:51:13.696 | DEBUG | django.db.backends.schema:124] ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT \"VARIO_EXA_UUID_8498C9C8_P\" PRIMARY KEY (\"UUID\"); (params ())\n[2018-06-15 08:51:13.775 | DEBUG | django.db.backends:111] (0.079) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT \"VARIO_EXA_UUID_8498C9C8_P\" PRIMARY KEY (\"UUID\")' - PARAMS = (); args=()\nTraceback (most recent call last):\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 85, in _execute\n    return self.cursor.execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/oracle/base.py\", line 500, in execute\n    return self.cursor.execute(query, self._param_generator(params))\ncx_Oracle.DatabaseError: ORA-02261: such unique or primary key already exists in the table\n\nThe above exception was the direct cause of the following exception:\n\nTraceback (most recent call last):\n  File \"./manage.py\", line 35, in <module>\n    execute_from_command_line(sys.argv)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/__init__.py\", line 371, in execute_from_command_line\n    utility.execute()\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/__init__.py\", line 365, in execute\n    self.fetch_command(subcommand).run_from_argv(self.argv)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/base.py\", line 288, in run_from_argv\n    self.execute(*args, **cmd_options)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/base.py\", line 335, in execute\n    output = self.handle(*args, **options)\n  File \"/var/www/forms/forms/common/management/commands/migrate.py\", line 54, in handle\n    super().handle(*args, **options)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/core/management/commands/migrate.py\", line 200, in handle\n    fake_initial=fake_initial,\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/executor.py\", line 117, in migrate\n    state = self._migrate_all_forwards(state, plan, full_plan, fake=fake, fake_initial=fake_initial)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/executor.py\", line 147, in _migrate_all_forwards\n    state = self.apply_migration(state, migration, fake=fake, fake_initial=fake_initial)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/executor.py\", line 244, in apply_migration\n    state = migration.apply(state, schema_editor)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/migration.py\", line 122, in apply\n    operation.database_forwards(self.app_label, schema_editor, old_state, project_state)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/migrations/operations/fields.py\", line 216, in database_forwards\n    schema_editor.alter_field(from_model, from_field, to_field)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/oracle/schema.py\", line 59, in alter_field\n    super().alter_field(model, old_field, new_field, strict)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/base/schema.py\", line 525, in alter_field\n    old_db_params, new_db_params, strict)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/base/schema.py\", line 725, in _alter_field\n    \"columns\": self.quote_name(new_field.column),\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/base/schema.py\", line 133, in execute\n    cursor.execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 100, in execute\n    return super().execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/raven/contrib/django/client.py\", line 127, in execute\n    return real_execute(self, sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 68, in execute\n    return self._execute_with_wrappers(sql, params, many=False, executor=self._execute)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 77, in _execute_with_wrappers\n    return executor(sql, params, many, context)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 85, in _execute\n    return self.cursor.execute(sql, params)\n  File \"/var/www/forms/forms/forms/db/backends/utils.py\", line 16, in execute\n    return super().execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/raven/contrib/django/client.py\", line 127, in execute\n    return real_execute(self, sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 68, in execute\n    return self._execute_with_wrappers(sql, params, many=False, executor=self._execute)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 77, in _execute_with_wrappers\n    return executor(sql, params, many, context)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 85, in _execute\n    return self.cursor.execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/utils.py\", line 89, in __exit__\n    raise dj_exc_value.with_traceback(traceback) from exc_value\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/utils.py\", line 85, in _execute\n    return self.cursor.execute(sql, params)\n  File \"/var/www/venv/lib/python3.5/site-packages/django/db/backends/oracle/base.py\", line 500, in execute\n    return self.cursor.execute(query, self._param_generator(params))\ndjango.db.utils.DatabaseError: ORA-02261: such unique or primary key already exists in the table\nNotice the two calls:\n[2018-06-15 08:51:11.437 | DEBUG | django.db.backends:111] (0.094) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT vario_exa_uuid_8498c9c8_u UNIQUE (\"UUID\")' - PARAMS = (); args=()\nand a bit below:\n[2018-06-15 08:51:13.775 | DEBUG | django.db.backends:111] (0.079) QUERY = 'ALTER TABLE \"VARIO_EXAMPLELAYOUT\" ADD CONSTRAINT \"VARIO_EXA_UUID_8498C9C8_P\" PRIMARY KEY (\"UUID\")' - PARAMS = (); args=()\n...which raises the error\nThe only other operations before this on this field that I have in one of the previous migrations is to actually add this UUID field as a nullable, non-unique field, and then in a data migration to populate it with a unique uuids for all ExampleLayouts:\n# prepare to migrate to UUID PKs for ExampleLayout and LayoutField\n        migrations.AddField(\n            model_name='examplelayout',\n            name='uuid',\n            field=models.UUIDField(null=True, default=uuid.uuid4),\n        ),\nFor completeness, the initial models look like this:\nclass\nExampleLayout\n(\nmodels\n.\nModel\n):\nimage\n=\nmodels\n.\nOneToOneField\n(\nImage\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'example_layout'\n)\ndoc_type\n=\nmodels\n.\nForeignKey\n(\nDocumentType\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'layouts'\n)\nclass\nLayoutField\n(\nmodels\n.\nModel\n):\ncore_field_type\n=\nmodels\n.\nForeignKey\n(\nCoreFieldType\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'layout_fields'\n)\nlayout\n=\nmodels\n.\nForeignKey\n(\nExampleLayout\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'fields'\n)\nbounding_box\n=\nBoundingBoxCharField\n()\ntranscription\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n128\n)\nwhile their final \"target\" version that I am trying to migrate to is this:\nclass\nExampleLayout\n(\nmodels\n.\nModel\n):\nuuid\n=\nmodels\n.\nUUIDField\n(\ndefault\n=\nuuid\n.\nuuid4\n,\nprimary_key\n=\nTrue\n,\neditable\n=\nFalse\n)\nimage\n=\nmodels\n.\nOneToOneField\n(\nImage\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'example_layout'\n)\ndoc_type\n=\nmodels\n.\nForeignKey\n(\nDocumentType\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'layouts'\n)\nclass\nLayoutField\n(\nmodels\n.\nModel\n):\nuuid\n=\nmodels\n.\nUUIDField\n(\ndefault\n=\nuuid\n.\nuuid4\n,\nprimary_key\n=\nTrue\n,\neditable\n=\nFalse\n)\ncore_field_type\n=\nmodels\n.\nForeignKey\n(\nCoreFieldType\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'layout_fields'\n)\nlayout\n=\nmodels\n.\nForeignKey\n(\nExampleLayout\n,\non_delete\n=\nmodels\n.\nCASCADE\n,\nrelated_name\n=\n'fields'\n)\nbounding_box\n=\nBoundingBoxCharField\n()\ntranscription\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n128\n)",
    "issue_closed_at": "2018-06-17T01:51:24",
    "base_commit": "e95008f2411d50929873f634c3e14ebac811fd28",
    "changes": [
      {
        "file": "django/db/backends/base/schema.py",
        "type": "function",
        "name": "_alter_field",
        "class_name": "BaseDatabaseSchemaEditor",
        "code": "def _alter_field(self, model, old_field, new_field, old_type, new_type,\n                     old_db_params, new_db_params, strict=False):\n        \"\"\"Perform a \"physical\" (non-ManyToMany) field update.\"\"\"\n        # Drop any FK constraints, we'll remake them later\n        fks_dropped = set()\n        if old_field.remote_field and old_field.db_constraint:\n            fk_names = self._constraint_names(model, [old_field.column], foreign_key=True)\n            if strict and len(fk_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of foreign key constraints for %s.%s\" % (\n                    len(fk_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for fk_name in fk_names:\n                fks_dropped.add((old_field.column,))\n                self.execute(self._delete_constraint_sql(self.sql_delete_fk, model, fk_name))\n        # Has unique been removed?\n        if old_field.unique and (not new_field.unique or (not old_field.primary_key and new_field.primary_key)):\n            # Find the unique constraint for this field\n            constraint_names = self._constraint_names(model, [old_field.column], unique=True, primary_key=False)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of unique constraints for %s.%s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_name))\n        # Drop incoming FK constraints if the field is a primary key or unique,\n        # which might be a to_field target, and things are going to change.\n        drop_foreign_keys = (\n            (\n                (old_field.primary_key and new_field.primary_key) or\n                (old_field.unique and new_field.unique)\n            ) and old_type != new_type\n        )\n        if drop_foreign_keys:\n            # '_meta.related_field' also contains M2M reverse fields, these\n            # will be filtered out\n            for _old_rel, new_rel in _related_non_m2m_objects(old_field, new_field):\n                rel_fk_names = self._constraint_names(\n                    new_rel.related_model, [new_rel.field.column], foreign_key=True\n                )\n                for fk_name in rel_fk_names:\n                    self.execute(self._delete_constraint_sql(self.sql_delete_fk, new_rel.related_model, fk_name))\n        # Removed an index? (no strict check, as multiple indexes are possible)\n        # Remove indexes if db_index switched to False or a unique constraint\n        # will now be used in lieu of an index. The following lines from the\n        # truth table show all True cases; the rest are False:\n        #\n        # old_field.db_index | old_field.unique | new_field.db_index | new_field.unique\n        # ------------------------------------------------------------------------------\n        # True               | False            | False              | False\n        # True               | False            | False              | True\n        # True               | False            | True               | True\n        if old_field.db_index and not old_field.unique and (not new_field.db_index or new_field.unique):\n            # Find the index for this field\n            meta_index_names = {index.name for index in model._meta.indexes}\n            # Retrieve only BTREE indexes since this is what's created with\n            # db_index=True.\n            index_names = self._constraint_names(model, [old_field.column], index=True, type_=Index.suffix)\n            for index_name in index_names:\n                if index_name not in meta_index_names:\n                    # The only way to check if an index was created with\n                    # db_index=True or with Index(['field'], name='foo')\n                    # is to look at its name (refs #28053).\n                    self.execute(self._delete_constraint_sql(self.sql_delete_index, model, index_name))\n        # Change check constraints?\n        if old_db_params['check'] != new_db_params['check'] and old_db_params['check']:\n            constraint_names = self._constraint_names(model, [old_field.column], check=True)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of check constraints for %s.%s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_check, model, constraint_name))\n        # Have they renamed the column?\n        if old_field.column != new_field.column:\n            self.execute(self._rename_field_sql(model._meta.db_table, old_field, new_field, new_type))\n            # Rename all references to the renamed column.\n            for sql in self.deferred_sql:\n                if isinstance(sql, Statement):\n                    sql.rename_column_references(model._meta.db_table, old_field.column, new_field.column)\n        # Next, start accumulating actions to do\n        actions = []\n        null_actions = []\n        post_actions = []\n        # Type change?\n        if old_type != new_type:\n            fragment, other_actions = self._alter_column_type_sql(model, old_field, new_field, new_type)\n            actions.append(fragment)\n            post_actions.extend(other_actions)\n        # When changing a column NULL constraint to NOT NULL with a given\n        # default value, we need to perform 4 steps:\n        #  1. Add a default for new incoming writes\n        #  2. Update existing NULL rows with new default\n        #  3. Replace NULL constraint with NOT NULL\n        #  4. Drop the default again.\n        # Default change?\n        old_default = self.effective_default(old_field)\n        new_default = self.effective_default(new_field)\n        needs_database_default = (\n            old_field.null and\n            not new_field.null and\n            old_default != new_default and\n            new_default is not None and\n            not self.skip_default(new_field)\n        )\n        if needs_database_default:\n            actions.append(self._alter_column_default_sql(model, old_field, new_field))\n        # Nullability change?\n        if old_field.null != new_field.null:\n            fragment = self._alter_column_null_sql(model, old_field, new_field)\n            if fragment:\n                null_actions.append(fragment)\n        # Only if we have a default and there is a change from NULL to NOT NULL\n        four_way_default_alteration = (\n            new_field.has_default() and\n            (old_field.null and not new_field.null)\n        )\n        if actions or null_actions:\n            if not four_way_default_alteration:\n                # If we don't have to do a 4-way default alteration we can\n                # directly run a (NOT) NULL alteration\n                actions = actions + null_actions\n            # Combine actions together if we can (e.g. postgres)\n            if self.connection.features.supports_combined_alters and actions:\n                sql, params = tuple(zip(*actions))\n                actions = [(\", \".join(sql), sum(params, []))]\n            # Apply those actions\n            for sql, params in actions:\n                self.execute(\n                    self.sql_alter_column % {\n                        \"table\": self.quote_name(model._meta.db_table),\n                        \"changes\": sql,\n                    },\n                    params,\n                )\n            if four_way_default_alteration:\n                # Update existing rows with default value\n                self.execute(\n                    self.sql_update_with_default % {\n                        \"table\": self.quote_name(model._meta.db_table),\n                        \"column\": self.quote_name(new_field.column),\n                        \"default\": \"%s\",\n                    },\n                    [new_default],\n                )\n                # Since we didn't run a NOT NULL change before we need to do it\n                # now\n                for sql, params in null_actions:\n                    self.execute(\n                        self.sql_alter_column % {\n                            \"table\": self.quote_name(model._meta.db_table),\n                            \"changes\": sql,\n                        },\n                        params,\n                    )\n        if post_actions:\n            for sql, params in post_actions:\n                self.execute(sql, params)\n        # If primary_key changed to False, delete the primary key constraint.\n        if old_field.primary_key and not new_field.primary_key:\n            self._delete_primary_key(model, strict)\n        # Added a unique?\n        if (not old_field.unique and new_field.unique) or (\n            old_field.primary_key and not new_field.primary_key and new_field.unique\n        ):\n            self.execute(self._create_unique_sql(model, [new_field.column]))\n        # Added an index? Add an index if db_index switched to True or a unique\n        # constraint will no longer be used in lieu of an index. The following\n        # lines from the truth table show all True cases; the rest are False:\n        #\n        # old_field.db_index | old_field.unique | new_field.db_index | new_field.unique\n        # ------------------------------------------------------------------------------\n        # False              | False            | True               | False\n        # False              | True             | True               | False\n        # True               | True             | True               | False\n        if (not old_field.db_index or old_field.unique) and new_field.db_index and not new_field.unique:\n            self.execute(self._create_index_sql(model, [new_field]))\n        # Type alteration on primary key? Then we need to alter the column\n        # referring to us.\n        rels_to_update = []\n        if old_field.primary_key and new_field.primary_key and old_type != new_type:\n            rels_to_update.extend(_related_non_m2m_objects(old_field, new_field))\n        # Changed to become primary key?\n        if not old_field.primary_key and new_field.primary_key:\n            # Make the new one\n            self.execute(\n                self.sql_create_pk % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"name\": self.quote_name(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_pk\")\n                    ),\n                    \"columns\": self.quote_name(new_field.column),\n                }\n            )\n            # Update all referencing columns\n            rels_to_update.extend(_related_non_m2m_objects(old_field, new_field))\n        # Handle our type alters on the other end of rels from the PK stuff above\n        for old_rel, new_rel in rels_to_update:\n            rel_db_params = new_rel.field.db_parameters(connection=self.connection)\n            rel_type = rel_db_params['type']\n            fragment, other_actions = self._alter_column_type_sql(\n                new_rel.related_model, old_rel.field, new_rel.field, rel_type\n            )\n            self.execute(\n                self.sql_alter_column % {\n                    \"table\": self.quote_name(new_rel.related_model._meta.db_table),\n                    \"changes\": fragment[0],\n                },\n                fragment[1],\n            )\n            for sql, params in other_actions:\n                self.execute(sql, params)\n        # Does it have a foreign key?\n        if (new_field.remote_field and\n                (fks_dropped or not old_field.remote_field or not old_field.db_constraint) and\n                new_field.db_constraint):\n            self.execute(self._create_fk_sql(model, new_field, \"_fk_%(to_table)s_%(to_column)s\"))\n        # Rebuild FKs that pointed to us if we previously had to drop them\n        if drop_foreign_keys:\n            for rel in new_field.model._meta.related_objects:\n                if _is_relevant_relation(rel, new_field) and rel.field.db_constraint:\n                    self.execute(self._create_fk_sql(rel.related_model, rel.field, \"_fk\"))\n        # Does it have check constraints we need to add?\n        if old_db_params['check'] != new_db_params['check'] and new_db_params['check']:\n            self.execute(\n                self.sql_create_check % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"name\": self.quote_name(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_check\")\n                    ),\n                    \"column\": self.quote_name(new_field.column),\n                    \"check\": new_db_params['check'],\n                }\n            )\n        # Drop the default if we need to\n        # (Django usually does not use in-database defaults)\n        if needs_database_default:\n            changes_sql, params = self._alter_column_default_sql(model, old_field, new_field, drop=True)\n            sql = self.sql_alter_column % {\n                \"table\": self.quote_name(model._meta.db_table),\n                \"changes\": changes_sql,\n            }\n            self.execute(sql, params)\n        # Reset connection if required\n        if self.connection.features.connection_persists_old_columns:\n            self.connection.close()"
      },
      {
        "file": "django/db/backends/base/schema.py",
        "type": "function",
        "name": "_alter_field",
        "class_name": "BaseDatabaseSchemaEditor",
        "code": "def _alter_field(self, model, old_field, new_field, old_type, new_type,\n                     old_db_params, new_db_params, strict=False):\n        \"\"\"Perform a \"physical\" (non-ManyToMany) field update.\"\"\"\n        # Drop any FK constraints, we'll remake them later\n        fks_dropped = set()\n        if old_field.remote_field and old_field.db_constraint:\n            fk_names = self._constraint_names(model, [old_field.column], foreign_key=True)\n            if strict and len(fk_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of foreign key constraints for %s.%s\" % (\n                    len(fk_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for fk_name in fk_names:\n                fks_dropped.add((old_field.column,))\n                self.execute(self._delete_constraint_sql(self.sql_delete_fk, model, fk_name))\n        # Has unique been removed?\n        if old_field.unique and (not new_field.unique or (not old_field.primary_key and new_field.primary_key)):\n            # Find the unique constraint for this field\n            constraint_names = self._constraint_names(model, [old_field.column], unique=True, primary_key=False)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of unique constraints for %s.%s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_name))\n        # Drop incoming FK constraints if the field is a primary key or unique,\n        # which might be a to_field target, and things are going to change.\n        drop_foreign_keys = (\n            (\n                (old_field.primary_key and new_field.primary_key) or\n                (old_field.unique and new_field.unique)\n            ) and old_type != new_type\n        )\n        if drop_foreign_keys:\n            # '_meta.related_field' also contains M2M reverse fields, these\n            # will be filtered out\n            for _old_rel, new_rel in _related_non_m2m_objects(old_field, new_field):\n                rel_fk_names = self._constraint_names(\n                    new_rel.related_model, [new_rel.field.column], foreign_key=True\n                )\n                for fk_name in rel_fk_names:\n                    self.execute(self._delete_constraint_sql(self.sql_delete_fk, new_rel.related_model, fk_name))\n        # Removed an index? (no strict check, as multiple indexes are possible)\n        # Remove indexes if db_index switched to False or a unique constraint\n        # will now be used in lieu of an index. The following lines from the\n        # truth table show all True cases; the rest are False:\n        #\n        # old_field.db_index | old_field.unique | new_field.db_index | new_field.unique\n        # ------------------------------------------------------------------------------\n        # True               | False            | False              | False\n        # True               | False            | False              | True\n        # True               | False            | True               | True\n        if old_field.db_index and not old_field.unique and (not new_field.db_index or new_field.unique):\n            # Find the index for this field\n            meta_index_names = {index.name for index in model._meta.indexes}\n            # Retrieve only BTREE indexes since this is what's created with\n            # db_index=True.\n            index_names = self._constraint_names(model, [old_field.column], index=True, type_=Index.suffix)\n            for index_name in index_names:\n                if index_name not in meta_index_names:\n                    # The only way to check if an index was created with\n                    # db_index=True or with Index(['field'], name='foo')\n                    # is to look at its name (refs #28053).\n                    self.execute(self._delete_constraint_sql(self.sql_delete_index, model, index_name))\n        # Change check constraints?\n        if old_db_params['check'] != new_db_params['check'] and old_db_params['check']:\n            constraint_names = self._constraint_names(model, [old_field.column], check=True)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of check constraints for %s.%s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_check, model, constraint_name))\n        # Have they renamed the column?\n        if old_field.column != new_field.column:\n            self.execute(self._rename_field_sql(model._meta.db_table, old_field, new_field, new_type))\n            # Rename all references to the renamed column.\n            for sql in self.deferred_sql:\n                if isinstance(sql, Statement):\n                    sql.rename_column_references(model._meta.db_table, old_field.column, new_field.column)\n        # Next, start accumulating actions to do\n        actions = []\n        null_actions = []\n        post_actions = []\n        # Type change?\n        if old_type != new_type:\n            fragment, other_actions = self._alter_column_type_sql(model, old_field, new_field, new_type)\n            actions.append(fragment)\n            post_actions.extend(other_actions)\n        # When changing a column NULL constraint to NOT NULL with a given\n        # default value, we need to perform 4 steps:\n        #  1. Add a default for new incoming writes\n        #  2. Update existing NULL rows with new default\n        #  3. Replace NULL constraint with NOT NULL\n        #  4. Drop the default again.\n        # Default change?\n        old_default = self.effective_default(old_field)\n        new_default = self.effective_default(new_field)\n        needs_database_default = (\n            old_field.null and\n            not new_field.null and\n            old_default != new_default and\n            new_default is not None and\n            not self.skip_default(new_field)\n        )\n        if needs_database_default:\n            actions.append(self._alter_column_default_sql(model, old_field, new_field))\n        # Nullability change?\n        if old_field.null != new_field.null:\n            fragment = self._alter_column_null_sql(model, old_field, new_field)\n            if fragment:\n                null_actions.append(fragment)\n        # Only if we have a default and there is a change from NULL to NOT NULL\n        four_way_default_alteration = (\n            new_field.has_default() and\n            (old_field.null and not new_field.null)\n        )\n        if actions or null_actions:\n            if not four_way_default_alteration:\n                # If we don't have to do a 4-way default alteration we can\n                # directly run a (NOT) NULL alteration\n                actions = actions + null_actions\n            # Combine actions together if we can (e.g. postgres)\n            if self.connection.features.supports_combined_alters and actions:\n                sql, params = tuple(zip(*actions))\n                actions = [(\", \".join(sql), sum(params, []))]\n            # Apply those actions\n            for sql, params in actions:\n                self.execute(\n                    self.sql_alter_column % {\n                        \"table\": self.quote_name(model._meta.db_table),\n                        \"changes\": sql,\n                    },\n                    params,\n                )\n            if four_way_default_alteration:\n                # Update existing rows with default value\n                self.execute(\n                    self.sql_update_with_default % {\n                        \"table\": self.quote_name(model._meta.db_table),\n                        \"column\": self.quote_name(new_field.column),\n                        \"default\": \"%s\",\n                    },\n                    [new_default],\n                )\n                # Since we didn't run a NOT NULL change before we need to do it\n                # now\n                for sql, params in null_actions:\n                    self.execute(\n                        self.sql_alter_column % {\n                            \"table\": self.quote_name(model._meta.db_table),\n                            \"changes\": sql,\n                        },\n                        params,\n                    )\n        if post_actions:\n            for sql, params in post_actions:\n                self.execute(sql, params)\n        # If primary_key changed to False, delete the primary key constraint.\n        if old_field.primary_key and not new_field.primary_key:\n            self._delete_primary_key(model, strict)\n        # Added a unique?\n        if (not old_field.unique and new_field.unique) or (\n            old_field.primary_key and not new_field.primary_key and new_field.unique\n        ):\n            self.execute(self._create_unique_sql(model, [new_field.column]))\n        # Added an index? Add an index if db_index switched to True or a unique\n        # constraint will no longer be used in lieu of an index. The following\n        # lines from the truth table show all True cases; the rest are False:\n        #\n        # old_field.db_index | old_field.unique | new_field.db_index | new_field.unique\n        # ------------------------------------------------------------------------------\n        # False              | False            | True               | False\n        # False              | True             | True               | False\n        # True               | True             | True               | False\n        if (not old_field.db_index or old_field.unique) and new_field.db_index and not new_field.unique:\n            self.execute(self._create_index_sql(model, [new_field]))\n        # Type alteration on primary key? Then we need to alter the column\n        # referring to us.\n        rels_to_update = []\n        if old_field.primary_key and new_field.primary_key and old_type != new_type:\n            rels_to_update.extend(_related_non_m2m_objects(old_field, new_field))\n        # Changed to become primary key?\n        if not old_field.primary_key and new_field.primary_key:\n            # Make the new one\n            self.execute(\n                self.sql_create_pk % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"name\": self.quote_name(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_pk\")\n                    ),\n                    \"columns\": self.quote_name(new_field.column),\n                }\n            )\n            # Update all referencing columns\n            rels_to_update.extend(_related_non_m2m_objects(old_field, new_field))\n        # Handle our type alters on the other end of rels from the PK stuff above\n        for old_rel, new_rel in rels_to_update:\n            rel_db_params = new_rel.field.db_parameters(connection=self.connection)\n            rel_type = rel_db_params['type']\n            fragment, other_actions = self._alter_column_type_sql(\n                new_rel.related_model, old_rel.field, new_rel.field, rel_type\n            )\n            self.execute(\n                self.sql_alter_column % {\n                    \"table\": self.quote_name(new_rel.related_model._meta.db_table),\n                    \"changes\": fragment[0],\n                },\n                fragment[1],\n            )\n            for sql, params in other_actions:\n                self.execute(sql, params)\n        # Does it have a foreign key?\n        if (new_field.remote_field and\n                (fks_dropped or not old_field.remote_field or not old_field.db_constraint) and\n                new_field.db_constraint):\n            self.execute(self._create_fk_sql(model, new_field, \"_fk_%(to_table)s_%(to_column)s\"))\n        # Rebuild FKs that pointed to us if we previously had to drop them\n        if drop_foreign_keys:\n            for rel in new_field.model._meta.related_objects:\n                if _is_relevant_relation(rel, new_field) and rel.field.db_constraint:\n                    self.execute(self._create_fk_sql(rel.related_model, rel.field, \"_fk\"))\n        # Does it have check constraints we need to add?\n        if old_db_params['check'] != new_db_params['check'] and new_db_params['check']:\n            self.execute(\n                self.sql_create_check % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"name\": self.quote_name(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_check\")\n                    ),\n                    \"column\": self.quote_name(new_field.column),\n                    \"check\": new_db_params['check'],\n                }\n            )\n        # Drop the default if we need to\n        # (Django usually does not use in-database defaults)\n        if needs_database_default:\n            changes_sql, params = self._alter_column_default_sql(model, old_field, new_field, drop=True)\n            sql = self.sql_alter_column % {\n                \"table\": self.quote_name(model._meta.db_table),\n                \"changes\": changes_sql,\n            }\n            self.execute(sql, params)\n        # Reset connection if required\n        if self.connection.features.connection_persists_old_columns:\n            self.connection.close()"
      },
      {
        "file": "django/db/backends/base/schema.py",
        "type": "function",
        "name": "_alter_field",
        "class_name": "BaseDatabaseSchemaEditor",
        "code": "def _alter_field(self, model, old_field, new_field, old_type, new_type,\n                     old_db_params, new_db_params, strict=False):\n        \"\"\"Perform a \"physical\" (non-ManyToMany) field update.\"\"\"\n        # Drop any FK constraints, we'll remake them later\n        fks_dropped = set()\n        if old_field.remote_field and old_field.db_constraint:\n            fk_names = self._constraint_names(model, [old_field.column], foreign_key=True)\n            if strict and len(fk_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of foreign key constraints for %s.%s\" % (\n                    len(fk_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for fk_name in fk_names:\n                fks_dropped.add((old_field.column,))\n                self.execute(self._delete_constraint_sql(self.sql_delete_fk, model, fk_name))\n        # Has unique been removed?\n        if old_field.unique and (not new_field.unique or (not old_field.primary_key and new_field.primary_key)):\n            # Find the unique constraint for this field\n            constraint_names = self._constraint_names(model, [old_field.column], unique=True, primary_key=False)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of unique constraints for %s.%s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_name))\n        # Drop incoming FK constraints if the field is a primary key or unique,\n        # which might be a to_field target, and things are going to change.\n        drop_foreign_keys = (\n            (\n                (old_field.primary_key and new_field.primary_key) or\n                (old_field.unique and new_field.unique)\n            ) and old_type != new_type\n        )\n        if drop_foreign_keys:\n            # '_meta.related_field' also contains M2M reverse fields, these\n            # will be filtered out\n            for _old_rel, new_rel in _related_non_m2m_objects(old_field, new_field):\n                rel_fk_names = self._constraint_names(\n                    new_rel.related_model, [new_rel.field.column], foreign_key=True\n                )\n                for fk_name in rel_fk_names:\n                    self.execute(self._delete_constraint_sql(self.sql_delete_fk, new_rel.related_model, fk_name))\n        # Removed an index? (no strict check, as multiple indexes are possible)\n        # Remove indexes if db_index switched to False or a unique constraint\n        # will now be used in lieu of an index. The following lines from the\n        # truth table show all True cases; the rest are False:\n        #\n        # old_field.db_index | old_field.unique | new_field.db_index | new_field.unique\n        # ------------------------------------------------------------------------------\n        # True               | False            | False              | False\n        # True               | False            | False              | True\n        # True               | False            | True               | True\n        if old_field.db_index and not old_field.unique and (not new_field.db_index or new_field.unique):\n            # Find the index for this field\n            meta_index_names = {index.name for index in model._meta.indexes}\n            # Retrieve only BTREE indexes since this is what's created with\n            # db_index=True.\n            index_names = self._constraint_names(model, [old_field.column], index=True, type_=Index.suffix)\n            for index_name in index_names:\n                if index_name not in meta_index_names:\n                    # The only way to check if an index was created with\n                    # db_index=True or with Index(['field'], name='foo')\n                    # is to look at its name (refs #28053).\n                    self.execute(self._delete_constraint_sql(self.sql_delete_index, model, index_name))\n        # Change check constraints?\n        if old_db_params['check'] != new_db_params['check'] and old_db_params['check']:\n            constraint_names = self._constraint_names(model, [old_field.column], check=True)\n            if strict and len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of check constraints for %s.%s\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    old_field.column,\n                ))\n            for constraint_name in constraint_names:\n                self.execute(self._delete_constraint_sql(self.sql_delete_check, model, constraint_name))\n        # Have they renamed the column?\n        if old_field.column != new_field.column:\n            self.execute(self._rename_field_sql(model._meta.db_table, old_field, new_field, new_type))\n            # Rename all references to the renamed column.\n            for sql in self.deferred_sql:\n                if isinstance(sql, Statement):\n                    sql.rename_column_references(model._meta.db_table, old_field.column, new_field.column)\n        # Next, start accumulating actions to do\n        actions = []\n        null_actions = []\n        post_actions = []\n        # Type change?\n        if old_type != new_type:\n            fragment, other_actions = self._alter_column_type_sql(model, old_field, new_field, new_type)\n            actions.append(fragment)\n            post_actions.extend(other_actions)\n        # When changing a column NULL constraint to NOT NULL with a given\n        # default value, we need to perform 4 steps:\n        #  1. Add a default for new incoming writes\n        #  2. Update existing NULL rows with new default\n        #  3. Replace NULL constraint with NOT NULL\n        #  4. Drop the default again.\n        # Default change?\n        old_default = self.effective_default(old_field)\n        new_default = self.effective_default(new_field)\n        needs_database_default = (\n            old_field.null and\n            not new_field.null and\n            old_default != new_default and\n            new_default is not None and\n            not self.skip_default(new_field)\n        )\n        if needs_database_default:\n            actions.append(self._alter_column_default_sql(model, old_field, new_field))\n        # Nullability change?\n        if old_field.null != new_field.null:\n            fragment = self._alter_column_null_sql(model, old_field, new_field)\n            if fragment:\n                null_actions.append(fragment)\n        # Only if we have a default and there is a change from NULL to NOT NULL\n        four_way_default_alteration = (\n            new_field.has_default() and\n            (old_field.null and not new_field.null)\n        )\n        if actions or null_actions:\n            if not four_way_default_alteration:\n                # If we don't have to do a 4-way default alteration we can\n                # directly run a (NOT) NULL alteration\n                actions = actions + null_actions\n            # Combine actions together if we can (e.g. postgres)\n            if self.connection.features.supports_combined_alters and actions:\n                sql, params = tuple(zip(*actions))\n                actions = [(\", \".join(sql), sum(params, []))]\n            # Apply those actions\n            for sql, params in actions:\n                self.execute(\n                    self.sql_alter_column % {\n                        \"table\": self.quote_name(model._meta.db_table),\n                        \"changes\": sql,\n                    },\n                    params,\n                )\n            if four_way_default_alteration:\n                # Update existing rows with default value\n                self.execute(\n                    self.sql_update_with_default % {\n                        \"table\": self.quote_name(model._meta.db_table),\n                        \"column\": self.quote_name(new_field.column),\n                        \"default\": \"%s\",\n                    },\n                    [new_default],\n                )\n                # Since we didn't run a NOT NULL change before we need to do it\n                # now\n                for sql, params in null_actions:\n                    self.execute(\n                        self.sql_alter_column % {\n                            \"table\": self.quote_name(model._meta.db_table),\n                            \"changes\": sql,\n                        },\n                        params,\n                    )\n        if post_actions:\n            for sql, params in post_actions:\n                self.execute(sql, params)\n        # If primary_key changed to False, delete the primary key constraint.\n        if old_field.primary_key and not new_field.primary_key:\n            self._delete_primary_key(model, strict)\n        # Added a unique?\n        if (not old_field.unique and new_field.unique) or (\n            old_field.primary_key and not new_field.primary_key and new_field.unique\n        ):\n            self.execute(self._create_unique_sql(model, [new_field.column]))\n        # Added an index? Add an index if db_index switched to True or a unique\n        # constraint will no longer be used in lieu of an index. The following\n        # lines from the truth table show all True cases; the rest are False:\n        #\n        # old_field.db_index | old_field.unique | new_field.db_index | new_field.unique\n        # ------------------------------------------------------------------------------\n        # False              | False            | True               | False\n        # False              | True             | True               | False\n        # True               | True             | True               | False\n        if (not old_field.db_index or old_field.unique) and new_field.db_index and not new_field.unique:\n            self.execute(self._create_index_sql(model, [new_field]))\n        # Type alteration on primary key? Then we need to alter the column\n        # referring to us.\n        rels_to_update = []\n        if old_field.primary_key and new_field.primary_key and old_type != new_type:\n            rels_to_update.extend(_related_non_m2m_objects(old_field, new_field))\n        # Changed to become primary key?\n        if not old_field.primary_key and new_field.primary_key:\n            # Make the new one\n            self.execute(\n                self.sql_create_pk % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"name\": self.quote_name(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_pk\")\n                    ),\n                    \"columns\": self.quote_name(new_field.column),\n                }\n            )\n            # Update all referencing columns\n            rels_to_update.extend(_related_non_m2m_objects(old_field, new_field))\n        # Handle our type alters on the other end of rels from the PK stuff above\n        for old_rel, new_rel in rels_to_update:\n            rel_db_params = new_rel.field.db_parameters(connection=self.connection)\n            rel_type = rel_db_params['type']\n            fragment, other_actions = self._alter_column_type_sql(\n                new_rel.related_model, old_rel.field, new_rel.field, rel_type\n            )\n            self.execute(\n                self.sql_alter_column % {\n                    \"table\": self.quote_name(new_rel.related_model._meta.db_table),\n                    \"changes\": fragment[0],\n                },\n                fragment[1],\n            )\n            for sql, params in other_actions:\n                self.execute(sql, params)\n        # Does it have a foreign key?\n        if (new_field.remote_field and\n                (fks_dropped or not old_field.remote_field or not old_field.db_constraint) and\n                new_field.db_constraint):\n            self.execute(self._create_fk_sql(model, new_field, \"_fk_%(to_table)s_%(to_column)s\"))\n        # Rebuild FKs that pointed to us if we previously had to drop them\n        if drop_foreign_keys:\n            for rel in new_field.model._meta.related_objects:\n                if _is_relevant_relation(rel, new_field) and rel.field.db_constraint:\n                    self.execute(self._create_fk_sql(rel.related_model, rel.field, \"_fk\"))\n        # Does it have check constraints we need to add?\n        if old_db_params['check'] != new_db_params['check'] and new_db_params['check']:\n            self.execute(\n                self.sql_create_check % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"name\": self.quote_name(\n                        self._create_index_name(model._meta.db_table, [new_field.column], suffix=\"_check\")\n                    ),\n                    \"column\": self.quote_name(new_field.column),\n                    \"check\": new_db_params['check'],\n                }\n            )\n        # Drop the default if we need to\n        # (Django usually does not use in-database defaults)\n        if needs_database_default:\n            changes_sql, params = self._alter_column_default_sql(model, old_field, new_field, drop=True)\n            sql = self.sql_alter_column % {\n                \"table\": self.quote_name(model._meta.db_table),\n                \"changes\": changes_sql,\n            }\n            self.execute(sql, params)\n        # Reset connection if required\n        if self.connection.features.connection_persists_old_columns:\n            self.connection.close()"
      },
      {
        "file": "django/db/backends/base/schema.py",
        "type": "function",
        "name": "_field_indexes_sql",
        "class_name": "BaseDatabaseSchemaEditor",
        "code": "def _field_indexes_sql(self, model, field):\n        \"\"\"\n        Return a list of all index SQL statements for the specified field.\n        \"\"\"\n        output = []\n        if self._field_should_be_indexed(model, field):\n            output.append(self._create_index_sql(model, [field]))\n        return output"
      },
      {
        "file": "django/db/backends/oracle/schema.py",
        "type": "function",
        "name": "_field_should_be_indexed",
        "class_name": "DatabaseSchemaEditor",
        "code": "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"
      }
    ]
  },
  "Justification": "Candidate C addresses a fundamental issue with migration handling, which is pivotal when dealing with Enum flags conversions. Both the CURRENT bug and Candidate C involve changing the attributes of fields within migrations, particularly focusing on the transition between field types and constraints. This comparison is valuable, as the Current bug requires an adaptation of values to a new serialization method, similar to the detailed process of changing a field to a primary key format in Candidate C. Furthermore, Candidate C's resolution may provide insights into handling exceptions thrown during migration operations, thus directly benefiting the debugging of the CURRENT bug.",
  "instance_id": "django__django-15996",
  "repo": "django/django",
  "created_at": "2022-08-25T04:49:14Z",
  "problem_statement": "Support for serialization of combination of Enum flags.\nDescription\n\t \n\t\t(last modified by Willem Van Onsem)\n\t \nIf we work with a field:\nregex_flags = models.IntegerField(default=re.UNICODE | re.IGNORECASE)\nThis is turned into a migration with:\ndefault=re.RegexFlag[None]\nThis is due to the fact that the EnumSerializer aims to work with the .name of the item, but if there is no single item for the given value, then there is no such name.\nIn that case, we can use enum._decompose to obtain a list of names, and create an expression to create the enum value by \"ORing\" the items together.\n",
  "patch": "diff --git a/django/db/migrations/serializer.py b/django/db/migrations/serializer.py\n--- a/django/db/migrations/serializer.py\n+++ b/django/db/migrations/serializer.py\n@@ -16,7 +16,7 @@\n from django.db.migrations.operations.base import Operation\n from django.db.migrations.utils import COMPILED_REGEX_TYPE, RegexObject\n from django.utils.functional import LazyObject, Promise\n-from django.utils.version import get_docs_version\n+from django.utils.version import PY311, get_docs_version\n \n \n class BaseSerializer:\n@@ -125,8 +125,21 @@ class EnumSerializer(BaseSerializer):\n     def serialize(self):\n         enum_class = self.value.__class__\n         module = enum_class.__module__\n+        if issubclass(enum_class, enum.Flag):\n+            if PY311:\n+                members = list(self.value)\n+            else:\n+                members, _ = enum._decompose(enum_class, self.value)\n+                members = reversed(members)\n+        else:\n+            members = (self.value,)\n         return (\n-            \"%s.%s[%r]\" % (module, enum_class.__qualname__, self.value.name),\n+            \" | \".join(\n+                [\n+                    f\"{module}.{enum_class.__qualname__}[{item.name!r}]\"\n+                    for item in members\n+                ]\n+            ),\n             {\"import %s\" % module},\n         )\n \n"
}