{
    "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 A is the most relevant because it deals with issues arising from the removal of `unique_together` constraints in migrations, which parallels the issue at hand with `index_together`. Both reports revolve around migration failures in Django, especially concerning indexes and unique constraints across different Django versions and databases. The structural similarity in their stack traces points to similar error handling in the migration process. Moreover, the patch in candidate A specifically addresses modifications to the underlying schema editor used in migrations, which could provide insights into handling the current bug's constraints more effectively."
}