{
  "instance_id": "django__django-12747",
  "repo": "django/django",
  "created_at": "2020-04-18T16:41:40Z",
  "problem_statement": "QuerySet.Delete - inconsistent result when zero objects deleted\nDescription\n\t\nThe result format of the QuerySet.Delete method is a tuple: (X, Y) \nX - is the total amount of deleted objects (including foreign key deleted objects)\nY - is a dictionary specifying counters of deleted objects for each specific model (the key is the _meta.label of the model and the value is counter of deleted objects of this model).\nExample: <class 'tuple'>: (2, {'my_app.FileAccess': 1, 'my_app.File': 1})\nWhen there are zero objects to delete in total - the result is inconsistent:\nFor models with foreign keys - the result will be: <class 'tuple'>: (0, {})\nFor \"simple\" models without foreign key - the result will be: <class 'tuple'>: (0, {'my_app.BlockLibrary': 0})\nI would expect there will be no difference between the two cases: Either both will have the empty dictionary OR both will have dictionary with model-label keys and zero value.\n",
  "patch": "diff --git a/django/db/models/deletion.py b/django/db/models/deletion.py\n--- a/django/db/models/deletion.py\n+++ b/django/db/models/deletion.py\n@@ -408,7 +408,8 @@ def delete(self):\n             # fast deletes\n             for qs in self.fast_deletes:\n                 count = qs._raw_delete(using=self.using)\n-                deleted_counter[qs.model._meta.label] += count\n+                if count:\n+                    deleted_counter[qs.model._meta.label] += count\n \n             # update fields\n             for model, instances_for_fieldvalues in self.field_updates.items():\n@@ -426,7 +427,8 @@ def delete(self):\n                 query = sql.DeleteQuery(model)\n                 pk_list = [obj.pk for obj in instances]\n                 count = query.delete_batch(pk_list, self.using)\n-                deleted_counter[model._meta.label] += count\n+                if count:\n+                    deleted_counter[model._meta.label] += count\n \n                 if not model._meta.auto_created:\n                     for obj in instances:\n",
  "similar_bug_items": [
    {
      "pr_number": 9985,
      "pr_title": "Fixed #23869 -- Made ModelAdmin.get_deleted_objects() use has_delete_permission() for permissions checking.",
      "pr_body": "Instead of checking user.has_perm get_deleted_objects checks\r\nhas_delete_permission.",
      "issue_id": 23869,
      "issue_title": "Make ModelAdmin.get_deleted_objects() use ModelAdmin.has_delete_permission() for permissions checking",
      "issue_body": "Considering\nget_deleted_objects\nin\ndjango.contrib.admin.utils\n, it checks for deleting permission using\nuser.has_perm(p)\n, bypassing the\nModelAdmin\nmethod\nhas_delete_permission\nassigned to the class for the\nModel\nto be deleted.\n​\nhttps://github.com/django/django/blob/stable/1.7.x/django/contrib/admin/utils.py#L141\nTherefore, even in a senario where\ndef has_delete_permission(self, request, obj=None):\n        return True\nthe user is not able to delete the object, if he doesn't have the permission explicitly assigned for the class by an auth backend.\nA tentative idea would be to replace\nif not user.has_perm(p):\nwith\nif admin_site._registry[obj.__class__].has_delete_permission(request, obj)\nThere are though two problems:\nrequest\nis not defined\nwhat about\nForeignKey\nobjects that ought to be deleted but they exist in the admin panel only as\nInlines\n? That is, they don't have their own\nModelAdmin\nclass assigned.",
      "issue_closed_at": "2018-06-15T09:54:48",
      "base_commit": "ec2c9c353113bb1db6e32ed3f0b6c28bc06ca2eb",
      "changes": [
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "get_deleted_objects",
          "class_name": "ModelAdmin",
          "code": "def get_deleted_objects(self, objs, request):\n        \"\"\"\n        Hook for customizing the delete process for the delete view and the\n        \"delete selected\" action.\n        \"\"\"\n        return get_deleted_objects(objs, request.user, self.admin_site)"
        },
        {
          "file": "django/contrib/admin/utils.py",
          "type": "line",
          "name": "line 2",
          "code": "import decimal\nfrom collections import defaultdict\n\nfrom django.contrib.auth import get_permission_codename\nfrom django.core.exceptions import FieldDoesNotExist\nfrom django.db import models, router\nfrom django.db.models.constants import LOOKUP_SEP"
        },
        {
          "file": "django/contrib/admin/utils.py",
          "type": "function",
          "name": "flatten_fieldsets",
          "class_name": null,
          "code": "def flatten_fieldsets(fieldsets):\n    \"\"\"Return a list of field names from an admin fieldsets structure.\"\"\"\n    field_names = []\n    for name, opts in fieldsets:\n        field_names.extend(\n            flatten(opts['fields'])\n        )\n    return field_names"
        },
        {
          "file": "django/contrib/admin/utils.py",
          "type": "function",
          "name": "get_deleted_objects",
          "class_name": null,
          "code": "def get_deleted_objects(objs, user, admin_site):\n    \"\"\"\n    Find all objects related to ``objs`` that should also be deleted. ``objs``\n    must be a homogeneous iterable of objects (e.g. a QuerySet).\n\n    Return a nested list of strings suitable for display in the\n    template with the ``unordered_list`` filter.\n    \"\"\"\n    try:\n        obj = objs[0]\n    except IndexError:\n        return [], {}, set(), []\n    else:\n        using = router.db_for_write(obj._meta.model)\n    collector = NestedObjects(using=using)\n    collector.collect(objs)\n    perms_needed = set()\n\n    def format_callback(obj):\n        has_admin = obj.__class__ in admin_site._registry\n        opts = obj._meta\n\n        no_edit_link = '%s: %s' % (capfirst(opts.verbose_name), obj)\n\n        if has_admin:\n            try:\n                admin_url = reverse('%s:%s_%s_change'\n                                    % (admin_site.name,\n                                       opts.app_label,\n                                       opts.model_name),\n                                    None, (quote(obj.pk),))\n            except NoReverseMatch:\n                # Change url doesn't exist -- don't display link to edit\n                return no_edit_link\n\n            if 'delete' in opts.default_permissions:\n                p = '%s.%s' % (opts.app_label, get_permission_codename('delete', opts))\n                if not user.has_perm(p):\n                    perms_needed.add(opts.verbose_name)\n            # Display a link to the admin page.\n            return format_html('{}: <a href=\"{}\">{}</a>',\n                               capfirst(opts.verbose_name),\n                               admin_url,\n                               obj)\n        else:\n            # Don't display link to edit, because it either has no\n            # admin or is edited inline.\n            return no_edit_link\n\n    to_delete = collector.nested(format_callback)\n\n    protected = [format_callback(obj) for obj in collector.protected]\n    model_count = {model._meta.verbose_name_plural: len(objs) for model, objs in collector.model_objs.items()}\n\n    return to_delete, model_count, perms_needed, protected"
        },
        {
          "file": "django/contrib/admin/utils.py",
          "type": "function",
          "name": "format_callback",
          "class_name": null,
          "code": "def format_callback(obj):\n        has_admin = obj.__class__ in admin_site._registry\n        opts = obj._meta\n\n        no_edit_link = '%s: %s' % (capfirst(opts.verbose_name), obj)\n\n        if has_admin:\n            try:\n                admin_url = reverse('%s:%s_%s_change'\n                                    % (admin_site.name,\n                                       opts.app_label,\n                                       opts.model_name),\n                                    None, (quote(obj.pk),))\n            except NoReverseMatch:\n                # Change url doesn't exist -- don't display link to edit\n                return no_edit_link\n\n            if 'delete' in opts.default_permissions:\n                p = '%s.%s' % (opts.app_label, get_permission_codename('delete', opts))\n                if not user.has_perm(p):\n                    perms_needed.add(opts.verbose_name)\n            # Display a link to the admin page.\n            return format_html('{}: <a href=\"{}\">{}</a>',\n                               capfirst(opts.verbose_name),\n                               admin_url,\n                               obj)\n        else:\n            # Don't display link to edit, because it either has no\n            # admin or is edited inline.\n            return no_edit_link"
        }
      ]
    },
    {
      "pr_number": 5826,
      "pr_title": "Fixed #25882 -- Prevented fast deletes matching no rows from crashing on MySQL.",
      "pr_body": "",
      "issue_id": 25882,
      "issue_title": "Deletion on ForeignKey raises TypeError",
      "issue_body": "Consider following model constellation:\nclass Catalog(models.Model):\n\tname = models.CharField(max_length=255)\n\t...\n\nclass Reader(models.Model):\n\tname = models.CharField(max_length=255)\n\tcatalog = models.ForeignKey(Catalog)\n\t...\n\nclass ReaderHasMedia(models.Model):\n\treader = models.ForeignKey(Reader)\n\t...\nNow in some cases I need to perform following command\nReaderHasMedia.objects.filter(catalog=Catalog.objects.get(pk=123)).delete()\nWith this command, I get following TypeError:\nTraceback (most recent call last):\n  File \"<console>\", line 1, in <module>\n  File \"/usr/local/lib/python2.7/site-packages/django/db/models/query.py\", line 600, in delete\n    deleted, _rows_count = collector.delete()\n  File \"/usr/local/lib/python2.7/site-packages/django/db/models/deletion.py\", line 293, in delete\n    deleted_counter[qs.model._meta.label] += count\nTypeError: unsupported operand type(s) for +=: 'int' and 'NoneType'\nWith Django 1.8.7 there wasn't a count variable in deletion.py and so I believe, this is a fresh bug with Django 1.9...",
      "issue_closed_at": "2015-12-14T12:12:22",
      "base_commit": "5233b70070f8979f41ca1da2c1b1d78c8e30944e",
      "changes": [
        {
          "file": "django/db/models/sql/subqueries.py",
          "type": "function",
          "name": "delete_qs",
          "class_name": "DeleteQuery",
          "code": "def delete_qs(self, query, using):\n        \"\"\"\n        Delete the queryset in one SQL query (if possible). For simple queries\n        this is done by copying the query.query.where to self.query, for\n        complex queries by using subquery.\n        \"\"\"\n        innerq = query.query\n        # Make sure the inner query has at least one table in use.\n        innerq.get_initial_alias()\n        # The same for our new query.\n        self.get_initial_alias()\n        innerq_used_tables = [t for t in innerq.tables\n                              if innerq.alias_refcount[t]]\n        if not innerq_used_tables or innerq_used_tables == self.tables:\n            # There is only the base table in use in the query.\n            self.where = innerq.where\n        else:\n            pk = query.model._meta.pk\n            if not connections[using].features.update_can_self_select:\n                # We can't do the delete using subquery.\n                values = list(query.values_list('pk', flat=True))\n                if not values:\n                    return\n                return self.delete_batch(values, using)\n            else:\n                innerq.clear_select_clause()\n                innerq.select = [\n                    pk.get_col(self.get_initial_alias())\n                ]\n                values = innerq\n            self.where = self.where_class()\n            self.add_q(Q(pk__in=values))\n        cursor = self.get_compiler(using).execute_sql(CURSOR)\n        return cursor.rowcount if cursor else 0"
        }
      ]
    },
    {
      "pr_number": 10773,
      "pr_title": "Fixed #30050 -- Fixed InlineModelAdmin.has_change_permission() being called with non-None obj during add.",
      "pr_body": "https://code.djangoproject.com/ticket/30050",
      "issue_id": 30050,
      "issue_title": "InlineModelAdmin.has_change_permission() incorrectly called with non-None obj during add",
      "issue_body": "Fine with Django 2.1.3, bug with Django 2.1.4.\nIf I have an admin Inline of a\nModel\n, with fk to a\nParentModel\n, when I try to add a new\nParentModel\nin the admin panel (\"/admin/foo/parentmodel/add/\"), the Inline\nhas_change_permission\nis called 3 times instead of 2, after this commit:\n​\nhttps://github.com/django/django/commit/27f5b0aff3442e5c25e84972dff4f5fe1edd4e68\nat line 1962\nhas_change_permission\nis passed an\nobj\nthat is not\nNone\n.\nBefore this commit,\nobj\nwas\nNone\nboth of the times, now it's\nNone\nthe first time, an empty instance of\nParentModel\nthe second time, and\nNone\nthe third time.\nLine 1962 should be something like:\npermission_obj = obj if change else None\nif not inline.has_change_permission(request, permission_obj):",
      "issue_closed_at": "2019-01-01T09:05:00",
      "base_commit": "0123b67f6b8304a5c32a0fe98f97ae506977454b",
      "changes": [
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "user_deleted_form",
          "class_name": "ModelAdmin",
          "code": "def user_deleted_form(request, obj, formset, index):\n                \"\"\"Return whether or not the user deleted the form.\"\"\"\n                return (\n                    inline.has_delete_permission(request, obj) and\n                    '{}-{}-DELETE'.format(formset.prefix, index) in request.POST\n                )"
        }
      ]
    },
    {
      "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, )"
        }
      ]
    },
    {
      "pr_number": 9585,
      "pr_title": "Fixed #29016 -- Prevent undesired nullification of foreign keys on foreign key deletion",
      "pr_body": "https://code.djangoproject.com/ticket/29016",
      "issue_id": 29016,
      "issue_title": "Reuse of UpdateQuery breaks some delete updates",
      "issue_body": "On a model A, when deleting a foreign key pointing to a model B, some other foreign key of the model A pointing to the same model B may be nullified.\nI have isolated this behaviour on a simple project:\nmodels.py:\nfrom\ndjango.db\nimport\nmodels\nclass\nChildModel\n(\nmodels\n.\nModel\n):\nname\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n200\n)\nclass\nParentModel\n(\nmodels\n.\nModel\n):\nname\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n200\n)\nchild_1\n=\nmodels\n.\nForeignKey\n(\nChildModel\n,\non_delete\n=\nmodels\n.\nSET_NULL\n,\nrelated_name\n=\n'parents_1'\n,\nnull\n=\nTrue\n)\nchild_2\n=\nmodels\n.\nForeignKey\n(\nChildModel\n,\non_delete\n=\nmodels\n.\nSET_NULL\n,\nrelated_name\n=\n'parents_2'\n,\nnull\n=\nTrue\n)\nDjango shell session:\nfrom\ntestapp.models\nimport\nParentModel\n,\nChildModel\nchild_1\n=\nChildModel\n.\nobjects\n.\ncreate\n(\nname\n=\n\"child_1\"\n)\nchild_2\n=\nChildModel\n.\nobjects\n.\ncreate\n(\nname\n=\n\"child_2\"\n)\nparent_1\n=\nParentModel\n.\nobjects\n.\ncreate\n(\nname\n=\n\"parent 1\"\n,\nchild_1\n=\nchild_1\n,\nchild_2\n=\nchild_2\n)\nparent_2\n=\nParentModel\n.\nobjects\n.\ncreate\n(\nname\n=\n\"parent 2\"\n,\nchild_1\n=\nchild_2\n,\nchild_2\n=\nchild_1\n)\nchild_1\n.\ndelete\n()\nparent_1\n=\nParentModel\n.\nobjects\n.\nget\n(\npk\n=\nparent_1\n.\npk\n)\nparent_2\n=\nParentModel\n.\nobjects\n.\nget\n(\npk\n=\nparent_2\n.\npk\n)\n# parent_1.child_2 and parent_2.child_1 should be normaly equal to child_2 but...\nparent_1\n.\nchild_2\nis\nnot\nNone\nand\nparent_2\n.\nchild_1\nis\nnot\nNone\n# False is returned\nThis simple project has been tested on an SQLite database. The same behaviour has been first discovered on a PostgreSQL database.\nA mis-reuse of an UpdateQuery seems to be the cause of this bug.\nAfter search on the django bug tracker I have found another issue with the same patch attached\n#28099\n.\nI have opened this new ticket because the issue seems to be more severe (I have experienced large data loss) and more general.\nThis issue has been found on version 1.11 and 2.0 of Django.\nI have created a new branch on my github account with patch and test associated:\n​\nhttps://github.com/Nimn/django/tree/ticket_29016",
      "issue_closed_at": "2018-01-13T12:09:01",
      "base_commit": "cea5fe94c6bb1b61e791f1375c246566c950b3e3",
      "changes": [
        {
          "file": "django/db/models/deletion.py",
          "type": "function",
          "name": "delete",
          "class_name": "Collector",
          "code": "def delete(self):\n        # sort instance collections\n        for model, instances in self.data.items():\n            self.data[model] = sorted(instances, key=attrgetter(\"pk\"))\n\n        # if possible, bring the models in an order suitable for databases that\n        # don't support transactions or cannot defer constraint checks until the\n        # end of a transaction.\n        self.sort()\n        # number of objects deleted for each model label\n        deleted_counter = Counter()\n\n        with transaction.atomic(using=self.using, savepoint=False):\n            # send pre_delete signals\n            for model, obj in self.instances_with_model():\n                if not model._meta.auto_created:\n                    signals.pre_delete.send(\n                        sender=model, instance=obj, using=self.using\n                    )\n\n            # fast deletes\n            for qs in self.fast_deletes:\n                count = qs._raw_delete(using=self.using)\n                deleted_counter[qs.model._meta.label] += count\n\n            # update fields\n            for model, instances_for_fieldvalues in self.field_updates.items():\n                query = sql.UpdateQuery(model)\n                for (field, value), instances in instances_for_fieldvalues.items():\n                    query.update_batch([obj.pk for obj in instances],\n                                       {field.name: value}, self.using)\n\n            # reverse instance collections\n            for instances in self.data.values():\n                instances.reverse()\n\n            # delete instances\n            for model, instances in self.data.items():\n                query = sql.DeleteQuery(model)\n                pk_list = [obj.pk for obj in instances]\n                count = query.delete_batch(pk_list, self.using)\n                deleted_counter[model._meta.label] += count\n\n                if not model._meta.auto_created:\n                    for obj in instances:\n                        signals.post_delete.send(\n                            sender=model, instance=obj, using=self.using\n                        )\n\n        # update collected instances\n        for instances_for_fieldvalues in self.field_updates.values():\n            for (field, value), instances in instances_for_fieldvalues.items():\n                for obj in instances:\n                    setattr(obj, field.attname, value)\n        for model, instances in self.data.items():\n            for instance in instances:\n                setattr(instance, model._meta.pk.attname, None)\n        return sum(deleted_counter.values()), dict(deleted_counter)"
        }
      ]
    }
  ]
}