
* enable PTH rule * fix import in test_call_script * fix units tests * reorder ruff rules * Update reflex/utils/build.py Co-authored-by: Masen Furer <m_github@0x26.net> * format pyproject.toml --------- Co-authored-by: Masen Furer <m_github@0x26.net>
241 lines
6.7 KiB
Python
241 lines
6.7 KiB
Python
import datetime
|
|
import json
|
|
from enum import Enum
|
|
from pathlib import Path
|
|
from typing import Any, Type
|
|
|
|
import pytest
|
|
|
|
from reflex.base import Base
|
|
from reflex.components.core.colors import Color
|
|
from reflex.utils import serializers
|
|
from reflex.utils.format import json_dumps
|
|
from reflex.vars.base import LiteralVar
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"type_,expected",
|
|
[(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",
|
|
[
|
|
(datetime.datetime, serializers.serialize_datetime),
|
|
(datetime.date, serializers.serialize_datetime),
|
|
(datetime.time, serializers.serialize_datetime),
|
|
(datetime.timedelta, serializers.serialize_datetime),
|
|
(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 FooBarEnum(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, None),
|
|
([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",
|
|
},
|
|
),
|
|
(FooBarEnum.FOO, "foo"),
|
|
([FooBarEnum.FOO, FooBarEnum.BAR], ["foo", "bar"]),
|
|
(
|
|
{"key1": FooBarEnum.FOO, "key2": FooBarEnum.BAR},
|
|
{
|
|
"key1": "foo",
|
|
"key2": "bar",
|
|
},
|
|
),
|
|
(
|
|
BaseSubclass(ts=datetime.timedelta(1, 1, 1)),
|
|
{
|
|
"ts": "1 day, 0:00:01.000001",
|
|
},
|
|
),
|
|
(
|
|
[1, LiteralVar.create("hi")],
|
|
[1, "hi"],
|
|
),
|
|
(
|
|
(1, LiteralVar.create("hi")),
|
|
[1, "hi"],
|
|
),
|
|
({1: 2, 3: 4}, {1: 2, 3: 4}),
|
|
(
|
|
{1: LiteralVar.create("hi")},
|
|
{1: "hi"},
|
|
),
|
|
(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 json.loads(json_dumps(value)) == json.loads(json_dumps(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,
|
|
),
|
|
(datetime.date(2021, 1, 1), '"2021-01-01"', 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 = LiteralVar.create(value)
|
|
assert str(v) == expected
|