2019-10-12 13:23:03 +01:00
|
|
|
"""YAML fields"""
|
|
|
|
import yaml
|
|
|
|
from django import forms
|
|
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
|
|
|
|
|
2020-05-20 12:00:45 +01:00
|
|
|
class CodeMirrorWidget(forms.Textarea):
|
|
|
|
"""Custom Textarea-based Widget that triggers a CodeMirror editor"""
|
|
|
|
|
|
|
|
# CodeMirror mode to enable
|
|
|
|
mode: str
|
|
|
|
|
|
|
|
def __init__(self, *args, mode="yaml", **kwargs):
|
|
|
|
super().__init__(*args, **kwargs)
|
|
|
|
self.mode = mode
|
|
|
|
|
|
|
|
def render(self, *args, **kwargs):
|
2020-09-16 20:54:35 +01:00
|
|
|
attrs = kwargs.setdefault("attrs", {})
|
|
|
|
attrs.setdefault("class", "")
|
2020-05-20 12:00:45 +01:00
|
|
|
attrs["class"] += " codemirror"
|
|
|
|
attrs["data-cm-mode"] = self.mode
|
|
|
|
return super().render(*args, **kwargs)
|
|
|
|
|
|
|
|
|
2019-10-12 13:23:03 +01:00
|
|
|
class InvalidYAMLInput(str):
|
|
|
|
"""Invalid YAML String type"""
|
|
|
|
|
|
|
|
|
|
|
|
class YAMLString(str):
|
|
|
|
"""YAML String type"""
|
|
|
|
|
|
|
|
|
2020-08-15 20:04:22 +01:00
|
|
|
class YAMLField(forms.JSONField):
|
2019-10-12 13:23:03 +01:00
|
|
|
"""Django's JSON Field converted to YAML"""
|
|
|
|
|
|
|
|
default_error_messages = {
|
2019-12-31 11:51:16 +00:00
|
|
|
"invalid": _("'%(value)s' value must be valid YAML."),
|
2019-10-12 13:23:03 +01:00
|
|
|
}
|
|
|
|
widget = forms.Textarea
|
|
|
|
|
|
|
|
def to_python(self, value):
|
|
|
|
if self.disabled:
|
|
|
|
return value
|
|
|
|
if value in self.empty_values:
|
|
|
|
return None
|
|
|
|
if isinstance(value, (list, dict, int, float, YAMLString)):
|
|
|
|
return value
|
|
|
|
try:
|
|
|
|
converted = yaml.safe_load(value)
|
|
|
|
except yaml.YAMLError:
|
|
|
|
raise forms.ValidationError(
|
2019-12-31 11:51:16 +00:00
|
|
|
self.error_messages["invalid"], code="invalid", params={"value": value},
|
2019-10-12 13:23:03 +01:00
|
|
|
)
|
|
|
|
if isinstance(converted, str):
|
|
|
|
return YAMLString(converted)
|
2020-09-18 23:00:55 +01:00
|
|
|
if converted is None:
|
|
|
|
return {}
|
2019-10-12 13:23:03 +01:00
|
|
|
return converted
|
|
|
|
|
|
|
|
def bound_data(self, data, initial):
|
|
|
|
if self.disabled:
|
|
|
|
return initial
|
|
|
|
try:
|
|
|
|
return yaml.safe_load(data)
|
|
|
|
except yaml.YAMLError:
|
|
|
|
return InvalidYAMLInput(data)
|
|
|
|
|
|
|
|
def prepare_value(self, value):
|
|
|
|
if isinstance(value, InvalidYAMLInput):
|
|
|
|
return value
|
2020-02-23 14:27:11 +00:00
|
|
|
return yaml.dump(value, explicit_start=True, default_flow_style=False)
|
2019-10-12 13:23:03 +01: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 yaml.dump(initial, sort_keys=True) != yaml.dump(data, sort_keys=True)
|