{
    "Selected_candidate": {
        "pr_number": 3782,
        "pr_title": "Fixed #24037 -- Prevented data loss possibility when changing Meta.managed.",
        "pr_body": "The migrations autodetector now issues AlterModelOptions operations for\nMeta.managed changes instead of DeleteModel + CreateModel.\n\nhttps://code.djangoproject.com/ticket/24037\n",
        "issue_id": 24037,
        "issue_title": "Migrating a legacy table results in data loss",
        "issue_body": "I created several models for our legacy database using\ninspectdb\nand hence all those models had their\nmanaged\nflag set to\nFalse\n. Now I need to make some changes to that model, hence I removed the\nmanaged\nflag and the generated migration wants to drop my table first and then create it again. Which leads to loss of all my data.\nSo I edited the migration file and removed the\nDeleteModel\noperation. And faked that migration to avoid the error raised by\nCreateModel\nas the table already exists.\nBut now I cannot go backward beyond that migration as the backward operation of\nCreateModel\nwill drop my table anyway and there is no way to make a migration irreversible or ask it Not to drop my table.\nWhat would help in such cases is a way to handle the backward migration. So that I can either choose not to drop my table or raise an exception to make the migration irreversible. Or at least make the\nCreateModel\noperation irreversible. I found that the\nOperation\nclass has a flag called\nreversible\n. But I could find no way to set it to\nFalse\n.\nFor now I've added a fake RunPython operation that returns None to make that migration irreversible, as suggested by Markus\n​\n/ here\n. But I'd love some way to handle it properly.",
        "issue_closed_at": "2014-12-23T13:25:44",
        "base_commit": "69ee7c8d76da72d1392ed2a2597ed094da05d57e",
        "changes": [
            {
                "file": "django/db/migrations/autodetector.py",
                "type": "function",
                "name": "generate_created_models",
                "class_name": "MigrationAutodetector",
                "code": "def generate_created_models(self):\n        \"\"\"\n        Find all new models (both managed and unmanaged) and make create\n        operations for them as well as separate operations to create any\n        foreign key or M2M relationships (we'll optimize these back in later\n        if we can).\n\n        We also defer any model options that refer to collections of fields\n        that might be deferred (e.g. unique_together, index_together).\n        \"\"\"\n        added_models = set(self.new_model_keys) - set(self.old_model_keys)\n        added_unmanaged_models = set(self.new_unmanaged_keys) - set(self.old_unmanaged_keys)\n        models = chain(\n            sorted(added_models, key=self.swappable_first_key, reverse=True),\n            sorted(added_unmanaged_models, key=self.swappable_first_key, reverse=True)\n        )\n        for app_label, model_name in models:\n            model_state = self.to_state.models[app_label, model_name]\n            model_opts = self.new_apps.get_model(app_label, model_name)._meta\n            # Gather related fields\n            related_fields = {}\n            primary_key_rel = None\n            for field in model_opts.local_fields:\n                if field.rel:\n                    if field.rel.to:\n                        if field.primary_key:\n                            primary_key_rel = field.rel.to\n                        elif not field.rel.parent_link:\n                            related_fields[field.name] = field\n                    # through will be none on M2Ms on swapped-out models;\n                    # we can treat lack of through as auto_created=True, though.\n                    if getattr(field.rel, \"through\", None) and not field.rel.through._meta.auto_created:\n                        related_fields[field.name] = field\n            for field in model_opts.local_many_to_many:\n                if field.rel.to:\n                    related_fields[field.name] = field\n                if getattr(field.rel, \"through\", None) and not field.rel.through._meta.auto_created:\n                    related_fields[field.name] = field\n            # Are there unique/index_together to defer?\n            unique_together = model_state.options.pop('unique_together', None)\n            index_together = model_state.options.pop('index_together', None)\n            order_with_respect_to = model_state.options.pop('order_with_respect_to', None)\n            # Depend on the deletion of any possible proxy version of us\n            dependencies = [\n                (app_label, model_name, None, False),\n            ]\n            # Depend on all bases\n            for base in model_state.bases:\n                if isinstance(base, six.string_types) and \".\" in base:\n                    base_app_label, base_name = base.split(\".\", 1)\n                    dependencies.append((base_app_label, base_name, None, True))\n            # Depend on the other end of the primary key if it's a relation\n            if primary_key_rel:\n                dependencies.append((\n                    primary_key_rel._meta.app_label,\n                    primary_key_rel._meta.object_name,\n                    None,\n                    True\n                ))\n            # Generate creation operation\n            self.add_operation(\n                app_label,\n                operations.CreateModel(\n                    name=model_state.name,\n                    fields=[d for d in model_state.fields if d[0] not in related_fields],\n                    options=model_state.options,\n                    bases=model_state.bases,\n                    managers=model_state.managers,\n                ),\n                dependencies=dependencies,\n                beginning=True,\n            )\n\n            # Don't add operations which modify the database for unmanaged models\n            if not model_opts.managed:\n                continue\n\n            # Generate operations for each related field\n            for name, field in sorted(related_fields.items()):\n                # Account for FKs to swappable models\n                swappable_setting = getattr(field, 'swappable_setting', None)\n                if swappable_setting is not None:\n                    dep_app_label = \"__setting__\"\n                    dep_object_name = swappable_setting\n                else:\n                    dep_app_label = field.rel.to._meta.app_label\n                    dep_object_name = field.rel.to._meta.object_name\n                dependencies = [(dep_app_label, dep_object_name, None, True)]\n                if getattr(field.rel, \"through\", None) and not field.rel.through._meta.auto_created:\n                    dependencies.append((\n                        field.rel.through._meta.app_label,\n                        field.rel.through._meta.object_name,\n                        None,\n                        True\n                    ))\n                # Depend on our own model being created\n                dependencies.append((app_label, model_name, None, True))\n                # Make operation\n                self.add_operation(\n                    app_label,\n                    operations.AddField(\n                        model_name=model_name,\n                        name=name,\n                        field=field,\n                    ),\n                    dependencies=list(set(dependencies)),\n                )\n            # Generate other opns\n            related_dependencies = [\n                (app_label, model_name, name, True)\n                for name, field in sorted(related_fields.items())\n            ]\n            related_dependencies.append((app_label, model_name, None, True))\n            if unique_together:\n                self.add_operation(\n                    app_label,\n                    operations.AlterUniqueTogether(\n                        name=model_name,\n                        unique_together=unique_together,\n                    ),\n                    dependencies=related_dependencies\n                )\n            if index_together:\n                self.add_operation(\n                    app_label,\n                    operations.AlterIndexTogether(\n                        name=model_name,\n                        index_together=index_together,\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                )"
            },
            {
                "file": "django/db/migrations/autodetector.py",
                "type": "function",
                "name": "generate_deleted_models",
                "class_name": "MigrationAutodetector",
                "code": "def generate_deleted_models(self):\n        \"\"\"\n        Find all deleted models (managed and unmanaged) and make delete\n        operations for them as well as separate operations to delete any\n        foreign key or M2M relationships (we'll optimize these back in later\n        if we can).\n\n        We also bring forward removal of any model options that refer to\n        collections of fields - the inverse of generate_created_models().\n        \"\"\"\n        deleted_models = set(self.old_model_keys) - set(self.new_model_keys)\n        deleted_unmanaged_models = set(self.old_unmanaged_keys) - set(self.new_unmanaged_keys)\n        models = chain(sorted(deleted_models), sorted(deleted_unmanaged_models))\n        for app_label, model_name in models:\n            model_state = self.from_state.models[app_label, model_name]\n            model = self.old_apps.get_model(app_label, model_name)\n            if not model._meta.managed:\n                self.add_operation(\n                    app_label,\n                    operations.DeleteModel(\n                        name=model_state.name,\n                    ),\n                )\n                # Skip here, no need to handle fields for unmanaged models\n                continue\n\n            # Gather related fields\n            related_fields = {}\n            for field in model._meta.local_fields:\n                if field.rel:\n                    if field.rel.to:\n                        related_fields[field.name] = field\n                    # through will be none on M2Ms on swapped-out models;\n                    # we can treat lack of through as auto_created=True, though.\n                    if getattr(field.rel, \"through\", None) and not field.rel.through._meta.auto_created:\n                        related_fields[field.name] = field\n            for field in model._meta.local_many_to_many:\n                if field.rel.to:\n                    related_fields[field.name] = field\n                if getattr(field.rel, \"through\", None) and not field.rel.through._meta.auto_created:\n                    related_fields[field.name] = field\n            # Generate option removal first\n            unique_together = model_state.options.pop('unique_together', None)\n            index_together = model_state.options.pop('index_together', None)\n            if unique_together:\n                self.add_operation(\n                    app_label,\n                    operations.AlterUniqueTogether(\n                        name=model_name,\n                        unique_together=None,\n                    )\n                )\n            if index_together:\n                self.add_operation(\n                    app_label,\n                    operations.AlterIndexTogether(\n                        name=model_name,\n                        index_together=None,\n                    )\n                )\n            # Then remove each related field\n            for name, field in sorted(related_fields.items()):\n                self.add_operation(\n                    app_label,\n                    operations.RemoveField(\n                        model_name=model_name,\n                        name=name,\n                    )\n                )\n            # Finally, remove the model.\n            # This depends on both the removal/alteration of all incoming fields\n            # and the removal of all its own related fields, and if it's\n            # a through model the field that references it.\n            dependencies = []\n            for related_object in model._meta.get_all_related_objects():\n                dependencies.append((\n                    related_object.model._meta.app_label,\n                    related_object.model._meta.object_name,\n                    related_object.field.name,\n                    False,\n                ))\n                dependencies.append((\n                    related_object.model._meta.app_label,\n                    related_object.model._meta.object_name,\n                    related_object.field.name,\n                    \"alter\",\n                ))\n            for related_object in model._meta.get_all_related_many_to_many_objects():\n                dependencies.append((\n                    related_object.model._meta.app_label,\n                    related_object.model._meta.object_name,\n                    related_object.field.name,\n                    False,\n                ))\n            for name, field in sorted(related_fields.items()):\n                dependencies.append((app_label, model_name, name, False))\n            # We're referenced in another field's through=\n            through_user = self.through_users.get((app_label, model_state.name.lower()), None)\n            if through_user:\n                dependencies.append((through_user[0], through_user[1], through_user[2], False))\n            # Finally, make the operation, deduping any dependencies\n            self.add_operation(\n                app_label,\n                operations.DeleteModel(\n                    name=model_state.name,\n                ),\n                dependencies=list(set(dependencies)),\n            )"
            },
            {
                "file": "django/db/migrations/autodetector.py",
                "type": "function",
                "name": "generate_altered_options",
                "class_name": "MigrationAutodetector",
                "code": "def generate_altered_options(self):\n        \"\"\"\n        Works out if any non-schema-affecting options have changed and\n        makes an operation to represent them in state changes (in case Python\n        code in migrations needs them)\n        \"\"\"\n        models_to_check = self.kept_model_keys.union(self.kept_proxy_keys).union(self.kept_unmanaged_keys)\n        for app_label, model_name in sorted(models_to_check):\n            old_model_name = self.renamed_models.get((app_label, model_name), model_name)\n            old_model_state = self.from_state.models[app_label, old_model_name]\n            new_model_state = self.to_state.models[app_label, model_name]\n            old_options = dict(\n                option for option in old_model_state.options.items()\n                if option[0] in AlterModelOptions.ALTER_OPTION_KEYS\n            )\n            new_options = dict(\n                option for option in new_model_state.options.items()\n                if option[0] in AlterModelOptions.ALTER_OPTION_KEYS\n            )\n            if old_options != new_options:\n                self.add_operation(\n                    app_label,\n                    operations.AlterModelOptions(\n                        name=model_name,\n                        options=new_options,\n                    )\n                )"
            },
            {
                "file": "django/db/migrations/operations/models.py",
                "type": "class",
                "name": "AlterModelOptions",
                "code": "class AlterModelOptions(Operation):\n    \"\"\"\n    Sets new model options that don't directly affect the database schema\n    (like verbose_name, permissions, ordering). Python code in migrations\n    may still need them.\n    \"\"\"\n\n    # Model options we want to compare and preserve in an AlterModelOptions op\n    ALTER_OPTION_KEYS = [\n        \"get_latest_by\",\n        \"ordering\",\n        \"permissions\",\n        \"default_permissions\",\n        \"select_on_save\",\n        \"verbose_name\",\n        \"verbose_name_plural\",\n    ]\n\n    def __init__(self, name, options):\n        self.name = name\n        self.options = options\n\n    def deconstruct(self):\n        kwargs = {\n            'name': self.name,\n            'options': self.options,\n        }\n        return (\n            self.__class__.__name__,\n            [],\n            kwargs\n        )\n\n    def state_forwards(self, app_label, state):\n        model_state = state.models[app_label, self.name.lower()]\n        model_state.options = dict(model_state.options)\n        model_state.options.update(self.options)\n        for key in self.ALTER_OPTION_KEYS:\n            if key not in self.options and key in model_state.options:\n                del model_state.options[key]\n\n    def database_forwards(self, app_label, schema_editor, from_state, to_state):\n        pass\n\n    def database_backwards(self, app_label, schema_editor, from_state, to_state):\n        pass\n\n    def references_model(self, name, app_label=None):\n        return name.lower() == self.name.lower()\n\n    def describe(self):\n        return \"Change Meta options on %s\" % (self.name, )"
            }
        ]
    },
    "Justification": "Candidate A is the most helpful as it deals directly with model deletion, which is the central issue of the CURRENT bug report. It highlights how migrations can impact the management of model instances, specifically addressing the deletion process that doesn't clear primary keys. The patch in Candidate A prevents data loss during migrations, which parallels the required behavior expected from the delete operation in the CURRENT bug. The structural and module similarities while handling Django models and databases underline its relevance significantly, making it the best choice for guiding the resolution of the CURRENT bug."
}