123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484 |
- from typing import Callable, List
- import pytest
- import reflex as rx
- from reflex.constants.compiler import Hooks, Imports
- from reflex.event import (
- Event,
- EventActionsMixin,
- EventChain,
- EventHandler,
- EventSpec,
- call_event_handler,
- event,
- fix_events,
- )
- from reflex.state import BaseState
- from reflex.utils import format
- from reflex.vars.base import Field, LiteralVar, Var, VarData, field
- def make_var(value) -> Var:
- """Make a variable.
- Args:
- value: The value of the var.
- Returns:
- The var.
- """
- return Var(_js_expr=value)
- def test_create_event():
- """Test creating an event."""
- event = Event(token="token", name="state.do_thing", payload={"arg": "value"})
- assert event.token == "token"
- assert event.name == "state.do_thing"
- assert event.payload == {"arg": "value"}
- def test_call_event_handler():
- """Test that calling an event handler creates an event spec."""
- def test_fn():
- pass
- test_fn.__qualname__ = "test_fn"
- def test_fn_with_args(_, arg1, arg2):
- pass
- test_fn_with_args.__qualname__ = "test_fn_with_args"
- handler = EventHandler(fn=test_fn)
- event_spec = handler()
- assert event_spec.handler == handler
- assert event_spec.args == ()
- assert format.format_event(event_spec) == 'Event("test_fn", {})'
- handler = EventHandler(fn=test_fn_with_args)
- event_spec = handler(make_var("first"), make_var("second"))
- # Test passing vars as args.
- assert event_spec.handler == handler
- assert event_spec.args[0][0].equals(Var(_js_expr="arg1"))
- assert event_spec.args[0][1].equals(Var(_js_expr="first"))
- assert event_spec.args[1][0].equals(Var(_js_expr="arg2"))
- assert event_spec.args[1][1].equals(Var(_js_expr="second"))
- assert (
- format.format_event(event_spec)
- == 'Event("test_fn_with_args", {arg1:first,arg2:second})'
- )
- # Passing args as strings should format differently.
- event_spec = handler("first", "second")
- assert (
- format.format_event(event_spec)
- == 'Event("test_fn_with_args", {arg1:"first",arg2:"second"})'
- )
- first, second = 123, "456"
- handler = EventHandler(fn=test_fn_with_args)
- event_spec = handler(first, second)
- assert (
- format.format_event(event_spec)
- == 'Event("test_fn_with_args", {arg1:123,arg2:"456"})'
- )
- assert event_spec.handler == handler
- assert event_spec.args[0][0].equals(Var(_js_expr="arg1"))
- assert event_spec.args[0][1].equals(LiteralVar.create(first))
- assert event_spec.args[1][0].equals(Var(_js_expr="arg2"))
- assert event_spec.args[1][1].equals(LiteralVar.create(second))
- handler = EventHandler(fn=test_fn_with_args)
- with pytest.raises(TypeError):
- handler(test_fn)
- def test_call_event_handler_partial():
- """Calling an EventHandler with incomplete args returns an EventSpec that can be extended."""
- def test_fn_with_args(_, arg1, arg2):
- pass
- test_fn_with_args.__qualname__ = "test_fn_with_args"
- def spec(a2: Var[str]) -> List[Var[str]]:
- return [a2]
- handler = EventHandler(fn=test_fn_with_args, state_full_name="BigState")
- event_spec = handler(make_var("first"))
- event_spec2 = call_event_handler(event_spec, spec)
- assert event_spec.handler == handler
- assert len(event_spec.args) == 1
- assert event_spec.args[0][0].equals(Var(_js_expr="arg1"))
- assert event_spec.args[0][1].equals(Var(_js_expr="first"))
- assert (
- format.format_event(event_spec)
- == 'Event("BigState.test_fn_with_args", {arg1:first})'
- )
- assert event_spec2 is not event_spec
- assert event_spec2.handler == handler
- assert len(event_spec2.args) == 2
- assert event_spec2.args[0][0].equals(Var(_js_expr="arg1"))
- assert event_spec2.args[0][1].equals(Var(_js_expr="first"))
- assert event_spec2.args[1][0].equals(Var(_js_expr="arg2"))
- assert event_spec2.args[1][1].equals(Var(_js_expr="_a2", _var_type=str))
- assert (
- format.format_event(event_spec2)
- == 'Event("BigState.test_fn_with_args", {arg1:first,arg2:_a2})'
- )
- @pytest.mark.parametrize(
- ("arg1", "arg2"),
- (
- (1, 2),
- (1, "2"),
- ({"a": 1}, {"b": 2}),
- ),
- )
- def test_fix_events(arg1, arg2):
- """Test that chaining an event handler with args formats the payload correctly.
- Args:
- arg1: The first arg passed to the handler.
- arg2: The second arg passed to the handler.
- """
- def test_fn_with_args(_, arg1, arg2):
- pass
- test_fn_with_args.__qualname__ = "test_fn_with_args"
- handler = EventHandler(fn=test_fn_with_args)
- event_spec = handler(arg1, arg2)
- event = fix_events([event_spec], token="foo")[0]
- assert event.name == test_fn_with_args.__qualname__
- assert event.token == "foo"
- assert event.payload == {"arg1": arg1, "arg2": arg2}
- @pytest.mark.parametrize(
- "input,output",
- [
- (
- ("/path", None, None),
- 'Event("_redirect", {path:"/path",external:false,replace:false})',
- ),
- (
- ("/path", True, None),
- 'Event("_redirect", {path:"/path",external:true,replace:false})',
- ),
- (
- ("/path", False, None),
- 'Event("_redirect", {path:"/path",external:false,replace:false})',
- ),
- (
- (Var(_js_expr="path"), None, None),
- 'Event("_redirect", {path:path,external:false,replace:false})',
- ),
- (
- ("/path", None, True),
- 'Event("_redirect", {path:"/path",external:false,replace:true})',
- ),
- (
- ("/path", True, True),
- 'Event("_redirect", {path:"/path",external:true,replace:true})',
- ),
- ],
- )
- def test_event_redirect(input, output):
- """Test the event redirect function.
- Args:
- input: The input for running the test.
- output: The expected output to validate the test.
- """
- path, is_external, replace = input
- kwargs = {}
- if is_external is not None:
- kwargs["is_external"] = is_external
- if replace is not None:
- kwargs["replace"] = replace
- spec = event.redirect(path, **kwargs)
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_redirect"
- assert format.format_event(spec) == output
- def test_event_console_log():
- """Test the event console log function."""
- spec = event.console_log("message")
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_call_function"
- assert spec.args[0][0].equals(Var(_js_expr="function"))
- assert spec.args[0][1].equals(
- Var('(() => (console["log"]("message")))', _var_type=Callable)
- )
- assert (
- format.format_event(spec)
- == 'Event("_call_function", {function:(() => (console["log"]("message"))),callback:null})'
- )
- spec = event.console_log(Var(_js_expr="message"))
- assert (
- format.format_event(spec)
- == 'Event("_call_function", {function:(() => (console["log"](message))),callback:null})'
- )
- spec2 = event.console_log(Var(_js_expr="message2")).add_args(Var("throwaway"))
- assert (
- format.format_event(spec2)
- == 'Event("_call_function", {function:(() => (console["log"](message2))),callback:null})'
- )
- def test_event_window_alert():
- """Test the event window alert function."""
- spec = event.window_alert("message")
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_call_function"
- assert spec.args[0][0].equals(Var(_js_expr="function"))
- assert spec.args[0][1].equals(
- Var('(() => (window["alert"]("message")))', _var_type=Callable)
- )
- assert (
- format.format_event(spec)
- == 'Event("_call_function", {function:(() => (window["alert"]("message"))),callback:null})'
- )
- spec = event.window_alert(Var(_js_expr="message"))
- assert (
- format.format_event(spec)
- == 'Event("_call_function", {function:(() => (window["alert"](message))),callback:null})'
- )
- spec2 = event.window_alert(Var(_js_expr="message2")).add_args(Var("throwaway"))
- assert (
- format.format_event(spec2)
- == 'Event("_call_function", {function:(() => (window["alert"](message2))),callback:null})'
- )
- def test_set_focus():
- """Test the event set focus function."""
- spec = event.set_focus("input1")
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_set_focus"
- assert spec.args[0][0].equals(Var(_js_expr="ref"))
- assert spec.args[0][1].equals(LiteralVar.create("ref_input1"))
- assert format.format_event(spec) == 'Event("_set_focus", {ref:"ref_input1"})'
- spec = event.set_focus("input1")
- assert format.format_event(spec) == 'Event("_set_focus", {ref:"ref_input1"})'
- def test_set_value():
- """Test the event window alert function."""
- spec = event.set_value("input1", "")
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_set_value"
- assert spec.args[0][0].equals(Var(_js_expr="ref"))
- assert spec.args[0][1].equals(LiteralVar.create("ref_input1"))
- assert spec.args[1][0].equals(Var(_js_expr="value"))
- assert spec.args[1][1].equals(LiteralVar.create(""))
- assert (
- format.format_event(spec) == 'Event("_set_value", {ref:"ref_input1",value:""})'
- )
- spec = event.set_value("input1", Var(_js_expr="message"))
- assert (
- format.format_event(spec)
- == 'Event("_set_value", {ref:"ref_input1",value:message})'
- )
- def test_remove_cookie():
- """Test the event remove_cookie."""
- spec = event.remove_cookie("testkey")
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_remove_cookie"
- assert spec.args[0][0].equals(Var(_js_expr="key"))
- assert spec.args[0][1].equals(LiteralVar.create("testkey"))
- assert spec.args[1][0].equals(Var(_js_expr="options"))
- assert spec.args[1][1].equals(LiteralVar.create({"path": "/"}))
- assert (
- format.format_event(spec)
- == 'Event("_remove_cookie", {key:"testkey",options:({ ["path"] : "/" })})'
- )
- def test_remove_cookie_with_options():
- """Test the event remove_cookie with options."""
- options = {
- "path": "/foo",
- "domain": "example.com",
- "secure": True,
- "sameSite": "strict",
- }
- spec = event.remove_cookie("testkey", options)
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_remove_cookie"
- assert spec.args[0][0].equals(Var(_js_expr="key"))
- assert spec.args[0][1].equals(LiteralVar.create("testkey"))
- assert spec.args[1][0].equals(Var(_js_expr="options"))
- assert spec.args[1][1].equals(LiteralVar.create(options))
- assert (
- format.format_event(spec)
- == f'Event("_remove_cookie", {{key:"testkey",options:{LiteralVar.create(options)!s}}})'
- )
- def test_clear_local_storage():
- """Test the event clear_local_storage."""
- spec = event.clear_local_storage()
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_clear_local_storage"
- assert not spec.args
- assert format.format_event(spec) == 'Event("_clear_local_storage", {})'
- def test_remove_local_storage():
- """Test the event remove_local_storage."""
- spec = event.remove_local_storage("testkey")
- assert isinstance(spec, EventSpec)
- assert spec.handler.fn.__qualname__ == "_remove_local_storage"
- assert spec.args[0][0].equals(Var(_js_expr="key"))
- assert spec.args[0][1].equals(LiteralVar.create("testkey"))
- assert (
- format.format_event(spec) == 'Event("_remove_local_storage", {key:"testkey"})'
- )
- def test_event_actions():
- """Test DOM event actions, like stopPropagation and preventDefault."""
- # EventHandler
- handler = EventHandler(fn=lambda: None)
- assert not handler.event_actions
- sp_handler = handler.stop_propagation
- assert handler is not sp_handler
- assert sp_handler.event_actions == {"stopPropagation": True}
- pd_handler = handler.prevent_default
- assert handler is not pd_handler
- assert pd_handler.event_actions == {"preventDefault": True}
- both_handler = sp_handler.prevent_default
- assert both_handler is not sp_handler
- assert both_handler.event_actions == {
- "stopPropagation": True,
- "preventDefault": True,
- }
- throttle_handler = handler.throttle(300)
- assert handler is not throttle_handler
- assert throttle_handler.event_actions == {"throttle": 300}
- debounce_handler = handler.debounce(300)
- assert handler is not debounce_handler
- assert debounce_handler.event_actions == {"debounce": 300}
- all_handler = handler.stop_propagation.prevent_default.throttle(200).debounce(100)
- assert handler is not all_handler
- assert all_handler.event_actions == {
- "stopPropagation": True,
- "preventDefault": True,
- "throttle": 200,
- "debounce": 100,
- }
- assert not handler.event_actions
- # Convert to EventSpec should carry event actions
- sp_handler2 = handler.stop_propagation.throttle(200)
- spec = sp_handler2()
- assert spec.event_actions == {"stopPropagation": True, "throttle": 200}
- assert spec.event_actions == sp_handler2.event_actions
- assert spec.event_actions is not sp_handler2.event_actions
- # But it should be a copy!
- assert spec.event_actions is not sp_handler2.event_actions
- spec2 = spec.prevent_default
- assert spec is not spec2
- assert spec2.event_actions == {
- "stopPropagation": True,
- "preventDefault": True,
- "throttle": 200,
- }
- assert spec2.event_actions != spec.event_actions
- # The original handler should still not be touched.
- assert not handler.event_actions
- def test_event_actions_on_state():
- class EventActionState(BaseState):
- @rx.event
- def handler(self):
- pass
- handler = EventActionState.handler
- assert isinstance(handler, EventHandler)
- assert not handler.event_actions
- sp_handler = EventActionState.handler.stop_propagation
- assert isinstance(sp_handler, EventActionsMixin)
- assert sp_handler.event_actions == {"stopPropagation": True}
- # should NOT affect other references to the handler
- assert not handler.event_actions
- def test_event_var_data():
- class S(BaseState):
- x: Field[int] = field(0)
- @event
- def s(self, value: int):
- pass
- # Handler doesn't have any _var_data because it's just a str
- handler_var = Var.create(S.s)
- assert handler_var._get_all_var_data() is None
- # Ensure spec carries _var_data
- spec_var = Var.create(S.s(S.x))
- assert spec_var._get_all_var_data() == S.x._get_all_var_data()
- # Needed to instantiate the EventChain
- def _args_spec(value: Var[int]) -> tuple[Var[int]]:
- return (value,)
- # Ensure chain carries _var_data
- chain_var = Var.create(
- EventChain(
- events=[S.s(S.x)],
- args_spec=_args_spec,
- invocation=rx.vars.FunctionStringVar.create(""),
- )
- )
- assert chain_var._get_all_var_data() == S.x._get_all_var_data()
- chain_var_data = Var.create(
- EventChain(
- events=[],
- args_spec=_args_spec,
- )
- )._get_all_var_data()
- assert chain_var_data is not None
- assert chain_var_data == VarData(
- imports=Imports.EVENTS,
- hooks={Hooks.EVENTS: None},
- )
- def test_event_bound_method() -> None:
- class S(BaseState):
- @event
- def e(self, arg: str):
- print(arg)
- class Wrapper:
- def get_handler(self, arg: Var[str]):
- return S.e(arg)
- w = Wrapper()
- _ = rx.input(on_change=w.get_handler)
|