{
    "Selected_candidate": {
        "pr_number": 11551,
        "pr_title": "Fixed #30543 -- Fixed checks of ModelAdmin.list_display for fields accessible only via instance.",
        "pr_body": "Continue [ticke 30543](https://code.djangoproject.com/ticket/30543).\r\nTest has been added.",
        "issue_id": 30543,
        "issue_title": "admin.E108 is raised on fields accessible only via instance.",
        "issue_body": "As part of startup django validates the ModelAdmin's list_display list/tuple for correctness (django.admin.contrib.checks._check_list_display). Having upgraded django from 2.07 to 2.2.1 I found that a ModelAdmin with a list display that used to pass the checks and work fine in admin now fails validation, preventing django from starting. A PositionField from the django-positions library triggers this bug, explanation why follows.\nfrom django.db import models\nfrom position.Fields import PositionField\n\nclass Thing(models.Model)\n  number = models.IntegerField(default=0)\n  order = PositionField()\nfrom django.contrib import admin\nfrom .models import Thing\n\n@admin.register(Thing)\nclass ThingAdmin(admin.ModelAdmin)\n  list_display = ['number', 'order']\nUnder 2.2.1 this raises an incorrect admin.E108 message saying \"The value of list_display\n[1]\nrefers to 'order' which is not a callable...\".\nUnder 2.0.7 django starts up successfully.\nIf you change 'number' to 'no_number' or 'order' to 'no_order' then the validation correctly complains about those.\nThe reason for this bug is commit\n​\nhttps://github.com/django/django/commit/47016adbf54b54143d4cf052eeb29fc72d27e6b1\nwhich was proposed and accepted as a fix for bug\nhttps://code.djangoproject.com/ticket/28490\n. The problem is while it fixed that bug it broke the functionality of _check_list_display_item in other cases. The rationale for that change was that after field=getattr(model, item) field could be None if item was a descriptor returning None, but subsequent code incorrectly interpreted field being None as meaning getattr raised an AttributeError. As this was done\nafter\ntrying field = model._meta.get_field(item) and that failing that meant the validation error should be returned. However, after the above change if hasattr(model, item) is false then we no longer even try field = model._meta.get_field(item) before returning an error. The reason hasattr(model, item) is false in the case of a PositionField is its\nget\nmethod throws an exception if called on an instance of the PositionField class on the Thing model class, rather than a Thing instance.\nFor clarity, here are the various logical tests that _check_list_display_item needs to deal with and the behaviour before the above change, after it, and the correct behaviour (which my suggested patch exhibits). Note this is assuming the first 2 tests callable(item) and hasattr(obj, item) are both false (corresponding to item is an actual function/lambda rather than string or an attribute of ThingAdmin).\nhasattr(model, item) returns True or  False (which is the same as seeing if getattr(model, item) raises AttributeError)\nmodel._meta.get_field(item) returns a field or raises FieldDoesNotExist\nGet a field from somewhere, could either be from getattr(model,item) if hasattr was True or from get_field.\nIs that field an instance of ManyToManyField?\nIs that field None? (True in case of bug 28490)\nhasattr\nget_field\nfield is None?\nfield ManyToMany?\n2.0 returns\n2.2 returns\nCorrect behaviour\nComments\nTrue\nok\nFalse\nFalse\n[]\n[]\n[]\n-\nTrue\nok\nFalse\nTrue\nE109\nE109\nE109\n-\nTrue\nok\nTrue\nFalse\nE108\n[]\n[]\ngood bit of 28490 fix, 2.0 was wrong\nTrue\nraises\nFalse\nFalse\n[]\n[]\n[]\n-\nTrue\nraises\nFalse\nTrue\nE109\n[]\nE109\nAnother bug introduced by 28490 fix, fails to check if ManyToMany in get_field raise case\nTrue\nraises\nTrue\nFalse\nE108\n[]\n[]\ngood bit of 28490 fix, 2.0 was wrong\nFalse\nok\nFalse\nFalse\n[]\nE108\n[]\nbad bit of 28490 fix, bug hit with PositionField\nFalse\nok\nFalse\nTrue\n[]\nE108\nE109\nboth 2.0 and 2.2 wrong\nFalse\nok\nTrue\nFalse\n[]\nE108\n[]\nbad 28490 fix\nFalse\nraises\nFalse\nFalse\nE108\nE108\nE108\n-\nFalse\nraises\nFalse\nTrue\nE108\nE108\nE108\nimpossible condition, we got no field assigned to be a ManyToMany\nFalse\nraises\nTrue\nFalse\nE108\nE108\nE108\nimpossible condition, we got no field assigned to be None\nThe following code exhibits the correct behaviour in all cases. The key changes are there is no longer a check for hasattr(model, item), as that being false should not prevent us form attempting to get the field via get_field, and only return an E108 in the case both of them fail. If either of those means or procuring it are successful then we need to check if it's a ManyToMany. Whether or not the field is None is irrelevant, and behaviour is contained within the exception catching blocks that should cause it instead of signalled through a variable being set to None which is a source of conflation of different cases.\ndef _check_list_display_item(self, obj, item, label):\n    if callable(item):\n        return []\n    elif hasattr(obj, item):\n        return []\n    else:\n        try:\n            field = obj.model._meta.get_field(item)\n        except FieldDoesNotExist:\n            try:\n                field = getattr(obj.model, item)\n            except AttributeError:\n                return [\n                    checks.Error(\n                        \"The value of '%s' refers to '%s', which is not a callable, \"\n                        \"an attribute of '%s', or an attribute or method on '%s.%s'.\" % (\n                            label, item, obj.__class__.__name__,\n                            obj.model._meta.app_label, obj.model._meta.object_name,\n                        ),\n                        obj=obj.__class__,\n                        id='admin.E108',\n                    )\n                ]\n\n        if isinstance(field, models.ManyToManyField):\n            return [\n                checks.Error(\n                    \"The value of '%s' must not be a ManyToManyField.\" % label,\n                    obj=obj.__class__,\n                    id='admin.E109',\n                )\n            ]\n        return []",
        "issue_closed_at": "2019-07-10T03:59:12",
        "base_commit": "7991111af12056ec9a856f35935d273526338c1f",
        "changes": [
            {
                "file": "django/contrib/admin/checks.py",
                "type": "function",
                "name": "_check_list_display_item",
                "class_name": "ModelAdminChecks",
                "code": "def _check_list_display_item(self, obj, item, label):\n        if callable(item):\n            return []\n        elif hasattr(obj, item):\n            return []\n        elif hasattr(obj.model, item):\n            try:\n                field = obj.model._meta.get_field(item)\n            except FieldDoesNotExist:\n                return []\n            else:\n                if isinstance(field, models.ManyToManyField):\n                    return [\n                        checks.Error(\n                            \"The value of '%s' must not be a ManyToManyField.\" % label,\n                            obj=obj.__class__,\n                            id='admin.E109',\n                        )\n                    ]\n                return []\n        else:\n            return [\n                checks.Error(\n                    \"The value of '%s' refers to '%s', which is not a callable, \"\n                    \"an attribute of '%s', or an attribute or method on '%s.%s'.\" % (\n                        label, item, obj.__class__.__name__,\n                        obj.model._meta.app_label, obj.model._meta.object_name,\n                    ),\n                    obj=obj.__class__,\n                    id='admin.E108',\n                )\n            ]"
            }
        ]
    },
    "Justification": "Candidate E is the most relevant bug report as it involves improvements within the Django admin related to value validation for fields. Specifically, it addresses ModelAdmin's list_display attribute for fields that are only accessible via an instance, which connects closely to the CURRENT bug about JSONField display issues in the admin environment. Both reports involve how values are displayed in the admin interface and can benefit from insights on handling validation and display logic for fields. Additionally, since the CURRENT bug involves specifically addressing how a field's value is prepared to be displayed, the structural and symptom similarities make this candidate particularly helpful for debugging purposes."
}