[3.1.x] Fixed #32080 -- Fixed displaying Unicode chars in forms.JSONField and read-only JSONField values in admin.

Backport of de81676b51 from master
This commit is contained in:
Qi Zhao 2020-10-08 21:51:14 +08:00 committed by Mariusz Felisiak
parent b0e5e7b671
commit 8c403b17f9
6 changed files with 16 additions and 2 deletions

View File

@ -740,6 +740,7 @@ answer newbie questions, and generally made Django that much better:
Priyansh Saxena <askpriyansh@gmail.com> Priyansh Saxena <askpriyansh@gmail.com>
Przemysław Buczkowski <przemub@przemub.pl> Przemysław Buczkowski <przemub@przemub.pl>
Przemysław Suliga <http://suligap.net> Przemysław Suliga <http://suligap.net>
Qi Zhao <zhaoqi99@outlook.com>
Rachel Tobin <rmtobin@me.com> Rachel Tobin <rmtobin@me.com>
Rachel Willmer <http://www.willmer.com/kb/> Rachel Willmer <http://www.willmer.com/kb/>
Radek Švarz <https://www.svarz.cz/translate/> Radek Švarz <https://www.svarz.cz/translate/>

View File

@ -1,5 +1,6 @@
import datetime import datetime
import decimal import decimal
import json
from collections import defaultdict from collections import defaultdict
from django.core.exceptions import FieldDoesNotExist from django.core.exceptions import FieldDoesNotExist
@ -400,7 +401,7 @@ def display_for_field(value, field, empty_value_display):
return format_html('<a href="{}">{}</a>', value.url, value) return format_html('<a href="{}">{}</a>', value.url, value)
elif isinstance(field, models.JSONField) and value: elif isinstance(field, models.JSONField) and value:
try: try:
return field.get_prep_value(value) return json.dumps(value, ensure_ascii=False, cls=field.encoder)
except TypeError: except TypeError:
return display_for_value(value, empty_value_display) return display_for_value(value, empty_value_display)
else: else:

View File

@ -1265,7 +1265,7 @@ class JSONField(CharField):
def prepare_value(self, value): def prepare_value(self, value):
if isinstance(value, InvalidJSONInput): if isinstance(value, InvalidJSONInput):
return value 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): def has_changed(self, initial, data):
if super().has_changed(initial, data): if super().has_changed(initial, data):

View File

@ -11,3 +11,8 @@ Bugfixes
* Fixed a regression in Django 3.1.2 that caused the incorrect height of the * Fixed a regression in Django 3.1.2 that caused the incorrect height of the
admin changelist search bar (:ticket:`32072`). admin changelist search bar (:ticket:`32072`).
* Fixed displaying Unicode characters in
:class:`forms.JSONField <django.forms.JSONField>` and read-only
:class:`models.JSONField <django.db.models.JSONField>` values in the admin
(:ticket:`32080`).

View File

@ -184,6 +184,7 @@ class UtilsTests(SimpleTestCase):
({'a': {'b': 'c'}}, '{"a": {"b": "c"}}'), ({'a': {'b': 'c'}}, '{"a": {"b": "c"}}'),
(['a', 'b'], '["a", "b"]'), (['a', 'b'], '["a", "b"]'),
('a', '"a"'), ('a', '"a"'),
({'a': '你好 世界'}, '{"a": "你好 世界"}'),
({('a', 'b'): 'c'}, "{('a', 'b'): 'c'}"), # Invalid JSON. ({('a', 'b'): 'c'}, "{('a', 'b'): 'c'}"), # Invalid JSON.
] ]
for value, display_value in tests: for value, display_value in tests:

View File

@ -29,6 +29,12 @@ class JSONFieldTest(SimpleTestCase):
self.assertEqual(field.prepare_value({'a': 'b'}), '{"a": "b"}') self.assertEqual(field.prepare_value({'a': 'b'}), '{"a": "b"}')
self.assertEqual(field.prepare_value(None), 'null') self.assertEqual(field.prepare_value(None), 'null')
self.assertEqual(field.prepare_value('foo'), '"foo"') 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): def test_widget(self):
field = JSONField() field = JSONField()