{
  "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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": "",
      "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            ]"
        }
      ]
    }
  ]
}