Если вы хотите закодировать произвольный enum.Enum
член в JSON, а затем декодировать его как тот же член перечисления (а не просто value
атрибут члена перечисления ), вы можете сделать это, написав собственный JSONEncoder
класс и функцию декодирования для передачи в качестве object_hook
аргумента в json.load()
или json.loads()
:
PUBLIC_ENUMS = {
'Status': Status,
}
class EnumEncoder(json.JSONEncoder):
def default(self, obj):
if type(obj) in PUBLIC_ENUMS.values():
return {"__enum__": str(obj)}
return json.JSONEncoder.default(self, obj)
def as_enum(d):
if "__enum__" in d:
name, member = d["__enum__"].split(".")
return getattr(PUBLIC_ENUMS[name], member)
else:
return d
as_enum
Функция опирается на JSON будучи закодирован с использованием EnumEncoder
, или что - то, ведет себя тождественно ему.
Ограничение для членов PUBLIC_ENUMS
необходимо, чтобы избежать использования злонамеренно созданного текста, например, для того, чтобы обмануть вызывающий код для сохранения личной информации (например, секретного ключа, используемого приложением) в несвязанном поле базы данных, откуда она может быть раскрыта. (см. http://chat.stackoverflow.com/transcript/message/35999686#35999686 ).
Пример использования:
>>> data = {
... "action": "frobnicate",
... "status": Status.success
... }
>>> text = json.dumps(data, cls=EnumEncoder)
>>> text
'{"status": {"__enum__": "Status.success"}, "action": "frobnicate"}'
>>> json.loads(text, object_hook=as_enum)
{'status': <Status.success: 0>, 'action': 'frobnicate'}