
* wip type transforming serializers * old python sucks * typing fixups * Expose the `to` parameter on `rx.serializer` for type conversion Serializers can also return a tuple of `(serialized_value, type)`, if both ways are specified, then the returned value MUST match the `to` parameter. When initializing a new rx.Var, if `_var_is_string` is not specified and the serializer returns a `str` type, then mark `_var_is_string=True` to indicate that the Var should be treated like a string literal. Include datetime, color, types, and paths as "serializing to str" type. Avoid other changes at this point to reduce fallout from this change: Notably, the `serialize_str` function does NOT cast to `str`, which would cause existing code to treat all Var initialized with a str as a str literal even though this was NOT the default before. Update test cases to accomodate these changes. * Raise deprecation warning for rx.Var.create with string literal In the future, we will treat strings as string literals in the JS code. To get a Var that is not treated like a string, pass _var_is_string=False. This will allow our serializers to automatically identify cast string literals with less special cases (and the special cases need to be explicitly identified). * Add test case for mismatched serialized types * fix old python * Remove serializer returning a tuple feature Simplify the logic; instead of making a wrapper function that returns a tuple, just save the type conversions in a separate global. * Reset the LRU cache when adding new serializers --------- Co-authored-by: Masen Furer <m_github@0x26.net>
237 lines
7.1 KiB
Python
237 lines
7.1 KiB
Python
import datetime
|
|
from enum import Enum
|
|
from pathlib import Path
|
|
from typing import Any, Dict, List, Type
|
|
|
|
import pytest
|
|
|
|
from reflex.base import Base
|
|
from reflex.components.core.colors import Color
|
|
from reflex.utils import serializers
|
|
from reflex.vars import Var
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"type_,expected",
|
|
[(str, True), (dict, True), (Dict[int, int], True), (Enum, True)],
|
|
)
|
|
def test_has_serializer(type_: Type, expected: bool):
|
|
"""Test that has_serializer returns the correct value.
|
|
|
|
|
|
Args:
|
|
type_: The type to check.
|
|
expected: The expected result.
|
|
"""
|
|
assert serializers.has_serializer(type_) == expected
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"type_,expected",
|
|
[
|
|
(str, serializers.serialize_str),
|
|
(list, serializers.serialize_list),
|
|
(tuple, serializers.serialize_list),
|
|
(set, serializers.serialize_list),
|
|
(dict, serializers.serialize_dict),
|
|
(List[str], serializers.serialize_list),
|
|
(Dict[int, int], serializers.serialize_dict),
|
|
(datetime.datetime, serializers.serialize_datetime),
|
|
(datetime.date, serializers.serialize_datetime),
|
|
(datetime.time, serializers.serialize_datetime),
|
|
(datetime.timedelta, serializers.serialize_datetime),
|
|
(int, serializers.serialize_primitive),
|
|
(float, serializers.serialize_primitive),
|
|
(bool, serializers.serialize_primitive),
|
|
(Enum, serializers.serialize_enum),
|
|
],
|
|
)
|
|
def test_get_serializer(type_: Type, expected: serializers.Serializer):
|
|
"""Test that get_serializer returns the correct value.
|
|
|
|
|
|
Args:
|
|
type_: The type to check.
|
|
expected: The expected result.
|
|
"""
|
|
assert serializers.get_serializer(type_) == expected
|
|
|
|
|
|
def test_add_serializer():
|
|
"""Test that adding a serializer works."""
|
|
|
|
class Foo:
|
|
"""A test class."""
|
|
|
|
def __init__(self, name: str):
|
|
self.name = name
|
|
|
|
def serialize_foo(value: Foo) -> str:
|
|
"""Serialize an foo to a string.
|
|
|
|
Args:
|
|
value: The value to serialize.
|
|
|
|
Returns:
|
|
The serialized value.
|
|
"""
|
|
return value.name
|
|
|
|
# Initially there should be no serializer for int.
|
|
assert not serializers.has_serializer(Foo)
|
|
assert serializers.serialize(Foo("hi")) is None
|
|
|
|
# Register the serializer.
|
|
assert serializers.serializer(serialize_foo) == serialize_foo
|
|
|
|
# There should now be a serializer for int.
|
|
assert serializers.has_serializer(Foo)
|
|
assert serializers.get_serializer(Foo) == serialize_foo
|
|
assert serializers.serialize(Foo("hi")) == "hi"
|
|
|
|
# Remove the serializer.
|
|
serializers.SERIALIZERS.pop(Foo)
|
|
# LRU cache will still have the serializer, so we need to clear it.
|
|
assert serializers.has_serializer(Foo)
|
|
serializers.get_serializer.cache_clear()
|
|
assert not serializers.has_serializer(Foo)
|
|
|
|
|
|
class StrEnum(str, Enum):
|
|
"""An enum also inheriting from str."""
|
|
|
|
FOO = "foo"
|
|
BAR = "bar"
|
|
|
|
|
|
class TestEnum(Enum):
|
|
"""A lone enum class."""
|
|
|
|
FOO = "foo"
|
|
BAR = "bar"
|
|
|
|
|
|
class EnumWithPrefix(Enum):
|
|
"""An enum with a serializer adding a prefix."""
|
|
|
|
FOO = "foo"
|
|
BAR = "bar"
|
|
|
|
|
|
@serializers.serializer
|
|
def serialize_EnumWithPrefix(enum: EnumWithPrefix) -> str:
|
|
return "prefix_" + enum.value
|
|
|
|
|
|
class BaseSubclass(Base):
|
|
"""A class inheriting from Base for testing."""
|
|
|
|
ts: datetime.timedelta = datetime.timedelta(1, 1, 1)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"value,expected",
|
|
[
|
|
("test", "test"),
|
|
(1, "1"),
|
|
(1.0, "1.0"),
|
|
(True, "true"),
|
|
(False, "false"),
|
|
(None, "null"),
|
|
([1, 2, 3], "[1, 2, 3]"),
|
|
([1, "2", 3.0], '[1, "2", 3.0]'),
|
|
([{"key": 1}, {"key": 2}], '[{"key": 1}, {"key": 2}]'),
|
|
(StrEnum.FOO, "foo"),
|
|
([StrEnum.FOO, StrEnum.BAR], '["foo", "bar"]'),
|
|
(
|
|
{"key1": [1, 2, 3], "key2": [StrEnum.FOO, StrEnum.BAR]},
|
|
'{"key1": [1, 2, 3], "key2": ["foo", "bar"]}',
|
|
),
|
|
(EnumWithPrefix.FOO, "prefix_foo"),
|
|
([EnumWithPrefix.FOO, EnumWithPrefix.BAR], '["prefix_foo", "prefix_bar"]'),
|
|
(
|
|
{"key1": EnumWithPrefix.FOO, "key2": EnumWithPrefix.BAR},
|
|
'{"key1": "prefix_foo", "key2": "prefix_bar"}',
|
|
),
|
|
(TestEnum.FOO, "foo"),
|
|
([TestEnum.FOO, TestEnum.BAR], '["foo", "bar"]'),
|
|
(
|
|
{"key1": TestEnum.FOO, "key2": TestEnum.BAR},
|
|
'{"key1": "foo", "key2": "bar"}',
|
|
),
|
|
(
|
|
BaseSubclass(ts=datetime.timedelta(1, 1, 1)),
|
|
'{"ts": "1 day, 0:00:01.000001"}',
|
|
),
|
|
(
|
|
[1, Var.create_safe("hi"), Var.create_safe("bye", _var_is_local=False)],
|
|
'[1, "hi", bye]',
|
|
),
|
|
(
|
|
(1, Var.create_safe("hi"), Var.create_safe("bye", _var_is_local=False)),
|
|
'[1, "hi", bye]',
|
|
),
|
|
({1: 2, 3: 4}, '{"1": 2, "3": 4}'),
|
|
(
|
|
{1: Var.create_safe("hi"), 3: Var.create_safe("bye", _var_is_local=False)},
|
|
'{"1": "hi", "3": bye}',
|
|
),
|
|
(datetime.datetime(2021, 1, 1, 1, 1, 1, 1), "2021-01-01 01:01:01.000001"),
|
|
(datetime.date(2021, 1, 1), "2021-01-01"),
|
|
(datetime.time(1, 1, 1, 1), "01:01:01.000001"),
|
|
(datetime.timedelta(1, 1, 1), "1 day, 0:00:01.000001"),
|
|
(
|
|
[datetime.timedelta(1, 1, 1), datetime.timedelta(1, 1, 2)],
|
|
'["1 day, 0:00:01.000001", "1 day, 0:00:01.000002"]',
|
|
),
|
|
(Color(color="slate", shade=1), "var(--slate-1)"),
|
|
(Color(color="orange", shade=1, alpha=True), "var(--orange-a1)"),
|
|
(Color(color="accent", shade=1, alpha=True), "var(--accent-a1)"),
|
|
],
|
|
)
|
|
def test_serialize(value: Any, expected: str):
|
|
"""Test that serialize returns the correct value.
|
|
|
|
|
|
Args:
|
|
value: The value to serialize.
|
|
expected: The expected result.
|
|
"""
|
|
assert serializers.serialize(value) == expected
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"value,expected,exp_var_is_string",
|
|
[
|
|
("test", "test", False),
|
|
(1, "1", False),
|
|
(1.0, "1.0", False),
|
|
(True, "true", False),
|
|
(False, "false", False),
|
|
([1, 2, 3], "[1, 2, 3]", False),
|
|
([{"key": 1}, {"key": 2}], '[{"key": 1}, {"key": 2}]', False),
|
|
(StrEnum.FOO, "foo", False),
|
|
([StrEnum.FOO, StrEnum.BAR], '["foo", "bar"]', False),
|
|
(
|
|
BaseSubclass(ts=datetime.timedelta(1, 1, 1)),
|
|
'{"ts": "1 day, 0:00:01.000001"}',
|
|
False,
|
|
),
|
|
(datetime.datetime(2021, 1, 1, 1, 1, 1, 1), "2021-01-01 01:01:01.000001", True),
|
|
(Color(color="slate", shade=1), "var(--slate-1)", True),
|
|
(BaseSubclass, "BaseSubclass", True),
|
|
(Path("."), ".", True),
|
|
],
|
|
)
|
|
def test_serialize_var_to_str(value: Any, expected: str, exp_var_is_string: bool):
|
|
"""Test that serialize with `to=str` passed to a Var is marked with _var_is_string.
|
|
|
|
Args:
|
|
value: The value to serialize.
|
|
expected: The expected result.
|
|
exp_var_is_string: The expected value of _var_is_string.
|
|
"""
|
|
v = Var.create_safe(value)
|
|
assert v._var_full_name == expected
|
|
assert v._var_is_string == exp_var_is_string
|