1
0

test_event.py 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484
  1. from typing import Callable, List
  2. import pytest
  3. import reflex as rx
  4. from reflex.constants.compiler import Hooks, Imports
  5. from reflex.event import (
  6. Event,
  7. EventActionsMixin,
  8. EventChain,
  9. EventHandler,
  10. EventSpec,
  11. call_event_handler,
  12. event,
  13. fix_events,
  14. )
  15. from reflex.state import BaseState
  16. from reflex.utils import format
  17. from reflex.vars.base import Field, LiteralVar, Var, VarData, field
  18. def make_var(value) -> Var:
  19. """Make a variable.
  20. Args:
  21. value: The value of the var.
  22. Returns:
  23. The var.
  24. """
  25. return Var(_js_expr=value)
  26. def test_create_event():
  27. """Test creating an event."""
  28. event = Event(token="token", name="state.do_thing", payload={"arg": "value"})
  29. assert event.token == "token"
  30. assert event.name == "state.do_thing"
  31. assert event.payload == {"arg": "value"}
  32. def test_call_event_handler():
  33. """Test that calling an event handler creates an event spec."""
  34. def test_fn():
  35. pass
  36. test_fn.__qualname__ = "test_fn"
  37. def test_fn_with_args(_, arg1, arg2):
  38. pass
  39. test_fn_with_args.__qualname__ = "test_fn_with_args"
  40. handler = EventHandler(fn=test_fn)
  41. event_spec = handler()
  42. assert event_spec.handler == handler
  43. assert event_spec.args == ()
  44. assert format.format_event(event_spec) == 'Event("test_fn", {})'
  45. handler = EventHandler(fn=test_fn_with_args)
  46. event_spec = handler(make_var("first"), make_var("second"))
  47. # Test passing vars as args.
  48. assert event_spec.handler == handler
  49. assert event_spec.args[0][0].equals(Var(_js_expr="arg1"))
  50. assert event_spec.args[0][1].equals(Var(_js_expr="first"))
  51. assert event_spec.args[1][0].equals(Var(_js_expr="arg2"))
  52. assert event_spec.args[1][1].equals(Var(_js_expr="second"))
  53. assert (
  54. format.format_event(event_spec)
  55. == 'Event("test_fn_with_args", {arg1:first,arg2:second})'
  56. )
  57. # Passing args as strings should format differently.
  58. event_spec = handler("first", "second")
  59. assert (
  60. format.format_event(event_spec)
  61. == 'Event("test_fn_with_args", {arg1:"first",arg2:"second"})'
  62. )
  63. first, second = 123, "456"
  64. handler = EventHandler(fn=test_fn_with_args)
  65. event_spec = handler(first, second)
  66. assert (
  67. format.format_event(event_spec)
  68. == 'Event("test_fn_with_args", {arg1:123,arg2:"456"})'
  69. )
  70. assert event_spec.handler == handler
  71. assert event_spec.args[0][0].equals(Var(_js_expr="arg1"))
  72. assert event_spec.args[0][1].equals(LiteralVar.create(first))
  73. assert event_spec.args[1][0].equals(Var(_js_expr="arg2"))
  74. assert event_spec.args[1][1].equals(LiteralVar.create(second))
  75. handler = EventHandler(fn=test_fn_with_args)
  76. with pytest.raises(TypeError):
  77. handler(test_fn)
  78. def test_call_event_handler_partial():
  79. """Calling an EventHandler with incomplete args returns an EventSpec that can be extended."""
  80. def test_fn_with_args(_, arg1, arg2):
  81. pass
  82. test_fn_with_args.__qualname__ = "test_fn_with_args"
  83. def spec(a2: Var[str]) -> List[Var[str]]:
  84. return [a2]
  85. handler = EventHandler(fn=test_fn_with_args, state_full_name="BigState")
  86. event_spec = handler(make_var("first"))
  87. event_spec2 = call_event_handler(event_spec, spec)
  88. assert event_spec.handler == handler
  89. assert len(event_spec.args) == 1
  90. assert event_spec.args[0][0].equals(Var(_js_expr="arg1"))
  91. assert event_spec.args[0][1].equals(Var(_js_expr="first"))
  92. assert (
  93. format.format_event(event_spec)
  94. == 'Event("BigState.test_fn_with_args", {arg1:first})'
  95. )
  96. assert event_spec2 is not event_spec
  97. assert event_spec2.handler == handler
  98. assert len(event_spec2.args) == 2
  99. assert event_spec2.args[0][0].equals(Var(_js_expr="arg1"))
  100. assert event_spec2.args[0][1].equals(Var(_js_expr="first"))
  101. assert event_spec2.args[1][0].equals(Var(_js_expr="arg2"))
  102. assert event_spec2.args[1][1].equals(Var(_js_expr="_a2", _var_type=str))
  103. assert (
  104. format.format_event(event_spec2)
  105. == 'Event("BigState.test_fn_with_args", {arg1:first,arg2:_a2})'
  106. )
  107. @pytest.mark.parametrize(
  108. ("arg1", "arg2"),
  109. (
  110. (1, 2),
  111. (1, "2"),
  112. ({"a": 1}, {"b": 2}),
  113. ),
  114. )
  115. def test_fix_events(arg1, arg2):
  116. """Test that chaining an event handler with args formats the payload correctly.
  117. Args:
  118. arg1: The first arg passed to the handler.
  119. arg2: The second arg passed to the handler.
  120. """
  121. def test_fn_with_args(_, arg1, arg2):
  122. pass
  123. test_fn_with_args.__qualname__ = "test_fn_with_args"
  124. handler = EventHandler(fn=test_fn_with_args)
  125. event_spec = handler(arg1, arg2)
  126. event = fix_events([event_spec], token="foo")[0]
  127. assert event.name == test_fn_with_args.__qualname__
  128. assert event.token == "foo"
  129. assert event.payload == {"arg1": arg1, "arg2": arg2}
  130. @pytest.mark.parametrize(
  131. "input,output",
  132. [
  133. (
  134. ("/path", None, None),
  135. 'Event("_redirect", {path:"/path",external:false,replace:false})',
  136. ),
  137. (
  138. ("/path", True, None),
  139. 'Event("_redirect", {path:"/path",external:true,replace:false})',
  140. ),
  141. (
  142. ("/path", False, None),
  143. 'Event("_redirect", {path:"/path",external:false,replace:false})',
  144. ),
  145. (
  146. (Var(_js_expr="path"), None, None),
  147. 'Event("_redirect", {path:path,external:false,replace:false})',
  148. ),
  149. (
  150. ("/path", None, True),
  151. 'Event("_redirect", {path:"/path",external:false,replace:true})',
  152. ),
  153. (
  154. ("/path", True, True),
  155. 'Event("_redirect", {path:"/path",external:true,replace:true})',
  156. ),
  157. ],
  158. )
  159. def test_event_redirect(input, output):
  160. """Test the event redirect function.
  161. Args:
  162. input: The input for running the test.
  163. output: The expected output to validate the test.
  164. """
  165. path, is_external, replace = input
  166. kwargs = {}
  167. if is_external is not None:
  168. kwargs["is_external"] = is_external
  169. if replace is not None:
  170. kwargs["replace"] = replace
  171. spec = event.redirect(path, **kwargs)
  172. assert isinstance(spec, EventSpec)
  173. assert spec.handler.fn.__qualname__ == "_redirect"
  174. assert format.format_event(spec) == output
  175. def test_event_console_log():
  176. """Test the event console log function."""
  177. spec = event.console_log("message")
  178. assert isinstance(spec, EventSpec)
  179. assert spec.handler.fn.__qualname__ == "_call_function"
  180. assert spec.args[0][0].equals(Var(_js_expr="function"))
  181. assert spec.args[0][1].equals(
  182. Var('(() => (console["log"]("message")))', _var_type=Callable)
  183. )
  184. assert (
  185. format.format_event(spec)
  186. == 'Event("_call_function", {function:(() => (console["log"]("message"))),callback:null})'
  187. )
  188. spec = event.console_log(Var(_js_expr="message"))
  189. assert (
  190. format.format_event(spec)
  191. == 'Event("_call_function", {function:(() => (console["log"](message))),callback:null})'
  192. )
  193. spec2 = event.console_log(Var(_js_expr="message2")).add_args(Var("throwaway"))
  194. assert (
  195. format.format_event(spec2)
  196. == 'Event("_call_function", {function:(() => (console["log"](message2))),callback:null})'
  197. )
  198. def test_event_window_alert():
  199. """Test the event window alert function."""
  200. spec = event.window_alert("message")
  201. assert isinstance(spec, EventSpec)
  202. assert spec.handler.fn.__qualname__ == "_call_function"
  203. assert spec.args[0][0].equals(Var(_js_expr="function"))
  204. assert spec.args[0][1].equals(
  205. Var('(() => (window["alert"]("message")))', _var_type=Callable)
  206. )
  207. assert (
  208. format.format_event(spec)
  209. == 'Event("_call_function", {function:(() => (window["alert"]("message"))),callback:null})'
  210. )
  211. spec = event.window_alert(Var(_js_expr="message"))
  212. assert (
  213. format.format_event(spec)
  214. == 'Event("_call_function", {function:(() => (window["alert"](message))),callback:null})'
  215. )
  216. spec2 = event.window_alert(Var(_js_expr="message2")).add_args(Var("throwaway"))
  217. assert (
  218. format.format_event(spec2)
  219. == 'Event("_call_function", {function:(() => (window["alert"](message2))),callback:null})'
  220. )
  221. def test_set_focus():
  222. """Test the event set focus function."""
  223. spec = event.set_focus("input1")
  224. assert isinstance(spec, EventSpec)
  225. assert spec.handler.fn.__qualname__ == "_set_focus"
  226. assert spec.args[0][0].equals(Var(_js_expr="ref"))
  227. assert spec.args[0][1].equals(LiteralVar.create("ref_input1"))
  228. assert format.format_event(spec) == 'Event("_set_focus", {ref:"ref_input1"})'
  229. spec = event.set_focus("input1")
  230. assert format.format_event(spec) == 'Event("_set_focus", {ref:"ref_input1"})'
  231. def test_set_value():
  232. """Test the event window alert function."""
  233. spec = event.set_value("input1", "")
  234. assert isinstance(spec, EventSpec)
  235. assert spec.handler.fn.__qualname__ == "_set_value"
  236. assert spec.args[0][0].equals(Var(_js_expr="ref"))
  237. assert spec.args[0][1].equals(LiteralVar.create("ref_input1"))
  238. assert spec.args[1][0].equals(Var(_js_expr="value"))
  239. assert spec.args[1][1].equals(LiteralVar.create(""))
  240. assert (
  241. format.format_event(spec) == 'Event("_set_value", {ref:"ref_input1",value:""})'
  242. )
  243. spec = event.set_value("input1", Var(_js_expr="message"))
  244. assert (
  245. format.format_event(spec)
  246. == 'Event("_set_value", {ref:"ref_input1",value:message})'
  247. )
  248. def test_remove_cookie():
  249. """Test the event remove_cookie."""
  250. spec = event.remove_cookie("testkey")
  251. assert isinstance(spec, EventSpec)
  252. assert spec.handler.fn.__qualname__ == "_remove_cookie"
  253. assert spec.args[0][0].equals(Var(_js_expr="key"))
  254. assert spec.args[0][1].equals(LiteralVar.create("testkey"))
  255. assert spec.args[1][0].equals(Var(_js_expr="options"))
  256. assert spec.args[1][1].equals(LiteralVar.create({"path": "/"}))
  257. assert (
  258. format.format_event(spec)
  259. == 'Event("_remove_cookie", {key:"testkey",options:({ ["path"] : "/" })})'
  260. )
  261. def test_remove_cookie_with_options():
  262. """Test the event remove_cookie with options."""
  263. options = {
  264. "path": "/foo",
  265. "domain": "example.com",
  266. "secure": True,
  267. "sameSite": "strict",
  268. }
  269. spec = event.remove_cookie("testkey", options)
  270. assert isinstance(spec, EventSpec)
  271. assert spec.handler.fn.__qualname__ == "_remove_cookie"
  272. assert spec.args[0][0].equals(Var(_js_expr="key"))
  273. assert spec.args[0][1].equals(LiteralVar.create("testkey"))
  274. assert spec.args[1][0].equals(Var(_js_expr="options"))
  275. assert spec.args[1][1].equals(LiteralVar.create(options))
  276. assert (
  277. format.format_event(spec)
  278. == f'Event("_remove_cookie", {{key:"testkey",options:{LiteralVar.create(options)!s}}})'
  279. )
  280. def test_clear_local_storage():
  281. """Test the event clear_local_storage."""
  282. spec = event.clear_local_storage()
  283. assert isinstance(spec, EventSpec)
  284. assert spec.handler.fn.__qualname__ == "_clear_local_storage"
  285. assert not spec.args
  286. assert format.format_event(spec) == 'Event("_clear_local_storage", {})'
  287. def test_remove_local_storage():
  288. """Test the event remove_local_storage."""
  289. spec = event.remove_local_storage("testkey")
  290. assert isinstance(spec, EventSpec)
  291. assert spec.handler.fn.__qualname__ == "_remove_local_storage"
  292. assert spec.args[0][0].equals(Var(_js_expr="key"))
  293. assert spec.args[0][1].equals(LiteralVar.create("testkey"))
  294. assert (
  295. format.format_event(spec) == 'Event("_remove_local_storage", {key:"testkey"})'
  296. )
  297. def test_event_actions():
  298. """Test DOM event actions, like stopPropagation and preventDefault."""
  299. # EventHandler
  300. handler = EventHandler(fn=lambda: None)
  301. assert not handler.event_actions
  302. sp_handler = handler.stop_propagation
  303. assert handler is not sp_handler
  304. assert sp_handler.event_actions == {"stopPropagation": True}
  305. pd_handler = handler.prevent_default
  306. assert handler is not pd_handler
  307. assert pd_handler.event_actions == {"preventDefault": True}
  308. both_handler = sp_handler.prevent_default
  309. assert both_handler is not sp_handler
  310. assert both_handler.event_actions == {
  311. "stopPropagation": True,
  312. "preventDefault": True,
  313. }
  314. throttle_handler = handler.throttle(300)
  315. assert handler is not throttle_handler
  316. assert throttle_handler.event_actions == {"throttle": 300}
  317. debounce_handler = handler.debounce(300)
  318. assert handler is not debounce_handler
  319. assert debounce_handler.event_actions == {"debounce": 300}
  320. all_handler = handler.stop_propagation.prevent_default.throttle(200).debounce(100)
  321. assert handler is not all_handler
  322. assert all_handler.event_actions == {
  323. "stopPropagation": True,
  324. "preventDefault": True,
  325. "throttle": 200,
  326. "debounce": 100,
  327. }
  328. assert not handler.event_actions
  329. # Convert to EventSpec should carry event actions
  330. sp_handler2 = handler.stop_propagation.throttle(200)
  331. spec = sp_handler2()
  332. assert spec.event_actions == {"stopPropagation": True, "throttle": 200}
  333. assert spec.event_actions == sp_handler2.event_actions
  334. assert spec.event_actions is not sp_handler2.event_actions
  335. # But it should be a copy!
  336. assert spec.event_actions is not sp_handler2.event_actions
  337. spec2 = spec.prevent_default
  338. assert spec is not spec2
  339. assert spec2.event_actions == {
  340. "stopPropagation": True,
  341. "preventDefault": True,
  342. "throttle": 200,
  343. }
  344. assert spec2.event_actions != spec.event_actions
  345. # The original handler should still not be touched.
  346. assert not handler.event_actions
  347. def test_event_actions_on_state():
  348. class EventActionState(BaseState):
  349. @rx.event
  350. def handler(self):
  351. pass
  352. handler = EventActionState.handler
  353. assert isinstance(handler, EventHandler)
  354. assert not handler.event_actions
  355. sp_handler = EventActionState.handler.stop_propagation
  356. assert isinstance(sp_handler, EventActionsMixin)
  357. assert sp_handler.event_actions == {"stopPropagation": True}
  358. # should NOT affect other references to the handler
  359. assert not handler.event_actions
  360. def test_event_var_data():
  361. class S(BaseState):
  362. x: Field[int] = field(0)
  363. @event
  364. def s(self, value: int):
  365. pass
  366. # Handler doesn't have any _var_data because it's just a str
  367. handler_var = Var.create(S.s)
  368. assert handler_var._get_all_var_data() is None
  369. # Ensure spec carries _var_data
  370. spec_var = Var.create(S.s(S.x))
  371. assert spec_var._get_all_var_data() == S.x._get_all_var_data()
  372. # Needed to instantiate the EventChain
  373. def _args_spec(value: Var[int]) -> tuple[Var[int]]:
  374. return (value,)
  375. # Ensure chain carries _var_data
  376. chain_var = Var.create(
  377. EventChain(
  378. events=[S.s(S.x)],
  379. args_spec=_args_spec,
  380. invocation=rx.vars.FunctionStringVar.create(""),
  381. )
  382. )
  383. assert chain_var._get_all_var_data() == S.x._get_all_var_data()
  384. chain_var_data = Var.create(
  385. EventChain(
  386. events=[],
  387. args_spec=_args_spec,
  388. )
  389. )._get_all_var_data()
  390. assert chain_var_data is not None
  391. assert chain_var_data == VarData(
  392. imports=Imports.EVENTS,
  393. hooks={Hooks.EVENTS: None},
  394. )
  395. def test_event_bound_method() -> None:
  396. class S(BaseState):
  397. @event
  398. def e(self, arg: str):
  399. print(arg)
  400. class Wrapper:
  401. def get_handler(self, arg: Var[str]):
  402. return S.e(arg)
  403. w = Wrapper()
  404. _ = rx.input(on_change=w.get_handler)