{
  "Selected_candidate": {
    "pr_number": 4655,
    "pr_title": "Fixed #24757 -- Recreated MySQL index when needed during unique_together removal",
    "pr_body": "",
    "issue_id": 24757,
    "issue_title": "Removing unique_together constraint make Django 1.8 migration fail with MySQL",
    "issue_body": "Migrations containing\nmigrations.AlterUniqueTogether(name='xxx', unique_together=set([]),)\nmay fail on Django 1.8 but not on Django 1.7 if using the backend mysql.\nGiven the database engine is mysql and the following\nmodels.py\n:\nfrom django.db import models\n\n\nclass MyModelA(models.Model):\n    s = models.CharField(max_length=120)\n\n\nclass MyModelB(models.Model):\n    a = models.ForeignKey(MyModelA)\n    b = models.IntegerField()\nAnd the 2 following migrations:\n# -*- coding: utf-8 -*-\nfrom __future__ import unicode_literals\n\nfrom django.db import models, migrations\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n    ]\n\n    operations = [\n        migrations.CreateModel(\n            name='MyModelA',\n            fields=[\n                ('id', models.AutoField(serialize=False, auto_created=True, verbose_name='ID', primary_key=True)),\n                ('s', models.CharField(max_length=120)),\n            ],\n        ),\n        migrations.CreateModel(\n            name='MyModelB',\n            fields=[\n                ('id', models.AutoField(serialize=False, auto_created=True, verbose_name='ID', primary_key=True)),\n                ('b', models.IntegerField()),\n                ('a', models.ForeignKey(to='mysite.MyModelA')),\n            ],\n        ),\n        migrations.AlterUniqueTogether(\n            name='mymodelb',\n            unique_together=set([('a', 'b')]),\n        ),\n    ]\n# -*- coding: utf-8 -*-\nfrom __future__ import unicode_literals\n\nfrom django.db import models, migrations\n\n\nclass Migration(migrations.Migration):\n\n    dependencies = [\n        ('mysite', '0001_initial'),\n    ]\n\n    operations = [\n        migrations.AlterUniqueTogether(\n            name='mymodelb',\n            unique_together=set([]),\n        ),\n    ]\nRunning\n./manage.py migrate\nworks fine on Django 1.7, but on Django 1.8 I get the following error:\n$ ./manage.py migrate                \nOperations to perform:\n  Synchronize unmigrated apps: staticfiles, messages\n  Apply all migrations: contenttypes, auth, mysite, sessions, admin\nSynchronizing apps without migrations:\n  Creating tables...\n    Running deferred SQL...\n  Installing custom SQL...\nRunning migrations:\n  Rendering model states... DONE\n  [..]\n  Applying mysite.0001_initial... OK\n  Applying mysite.0002_auto...Traceback (most recent call last):\n  [..]\ndjango.db.utils.OperationalError: (1553, \"Cannot drop index 'mysite_mymodelb_a_id_b53c781c93aab9a_uniq': needed in a foreign key constraint\")\nThe same migrations work on psql, I did not have the chance to test it against sqlite.\nInspecting the generated sql I found Django 1.7 add an index in addition to the unique constraint and Django 1.8 does not.\nOn Django 1.7\nBEGIN;\nCREATE TABLE `mysite_mymodela` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `s` varchar(120) NOT NULL);\nCREATE TABLE `mysite_mymodelb` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `b` integer NOT NULL, `a_id` integer NOT NULL);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_48275010b87402ed_uniq` UNIQUE (`a_id`, `b`);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_4981d3920f0fef1e_fk_mysite_mymodela_id` FOREIGN KEY (`a_id`) REFERENCES `mysite_mymodela` (`id`);\nCREATE INDEX `mysite_mymodelb_e2b453f4` ON `mysite_mymodelb` (`a_id`);\n\nCOMMIT;\nNote the\nCREATE INDEX `mysite_mymodelb_e2b453f4` ON `mysite_mymodelb` (`a_id`);\n.\nAnd on Django 1.8\nBEGIN;\nCREATE TABLE `mysite_mymodela` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `s` varchar(120) NOT NULL);\nCREATE TABLE `mysite_mymodelb` (`id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY, `b` integer NOT NULL, `a_id` integer NOT NULL);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_784bb266d0e363ab_uniq` UNIQUE (`a_id`, `b`);\nALTER TABLE `mysite_mymodelb` ADD CONSTRAINT `mysite_mymodelb_a_id_50892c78e36678b8_fk_mysite_mymodela_id` FOREIGN KEY (`a_id`) REFERENCES `mysite_mymodela` (`id`);\n\nCOMMIT;\nI joined the full traceback.",
    "issue_closed_at": "2015-05-15T10:06:22",
    "base_commit": "0eaef8e527af556c0c517a64035929404cf94a39",
    "changes": [
      {
        "file": "django/db/backends/base/schema.py",
        "type": "function",
        "name": "alter_unique_together",
        "class_name": "BaseDatabaseSchemaEditor",
        "code": "def alter_unique_together(self, model, old_unique_together, new_unique_together):\n        \"\"\"\n        Deals with a model changing its unique_together.\n        Note: The input unique_togethers must be doubly-nested, not the single-\n        nested [\"foo\", \"bar\"] format.\n        \"\"\"\n        olds = set(tuple(fields) for fields in old_unique_together)\n        news = set(tuple(fields) for fields in new_unique_together)\n        # Deleted uniques\n        for fields in olds.difference(news):\n            columns = [model._meta.get_field(field).column for field in fields]\n            constraint_names = self._constraint_names(model, columns, unique=True)\n            if len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of constraints for %s(%s)\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    \", \".join(columns),\n                ))\n            self.execute(self._delete_constraint_sql(self.sql_delete_unique, model, constraint_names[0]))\n        # Created uniques\n        for fields in news.difference(olds):\n            columns = [model._meta.get_field(field).column for field in fields]\n            self.execute(self._create_unique_sql(model, columns))"
      },
      {
        "file": "django/db/backends/base/schema.py",
        "type": "function",
        "name": "alter_index_together",
        "class_name": "BaseDatabaseSchemaEditor",
        "code": "def alter_index_together(self, model, old_index_together, new_index_together):\n        \"\"\"\n        Deals with a model changing its index_together.\n        Note: The input index_togethers must be doubly-nested, not the single-\n        nested [\"foo\", \"bar\"] format.\n        \"\"\"\n        olds = set(tuple(fields) for fields in old_index_together)\n        news = set(tuple(fields) for fields in new_index_together)\n        # Deleted indexes\n        for fields in olds.difference(news):\n            columns = [model._meta.get_field(field).column for field in fields]\n            constraint_names = self._constraint_names(model, list(columns), index=True)\n            if len(constraint_names) != 1:\n                raise ValueError(\"Found wrong number (%s) of constraints for %s(%s)\" % (\n                    len(constraint_names),\n                    model._meta.db_table,\n                    \", \".join(columns),\n                ))\n            self.execute(self._delete_constraint_sql(self.sql_delete_index, model, constraint_names[0]))\n        # Created indexes\n        for field_names in news.difference(olds):\n            fields = [model._meta.get_field(field) for field in field_names]\n            self.execute(self._create_index_sql(model, fields, suffix=\"_idx\"))"
      },
      {
        "file": "django/db/backends/mysql/schema.py",
        "type": "function",
        "name": "_model_indexes_sql",
        "class_name": "DatabaseSchemaEditor",
        "code": "def _model_indexes_sql(self, model):\n        storage = self.connection.introspection.get_storage_engine(\n            self.connection.cursor(), model._meta.db_table\n        )\n        if storage == \"InnoDB\":\n            for field in model._meta.local_fields:\n                if field.db_index and not field.unique and field.get_internal_type() == \"ForeignKey\":\n                    # Temporary setting db_index to False (in memory) to disable\n                    # index creation for FKs (index automatically created by MySQL)\n                    field.db_index = False\n        return super(DatabaseSchemaEditor, self)._model_indexes_sql(model)"
      }
    ]
  },
  "Justification": "Candidate D is the most relevant to the CURRENT bug report because both issues involve the handling of unique constraints and indexes within migrations on a MySQL database. The CURRENT bug crashes when re-applying an index operation (RenameIndex), leading to a conflict, similar to how the removal of a unique_together constraint in Candidate D causes a migration to fail due to existing requirements in the index. Both cases suggest common underlying issues in managing database constraints during schema changes, especially with MySQL migrations, making the insights from Candidate D particularly useful for debugging the CURRENT issue.",
  "instance_id": "django__django-15695",
  "repo": "django/django",
  "created_at": "2022-05-16T07:58:51Z",
  "problem_statement": "RenameIndex() crashes when unnamed index is moving backward and forward.\nDescription\n\t\nRenameIndex() should restore the old auto-generated name when an unnamed index for unique_together is moving backward. Now re-applying RenameIndex() crashes. For example:\ntests/migrations/test_operations.py\ndiff --git a/tests/migrations/test_operations.py b/tests/migrations/test_operations.py\nindex cfd28b1b39..c0a55023bb 100644\n\t\t\t\t\t\n\t\t\t\t\t a\n\t\t\t\t \n\t\t\t\t\t\n\t\t\t\t\t b\n\t\t\t\t \n class OperationTests(OperationTestBase): \n29882988        with connection.schema_editor() as editor, self.assertNumQueries(0):\n29892989            operation.database_backwards(app_label, editor, new_state, project_state)\n29902990        self.assertIndexNameExists(table_name, \"new_pony_test_idx\")\n 2991        # Re-apply renaming.\n 2992        with connection.schema_editor() as editor:\n 2993            operation.database_forwards(app_label, editor, project_state, new_state)\n 2994        self.assertIndexNameExists(table_name, \"new_pony_test_idx\")\n29912995        # Deconstruction.\n29922996        definition = operation.deconstruct()\n29932997        self.assertEqual(definition[0], \"RenameIndex\")\ncrashes on PostgreSQL:\ndjango.db.utils.ProgrammingError: relation \"new_pony_test_idx\" already exists\n",
  "patch": "diff --git a/django/db/migrations/operations/models.py b/django/db/migrations/operations/models.py\n--- a/django/db/migrations/operations/models.py\n+++ b/django/db/migrations/operations/models.py\n@@ -960,6 +960,9 @@ def database_forwards(self, app_label, schema_editor, from_state, to_state):\n         else:\n             from_model_state = from_state.models[app_label, self.model_name_lower]\n             old_index = from_model_state.get_index_by_name(self.old_name)\n+        # Don't alter when the index name is not changed.\n+        if old_index.name == self.new_name:\n+            return\n \n         to_model_state = to_state.models[app_label, self.model_name_lower]\n         new_index = to_model_state.get_index_by_name(self.new_name)\n"
}