{
  "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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)"
        }
      ]
    }
  ]
}