{
  "Selected_candidate": {
    "pr_number": 4462,
    "pr_title": "Fixed #24578 -- Fixed crash with QuerySet.update() on FK to O2O fields.",
    "pr_body": "",
    "issue_id": 24578,
    "issue_title": "prepare_database_save breaks some OneToOneField's in 1.8",
    "issue_body": "With 1.8 the following doesn't work:\n# models.py\nfrom django.db import models\n\nclass A(models.Model):\n    pass\n\nclass Aprime(models.Model):\n\n    original = models.OneToOneField(A, primary_key=True)\n\nclass B(models.Model):\n    aprime = models.ForeignKey(Aprime)\n\n# tests.py\n\nfrom django.test import TestCase\n\nfrom models import *\n\nclass TestTestCase(TestCase):\n\n    def test_ref(self):\n        a_one = A.objects.create()\n        a_two = A.objects.create()\n\n        self.assertNotEqual(a_one, a_two)\n\n        aprime = Aprime.objects.create(original=a_one)\n        aprime2 = Aprime.objects.create(original=a_two)\n\n        b = B.objects.create(aprime=aprime)\n\n        self.assertEqual(b.aprime, aprime)\n\n        B.objects.update(aprime=aprime2)\n\n        b = B.objects.get(pk=b.id)\n\n        self.assertEqual(b.aprime, aprime2)\nThis throws a\nInterfaceError: Error binding parameter 0 - probably unsupported type.\nwhen calling\nB.objects.update(aprime=aprime2)\n.\nThe bug disappears when overriding\nprepare_database_save\nto just return\nself.pk\ninstead of\ngetattr(self, field.rel.field_name)\n.  (This was the behavior as of 1.7 and the bug does not occur there.)\nNot sure why\nprepare_database_save\nwas modified but this bug actually affects a project of mine so would suggest rolling back to the 1.7 version.",
    "issue_closed_at": "2015-04-09T07:21:57",
    "base_commit": "20a98d863f00fc48f9c7fd783d8d0539c6be41f5",
    "changes": [
      {
        "file": "django/db/models/base.py",
        "type": "function",
        "name": "_get_next_or_previous_in_order",
        "class_name": "Model",
        "code": "def _get_next_or_previous_in_order(self, is_next):\n        cachename = \"__%s_order_cache\" % is_next\n        if not hasattr(self, cachename):\n            op = 'gt' if is_next else 'lt'\n            order = '_order' if is_next else '-_order'\n            order_field = self._meta.order_with_respect_to\n            obj = self._default_manager.filter(**{\n                order_field.name: getattr(self, order_field.attname)\n            }).filter(**{\n                '_order__%s' % op: self._default_manager.values('_order').filter(**{\n                    self._meta.pk.name: self.pk\n                })\n            }).order_by(order)[:1].get()\n            setattr(self, cachename, obj)\n        return getattr(self, cachename)"
      }
    ]
  },
  "Justification": "Candidate C is particularly relevant because it deals with issues related to OneToOne relationships and foreign keys, which ties directly to the use case of the Current Bug Report involving self-referencing foreign keys. Both reports concern model relationships and how they impact query outcomes, specifically under different Django versions. The structural issues with joining and the differences in behavior regarding relationships could provide valuable insights into fixing the current issue. Additionally, the patch for Candidate C addresses underlying problems with QuerySet operations, making it a strong candidate for debugging the current problem.",
  "instance_id": "django__django-13033",
  "repo": "django/django",
  "created_at": "2020-06-07T14:52:19Z",
  "problem_statement": "Self referencing foreign key doesn't correctly order by a relation \"_id\" field.\nDescription\n\t\nInitially discovered on 2.2.10 but verified still happens on 3.0.6. Given the following models:\nclass OneModel(models.Model):\n\tclass Meta:\n\t\tordering = (\"-id\",)\n\tid = models.BigAutoField(primary_key=True)\n\troot = models.ForeignKey(\"OneModel\", on_delete=models.CASCADE, null=True)\n\toneval = models.BigIntegerField(null=True)\nclass TwoModel(models.Model):\n\tid = models.BigAutoField(primary_key=True)\n\trecord = models.ForeignKey(OneModel, on_delete=models.CASCADE)\n\ttwoval = models.BigIntegerField(null=True)\nThe following queryset gives unexpected results and appears to be an incorrect SQL query:\nqs = TwoModel.objects.filter(record__oneval__in=[1,2,3])\nqs = qs.order_by(\"record__root_id\")\nprint(qs.query)\nSELECT \"orion_twomodel\".\"id\", \"orion_twomodel\".\"record_id\", \"orion_twomodel\".\"twoval\" FROM \"orion_twomodel\" INNER JOIN \"orion_onemodel\" ON (\"orion_twomodel\".\"record_id\" = \"orion_onemodel\".\"id\") LEFT OUTER JOIN \"orion_onemodel\" T3 ON (\"orion_onemodel\".\"root_id\" = T3.\"id\") WHERE \"orion_onemodel\".\"oneval\" IN (1, 2, 3) ORDER BY T3.\"id\" DESC\nThe query has an unexpected DESCENDING sort. That appears to come from the default sort order on the OneModel class, but I would expect the order_by() to take prececence. The the query has two JOINS, which is unnecessary. It appears that, since OneModel.root is a foreign key to itself, that is causing it to do the unnecessary extra join. In fact, testing a model where root is a foreign key to a third model doesn't show the problem behavior.\nNote also that the queryset with order_by(\"record__root\") gives the exact same SQL.\nThis queryset gives correct results and what looks like a pretty optimal SQL:\nqs = TwoModel.objects.filter(record__oneval__in=[1,2,3])\nqs = qs.order_by(\"record__root__id\")\nprint(qs.query)\nSELECT \"orion_twomodel\".\"id\", \"orion_twomodel\".\"record_id\", \"orion_twomodel\".\"twoval\" FROM \"orion_twomodel\" INNER JOIN \"orion_onemodel\" ON (\"orion_twomodel\".\"record_id\" = \"orion_onemodel\".\"id\") WHERE \"orion_onemodel\".\"oneval\" IN (1, 2, 3) ORDER BY \"orion_onemodel\".\"root_id\" ASC\nSo is this a potential bug or a misunderstanding on my part?\nAnother queryset that works around the issue and gives a reasonable SQL query and expected results:\nqs = TwoModel.objects.filter(record__oneval__in=[1,2,3])\nqs = qs.annotate(root_id=F(\"record__root_id\"))\nqs = qs.order_by(\"root_id\")\nprint(qs.query)\nSELECT \"orion_twomodel\".\"id\", \"orion_twomodel\".\"record_id\", \"orion_twomodel\".\"twoval\" FROM \"orion_twomodel\" INNER JOIN \"orion_onemodel\" ON (\"orion_twomodel\".\"record_id\" = \"orion_onemodel\".\"id\") WHERE \"orion_onemodel\".\"oneval\" IN (1, 2, 3) ORDER BY \"orion_onemodel\".\"zero_id\" ASC\nASCENDING sort, and a single INNER JOIN, as I'd expect. That actually works for my use because I need that output column anyway.\nOne final oddity; with the original queryset but the inverted sort order_by():\nqs = TwoModel.objects.filter(record__oneval__in=[1,2,3])\nqs = qs.order_by(\"-record__root_id\")\nprint(qs.query)\nSELECT \"orion_twomodel\".\"id\", \"orion_twomodel\".\"record_id\", \"orion_twomodel\".\"twoval\" FROM \"orion_twomodel\" INNER JOIN \"orion_onemodel\" ON (\"orion_twomodel\".\"record_id\" = \"orion_onemodel\".\"id\") LEFT OUTER JOIN \"orion_onemodel\" T3 ON (\"orion_onemodel\".\"root_id\" = T3.\"id\") WHERE \"orion_onemodel\".\"oneval\" IN (1, 2, 3) ORDER BY T3.\"id\" ASC\nOne gets the query with the two JOINs but an ASCENDING sort order. I was not under the impression that sort orders are somehow relative to the class level sort order, eg: does specifing order_by(\"-record__root_id\") invert the class sort order? Testing that on a simple case doesn't show that behavior at all.\nThanks for any assistance and clarification.\n",
  "patch": "diff --git a/django/db/models/sql/compiler.py b/django/db/models/sql/compiler.py\n--- a/django/db/models/sql/compiler.py\n+++ b/django/db/models/sql/compiler.py\n@@ -727,7 +727,12 @@ def find_ordering_name(self, name, opts, alias=None, default_order='ASC',\n         # If we get to this point and the field is a relation to another model,\n         # append the default ordering for that model unless it is the pk\n         # shortcut or the attribute name of the field that is specified.\n-        if field.is_relation and opts.ordering and getattr(field, 'attname', None) != name and name != 'pk':\n+        if (\n+            field.is_relation and\n+            opts.ordering and\n+            getattr(field, 'attname', None) != pieces[-1] and\n+            name != 'pk'\n+        ):\n             # Firstly, avoid infinite loops.\n             already_seen = already_seen or set()\n             join_tuple = tuple(getattr(self.query.alias_map[j], 'join_cols', None) for j in joins)\n"
}