{
  "Selected_candidate": {
    "pr_number": 10541,
    "pr_title": "Fixed #27595 -- Made ForeignKey.get_col() follow target chains.",
    "pr_body": "We were previously only following foreign relationships one level deep which\r\nwas preventing foreign keys to foreign keys from being resolved appropriately.\r\n\r\nThis was causing all sorts of weirdness such as improper database value\r\nconversion for UUIDField on SQLite because the resolved expression's output\r\nfield's internal type was not correct.\r\n\r\nAlso added tests to make sure unlikely foreign reference cycles do not cause\r\nrecursion errors.\r\n\r\nRefs [#24343](https://code.djangoproject.com/ticket/24343) ([PR](https://github.com/django/django/pull/4171))\r\n\r\nThanks oyooyo for the report and Wayne Merry for the investigation.\r\n\r\nhttps://code.djangoproject.com/ticket/27595",
    "issue_id": 27595,
    "issue_title": "Database converters are not run for related fields referencing related fields",
    "issue_body": "Sample models.py:\nfrom django.db import models\n\nimport uuid\n\nclass Base_Model(models.Model):\n\tid = models.UUIDField(primary_key = True, default=uuid.uuid4, editable=False)\n\t# id = models.AutoField(primary_key = True)\n\tprototype = models.ForeignKey('Prototype_Model', blank=True, null=True)\n\tdef __str__(self):\n\t\treturn str(self.id)\n\nclass Prototype_Model(Base_Model):\n\tpass\nSample admin.py:\nfrom django.contrib import admin\n\nfrom .models import Prototype_Model\n\nclass Base_Model_Admin(admin.ModelAdmin):\n\tlist_display = ['id', 'prototype']\n\nadmin.site.register(Prototype_Model, Base_Model_Admin)\nSteps to reproduce:\nCreate a Prototype_Model via the admin. Leave the \"prototype\" ForeignKey field empty.\nCreate another Prototype_Model via the admin. For the \"prototype\" ForeignKey field, choose the first created Prototype_Model from the dropdown\nSwitch to the Prototype_Model list view, it should show that the \"prototype\" ForeignKey field of the second Prototype_Model indeed references the first model\nClick on this second Prototype_Model in order to access the change/edit model form and look at the dropdown for the \"prototype\" field. The expected behaviour is that the dropdown should have the referenced first model automatically selected, but instead, nothing is automatically selected.\nIf an AutoField is used instead of a UUIDField as the primary key \"id\" field (see commented out line in Base_Model), the behaviour is as expected.",
    "issue_closed_at": "2018-10-22T10:22:55",
    "base_commit": "19126339f307e589f99259ab0176c4367a8055f0",
    "changes": [
      {
        "file": "django/db/models/fields/related.py",
        "type": "function",
        "name": "get_db_converters",
        "class_name": "ForeignKey",
        "code": "def get_db_converters(self, connection):\n        converters = super().get_db_converters(connection)\n        if connection.features.interprets_empty_strings_as_nulls:\n            converters += [self.convert_empty_strings]\n        return converters"
      }
    ]
  },
  "Justification": "Candidate E is highly relevant because it addresses issues related to handling foreign key references in Django models. While not directly connected to JS media order conflicts, it does share similarities in the sense that both involve web form behavior and expected outcomes in the admin interface. The bug with the foreign keys impacting dropdown selections reflects a deeper problem in how Django handles linked data, akin to how media files are resolved and ordered. Understanding the fix in relation to the handling of foreign keys could offer insights into why MediaOrderConflictWarnings arise, especially regarding the expected order of usage within widgets/forms.",
  "instance_id": "django__django-11019",
  "repo": "django/django",
  "created_at": "2019-02-23T15:51:14Z",
  "problem_statement": "Merging 3 or more media objects can throw unnecessary MediaOrderConflictWarnings\nDescription\n\t\nConsider the following form definition, where text-editor-extras.js depends on text-editor.js but all other JS files are independent:\nfrom django import forms\nclass ColorPicker(forms.Widget):\n\tclass Media:\n\t\tjs = ['color-picker.js']\nclass SimpleTextWidget(forms.Widget):\n\tclass Media:\n\t\tjs = ['text-editor.js']\nclass FancyTextWidget(forms.Widget):\n\tclass Media:\n\t\tjs = ['text-editor.js', 'text-editor-extras.js', 'color-picker.js']\nclass MyForm(forms.Form):\n\tbackground_color = forms.CharField(widget=ColorPicker())\n\tintro = forms.CharField(widget=SimpleTextWidget())\n\tbody = forms.CharField(widget=FancyTextWidget())\nDjango should be able to resolve the JS files for the final form into the order text-editor.js, text-editor-extras.js, color-picker.js. However, accessing MyForm().media results in:\n/projects/django/django/forms/widgets.py:145: MediaOrderConflictWarning: Detected duplicate Media files in an opposite order:\ntext-editor-extras.js\ntext-editor.js\n MediaOrderConflictWarning,\nMedia(css={}, js=['text-editor-extras.js', 'color-picker.js', 'text-editor.js'])\nThe MediaOrderConflictWarning is a result of the order that the additions happen in: ColorPicker().media + SimpleTextWidget().media produces Media(css={}, js=['color-picker.js', 'text-editor.js']), which (wrongly) imposes the constraint that color-picker.js must appear before text-editor.js.\nThe final result is particularly unintuitive here, as it's worse than the \"naïve\" result produced by Django 1.11 before order-checking was added (color-picker.js, text-editor.js, text-editor-extras.js), and the pair of files reported in the warning message seems wrong too (aren't color-picker.js and text-editor.js the wrong-ordered ones?)\n",
  "patch": "diff --git a/django/forms/widgets.py b/django/forms/widgets.py\n--- a/django/forms/widgets.py\n+++ b/django/forms/widgets.py\n@@ -6,16 +6,21 @@\n import datetime\n import re\n import warnings\n+from collections import defaultdict\n from itertools import chain\n \n from django.conf import settings\n from django.forms.utils import to_current_timezone\n from django.templatetags.static import static\n from django.utils import datetime_safe, formats\n+from django.utils.datastructures import OrderedSet\n from django.utils.dates import MONTHS\n from django.utils.formats import get_format\n from django.utils.html import format_html, html_safe\n from django.utils.safestring import mark_safe\n+from django.utils.topological_sort import (\n+    CyclicDependencyError, stable_topological_sort,\n+)\n from django.utils.translation import gettext_lazy as _\n \n from .renderers import get_default_renderer\n@@ -59,22 +64,15 @@ def __str__(self):\n \n     @property\n     def _css(self):\n-        css = self._css_lists[0]\n-        # filter(None, ...) avoids calling merge with empty dicts.\n-        for obj in filter(None, self._css_lists[1:]):\n-            css = {\n-                medium: self.merge(css.get(medium, []), obj.get(medium, []))\n-                for medium in css.keys() | obj.keys()\n-            }\n-        return css\n+        css = defaultdict(list)\n+        for css_list in self._css_lists:\n+            for medium, sublist in css_list.items():\n+                css[medium].append(sublist)\n+        return {medium: self.merge(*lists) for medium, lists in css.items()}\n \n     @property\n     def _js(self):\n-        js = self._js_lists[0]\n-        # filter(None, ...) avoids calling merge() with empty lists.\n-        for obj in filter(None, self._js_lists[1:]):\n-            js = self.merge(js, obj)\n-        return js\n+        return self.merge(*self._js_lists)\n \n     def render(self):\n         return mark_safe('\\n'.join(chain.from_iterable(getattr(self, 'render_' + name)() for name in MEDIA_TYPES)))\n@@ -115,39 +113,37 @@ def __getitem__(self, name):\n         raise KeyError('Unknown media type \"%s\"' % name)\n \n     @staticmethod\n-    def merge(list_1, list_2):\n+    def merge(*lists):\n         \"\"\"\n-        Merge two lists while trying to keep the relative order of the elements.\n-        Warn if the lists have the same two elements in a different relative\n-        order.\n+        Merge lists while trying to keep the relative order of the elements.\n+        Warn if the lists have the same elements in a different relative order.\n \n         For static assets it can be important to have them included in the DOM\n         in a certain order. In JavaScript you may not be able to reference a\n         global or in CSS you might want to override a style.\n         \"\"\"\n-        # Start with a copy of list_1.\n-        combined_list = list(list_1)\n-        last_insert_index = len(list_1)\n-        # Walk list_2 in reverse, inserting each element into combined_list if\n-        # it doesn't already exist.\n-        for path in reversed(list_2):\n-            try:\n-                # Does path already exist in the list?\n-                index = combined_list.index(path)\n-            except ValueError:\n-                # Add path to combined_list since it doesn't exist.\n-                combined_list.insert(last_insert_index, path)\n-            else:\n-                if index > last_insert_index:\n-                    warnings.warn(\n-                        'Detected duplicate Media files in an opposite order:\\n'\n-                        '%s\\n%s' % (combined_list[last_insert_index], combined_list[index]),\n-                        MediaOrderConflictWarning,\n-                    )\n-                # path already exists in the list. Update last_insert_index so\n-                # that the following elements are inserted in front of this one.\n-                last_insert_index = index\n-        return combined_list\n+        dependency_graph = defaultdict(set)\n+        all_items = OrderedSet()\n+        for list_ in filter(None, lists):\n+            head = list_[0]\n+            # The first items depend on nothing but have to be part of the\n+            # dependency graph to be included in the result.\n+            dependency_graph.setdefault(head, set())\n+            for item in list_:\n+                all_items.add(item)\n+                # No self dependencies\n+                if head != item:\n+                    dependency_graph[item].add(head)\n+                head = item\n+        try:\n+            return stable_topological_sort(all_items, dependency_graph)\n+        except CyclicDependencyError:\n+            warnings.warn(\n+                'Detected duplicate Media files in an opposite order: {}'.format(\n+                    ', '.join(repr(l) for l in lists)\n+                ), MediaOrderConflictWarning,\n+            )\n+            return list(all_items)\n \n     def __add__(self, other):\n         combined = Media()\n"
}