2015-05-31 05:13:58 +08:00
|
|
|
import json
|
|
|
|
|
|
|
|
from django import forms
|
2017-01-27 03:58:33 +08:00
|
|
|
from django.utils.translation import gettext_lazy as _
|
2015-05-31 05:13:58 +08:00
|
|
|
|
|
|
|
__all__ = ['JSONField']
|
|
|
|
|
|
|
|
|
2016-12-29 23:27:49 +08:00
|
|
|
class InvalidJSONInput(str):
|
2016-03-27 03:11:57 +08:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
2016-12-29 23:27:49 +08:00
|
|
|
class JSONString(str):
|
2016-11-12 09:07:31 +08:00
|
|
|
pass
|
|
|
|
|
|
|
|
|
2015-05-31 05:13:58 +08:00
|
|
|
class JSONField(forms.CharField):
|
|
|
|
default_error_messages = {
|
2019-06-28 00:39:47 +08:00
|
|
|
'invalid': _('“%(value)s” value must be valid JSON.'),
|
2015-05-31 05:13:58 +08:00
|
|
|
}
|
2016-06-13 20:09:54 +08:00
|
|
|
widget = forms.Textarea
|
2015-05-31 05:13:58 +08:00
|
|
|
|
|
|
|
def to_python(self, value):
|
2016-07-26 20:18:08 +08:00
|
|
|
if self.disabled:
|
|
|
|
return value
|
2015-05-31 05:13:58 +08:00
|
|
|
if value in self.empty_values:
|
|
|
|
return None
|
2016-11-12 09:07:31 +08:00
|
|
|
elif isinstance(value, (list, dict, int, float, JSONString)):
|
|
|
|
return value
|
2015-05-31 05:13:58 +08:00
|
|
|
try:
|
2016-11-12 09:07:31 +08:00
|
|
|
converted = json.loads(value)
|
2017-06-21 03:16:37 +08:00
|
|
|
except json.JSONDecodeError:
|
2015-05-31 05:13:58 +08:00
|
|
|
raise forms.ValidationError(
|
|
|
|
self.error_messages['invalid'],
|
|
|
|
code='invalid',
|
|
|
|
params={'value': value},
|
|
|
|
)
|
2016-12-29 23:27:49 +08:00
|
|
|
if isinstance(converted, str):
|
2016-11-12 09:07:31 +08:00
|
|
|
return JSONString(converted)
|
|
|
|
else:
|
|
|
|
return converted
|
2015-05-31 05:13:58 +08:00
|
|
|
|
2016-03-27 03:11:57 +08:00
|
|
|
def bound_data(self, data, initial):
|
|
|
|
if self.disabled:
|
|
|
|
return initial
|
|
|
|
try:
|
|
|
|
return json.loads(data)
|
2017-06-21 03:16:37 +08:00
|
|
|
except json.JSONDecodeError:
|
2016-03-27 03:11:57 +08:00
|
|
|
return InvalidJSONInput(data)
|
|
|
|
|
2015-05-31 05:13:58 +08:00
|
|
|
def prepare_value(self, value):
|
2016-03-27 03:11:57 +08:00
|
|
|
if isinstance(value, InvalidJSONInput):
|
|
|
|
return value
|
2015-05-31 05:13:58 +08:00
|
|
|
return json.dumps(value)
|
2017-09-01 14:26:09 +08:00
|
|
|
|
|
|
|
def has_changed(self, initial, data):
|
|
|
|
if super().has_changed(initial, data):
|
|
|
|
return True
|
|
|
|
# For purposes of seeing whether something has changed, True isn't the
|
|
|
|
# same as 1 and the order of keys doesn't matter.
|
|
|
|
data = self.to_python(data)
|
|
|
|
return json.dumps(initial, sort_keys=True) != json.dumps(data, sort_keys=True)
|