{
  "Selected_candidate": {
    "pr_number": 16509,
    "pr_title": "Fixed #34304 -- Made MySQL's SchemaEditor.remove_constraint() don't create foreign key index when unique constraint is ignored.",
    "pr_body": "Fixes ticket-34304.",
    "issue_id": 34304,
    "issue_title": "Adding and removing a conditional UniqueConstraint to ForeignKey multiple times crashes on MySQL",
    "issue_body": "Adding and removing a\nUniqueConstraint\nto\nForeignKey\nwith a\ncondition\nmore than once on MySQL will crash the schema editor an\nOperationalError\n, e.g.\ndjango.db.utils.OperationalError: (1061, \"Duplicate key name 'schema_book_author_id_c80c8297'\")\nTest in\ntests.schema.tests\n:\ndef test_unique_with_fk_and_condition_multiple_times(self):\n        \"\"\"\n        Tests adding and removing a unique constraint to ForeignKey\n        with a condition multiple times.\n        \"\"\"\n        with connection.schema_editor() as editor:\n            editor.create_model(Author)\n            editor.create_model(Book)\n        constraint = UniqueConstraint(\n            \"author\",\n            condition=Q(title__in=[\"tHGttG\", \"tRatEotU\"]),\n            name=\"book_author_condition_uniq\",\n        )\n\n        for i in range(2):\n            # Add constraint.\n            with connection.schema_editor() as editor:\n                editor.add_constraint(Book, constraint)\n                sql = constraint.create_sql(Book, editor)\n            book_table = Book._meta.db_table\n            constraints = self.get_constraints(book_table)\n            if connection.features.supports_partial_indexes:\n                self.assertIn(constraint.name, constraints)\n                self.assertIs(constraints[constraint.name][\"unique\"], True)\n                self.assertIn(\"WHERE %s IN\" % editor.quote_name(\"title\"), str(sql))\n            else:\n                self.assertNotIn(constraint.name, constraints)\n                self.assertIsNone(sql)\n            # Remove constraint.\n            with connection.schema_editor() as editor:\n                editor.remove_constraint(Book, constraint)\n            self.assertNotIn(constraint.name, self.get_constraints(book_table))\nThis is a regression in\nb731e8841558ee4caaba766c83f34ea9c7004f8b\n.\nUse case:\n​\nWagtail\nsupports multiple database backends. We have a migration that adds a unique constraint to a foreign key with a condition. In addition, we also have migrations that alter the field referenced in the condition. Before making an alter field operation, we need to remove the unique constraint to avoid violating the constraint, then re-add the constraint after the field has been altered. We have more than one migrations that do these operations, thus triggering the error. This issue only happens on MySQL,  specifically after the above commit was merged into Django. Even though conditional indexes aren't supported on MySQL, I believe it should just be ignored (as with previous releases) instead of crashing the schema editor.\nRelevant migrations in Wagtail:\n​\n0050_workflow_rejected_to_needs_changes\n: First instance of\nRemoveConstraint\nand\nAddConstraint\n​\n0060_fix_workflow_unique_constraint\n: another instance, this fails to run. (Though, I know this migration doesn't really do anything.)",
    "issue_closed_at": "2023-01-31T04:52:54",
    "base_commit": "40217d1a82b0c16cddba377325d12b2c253f402a",
    "changes": [
      {
        "file": "django/db/backends/mysql/schema.py",
        "type": "function",
        "name": "add_field",
        "class_name": "DatabaseSchemaEditor",
        "code": "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(\n                \"UPDATE %(table)s SET %(column)s = %%s\"\n                % {\n                    \"table\": self.quote_name(model._meta.db_table),\n                    \"column\": self.quote_name(field.column),\n                },\n                [effective_default],\n            )"
      }
    ]
  },
  "Justification": "Candidate A is the most relevant since it deals with the functionality of adding and removing constraints in Django migrations, closely related to the operations specified in the CURRENT bug report regarding AlterField optimization. Both bugs deal with migration operations in Django and involve the handling of fields, specifically how those fields are altered. The patch made to handle the duplicate key error brings insights into efficiently managing database schema changes, which could provide crucial perspective on understanding the optimizer's failure to handle multiple AlterField operations gracefully.",
  "instance_id": "django__django-16595",
  "repo": "django/django",
  "created_at": "2023-02-24T10:30:35Z",
  "problem_statement": "Migration optimizer does not reduce multiple AlterField\nDescription\n\t\nLet's consider the following operations: \noperations = [\n\tmigrations.AddField(\n\t\tmodel_name=\"book\",\n\t\tname=\"title\",\n\t\tfield=models.CharField(max_length=256, null=True),\n\t),\n\tmigrations.AlterField(\n\t\tmodel_name=\"book\",\n\t\tname=\"title\",\n\t\tfield=models.CharField(max_length=128, null=True),\n\t),\n\tmigrations.AlterField(\n\t\tmodel_name=\"book\",\n\t\tname=\"title\",\n\t\tfield=models.CharField(max_length=128, null=True, help_text=\"help\"),\n\t),\n\tmigrations.AlterField(\n\t\tmodel_name=\"book\",\n\t\tname=\"title\",\n\t\tfield=models.CharField(max_length=128, null=True, help_text=\"help\", default=None),\n\t),\n]\nIf I run the optimizer, I get only the AddField, as we could expect. However, if the AddField model is separated from the AlterField (e.g. because of a non-elidable migration, or inside a non-squashed migration), none of the AlterField are reduced:\noptimizer.optimize(operations[1:], \"books\") \n[<AlterField model_name='book', name='title', field=<django.db.models.fields.CharField>>,\n <AlterField model_name='book', name='title', field=<django.db.models.fields.CharField>>,\n <AlterField model_name='book', name='title', field=<django.db.models.fields.CharField>>]\nIndeed, the AlterField.reduce does not consider the the case where operation is also an AlterField. \nIs this behaviour intended? If so, could it be documented? \nOtherwise, would it make sense to add something like\n\t\tif isinstance(operation, AlterField) and self.is_same_field_operation(\n\t\t\toperation\n\t\t):\n\t\t\treturn [operation]\n",
  "patch": "diff --git a/django/db/migrations/operations/fields.py b/django/db/migrations/operations/fields.py\n--- a/django/db/migrations/operations/fields.py\n+++ b/django/db/migrations/operations/fields.py\n@@ -247,9 +247,9 @@ def migration_name_fragment(self):\n         return \"alter_%s_%s\" % (self.model_name_lower, self.name_lower)\n \n     def reduce(self, operation, app_label):\n-        if isinstance(operation, RemoveField) and self.is_same_field_operation(\n-            operation\n-        ):\n+        if isinstance(\n+            operation, (AlterField, RemoveField)\n+        ) and self.is_same_field_operation(operation):\n             return [operation]\n         elif (\n             isinstance(operation, RenameField)\n"
}