Source code for chatsky.utils.devel.json_serialization
"""
Serialization
-------------
Tools that provide JSON serialization via Pickle for unserializable objects.
- :py:data:`~.PickleEncodedValue`:
A field annotated with this will be pickled/unpickled during JSON-serialization/validation.
- :py:data:`~.JSONSerializableDict`:
A dictionary field annotated with this will make all its items smart-serializable:
If an item is serializable -- nothing would change.
Otherwise -- it will be serialized via pickle.
- :py:class:`~.JSONSerializableExtras`:
A pydantic base class that makes its extra fields a `JSONSerializableDict`.
"""
from base64 import decodebytes, encodebytes
from copy import deepcopy
from pickle import dumps, loads
from typing import Any, Dict, List, Union
from typing_extensions import TypeAlias
from pydantic import (
JsonValue,
RootModel,
BaseModel,
model_validator,
model_serializer,
)
from pydantic_core import PydanticSerializationError
_JSON_EXTRA_FIELDS_KEYS = "__pickled_extra_fields__"
"""
This key is used in :py:data:`~.JSONSerializableDict` to remember pickled items.
"""
Serializable: TypeAlias = Dict[str, Union[JsonValue, List[Any], Dict[str, Any], Any]]
"""Type annotation for objects supported by :py:func:`~.json_pickle_serializer`."""
[docs]
class _WrapperModel(RootModel):
"""
Wrapper model for testing whether an object is serializable to JSON.
"""
root: Any
[docs]
def pickle_serializer(value: Any) -> str:
"""
Serializer function that serializes any pickle-serializable value into JSON-serializable.
Serializes value with pickle and encodes bytes as base64 string.
:param value: Pickle-serializable object.
:return: String-encoded object.
"""
return encodebytes(dumps(value)).decode()
[docs]
def pickle_validator(value: str) -> Any:
"""
Validator function that validates base64 string encoded bytes as a pickle-serializable value.
Decodes base64 string and validates value with pickle.
:param value: String-encoded string.
:return: Pickle-serializable object.
"""
return loads(decodebytes(value.encode()))
[docs]
def json_pickle_serializer(model: Serializable) -> Serializable:
"""
Serializer function that serializes a dictionary or Pydantic object to JSON.
For every object field, it checks whether the field is JSON serializable,
and if it's not, serializes it using pickle.
It also keeps track of pickle-serializable field names in a special list.
:param model: Pydantic model object or a dictionary.
:original_serializer: Original serializer function for model.
:return: model with all the fields serialized to JSON.
"""
extra_fields = list()
model_copy = deepcopy(model)
for field_name, field_value in model_copy.items():
try:
if isinstance(field_value, bytes):
raise PydanticSerializationError("")
else:
model_copy[field_name] = _WrapperModel(root=field_value).model_dump(mode="json")
except PydanticSerializationError:
model_copy[field_name] = pickle_serializer(field_value)
extra_fields += [field_name]
if len(extra_fields) > 0:
model_copy[_JSON_EXTRA_FIELDS_KEYS] = extra_fields
return model_copy
[docs]
def json_pickle_validator(model: Serializable) -> Serializable:
"""
Validator function that validates a JSON dictionary to a python dictionary.
For every object field, it checks if the field is pickle-serialized,
and if it is, validates it using pickle.
:param model: Pydantic model object or a dictionary.
:return: model with all the fields serialized to JSON.
"""
model_copy = deepcopy(model)
if _JSON_EXTRA_FIELDS_KEYS in model.keys():
for extra_key in model[_JSON_EXTRA_FIELDS_KEYS]:
extra_value = model[extra_key]
model_copy[extra_key] = pickle_validator(extra_value)
del model_copy[_JSON_EXTRA_FIELDS_KEYS]
return model_copy