{
  "Selected_candidate": {
    "pr_number": 8926,
    "pr_title": "Fixed #28493 -- Made migrations autodetector find dependencies for model renaming.",
    "pr_body": "https://code.djangoproject.com/ticket/28493",
    "issue_id": 28493,
    "issue_title": "Foreign keys break on migration if models are renamed in a different app",
    "issue_body": "Steps to reproduce:\nCreate app\na\n, with a model called\nA\n. Run\nmakemigrations a\nCreate app\nb\n, with a model called\nB\n. Give\nB\na\nForeignKey\nto\nA\n. Run\nmakemigrations b\nRename model\nA\nto\nAa\n. Run\nmakemigrations a\nRun\nmigrate\nExpected behaviour:\nThe migration completes successfully\nActual behaviour:\nMigration\n0001\non\nb\nis run after migration\n0002\non\na\n, and the migration fails with the following error:\nraise ValueError('Related model %r cannot be resolved' % self.remote_field.model)\nValueError: Related model u'a.A' cannot be resolved",
    "issue_closed_at": "2017-09-04T14:40:14",
    "base_commit": "3f2b1d926bb3a72b4c3d2cd958455ebb9b4ca458",
    "changes": [
      {
        "file": "django/db/migrations/autodetector.py",
        "type": "function",
        "name": "generate_renamed_models",
        "class_name": "MigrationAutodetector",
        "code": "def generate_renamed_models(self):\n        \"\"\"\n        Find any renamed models, generate the operations for them, and remove\n        the old entry from the model lists. Must be run before other\n        model-level generation.\n        \"\"\"\n        self.renamed_models = {}\n        self.renamed_models_rel = {}\n        added_models = self.new_model_keys - self.old_model_keys\n        for app_label, model_name in sorted(added_models):\n            model_state = self.to_state.models[app_label, model_name]\n            model_fields_def = self.only_relation_agnostic_fields(model_state.fields)\n\n            removed_models = self.old_model_keys - self.new_model_keys\n            for rem_app_label, rem_model_name in removed_models:\n                if rem_app_label == app_label:\n                    rem_model_state = self.from_state.models[rem_app_label, rem_model_name]\n                    rem_model_fields_def = self.only_relation_agnostic_fields(rem_model_state.fields)\n                    if model_fields_def == rem_model_fields_def:\n                        if self.questioner.ask_rename_model(rem_model_state, model_state):\n                            self.add_operation(\n                                app_label,\n                                operations.RenameModel(\n                                    old_name=rem_model_state.name,\n                                    new_name=model_state.name,\n                                )\n                            )\n                            self.renamed_models[app_label, model_name] = rem_model_name\n                            renamed_models_rel_key = '%s.%s' % (rem_model_state.app_label, rem_model_state.name)\n                            self.renamed_models_rel[renamed_models_rel_key] = '%s.%s' % (\n                                model_state.app_label,\n                                model_state.name,\n                            )\n                            self.old_model_keys.remove((rem_app_label, rem_model_name))\n                            self.old_model_keys.add((app_label, model_name))\n                            break"
      }
    ]
  },
  "Justification": "Candidate E is the most helpful report as it specifically addresses an issue that arises when models are renamed in a different app, which closely relates to the CURRENT bug involving migration failures due to existing entries in the `auth_permission` table after a model has been renamed. Both bug reports are concerned with how Django handles migrations when the database schema has changed, and the fix described in Candidate E directly improves the migration process for scenarios similar to the one described in the CURRENT bug. These shared themes of model renaming and migration integrity make Candidate E particularly relevant for debugging the CURRENT bug.",
  "instance_id": "django__django-11283",
  "repo": "django/django",
  "created_at": "2019-04-26T07:02:50Z",
  "problem_statement": "Migration auth.0011_update_proxy_permissions fails for models recreated as a proxy.\nDescription\n\t \n\t\t(last modified by Mariusz Felisiak)\n\t \nI am trying to update my project to Django 2.2. When I launch python manage.py migrate, I get this error message when migration auth.0011_update_proxy_permissions is applying (full stacktrace is available ​here):\ndjango.db.utils.IntegrityError: duplicate key value violates unique constraint \"idx_18141_auth_permission_content_type_id_01ab375a_uniq\" DETAIL: Key (co.ntent_type_id, codename)=(12, add_agency) already exists.\nIt looks like the migration is trying to re-create already existing entries in the auth_permission table. At first I though it cloud because we recently renamed a model. But after digging and deleting the entries associated with the renamed model from our database in the auth_permission table, the problem still occurs with other proxy models.\nI tried to update directly from 2.0.13 and 2.1.8. The issues appeared each time. I also deleted my venv and recreated it without an effect.\nI searched for a ticket about this on the bug tracker but found nothing. I also posted this on ​django-users and was asked to report this here.\n",
  "patch": "diff --git a/django/contrib/auth/migrations/0011_update_proxy_permissions.py b/django/contrib/auth/migrations/0011_update_proxy_permissions.py\n--- a/django/contrib/auth/migrations/0011_update_proxy_permissions.py\n+++ b/django/contrib/auth/migrations/0011_update_proxy_permissions.py\n@@ -1,5 +1,18 @@\n-from django.db import migrations\n+import sys\n+\n+from django.core.management.color import color_style\n+from django.db import migrations, transaction\n from django.db.models import Q\n+from django.db.utils import IntegrityError\n+\n+WARNING = \"\"\"\n+    A problem arose migrating proxy model permissions for {old} to {new}.\n+\n+      Permission(s) for {new} already existed.\n+      Codenames Q: {query}\n+\n+    Ensure to audit ALL permissions for {old} and {new}.\n+\"\"\"\n \n \n def update_proxy_model_permissions(apps, schema_editor, reverse=False):\n@@ -7,6 +20,7 @@ def update_proxy_model_permissions(apps, schema_editor, reverse=False):\n     Update the content_type of proxy model permissions to use the ContentType\n     of the proxy model.\n     \"\"\"\n+    style = color_style()\n     Permission = apps.get_model('auth', 'Permission')\n     ContentType = apps.get_model('contenttypes', 'ContentType')\n     for Model in apps.get_models():\n@@ -24,10 +38,16 @@ def update_proxy_model_permissions(apps, schema_editor, reverse=False):\n         proxy_content_type = ContentType.objects.get_for_model(Model, for_concrete_model=False)\n         old_content_type = proxy_content_type if reverse else concrete_content_type\n         new_content_type = concrete_content_type if reverse else proxy_content_type\n-        Permission.objects.filter(\n-            permissions_query,\n-            content_type=old_content_type,\n-        ).update(content_type=new_content_type)\n+        try:\n+            with transaction.atomic():\n+                Permission.objects.filter(\n+                    permissions_query,\n+                    content_type=old_content_type,\n+                ).update(content_type=new_content_type)\n+        except IntegrityError:\n+            old = '{}_{}'.format(old_content_type.app_label, old_content_type.model)\n+            new = '{}_{}'.format(new_content_type.app_label, new_content_type.model)\n+            sys.stdout.write(style.WARNING(WARNING.format(old=old, new=new, query=permissions_query)))\n \n \n def revert_proxy_model_permissions(apps, schema_editor):\n"
}