reflex/tests/units/utils/test_serializers.py
Thomas Brandého 59dd54c049
add workflow to check dependencies on release branch (#4050)
* add workflow to check dependencies on release branch

* rename action to follow convention of other actions

* update workflow

* bump poetry version

* relock deps

* update check to ignore pyright and ruff

* oops, you saw nothing

* split dep check in two job

* fix frontend dep check

* fix stuff

* hmm yeah

* nope nope nope

* sigh

* bump js versions for some packages

* fix some warnings in tests

* fix tests

* try some options

* try to set asyncio policy

* debug dep check

* fix attempt for backend dep

* clean up output for backend check

* run bun outdated on reflex-web to catch most of the packages

* fix python version

* fix python version

* add missing env

* fix bun command

* fix workdir of frontend check

* update packages version

* up-pin plotly.js version

* add debug ouput

* clean frontend dep check output

* fix output

* fix async tests for redis

* relock poetry.lock

* Non-async functions do not need pytest_asyncio.fixture

* test_state: close StateManagerRedis connection in test to avoid warning

---------

Co-authored-by: Masen Furer <m_github@0x26.net>
2024-10-07 20:57:02 +02:00

243 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,
),
(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