{
  "id": "django__django-13964",
  "question": "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",
  "answer": "{\"task_id\": \"79c15c84-ee07-4651-8b1e-20e69716e17a\", \"workflow_run_id\": \"0e9d4f25-a9be-4bb1-bfa6-2432c4880c30\", \"data\": {\"id\": \"0e9d4f25-a9be-4bb1-bfa6-2432c4880c30\", \"workflow_id\": \"bc5a1a30-852f-4fd7-98f0-8fca2f23c564\", \"status\": \"succeeded\", \"outputs\": {\"ranked_patches\": \"```json\\n{\\n  \\\"ranked_patches\\\": [\\n    {\\n      \\\"patch_id\\\": 2,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/base.py b/django/db/models/base.py\\\\nindex 123abc..456def 100644\\\\n--- a/django/db/models/base.py\\\\n+++ b/django/db/models/base.py\\\\n@@ -600,6 +600,14 @@ class Model:\\\\n                 # An unsaved object.\\\\n                 raise ValueError(\\\\\\\"Cannot save an instance without a primary key\\\\\\\")\\\\n \\\\n+        # Process pending FK updates if this instance is being saved with a PK now\\\\n+        if self._state.adding and self.pk and hasattr(self, '_pending_fk_updates'):\\\\n+            for pending_instance, fk_field in self._pending_fk_updates:\\\\n+                related_pk = getattr(self, self._meta.pk.name)\\\\n+                setattr(pending_instance, fk_field.attname, related_pk)\\\\n+                pending_instance.save(update_fields=[fk_field.attname])\\\\n+            del self._pending_fk_updates\\\\n+\\\\n         # Pre-save operations\\\\n         using = using or router.db_for_write(self.__class__, instance=self)\\\\n         context = {'using': using, 'instance': self, 'raw': raw, 'update_fields': update_fields}\\\\n\\\",\\n      \\\"rank\\\": 1,\\n      \\\"score\\\": 0.95,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.95,\\n        \\\"code_quality\\\": 0.95,\\n        \\\"risk_level\\\": 0.1,\\n        \\\"best_practices\\\": 0.9,\\n        \\\"test_coverage\\\": 0.9\\n      },\\n      \\\"reasoning\\\": \\\"This patch correctly handles the pending foreign key updates by processing them when the primary key becomes available. It ensures that the foreign key is set correctly and maintains the integrity of the database. The approach is clean and aligns well with Django's model save operations.\\\",\\n      \\\"recommendation\\\": \\\"Highly recommended\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 1,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/fields/related.py b/django/db/models/fields/related.py\\\\nindex abc123..def456 100644\\\\n--- a/django/db/models/fields/related.py\\\\n+++ b/django/db/models/fields/related.py\\\\n@@ -200,6 +200,13 @@ class ForwardManyToOneDescriptor:\\\\n         if instance is None:\\\\n             instance = self.related.related_model()\\\\n \\\\n+        # If the related object is unsaved and lacks a primary key, defer the FK setting\\\\n+        if instance.pk is None:\\\\n+            # Register for pending update\\\\n+            instance._pending_fk_updates = getattr(instance, '_pending_fk_updates', [])\\\\n+            instance._pending_fk_updates.append((instance, instance._meta.get_field(self.field.name)))\\\\n+            return\\\\n+\\\\n         setattr(instance, self.field.attname, value)\\\\n \\\\n     def get_object_cache(self, instance):\\\\n\\\",\\n      \\\"rank\\\": 2,\\n      \\\"score\\\": 0.9,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.9,\\n        \\\"code_quality\\\": 0.9,\\n        \\\"risk_level\\\": 0.15,\\n        \\\"best_practices\\\": 0.85,\\n        \\\"test_coverage\\\": 0.85\\n      },\\n      \\\"reasoning\\\": \\\"This patch defers the foreign key setting when the related object is unsaved and lacks a primary key. It registers pending updates, which is a good approach, but it relies on another mechanism to process these updates, which is not included in this patch.\\\",\\n      \\\"recommendation\\\": \\\"Recommended with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 3,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/fields/related.py b/django/db/models/fields/related.py\\\\nindex abc123..789ghi 100644\\\\n--- a/django/db/models/fields/related.py\\\\n+++ b/django/db/models/fields/related.py\\\\n@@ -200,6 +200,16 @@ class ForwardManyToOneDescriptor:\\\\n         if instance is None:\\\\n             instance = self.related.related_model()\\\\n \\\\n+        # If the related object has a pending key, avoid setting the foreign key immediately\\\\n+        if instance.pk is None:\\\\n+            def update_fk(instance, value):\\\\n+                setattr(instance, self.field.attname, value.pk)\\\\n+\\\\n+            # Register a post-save signal to update the foreign key\\\\n+            models.signals.post_save.connect(update_fk, sender=instance.__class__)\\\\n+            return\\\\n+\\\\n         setattr(instance, self.field.attname, value)\\\\n \\\\n     def get_object_cache(self, instance):\\\\n\\\",\\n      \\\"rank\\\": 3,\\n      \\\"score\\\": 0.85,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.85,\\n        \\\"code_quality\\\": 0.85,\\n        \\\"risk_level\\\": 0.2,\\n        \\\"best_practices\\\": 0.8,\\n        \\\"test_coverage\\\": 0.8\\n      },\\n      \\\"reasoning\\\": \\\"This patch uses a post-save signal to update the foreign key, which is a valid approach but introduces additional complexity and potential for signal-related issues. It may not be the most efficient solution.\\\",\\n      \\\"recommendation\\\": \\\"Consider with caution\\\"\\n    },\\n    {\\n      \\\"patch_id\\\": 4,\\n      \\\"patch\\\": \\\"diff --git a/django/db/models/base.py b/django/db/models/base.py\\\\nindex 123abc..789ghi 100644\\\\n--- a/django/db/models/base.py\\\\n+++ b/django/db/models/base.py\\\\n@@ -600,6 +600,16 @@ class Model:\\\\n                 # An unsaved object.\\\\n                 raise ValueError(\\\\\\\"Cannot save an instance without a primary key\\\\\\\")\\\\n \\\\n+        # For instances with pending FK updates, process them\\\\n+        if self.pk is None and hasattr(self, '_pending_fk_updates'):\\\\n+            pending_updates = self._pending_fk_updates\\\\n+            for pending_instance in pending_updates:\\\\n+                pending_instance.save()  # Implicitly updates the FK in parent\\\\n+            del self._pending_fk_updates\\\\n+\\\\n+        models.signals.post_save.send(sender=self.__class__, instance=self, created=self._state.adding, update_fields=update_fields)\\\\n+\\\\n         # Pre-save operations\\\\n         using = using or router.db_for_write(self.__class__, instance=self)\\\\n         context = {'using': using, 'instance': self, 'raw': raw, 'update_fields': update_fields}\\\\n\\\",\\n      \\\"rank\\\": 4,\\n      \\\"score\\\": 0.8,\\n      \\\"evaluation\\\": {\\n        \\\"correctness\\\": 0.8,\\n        \\\"code_quality\\\": 0.8,\\n        \\\"risk_level\\\": 0.25,\\n        \\\"best_practices\\\": 0.75,\\n        \\\"test_coverage\\\": 0.75\\n      },\\n      \\\"reasoning\\\": \\\"This patch attempts to process pending foreign key updates but does so in a way that may not handle all edge cases effectively. It also introduces a post-save signal, which can complicate the flow and increase the risk of unintended side effects.\\\",\\n      \\\"recommendation\\\": \\\"Least recommended\\\"\\n    }\\n  ],\\n  \\\"evaluation_summary\\\": \\\"Patch 2 is the most effective solution, providing a clean and efficient way to handle pending foreign key updates by processing them when the primary key becomes available. Patch 1 is a good approach but requires additional mechanisms to complete the process. Patches 3 and 4 introduce signals, which add complexity and potential risks, making them less desirable solutions.\\\"\\n}\\n```\", \"generated_tests\": \"{\\n  \\\"reproduction_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue_with_empty_instance\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue_with_empty_instance():\\\\n    with transaction.atomic():\\\\n        order = Order()\\\\n        order.product = Product()\\\\n        order.product.sku = 'foo'\\\\n        order.product.save()\\\\n        order.save()\\\\n    assert Order.objects.filter(product_id='').exists() == True\\\\n    assert Order.objects.filter(product=order.product).exists() == False\\\",\\n      \\\"description\\\": \\\"This test reproduces the original issue when initializing the related unsaved instance with an empty primary key before assigning it to the parent\\\",\\n      \\\"expected_behavior\\\": \\\"The assertions should pass before applying the patches\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_reproduce_original_issue_with_initialized_instance\\\",\\n      \\\"test_code\\\": \\\"def test_reproduce_original_issue_with_initialized_instance():\\\\n    with transaction.atomic():\\\\n        order = Order()\\\\n        order.product = Product()\\\\n        order.product.sku = 'foo'\\\\n        order.product.save()\\\\n        order.save()\\\\n    assert Order.objects.filter(product=order.product).exists() == True\\\",\\n      \\\"description\\\": \\\"This test reproduces the issue when initializing the related unsaved instance with a primary key before assigning it to the parent\\\",\\n      \\\"expected_behavior\\\": \\\"The assertion should fail before applying the patches\\\"\\n    }\\n  ],\\n  \\\"validation_tests\\\": [\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_with_empty_instance\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_with_empty_instance():\\\\n    with transaction.atomic():\\\\n        order = Order()\\\\n        order.product = Product()\\\\n        order.product.sku = 'foo'\\\\n        order.product.save()\\\\n        order.save()\\\\n    assert Order.objects.filter(product_id='').exists() == False\\\\n    assert Order.objects.filter(product=order.product).exists() == True\\\",\\n      \\\"description\\\": \\\"This test validates that the patch correctly handles the issue when initializing the related unsaved instance with an empty primary key\\\",\\n      \\\"expected_behavior\\\": \\\"The assertions should pass after applying the patches\\\"\\n    },\\n    {\\n      \\\"test_name\\\": \\\"test_patch_validation_with_initialized_instance\\\",\\n      \\\"test_code\\\": \\\"def test_patch_validation_with_initialized_instance():\\\\n    with transaction.atomic():\\\\n        order = Order()\\\\n        order.product = Product(sku='foo')\\\\n        order.product.save()\\\\n        order.save()\\\\n    assert Order.objects.filter(product=order.product).exists() == True\\\",\\n      \\\"description\\\": \\\"This test validates that the patch fixes the issue when initializing the related unsaved instance with a primary key\\\",\\n      \\\"expected_behavior\\\": \\\"The assertion should pass after applying the patches\\\"\\n    }\\n  ],\\n  \\\"test_summary\\\": \\\"Generated comprehensive test cases for reproducing the original issue and validating the patches\\\"\\n}\"}, \"error\": \"\", \"elapsed_time\": 582.981071, \"total_tokens\": 27553, \"total_steps\": 9, \"created_at\": 1753303307, \"finished_at\": 1753303890}}"
}