{
  "instance_id": "django__django-12308",
  "repo": "django/django",
  "created_at": "2020-01-12T04:21:15Z",
  "problem_statement": "JSONField are not properly displayed in admin when they are readonly.\nDescription\n\t\nJSONField values are displayed as dict when readonly in the admin.\nFor example, {\"foo\": \"bar\"} would be displayed as {'foo': 'bar'}, which is not valid JSON.\nI believe the fix would be to add a special case in django.contrib.admin.utils.display_for_field to call the prepare_value of the JSONField (not calling json.dumps directly to take care of the InvalidJSONInput case).\n",
  "patch": "diff --git a/django/contrib/admin/utils.py b/django/contrib/admin/utils.py\n--- a/django/contrib/admin/utils.py\n+++ b/django/contrib/admin/utils.py\n@@ -398,6 +398,11 @@ def display_for_field(value, field, empty_value_display):\n         return formats.number_format(value)\n     elif isinstance(field, models.FileField) and value:\n         return format_html('<a href=\"{}\">{}</a>', value.url, value)\n+    elif isinstance(field, models.JSONField) and value:\n+        try:\n+            return field.get_prep_value(value)\n+        except TypeError:\n+            return display_for_value(value, empty_value_display)\n     else:\n         return display_for_value(value, empty_value_display)\n \n",
  "similar_bug_items": [
    {
      "pr_number": 9491,
      "pr_title": "Fixed #28958 -- Fixed admin changelist crash when using a query expression in the page's ordering.",
      "pr_body": "https://code.djangoproject.com/ticket/28958",
      "issue_id": 28958,
      "issue_title": "Admin changelist crashes when using query expression in the model's Meta.ordering or ModelAdmin.ordering",
      "issue_body": "Using the newly introduced query-expression ordering feature in either the model's Meta.ordering or in its ModelAdmin causes errors.\nSteps to reproduce:\nFirst variant - using the model's Meta.ordering option -\n//models.py\nclass Game(models.Model):\n    score = models.IntegerField(null=True)\n\n     class Meta:\n         ordering = [F(\"score\").desc(nulls_last=True)]\n...\n// admin.py\nadmin.site.register(Game)\nAccessing the admin site will now give an\nAttributeError: 'OrderBy' object has no attribute 'startswith'\nVariant 2 - Using the ModelAdmin ordering attribute:\n//admin.py\nclass GameAdmin(models.ModelAdmin):\n    ordering = [F(\"score\").desc(nulls_last=True)]\nrunserver\nfails with error:\nTypeError: argument of type 'OrderBy' is not iterable",
      "issue_closed_at": "2017-12-27T11:38:54",
      "base_commit": "ef6c680f60a8e2ab45c936ec91f856fb3d7dc651",
      "changes": [
        {
          "file": "django/contrib/admin/views/main.py",
          "type": "line",
          "name": "line 15",
          "code": ")\nfrom django.core.paginator import InvalidPage\nfrom django.db import models\nfrom django.urls import reverse\nfrom django.utils.http import urlencode\nfrom django.utils.translation import gettext"
        },
        {
          "file": "django/contrib/admin/views/main.py",
          "type": "function",
          "name": "get_ordering_field_columns",
          "class_name": "ChangeList",
          "code": "def get_ordering_field_columns(self):\n        \"\"\"\n        Return an OrderedDict of ordering field column numbers and asc/desc.\n        \"\"\"\n        # We must cope with more than one column having the same underlying sort\n        # field, so we base things on column numbers.\n        ordering = self._get_default_ordering()\n        ordering_fields = OrderedDict()\n        if ORDER_VAR not in self.params:\n            # for ordering specified on ModelAdmin or model Meta, we don't know\n            # the right column numbers absolutely, because there might be more\n            # than one column associated with that ordering, so we guess.\n            for field in ordering:\n                if field.startswith('-'):\n                    field = field[1:]\n                    order_type = 'desc'\n                else:\n                    order_type = 'asc'\n                for index, attr in enumerate(self.list_display):\n                    if self.get_ordering_field(attr) == field:\n                        ordering_fields[index] = order_type\n                        break\n        else:\n            for p in self.params[ORDER_VAR].split('.'):\n                none, pfx, idx = p.rpartition('-')\n                try:\n                    idx = int(idx)\n                except ValueError:\n                    continue  # skip it\n                ordering_fields[idx] = 'desc' if pfx == '-' else 'asc'\n        return ordering_fields"
        }
      ]
    },
    {
      "pr_number": 9406,
      "pr_title": "Fixed #28871 -- Fixed initialization of autocomplete widgets in \"Add another\" inlines.",
      "pr_body": "https://code.djangoproject.com/ticket/28871",
      "issue_id": 28871,
      "issue_title": "Autocomplete select for new lines in inline model do not open",
      "issue_body": "When a model has an Inline model, autocomplete select does not show for new rows.\nSelect box is is replaced with Select2 style, but clicking it does not show the select list.\nmodel.py\nclass\nItem\n(\nmodels\n.\nModel\n):\nname\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n100\n)\nclass\nCategory\n(\nmodels\n.\nModel\n):\nname\n=\nmodels\n.\nCharField\n(\nmax_length\n=\n100\n)\ndef\n__str__\n(\nself\n):\nreturn\nself\n.\nname\nclass\nLineItem\n(\nmodels\n.\nModel\n):\nitem\n=\nmodels\n.\nForeignKey\n(\n'Item'\n,\non_delete\n=\nmodels\n.\nCASCADE\n)\ncategory\n=\nmodels\n.\nForeignKey\n(\n'Category'\n,\non_delete\n=\nmodels\n.\nCASCADE\n)\nadmin.py\nclass\nLineItemInline\n(\nadmin\n.\nTabularInline\n):\nmodel\n=\nLineItem\nextra\n=\n1\nautocomplete_fields\n=\n(\n'category'\n,)\n@admin\n.\nregister\n(\nItem\n)\nclass\nItemAdmin\n(\nadmin\n.\nModelAdmin\n):\nlist_display\n=\n(\n'name'\n,)\nsearch_fields\n=\n(\n'name'\n,)\ninlines\n=\n[\nLineItemInline\n]\n@admin\n.\nregister\n(\nCategory\n)\nclass\nCategoryAdmin\n(\nadmin\n.\nModelAdmin\n):\nlist_display\n=\n(\n'name'\n,)\nsearch_fields\n=\n(\n'name'\n,)\nGithub repo\nWith demo and instructions on how to reproduce\n​\nhttps://github.com/gotling/bug-django2-autocomplete",
      "issue_closed_at": "2017-12-01T21:42:03",
      "base_commit": "095c1aaa898bed40568009db836aa8434f1b983d",
      "changes": [
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "formfield_for_foreignkey",
          "class_name": "BaseModelAdmin",
          "code": "def formfield_for_foreignkey(self, db_field, request, **kwargs):\n        \"\"\"\n        Get a form Field for a ForeignKey.\n        \"\"\"\n        db = kwargs.get('using')\n\n        if db_field.name in self.get_autocomplete_fields(request):\n            kwargs['widget'] = AutocompleteSelect(db_field.remote_field, using=db)\n        elif db_field.name in self.raw_id_fields:\n            kwargs['widget'] = widgets.ForeignKeyRawIdWidget(db_field.remote_field, self.admin_site, using=db)\n        elif db_field.name in self.radio_fields:\n            kwargs['widget'] = widgets.AdminRadioSelect(attrs={\n                'class': get_ul_class(self.radio_fields[db_field.name]),\n            })\n            kwargs['empty_label'] = _('None') if db_field.blank else None\n\n        if 'queryset' not in kwargs:\n            queryset = self.get_field_queryset(db, db_field, request)\n            if queryset is not None:\n                kwargs['queryset'] = queryset\n\n        return db_field.formfield(**kwargs)"
        },
        {
          "file": "django/contrib/admin/options.py",
          "type": "function",
          "name": "formfield_for_manytomany",
          "class_name": "BaseModelAdmin",
          "code": "def formfield_for_manytomany(self, db_field, request, **kwargs):\n        \"\"\"\n        Get a form Field for a ManyToManyField.\n        \"\"\"\n        # If it uses an intermediary model that isn't auto created, don't show\n        # a field in admin.\n        if not db_field.remote_field.through._meta.auto_created:\n            return None\n        db = kwargs.get('using')\n\n        autocomplete_fields = self.get_autocomplete_fields(request)\n        if db_field.name in autocomplete_fields:\n            kwargs['widget'] = AutocompleteSelectMultiple(db_field.remote_field, using=db)\n        elif db_field.name in self.raw_id_fields:\n            kwargs['widget'] = widgets.ManyToManyRawIdWidget(db_field.remote_field, self.admin_site, using=db)\n        elif db_field.name in list(self.filter_vertical) + list(self.filter_horizontal):\n            kwargs['widget'] = widgets.FilteredSelectMultiple(\n                db_field.verbose_name,\n                db_field.name in self.filter_vertical\n            )\n\n        if 'queryset' not in kwargs:\n            queryset = self.get_field_queryset(db, db_field, request)\n            if queryset is not None:\n                kwargs['queryset'] = queryset\n\n        form_field = db_field.formfield(**kwargs)\n        if (isinstance(form_field.widget, SelectMultiple) and\n                not isinstance(form_field.widget, (CheckboxSelectMultiple, AutocompleteSelectMultiple))):\n            msg = _('Hold down \"Control\", or \"Command\" on a Mac, to select more than one.')\n            help_text = form_field.help_text\n            form_field.help_text = format_lazy('{} {}', help_text, msg) if help_text else msg\n        return form_field"
        },
        {
          "file": "django/contrib/admin/widgets.py",
          "type": "function",
          "name": "__init__",
          "class_name": "AutocompleteMixin",
          "code": "def __init__(self, rel, attrs=None, choices=(), using=None):\n        self.rel = rel\n        self.db = using\n        self.choices = choices\n        if attrs is not None:\n            self.attrs = attrs.copy()\n        else:\n            self.attrs = {}"
        }
      ]
    },
    {
      "pr_number": 9632,
      "pr_title": "Fixed #29036 -- Change SelectDateWidget's empty value",
      "pr_body": "https://code.djangoproject.com/ticket/29036",
      "issue_id": 29036,
      "issue_title": "HTML5 required validation for SelectDateWidget doesn't work if the attribute is added by JavaScript",
      "issue_body": "SelectDateWidget\nuses\n0\nas a empty value. When the field uses HTML5 required attribute to make browser check the input, it does not work. Browsers (tested with Firefox 52.5 and Chromium 62) consider selected\n0\nas filled, thus\nrequired\ncheck incorrectly passes.\nI suggest to use empty string as a value for\nSelectDateWidget.none_value\n.",
      "issue_closed_at": "2018-01-30T18:43:14",
      "base_commit": "5538729e4ec1adf1c79d94c2b47b5dcf591ad94b",
      "changes": [
        {
          "file": "django/forms/widgets.py",
          "type": "class",
          "name": "SelectDateWidget",
          "code": "class SelectDateWidget(Widget):\n    \"\"\"\n    A widget that splits date input into three <select> boxes.\n\n    This also serves as an example of a Widget that has more than one HTML\n    element and hence implements value_from_datadict.\n    \"\"\"\n    none_value = (0, '---')\n    month_field = '%s_month'\n    day_field = '%s_day'\n    year_field = '%s_year'\n    template_name = 'django/forms/widgets/select_date.html'\n    input_type = 'select'\n    select_widget = Select\n    date_re = re.compile(r'(\\d{4}|0)-(\\d\\d?)-(\\d\\d?)$')\n\n    def __init__(self, attrs=None, years=None, months=None, empty_label=None):\n        self.attrs = attrs or {}\n\n        # Optional list or tuple of years to use in the \"year\" select box.\n        if years:\n            self.years = years\n        else:\n            this_year = datetime.date.today().year\n            self.years = range(this_year, this_year + 10)\n\n        # Optional dict of months to use in the \"month\" select box.\n        if months:\n            self.months = months\n        else:\n            self.months = MONTHS\n\n        # Optional string, list, or tuple to use as empty_label.\n        if isinstance(empty_label, (list, tuple)):\n            if not len(empty_label) == 3:\n                raise ValueError('empty_label list/tuple must have 3 elements.')\n\n            self.year_none_value = (0, empty_label[0])\n            self.month_none_value = (0, empty_label[1])\n            self.day_none_value = (0, empty_label[2])\n        else:\n            if empty_label is not None:\n                self.none_value = (0, empty_label)\n\n            self.year_none_value = self.none_value\n            self.month_none_value = self.none_value\n            self.day_none_value = self.none_value\n\n    def get_context(self, name, value, attrs):\n        context = super().get_context(name, value, attrs)\n        date_context = {}\n        year_choices = [(i, str(i)) for i in self.years]\n        if not self.is_required:\n            year_choices.insert(0, self.year_none_value)\n        year_attrs = context['widget']['attrs'].copy()\n        year_name = self.year_field % name\n        year_attrs['id'] = 'id_%s' % year_name\n        date_context['year'] = self.select_widget(attrs, choices=year_choices).get_context(\n            name=year_name,\n            value=context['widget']['value']['year'],\n            attrs=year_attrs,\n        )\n        month_choices = list(self.months.items())\n        if not self.is_required:\n            month_choices.insert(0, self.month_none_value)\n        month_attrs = context['widget']['attrs'].copy()\n        month_name = self.month_field % name\n        month_attrs['id'] = 'id_%s' % month_name\n        date_context['month'] = self.select_widget(attrs, choices=month_choices).get_context(\n            name=month_name,\n            value=context['widget']['value']['month'],\n            attrs=month_attrs,\n        )\n        day_choices = [(i, i) for i in range(1, 32)]\n        if not self.is_required:\n            day_choices.insert(0, self.day_none_value)\n        day_attrs = context['widget']['attrs'].copy()\n        day_name = self.day_field % name\n        day_attrs['id'] = 'id_%s' % day_name\n        date_context['day'] = self.select_widget(attrs, choices=day_choices,).get_context(\n            name=day_name,\n            value=context['widget']['value']['day'],\n            attrs=day_attrs,\n        )\n        subwidgets = []\n        for field in self._parse_date_fmt():\n            subwidgets.append(date_context[field]['widget'])\n        context['widget']['subwidgets'] = subwidgets\n        return context\n\n    def format_value(self, value):\n        \"\"\"\n        Return a dict containing the year, month, and day of the current value.\n        Use dict instead of a datetime to allow invalid dates such as February\n        31 to display correctly.\n        \"\"\"\n        year, month, day = None, None, None\n        if isinstance(value, (datetime.date, datetime.datetime)):\n            year, month, day = value.year, value.month, value.day\n        elif isinstance(value, str):\n            match = self.date_re.match(value)\n            if match:\n                year, month, day = [int(val) for val in match.groups()]\n            elif settings.USE_L10N:\n                input_format = get_format('DATE_INPUT_FORMATS')[0]\n                try:\n                    d = datetime.datetime.strptime(value, input_format)\n                except ValueError:\n                    pass\n                else:\n                    year, month, day = d.year, d.month, d.day\n        return {'year': year, 'month': month, 'day': day}\n\n    @staticmethod\n    def _parse_date_fmt():\n        fmt = get_format('DATE_FORMAT')\n        escaped = False\n        for char in fmt:\n            if escaped:\n                escaped = False\n            elif char == '\\\\':\n                escaped = True\n            elif char in 'Yy':\n                yield 'year'\n            elif char in 'bEFMmNn':\n                yield 'month'\n            elif char in 'dj':\n                yield 'day'\n\n    def id_for_label(self, id_):\n        for first_select in self._parse_date_fmt():\n            return '%s_%s' % (id_, first_select)\n        return '%s_month' % id_\n\n    def value_from_datadict(self, data, files, name):\n        y = data.get(self.year_field % name)\n        m = data.get(self.month_field % name)\n        d = data.get(self.day_field % name)\n        if y == m == d == \"0\":\n            return None\n        if y and m and d:\n            if settings.USE_L10N:\n                input_format = get_format('DATE_INPUT_FORMATS')[0]\n                try:\n                    date_value = datetime.date(int(y), int(m), int(d))\n                except ValueError:\n                    return '%s-%s-%s' % (y, m, d)\n                else:\n                    date_value = datetime_safe.new_date(date_value)\n                    return date_value.strftime(input_format)\n            else:\n                return '%s-%s-%s' % (y, m, d)\n        return data.get(name)\n\n    def value_omitted_from_data(self, data, files, name):\n        return not any(\n            ('{}_{}'.format(name, interval) in data)\n            for interval in ('year', 'month', 'day')\n        )"
        },
        {
          "file": "django/forms/widgets.py",
          "type": "function",
          "name": "__init__",
          "class_name": "SelectDateWidget",
          "code": "def __init__(self, attrs=None, years=None, months=None, empty_label=None):\n        self.attrs = attrs or {}\n\n        # Optional list or tuple of years to use in the \"year\" select box.\n        if years:\n            self.years = years\n        else:\n            this_year = datetime.date.today().year\n            self.years = range(this_year, this_year + 10)\n\n        # Optional dict of months to use in the \"month\" select box.\n        if months:\n            self.months = months\n        else:\n            self.months = MONTHS\n\n        # Optional string, list, or tuple to use as empty_label.\n        if isinstance(empty_label, (list, tuple)):\n            if not len(empty_label) == 3:\n                raise ValueError('empty_label list/tuple must have 3 elements.')\n\n            self.year_none_value = (0, empty_label[0])\n            self.month_none_value = (0, empty_label[1])\n            self.day_none_value = (0, empty_label[2])\n        else:\n            if empty_label is not None:\n                self.none_value = (0, empty_label)\n\n            self.year_none_value = self.none_value\n            self.month_none_value = self.none_value\n            self.day_none_value = self.none_value"
        },
        {
          "file": "django/forms/widgets.py",
          "type": "function",
          "name": "format_value",
          "class_name": "SelectDateWidget",
          "code": "def format_value(self, value):\n        \"\"\"\n        Return a dict containing the year, month, and day of the current value.\n        Use dict instead of a datetime to allow invalid dates such as February\n        31 to display correctly.\n        \"\"\"\n        year, month, day = None, None, None\n        if isinstance(value, (datetime.date, datetime.datetime)):\n            year, month, day = value.year, value.month, value.day\n        elif isinstance(value, str):\n            match = self.date_re.match(value)\n            if match:\n                year, month, day = [int(val) for val in match.groups()]\n            elif settings.USE_L10N:\n                input_format = get_format('DATE_INPUT_FORMATS')[0]\n                try:\n                    d = datetime.datetime.strptime(value, input_format)\n                except ValueError:\n                    pass\n                else:\n                    year, month, day = d.year, d.month, d.day\n        return {'year': year, 'month': month, 'day': day}"
        },
        {
          "file": "django/forms/widgets.py",
          "type": "function",
          "name": "value_from_datadict",
          "class_name": "SelectDateWidget",
          "code": "def value_from_datadict(self, data, files, name):\n        y = data.get(self.year_field % name)\n        m = data.get(self.month_field % name)\n        d = data.get(self.day_field % name)\n        if y == m == d == \"0\":\n            return None\n        if y and m and d:\n            if settings.USE_L10N:\n                input_format = get_format('DATE_INPUT_FORMATS')[0]\n                try:\n                    date_value = datetime.date(int(y), int(m), int(d))\n                except ValueError:\n                    return '%s-%s-%s' % (y, m, d)\n                else:\n                    date_value = datetime_safe.new_date(date_value)\n                    return date_value.strftime(input_format)\n            else:\n                return '%s-%s-%s' % (y, m, d)\n        return data.get(name)"
        }
      ]
    },
    {
      "pr_number": 7097,
      "pr_title": "Fixed #27068 -- Unified form field initial data retrieval.",
      "pr_body": "https://code.djangoproject.com/ticket/27068\n",
      "issue_id": 27068,
      "issue_title": "Acquire form's initial data more consistently",
      "issue_body": "In\nBoundField\n, sometimes initial data\n​\nhandles callables\n. While other times,\n​\nit doesn't\n. This can lead to a theoretical bug when the initial data is a callable, but the field is disabled.\nOther times, the initial callable is handled, but microseconds are\n​\nnot chopped\nleading to theoretical false positives that data has changed.\nInitial data should be handled more consistently across all cases.\nMarking as a bug due to the theoretical edge cases that can produce unexpected results. Noticed these inconsistencies while working on\n#27037\n.",
      "issue_closed_at": "2016-08-18T19:51:32",
      "base_commit": "13857b45ca54a519a58361238442af84262c0d23",
      "changes": [
        {
          "file": "django/forms/boundfield.py",
          "type": "function",
          "name": "value",
          "class_name": "BoundField",
          "code": "def value(self):\n        \"\"\"\n        Returns the value for this BoundField, using the initial value if\n        the form is not bound or the data otherwise.\n        \"\"\"\n        if not self.form.is_bound:\n            data = self.initial\n        else:\n            data = self.field.bound_data(\n                self.data, self.form.initial.get(self.name, self.field.initial)\n            )\n        return self.field.prepare_value(data)"
        },
        {
          "file": "django/forms/boundfield.py",
          "type": "function",
          "name": "id_for_label",
          "class_name": "BoundField",
          "code": "def id_for_label(self):\n        \"\"\"\n        Wrapper around the field widget's `id_for_label` method.\n        Useful, for example, for focusing on this field regardless of whether\n        it has a single widget or a MultiWidget.\n        \"\"\"\n        widget = self.field.widget\n        id_ = widget.attrs.get('id') or self.auto_id\n        return widget.id_for_label(id_)"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "_clean_fields",
          "class_name": "BaseForm",
          "code": "def _clean_fields(self):\n        for name, field in self.fields.items():\n            # value_from_datadict() gets the data from the data dictionaries.\n            # Each widget type knows how to retrieve its own data, because some\n            # widgets split data over several HTML fields.\n            if field.disabled:\n                value = self.initial.get(name, field.initial)\n            else:\n                value = field.widget.value_from_datadict(self.data, self.files, self.add_prefix(name))\n            try:\n                if isinstance(field, FileField):\n                    initial = self.initial.get(name, field.initial)\n                    value = field.clean(value, initial)\n                else:\n                    value = field.clean(value)\n                self.cleaned_data[name] = value\n                if hasattr(self, 'clean_%s' % name):\n                    value = getattr(self, 'clean_%s' % name)()\n                    self.cleaned_data[name] = value\n            except ValidationError as e:\n                self.add_error(name, e)"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "changed_data",
          "class_name": "BaseForm",
          "code": "def changed_data(self):\n        data = []\n        for name, field in self.fields.items():\n            prefixed_name = self.add_prefix(name)\n            data_value = field.widget.value_from_datadict(self.data, self.files, prefixed_name)\n            if not field.show_hidden_initial:\n                initial_value = self.initial.get(name, field.initial)\n                if callable(initial_value):\n                    initial_value = initial_value()\n            else:\n                initial_prefixed_name = self.add_initial_prefix(name)\n                hidden_widget = field.hidden_widget()\n                try:\n                    initial_value = field.to_python(hidden_widget.value_from_datadict(\n                        self.data, self.files, initial_prefixed_name))\n                except ValidationError:\n                    # Always assume data has changed if validation fails.\n                    data.append(name)\n                    continue\n            if field.has_changed(initial_value, data_value):\n                data.append(name)\n        return data"
        },
        {
          "file": "django/forms/forms.py",
          "type": "function",
          "name": "visible_fields",
          "class_name": "BaseForm",
          "code": "def visible_fields(self):\n        \"\"\"\n        Returns a list of BoundField objects that aren't hidden fields.\n        The opposite of the hidden_fields() method.\n        \"\"\"\n        return [field for field in self if not field.is_hidden]"
        }
      ]
    },
    {
      "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            ]"
        }
      ]
    }
  ]
}