{
  "Selected_candidate": {
    "pr_number": 8626,
    "pr_title": "Fixed #26362 -- Update the inherited id field of an object when its ancestor changes",
    "pr_body": "https://code.djangoproject.com/ticket/26362",
    "issue_id": 26362,
    "issue_title": "Manually changing the parent of a child model is silently failing.",
    "issue_body": "To be honest, I have no idea if it is supposed to be allowed to set the parent via the parent_ptr, but it isn't working. If it isn't allowed, it should be reporting an error.\nExample:\nclass Parent(models.Model):\n    field1 = models.TextField(default=\"abc\")\n\nclass Child(Parent):\n    field2 = models.TextField(default=\"abc\")\n\ndef change_parent():\n    child = Child.objects.get(id=1)\n    new_parent = Parent.objects.create()\n\n    print child.id  # Prints 1\n    print child.parent_ptr.id  # Prints 1\n    print new_parent.id  # We'll say it is number 5\n    child.parent_ptr = new_parent\n\n    print child.id  # Prints 1\n    print child.parent_ptr.id  # Prints 5\n    child.save()\n\n    print child.id  # Prints 1\n    print child.parent_ptr.id  # Prints 1\nThis applies to the update function as well. I tried:\nChild.objects.filter(id=1).update(parent_ptr=new_parent)\n    child = Child.objects.get(id=1)\n\n    print child.id  # Prints 1\n    print child.parent_ptr.id  # Prints 1",
    "issue_closed_at": "2017-06-26T11:11:44",
    "base_commit": "fa283067c94546e38a7c4c3748c5fd31c58b7f22",
    "changes": [
      {
        "file": "django/db/models/fields/related_descriptors.py",
        "type": "function",
        "name": "get_object",
        "class_name": "ForwardOneToOneDescriptor",
        "code": "def get_object(self, instance):\n        if self.field.remote_field.parent_link:\n            deferred = instance.get_deferred_fields()\n            # Because it's a parent link, all the data is available in the\n            # instance, so populate the parent model with this data.\n            rel_model = self.field.remote_field.model\n            fields = [field.attname for field in rel_model._meta.concrete_fields]\n\n            # If any of the related model's fields are deferred, fallback to\n            # fetching all fields from the related model. This avoids a query\n            # on the related model for every deferred field.\n            if not any(field in fields for field in deferred):\n                kwargs = {field: getattr(instance, field) for field in fields}\n                obj = rel_model(**kwargs)\n                obj._state.adding = instance._state.adding\n                obj._state.db = instance._state.db\n                return obj\n        return super().get_object(instance)"
      }
    ]
  },
  "Justification": "Candidate D is the most relevant due to its similarity in context. Both the CURRENT bug and D involve issues related to parent-child relationships in a model hierarchy and saving those relationships correctly. While the CURRENT bug focuses on foreign key constraints and primary key assignment, Candidate D addresses failures when changing a parent reference in a child model and underscores the need for proper handling in relational model structures. The fix in Candidate D emphasizes the importance of managing inherited properties and constraints, which directly relates to the handling of the foreign key in the CURRENT bug report. This shared focus makes Candidate D particularly useful for debugging the CURRENT bug.",
  "instance_id": "django__django-13964",
  "repo": "django/django",
  "created_at": "2021-02-02T17:07:43Z",
  "problem_statement": "Saving parent object after setting on child leads to data loss for parents with non-numeric primary key.\nDescription\n\t \n\t\t(last modified by Charlie DeTar)\n\t \nGiven a model with a foreign key relation to another model that has a non-auto CharField as its primary key:\nclass Product(models.Model):\n\tsku = models.CharField(primary_key=True, max_length=50)\nclass Order(models.Model):\n\tproduct = models.ForeignKey(Product, on_delete=models.CASCADE)\nIf the relation is initialized on the parent with an empty instance that does not yet specify its primary key, and the primary key is subsequently defined, the parent does not \"see\" the primary key's change:\nwith transaction.atomic():\n\torder = Order()\n\torder.product = Product()\n\torder.product.sku = \"foo\"\n\torder.product.save()\n\torder.save()\n\tassert Order.objects.filter(product_id=\"\").exists() # Succeeds, but shouldn't\n\tassert Order.objects.filter(product=order.product).exists() # Fails\nInstead of product_id being populated with product.sku, it is set to emptystring. The foreign key constraint which would enforce the existence of a product with sku=\"\" is deferred until the transaction commits. The transaction does correctly fail on commit with a ForeignKeyViolation due to the non-existence of a product with emptystring as its primary key.\nOn the other hand, if the related unsaved instance is initialized with its primary key before assignment to the parent, it is persisted correctly:\nwith transaction.atomic():\n\torder = Order()\n\torder.product = Product(sku=\"foo\")\n\torder.product.save()\n\torder.save()\n\tassert Order.objects.filter(product=order.product).exists() # succeeds\nCommitting the transaction also succeeds.\nThis may have something to do with how the Order.product_id field is handled at assignment, together with something about handling fetching of auto vs non-auto primary keys from the related instance.\n",
  "patch": "diff --git a/django/db/models/base.py b/django/db/models/base.py\n--- a/django/db/models/base.py\n+++ b/django/db/models/base.py\n@@ -933,7 +933,7 @@ def _prepare_related_fields_for_save(self, operation_name):\n                         \"%s() prohibited to prevent data loss due to unsaved \"\n                         \"related object '%s'.\" % (operation_name, field.name)\n                     )\n-                elif getattr(self, field.attname) is None:\n+                elif getattr(self, field.attname) in field.empty_values:\n                     # Use pk from related object if it has been saved after\n                     # an assignment.\n                     setattr(self, field.attname, obj.pk)\n"
}