{
  "Selected_candidate": {
    "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)"
      }
    ]
  },
  "Justification": "Candidate E is the most relevant because it deals with the unintended consequences of deleting foreign key objects, which is closely related to the issue of inconsistent results from the QuerySet.Delete method in the CURRENT bug report. Both bugs focus on how deletion operations affect related models, highlighting the importance of accurate deletion logic. The patch for Candidate E specifically addresses the problem of nullification of foreign keys during deletion, which can provide valuable insights into ensuring consistent deletion behavior in the CURRENT bug report as well.",
  "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"
}