serializers.py 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516
  1. """Serializers used to convert Var types to JSON strings."""
  2. from __future__ import annotations
  3. import contextlib
  4. import dataclasses
  5. import functools
  6. import inspect
  7. import json
  8. import warnings
  9. from datetime import date, datetime, time, timedelta
  10. from enum import Enum
  11. from pathlib import Path
  12. from typing import (
  13. Any,
  14. Callable,
  15. Literal,
  16. Sequence,
  17. Set,
  18. Type,
  19. TypeVar,
  20. Union,
  21. get_type_hints,
  22. overload,
  23. )
  24. from uuid import UUID
  25. from pydantic import BaseModel as BaseModelV2
  26. from pydantic.v1 import BaseModel as BaseModelV1
  27. from reflex.base import Base
  28. from reflex.constants.colors import Color, format_color
  29. from reflex.utils import console, types
  30. # Mapping from type to a serializer.
  31. # The serializer should convert the type to a JSON object.
  32. SerializedType = Union[str, bool, int, float, list, dict, None]
  33. Serializer = Callable[[Any], SerializedType]
  34. SERIALIZERS: dict[Type, Serializer] = {}
  35. SERIALIZER_TYPES: dict[Type, Type] = {}
  36. SERIALIZED_FUNCTION = TypeVar("SERIALIZED_FUNCTION", bound=Serializer)
  37. @overload
  38. def serializer(
  39. fn: None = None,
  40. to: Type[SerializedType] | None = None,
  41. overwrite: bool | None = None,
  42. ) -> Callable[[SERIALIZED_FUNCTION], SERIALIZED_FUNCTION]: ...
  43. @overload
  44. def serializer(
  45. fn: SERIALIZED_FUNCTION,
  46. to: Type[SerializedType] | None = None,
  47. overwrite: bool | None = None,
  48. ) -> SERIALIZED_FUNCTION: ...
  49. def serializer(
  50. fn: SERIALIZED_FUNCTION | None = None,
  51. to: Any = None,
  52. overwrite: bool | None = None,
  53. ) -> SERIALIZED_FUNCTION | Callable[[SERIALIZED_FUNCTION], SERIALIZED_FUNCTION]:
  54. """Decorator to add a serializer for a given type.
  55. Args:
  56. fn: The function to decorate.
  57. to: The type returned by the serializer. If this is `str`, then any Var created from this type will be treated as a string.
  58. overwrite: Whether to overwrite the existing serializer.
  59. Returns:
  60. The decorated function.
  61. """
  62. def wrapper(fn: SERIALIZED_FUNCTION) -> SERIALIZED_FUNCTION:
  63. # Check the type hints to get the type of the argument.
  64. type_hints = get_type_hints(fn)
  65. args = [arg for arg in type_hints if arg != "return"]
  66. # Make sure the function takes a single argument.
  67. if len(args) != 1:
  68. raise ValueError("Serializer must take a single argument.")
  69. # Get the type of the argument.
  70. type_ = type_hints[args[0]]
  71. # Make sure the type is not already registered.
  72. registered_fn = SERIALIZERS.get(type_)
  73. if registered_fn is not None and registered_fn != fn and overwrite is not True:
  74. message = f"Overwriting serializer for type {type_} from {registered_fn.__module__}:{registered_fn.__qualname__} to {fn.__module__}:{fn.__qualname__}."
  75. if overwrite is False:
  76. raise ValueError(message)
  77. caller_frame = next(
  78. filter(
  79. lambda frame: frame.filename != __file__,
  80. inspect.getouterframes(inspect.currentframe()),
  81. ),
  82. None,
  83. )
  84. file_info = (
  85. f"(at {caller_frame.filename}:{caller_frame.lineno})"
  86. if caller_frame
  87. else ""
  88. )
  89. console.warn(
  90. f"{message} Call rx.serializer with `overwrite=True` if this is intentional. {file_info}"
  91. )
  92. to_type = to or type_hints.get("return")
  93. # Apply type transformation if requested
  94. if to_type:
  95. SERIALIZER_TYPES[type_] = to_type
  96. get_serializer_type.cache_clear()
  97. # Register the serializer.
  98. SERIALIZERS[type_] = fn
  99. get_serializer.cache_clear()
  100. # Return the function.
  101. return fn
  102. if fn is not None:
  103. return wrapper(fn)
  104. return wrapper
  105. @overload
  106. def serialize(
  107. value: Any, get_type: Literal[True]
  108. ) -> tuple[SerializedType | None, types.GenericType | None]: ...
  109. @overload
  110. def serialize(value: Any, get_type: Literal[False]) -> SerializedType | None: ...
  111. @overload
  112. def serialize(value: Any) -> SerializedType | None: ...
  113. def serialize(
  114. value: Any, get_type: bool = False
  115. ) -> Union[
  116. SerializedType | None,
  117. tuple[SerializedType | None, types.GenericType | None],
  118. ]:
  119. """Serialize the value to a JSON string.
  120. Args:
  121. value: The value to serialize.
  122. get_type: Whether to return the type of the serialized value.
  123. Returns:
  124. The serialized value, or None if a serializer is not found.
  125. """
  126. # Get the serializer for the type.
  127. serializer = get_serializer(type(value))
  128. # If there is no serializer, return None.
  129. if serializer is None:
  130. if dataclasses.is_dataclass(value) and not isinstance(value, type):
  131. return {k.name: getattr(value, k.name) for k in dataclasses.fields(value)}
  132. if get_type:
  133. return None, None
  134. return None
  135. # Serialize the value.
  136. serialized = serializer(value)
  137. # Return the serialized value and the type.
  138. if get_type:
  139. return serialized, get_serializer_type(type(value))
  140. else:
  141. return serialized
  142. @functools.lru_cache
  143. def get_serializer(type_: Type) -> Serializer | None:
  144. """Get the serializer for the type.
  145. Args:
  146. type_: The type to get the serializer for.
  147. Returns:
  148. The serializer for the type, or None if there is no serializer.
  149. """
  150. # First, check if the type is registered.
  151. serializer = SERIALIZERS.get(type_)
  152. if serializer is not None:
  153. return serializer
  154. # If the type is not registered, check if it is a subclass of a registered type.
  155. for registered_type, serializer in reversed(SERIALIZERS.items()):
  156. if types._issubclass(type_, registered_type):
  157. return serializer
  158. # If there is no serializer, return None.
  159. return None
  160. @functools.lru_cache
  161. def get_serializer_type(type_: Type) -> Type | None:
  162. """Get the converted type for the type after serializing.
  163. Args:
  164. type_: The type to get the serializer type for.
  165. Returns:
  166. The serialized type for the type, or None if there is no type conversion registered.
  167. """
  168. # First, check if the type is registered.
  169. serializer = SERIALIZER_TYPES.get(type_)
  170. if serializer is not None:
  171. return serializer
  172. # If the type is not registered, check if it is a subclass of a registered type.
  173. for registered_type, serializer in reversed(SERIALIZER_TYPES.items()):
  174. if types._issubclass(type_, registered_type):
  175. return serializer
  176. # If there is no serializer, return None.
  177. return None
  178. def has_serializer(type_: Type, into_type: Type | None = None) -> bool:
  179. """Check if there is a serializer for the type.
  180. Args:
  181. type_: The type to check.
  182. into_type: The type to serialize into.
  183. Returns:
  184. Whether there is a serializer for the type.
  185. """
  186. serializer_for_type = get_serializer(type_)
  187. return serializer_for_type is not None and (
  188. into_type is None or get_serializer_type(type_) == into_type
  189. )
  190. def can_serialize(type_: Type, into_type: Type | None = None) -> bool:
  191. """Check if there is a serializer for the type.
  192. Args:
  193. type_: The type to check.
  194. into_type: The type to serialize into.
  195. Returns:
  196. Whether there is a serializer for the type.
  197. """
  198. return has_serializer(type_, into_type) or (
  199. isinstance(type_, type)
  200. and dataclasses.is_dataclass(type_)
  201. and (into_type is None or into_type is dict)
  202. )
  203. @serializer(to=str)
  204. def serialize_type(value: type) -> str:
  205. """Serialize a python type.
  206. Args:
  207. value: the type to serialize.
  208. Returns:
  209. The serialized type.
  210. """
  211. return value.__name__
  212. @serializer(to=dict)
  213. def serialize_base(value: Base) -> dict:
  214. """Serialize a Base instance.
  215. Args:
  216. value : The Base to serialize.
  217. Returns:
  218. The serialized Base.
  219. """
  220. from reflex.vars.base import Var
  221. return {
  222. k: v for k, v in value.dict().items() if isinstance(v, Var) or not callable(v)
  223. }
  224. @serializer(to=dict)
  225. def serialize_base_model_v1(model: BaseModelV1) -> dict:
  226. """Serialize a pydantic v1 BaseModel instance.
  227. Args:
  228. model: The BaseModel to serialize.
  229. Returns:
  230. The serialized BaseModel.
  231. """
  232. return model.dict()
  233. if BaseModelV1 is not BaseModelV2:
  234. @serializer(to=dict)
  235. def serialize_base_model_v2(model: BaseModelV2) -> dict:
  236. """Serialize a pydantic v2 BaseModel instance.
  237. Args:
  238. model: The BaseModel to serialize.
  239. Returns:
  240. The serialized BaseModel.
  241. """
  242. return model.model_dump()
  243. @serializer
  244. def serialize_set(value: Set) -> list:
  245. """Serialize a set to a JSON serializable list.
  246. Args:
  247. value: The set to serialize.
  248. Returns:
  249. The serialized list.
  250. """
  251. return list(value)
  252. @serializer
  253. def serialize_sequence(value: Sequence) -> list:
  254. """Serialize a sequence to a JSON serializable list.
  255. Args:
  256. value: The sequence to serialize.
  257. Returns:
  258. The serialized list.
  259. """
  260. return list(value)
  261. @serializer(to=str)
  262. def serialize_datetime(dt: Union[date, datetime, time, timedelta]) -> str:
  263. """Serialize a datetime to a JSON string.
  264. Args:
  265. dt: The datetime to serialize.
  266. Returns:
  267. The serialized datetime.
  268. """
  269. return str(dt)
  270. @serializer(to=str)
  271. def serialize_path(path: Path) -> str:
  272. """Serialize a pathlib.Path to a JSON string.
  273. Args:
  274. path: The path to serialize.
  275. Returns:
  276. The serialized path.
  277. """
  278. return str(path.as_posix())
  279. @serializer
  280. def serialize_enum(en: Enum) -> str:
  281. """Serialize a enum to a JSON string.
  282. Args:
  283. en: The enum to serialize.
  284. Returns:
  285. The serialized enum.
  286. """
  287. return en.value
  288. @serializer(to=str)
  289. def serialize_uuid(uuid: UUID) -> str:
  290. """Serialize a UUID to a JSON string.
  291. Args:
  292. uuid: The UUID to serialize.
  293. Returns:
  294. The serialized UUID.
  295. """
  296. return str(uuid)
  297. @serializer(to=str)
  298. def serialize_color(color: Color) -> str:
  299. """Serialize a color.
  300. Args:
  301. color: The color to serialize.
  302. Returns:
  303. The serialized color.
  304. """
  305. return format_color(color.color, color.shade, color.alpha)
  306. with contextlib.suppress(ImportError):
  307. from pandas import DataFrame
  308. def format_dataframe_values(df: DataFrame) -> list[list[Any]]:
  309. """Format dataframe values to a list of lists.
  310. Args:
  311. df: The dataframe to format.
  312. Returns:
  313. The dataframe as a list of lists.
  314. """
  315. return [
  316. [str(d) if isinstance(d, (list, tuple)) else d for d in data]
  317. for data in list(df.values.tolist())
  318. ]
  319. @serializer
  320. def serialize_dataframe(df: DataFrame) -> dict:
  321. """Serialize a pandas dataframe.
  322. Args:
  323. df: The dataframe to serialize.
  324. Returns:
  325. The serialized dataframe.
  326. """
  327. return {
  328. "columns": df.columns.tolist(),
  329. "data": format_dataframe_values(df),
  330. }
  331. with contextlib.suppress(ImportError):
  332. from plotly.graph_objects import Figure, layout
  333. from plotly.io import to_json
  334. @serializer
  335. def serialize_figure(figure: Figure) -> dict:
  336. """Serialize a plotly figure.
  337. Args:
  338. figure: The figure to serialize.
  339. Returns:
  340. The serialized figure.
  341. """
  342. return json.loads(str(to_json(figure)))
  343. @serializer
  344. def serialize_template(template: layout.Template) -> dict:
  345. """Serialize a plotly template.
  346. Args:
  347. template: The template to serialize.
  348. Returns:
  349. The serialized template.
  350. """
  351. return {
  352. "data": json.loads(str(to_json(template.data))),
  353. "layout": json.loads(str(to_json(template.layout))),
  354. }
  355. with contextlib.suppress(ImportError):
  356. import base64
  357. import io
  358. from PIL.Image import MIME
  359. from PIL.Image import Image as Img
  360. @serializer
  361. def serialize_image(image: Img) -> str:
  362. """Serialize a plotly figure.
  363. Args:
  364. image: The image to serialize.
  365. Returns:
  366. The serialized image.
  367. """
  368. buff = io.BytesIO()
  369. image_format = getattr(image, "format", None) or "PNG"
  370. image.save(buff, format=image_format)
  371. image_bytes = buff.getvalue()
  372. base64_image = base64.b64encode(image_bytes).decode("utf-8")
  373. try:
  374. # Newer method to get the mime type, but does not always work.
  375. mime_type = image.get_format_mimetype() # pyright: ignore [reportAttributeAccessIssue]
  376. except AttributeError:
  377. try:
  378. # Fallback method
  379. mime_type = MIME[image_format]
  380. except KeyError:
  381. # Unknown mime_type: warn and return image/png and hope the browser can sort it out.
  382. warnings.warn( # noqa: B028
  383. f"Unknown mime type for {image} {image_format}. Defaulting to image/png"
  384. )
  385. mime_type = "image/png"
  386. return f"data:{mime_type};base64,{base64_image}"