test_serializers.py 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. import datetime
  2. import json
  3. from enum import Enum
  4. from pathlib import Path
  5. from typing import Any, Type
  6. import pytest
  7. from reflex.base import Base
  8. from reflex.components.core.colors import Color
  9. from reflex.utils import serializers
  10. from reflex.utils.format import json_dumps
  11. from reflex.vars.base import LiteralVar
  12. @pytest.mark.parametrize(
  13. "type_,expected",
  14. [(Enum, True)],
  15. )
  16. def test_has_serializer(type_: Type, expected: bool):
  17. """Test that has_serializer returns the correct value.
  18. Args:
  19. type_: The type to check.
  20. expected: The expected result.
  21. """
  22. assert serializers.has_serializer(type_) == expected
  23. @pytest.mark.parametrize(
  24. "type_,expected",
  25. [
  26. (datetime.datetime, serializers.serialize_datetime),
  27. (datetime.date, serializers.serialize_datetime),
  28. (datetime.time, serializers.serialize_datetime),
  29. (datetime.timedelta, serializers.serialize_datetime),
  30. (Enum, serializers.serialize_enum),
  31. ],
  32. )
  33. def test_get_serializer(type_: Type, expected: serializers.Serializer):
  34. """Test that get_serializer returns the correct value.
  35. Args:
  36. type_: The type to check.
  37. expected: The expected result.
  38. """
  39. assert serializers.get_serializer(type_) == expected
  40. def test_add_serializer():
  41. """Test that adding a serializer works."""
  42. class Foo:
  43. """A test class."""
  44. def __init__(self, name: str):
  45. self.name = name
  46. def serialize_foo(value: Foo) -> str:
  47. """Serialize an foo to a string.
  48. Args:
  49. value: The value to serialize.
  50. Returns:
  51. The serialized value.
  52. """
  53. return value.name
  54. # Initially there should be no serializer for int.
  55. assert not serializers.has_serializer(Foo)
  56. assert serializers.serialize(Foo("hi")) is None
  57. # Register the serializer.
  58. assert serializers.serializer(serialize_foo) == serialize_foo
  59. # There should now be a serializer for int.
  60. assert serializers.has_serializer(Foo)
  61. assert serializers.get_serializer(Foo) == serialize_foo
  62. assert serializers.serialize(Foo("hi")) == "hi"
  63. # Remove the serializer.
  64. serializers.SERIALIZERS.pop(Foo)
  65. # LRU cache will still have the serializer, so we need to clear it.
  66. assert serializers.has_serializer(Foo)
  67. serializers.get_serializer.cache_clear()
  68. assert not serializers.has_serializer(Foo)
  69. class StrEnum(str, Enum):
  70. """An enum also inheriting from str."""
  71. FOO = "foo"
  72. BAR = "bar"
  73. class FooBarEnum(Enum):
  74. """A lone enum class."""
  75. FOO = "foo"
  76. BAR = "bar"
  77. class EnumWithPrefix(Enum):
  78. """An enum with a serializer adding a prefix."""
  79. FOO = "foo"
  80. BAR = "bar"
  81. @serializers.serializer
  82. def serialize_EnumWithPrefix(enum: EnumWithPrefix) -> str:
  83. return "prefix_" + enum.value
  84. class BaseSubclass(Base):
  85. """A class inheriting from Base for testing."""
  86. ts: datetime.timedelta = datetime.timedelta(1, 1, 1)
  87. @pytest.mark.parametrize(
  88. "value,expected",
  89. [
  90. ("test", "test"),
  91. (1, 1),
  92. (1.0, 1.0),
  93. (True, True),
  94. (False, False),
  95. (None, None),
  96. ([1, 2, 3], [1, 2, 3]),
  97. ([1, "2", 3.0], [1, "2", 3.0]),
  98. ([{"key": 1}, {"key": 2}], [{"key": 1}, {"key": 2}]),
  99. (StrEnum.FOO, "foo"),
  100. ([StrEnum.FOO, StrEnum.BAR], ["foo", "bar"]),
  101. (
  102. {"key1": [1, 2, 3], "key2": [StrEnum.FOO, StrEnum.BAR]},
  103. {
  104. "key1": [1, 2, 3],
  105. "key2": ["foo", "bar"],
  106. },
  107. ),
  108. (EnumWithPrefix.FOO, "prefix_foo"),
  109. ([EnumWithPrefix.FOO, EnumWithPrefix.BAR], ["prefix_foo", "prefix_bar"]),
  110. (
  111. {"key1": EnumWithPrefix.FOO, "key2": EnumWithPrefix.BAR},
  112. {
  113. "key1": "prefix_foo",
  114. "key2": "prefix_bar",
  115. },
  116. ),
  117. (FooBarEnum.FOO, "foo"),
  118. ([FooBarEnum.FOO, FooBarEnum.BAR], ["foo", "bar"]),
  119. (
  120. {"key1": FooBarEnum.FOO, "key2": FooBarEnum.BAR},
  121. {
  122. "key1": "foo",
  123. "key2": "bar",
  124. },
  125. ),
  126. (
  127. BaseSubclass(ts=datetime.timedelta(1, 1, 1)),
  128. {
  129. "ts": "1 day, 0:00:01.000001",
  130. },
  131. ),
  132. (
  133. [1, LiteralVar.create("hi")],
  134. [1, "hi"],
  135. ),
  136. (
  137. (1, LiteralVar.create("hi")),
  138. [1, "hi"],
  139. ),
  140. ({1: 2, 3: 4}, {1: 2, 3: 4}),
  141. (
  142. {1: LiteralVar.create("hi")},
  143. {1: "hi"},
  144. ),
  145. (datetime.datetime(2021, 1, 1, 1, 1, 1, 1), "2021-01-01 01:01:01.000001"),
  146. (datetime.date(2021, 1, 1), "2021-01-01"),
  147. (datetime.time(1, 1, 1, 1), "01:01:01.000001"),
  148. (datetime.timedelta(1, 1, 1), "1 day, 0:00:01.000001"),
  149. (
  150. [datetime.timedelta(1, 1, 1), datetime.timedelta(1, 1, 2)],
  151. ["1 day, 0:00:01.000001", "1 day, 0:00:01.000002"],
  152. ),
  153. (Color(color="slate", shade=1), "var(--slate-1)"),
  154. (Color(color="orange", shade=1, alpha=True), "var(--orange-a1)"),
  155. (Color(color="accent", shade=1, alpha=True), "var(--accent-a1)"),
  156. ],
  157. )
  158. def test_serialize(value: Any, expected: str):
  159. """Test that serialize returns the correct value.
  160. Args:
  161. value: The value to serialize.
  162. expected: The expected result.
  163. """
  164. assert json.loads(json_dumps(value)) == json.loads(json_dumps(expected))
  165. @pytest.mark.parametrize(
  166. "value,expected,exp_var_is_string",
  167. [
  168. ("test", '"test"', False),
  169. (1, "1", False),
  170. (1.0, "1.0", False),
  171. (True, "true", False),
  172. (False, "false", False),
  173. ([1, 2, 3], "[1, 2, 3]", False),
  174. ([{"key": 1}, {"key": 2}], '[({ ["key"] : 1 }), ({ ["key"] : 2 })]', False),
  175. (StrEnum.FOO, '"foo"', False),
  176. ([StrEnum.FOO, StrEnum.BAR], '["foo", "bar"]', False),
  177. (
  178. BaseSubclass(ts=datetime.timedelta(1, 1, 1)),
  179. '({ ["ts"] : "1 day, 0:00:01.000001" })',
  180. False,
  181. ),
  182. (
  183. datetime.datetime(2021, 1, 1, 1, 1, 1, 1),
  184. '"2021-01-01 01:01:01.000001"',
  185. True,
  186. ),
  187. (datetime.date(2021, 1, 1), '"2021-01-01"', True),
  188. (Color(color="slate", shade=1), '"var(--slate-1)"', True),
  189. (BaseSubclass, '"BaseSubclass"', True),
  190. (Path(), '"."', True),
  191. ],
  192. )
  193. def test_serialize_var_to_str(value: Any, expected: str, exp_var_is_string: bool):
  194. """Test that serialize with `to=str` passed to a Var is marked with _var_is_string.
  195. Args:
  196. value: The value to serialize.
  197. expected: The expected result.
  198. exp_var_is_string: The expected value of _var_is_string.
  199. """
  200. v = LiteralVar.create(value)
  201. assert str(v) == expected