From 8c403b17f94a4ace87bfdd7d40bbc3afb18f7577 Mon Sep 17 00:00:00 2001 From: Qi Zhao Date: Thu, 8 Oct 2020 21:51:14 +0800 Subject: [PATCH] [3.1.x] Fixed #32080 -- Fixed displaying Unicode chars in forms.JSONField and read-only JSONField values in admin. Backport of de81676b51e4dad510ef387c3ae625f9091fe57f from master --- AUTHORS | 1 + django/contrib/admin/utils.py | 3 ++- django/forms/fields.py | 2 +- docs/releases/3.1.3.txt | 5 +++++ tests/admin_utils/tests.py | 1 + tests/forms_tests/field_tests/test_jsonfield.py | 6 ++++++ 6 files changed, 16 insertions(+), 2 deletions(-) diff --git a/AUTHORS b/AUTHORS index 80748781cbd..4cdfa6b1b03 100644 --- a/AUTHORS +++ b/AUTHORS @@ -740,6 +740,7 @@ answer newbie questions, and generally made Django that much better: Priyansh Saxena Przemysław Buczkowski Przemysław Suliga + Qi Zhao Rachel Tobin Rachel Willmer Radek Švarz diff --git a/django/contrib/admin/utils.py b/django/contrib/admin/utils.py index 446083e6593..ba06da52110 100644 --- a/django/contrib/admin/utils.py +++ b/django/contrib/admin/utils.py @@ -1,5 +1,6 @@ import datetime import decimal +import json from collections import defaultdict from django.core.exceptions import FieldDoesNotExist @@ -400,7 +401,7 @@ def display_for_field(value, field, empty_value_display): return format_html('{}', value.url, value) elif isinstance(field, models.JSONField) and value: try: - return field.get_prep_value(value) + return json.dumps(value, ensure_ascii=False, cls=field.encoder) except TypeError: return display_for_value(value, empty_value_display) else: diff --git a/django/forms/fields.py b/django/forms/fields.py index 36dad72704c..0efeb298cd6 100644 --- a/django/forms/fields.py +++ b/django/forms/fields.py @@ -1265,7 +1265,7 @@ class JSONField(CharField): def prepare_value(self, value): if isinstance(value, InvalidJSONInput): return value - return json.dumps(value, cls=self.encoder) + return json.dumps(value, ensure_ascii=False, cls=self.encoder) def has_changed(self, initial, data): if super().has_changed(initial, data): diff --git a/docs/releases/3.1.3.txt b/docs/releases/3.1.3.txt index 853ec7a48d5..e2bb2d0df3d 100644 --- a/docs/releases/3.1.3.txt +++ b/docs/releases/3.1.3.txt @@ -11,3 +11,8 @@ Bugfixes * Fixed a regression in Django 3.1.2 that caused the incorrect height of the admin changelist search bar (:ticket:`32072`). + +* Fixed displaying Unicode characters in + :class:`forms.JSONField ` and read-only + :class:`models.JSONField ` values in the admin + (:ticket:`32080`). diff --git a/tests/admin_utils/tests.py b/tests/admin_utils/tests.py index cf0d3e064e3..da9d24bed57 100644 --- a/tests/admin_utils/tests.py +++ b/tests/admin_utils/tests.py @@ -184,6 +184,7 @@ class UtilsTests(SimpleTestCase): ({'a': {'b': 'c'}}, '{"a": {"b": "c"}}'), (['a', 'b'], '["a", "b"]'), ('a', '"a"'), + ({'a': '你好 世界'}, '{"a": "你好 世界"}'), ({('a', 'b'): 'c'}, "{('a', 'b'): 'c'}"), # Invalid JSON. ] for value, display_value in tests: diff --git a/tests/forms_tests/field_tests/test_jsonfield.py b/tests/forms_tests/field_tests/test_jsonfield.py index 892a23a6a7a..f9e4281a019 100644 --- a/tests/forms_tests/field_tests/test_jsonfield.py +++ b/tests/forms_tests/field_tests/test_jsonfield.py @@ -29,6 +29,12 @@ class JSONFieldTest(SimpleTestCase): self.assertEqual(field.prepare_value({'a': 'b'}), '{"a": "b"}') self.assertEqual(field.prepare_value(None), 'null') self.assertEqual(field.prepare_value('foo'), '"foo"') + self.assertEqual(field.prepare_value('你好,世界'), '"你好,世界"') + self.assertEqual(field.prepare_value({'a': '😀🐱'}), '{"a": "😀🐱"}') + self.assertEqual( + field.prepare_value(["你好,世界", "jaźń"]), + '["你好,世界", "jaźń"]', + ) def test_widget(self): field = JSONField()