{
  "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 as it discusses issues related to migrations and foreign key constraints in MySQL, which directly relates to the CURRENT bug involving AlterOrderWithRespectTo and AddIndex operations. Both bugs involve migration processes that truck with the ordering and indexing of fields, indicating similar structural and module issues. The documented fix for Candidate D demonstrates a critical resolution approach that could provide insights into alleviating the CURRENT bug's failure triggered by the order of operations in migrations.",
  "instance_id": "django__django-13265",
  "repo": "django/django",
  "created_at": "2020-08-02T10:02:11Z",
  "problem_statement": "AlterOrderWithRespectTo() with ForeignKey crash when _order is included in Index().\nDescription\n\t\n\tclass Meta:\n\t\tdb_table = 'look_image'\n\t\torder_with_respect_to = 'look'\n\t\tindexes = [\n\t\t\tmodels.Index(fields=['look', '_order']),\n\t\t\tmodels.Index(fields=['created_at']),\n\t\t\tmodels.Index(fields=['updated_at']),\n\t\t]\nmigrations.CreateModel(\n\t\t\tname='LookImage',\n\t\t\tfields=[\n\t\t\t\t('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),\n\t\t\t\t('look', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='images', to='posts.Look', verbose_name='LOOK')),\n\t\t\t\t('image_url', models.URLField(blank=True, max_length=10000, null=True)),\n\t\t\t\t('image', models.ImageField(max_length=2000, upload_to='')),\n\t\t\t\t('deleted', models.DateTimeField(editable=False, null=True)),\n\t\t\t\t('created_at', models.DateTimeField(auto_now_add=True)),\n\t\t\t\t('updated_at', models.DateTimeField(auto_now=True)),\n\t\t\t],\n\t\t),\n\t\tmigrations.AddIndex(\n\t\t\tmodel_name='lookimage',\n\t\t\tindex=models.Index(fields=['look', '_order'], name='look_image_look_id_eaff30_idx'),\n\t\t),\n\t\tmigrations.AddIndex(\n\t\t\tmodel_name='lookimage',\n\t\t\tindex=models.Index(fields=['created_at'], name='look_image_created_f746cf_idx'),\n\t\t),\n\t\tmigrations.AddIndex(\n\t\t\tmodel_name='lookimage',\n\t\t\tindex=models.Index(fields=['updated_at'], name='look_image_updated_aceaf9_idx'),\n\t\t),\n\t\tmigrations.AlterOrderWithRespectTo(\n\t\t\tname='lookimage',\n\t\t\torder_with_respect_to='look',\n\t\t),\nI added orders_with_respect_to in new model class's Meta class and also made index for '_order' field by combining with other field. And a new migration file based on the model looks like the code above.\nThe problem is operation AlterOrderWithRespectTo after AddIndex of '_order' raising error because '_order' field had not been created yet.\nIt seems to be AlterOrderWithRespectTo has to proceed before AddIndex of '_order'.\n",
  "patch": "diff --git a/django/db/migrations/autodetector.py b/django/db/migrations/autodetector.py\n--- a/django/db/migrations/autodetector.py\n+++ b/django/db/migrations/autodetector.py\n@@ -182,12 +182,12 @@ def _detect_changes(self, convert_apps=None, graph=None):\n         self.generate_removed_fields()\n         self.generate_added_fields()\n         self.generate_altered_fields()\n+        self.generate_altered_order_with_respect_to()\n         self.generate_altered_unique_together()\n         self.generate_altered_index_together()\n         self.generate_added_indexes()\n         self.generate_added_constraints()\n         self.generate_altered_db_table()\n-        self.generate_altered_order_with_respect_to()\n \n         self._sort_migrations()\n         self._build_migration_list(graph)\n@@ -613,6 +613,18 @@ def generate_created_models(self):\n                     dependencies=list(set(dependencies)),\n                 )\n             # Generate other opns\n+            if order_with_respect_to:\n+                self.add_operation(\n+                    app_label,\n+                    operations.AlterOrderWithRespectTo(\n+                        name=model_name,\n+                        order_with_respect_to=order_with_respect_to,\n+                    ),\n+                    dependencies=[\n+                        (app_label, model_name, order_with_respect_to, True),\n+                        (app_label, model_name, None, True),\n+                    ]\n+                )\n             related_dependencies = [\n                 (app_label, model_name, name, True)\n                 for name in sorted(related_fields)\n@@ -654,19 +666,6 @@ def generate_created_models(self):\n                     ),\n                     dependencies=related_dependencies\n                 )\n-            if order_with_respect_to:\n-                self.add_operation(\n-                    app_label,\n-                    operations.AlterOrderWithRespectTo(\n-                        name=model_name,\n-                        order_with_respect_to=order_with_respect_to,\n-                    ),\n-                    dependencies=[\n-                        (app_label, model_name, order_with_respect_to, True),\n-                        (app_label, model_name, None, True),\n-                    ]\n-                )\n-\n             # Fix relationships if the model changed from a proxy model to a\n             # concrete model.\n             if (app_label, model_name) in self.old_proxy_keys:\n"
}