test_event_chain.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572
  1. """Ensure that Event Chains are properly queued and handled between frontend and backend."""
  2. from __future__ import annotations
  3. from typing import Generator
  4. import pytest
  5. from selenium.webdriver.common.by import By
  6. from reflex.testing import AppHarness, WebDriver
  7. MANY_EVENTS = 50
  8. def EventChain():
  9. """App with chained event handlers."""
  10. import asyncio
  11. import time
  12. from typing import List
  13. import reflex as rx
  14. # repeated here since the outer global isn't exported into the App module
  15. MANY_EVENTS = 50
  16. class State(rx.State):
  17. event_order: List[str] = []
  18. interim_value: str = ""
  19. def event_no_args(self):
  20. self.event_order.append("event_no_args")
  21. def event_arg(self, arg):
  22. self.event_order.append(f"event_arg:{arg}")
  23. def event_arg_repr_type(self, arg):
  24. self.event_order.append(f"event_arg_repr:{arg!r}_{type(arg).__name__}")
  25. def event_nested_1(self):
  26. self.event_order.append("event_nested_1")
  27. yield State.event_nested_2
  28. yield State.event_arg("nested_1") # type: ignore
  29. def event_nested_2(self):
  30. self.event_order.append("event_nested_2")
  31. yield State.event_nested_3
  32. yield rx.console_log("event_nested_2")
  33. yield State.event_arg("nested_2") # type: ignore
  34. def event_nested_3(self):
  35. self.event_order.append("event_nested_3")
  36. yield State.event_no_args
  37. yield State.event_arg("nested_3") # type: ignore
  38. def on_load_return_chain(self):
  39. self.event_order.append("on_load_return_chain")
  40. return [State.event_arg(1), State.event_arg(2), State.event_arg(3)] # type: ignore
  41. def on_load_yield_chain(self):
  42. self.event_order.append("on_load_yield_chain")
  43. yield State.event_arg(4) # type: ignore
  44. yield State.event_arg(5) # type: ignore
  45. yield State.event_arg(6) # type: ignore
  46. def click_return_event(self):
  47. self.event_order.append("click_return_event")
  48. return State.event_no_args
  49. def click_return_events(self):
  50. self.event_order.append("click_return_events")
  51. return [
  52. State.event_arg(7), # type: ignore
  53. rx.console_log("click_return_events"),
  54. State.event_arg(8), # type: ignore
  55. State.event_arg(9), # type: ignore
  56. ]
  57. def click_yield_chain(self):
  58. self.event_order.append("click_yield_chain:0")
  59. yield State.event_arg(10) # type: ignore
  60. self.event_order.append("click_yield_chain:1")
  61. yield rx.console_log("click_yield_chain")
  62. yield State.event_arg(11) # type: ignore
  63. self.event_order.append("click_yield_chain:2")
  64. yield State.event_arg(12) # type: ignore
  65. self.event_order.append("click_yield_chain:3")
  66. def click_yield_many_events(self):
  67. self.event_order.append("click_yield_many_events")
  68. for ix in range(MANY_EVENTS):
  69. yield State.event_arg(ix) # type: ignore
  70. yield rx.console_log(f"many_events_{ix}")
  71. self.event_order.append("click_yield_many_events_done")
  72. def click_yield_nested(self):
  73. self.event_order.append("click_yield_nested")
  74. yield State.event_nested_1
  75. yield State.event_arg("yield_nested") # type: ignore
  76. def redirect_return_chain(self):
  77. self.event_order.append("redirect_return_chain")
  78. yield rx.redirect("/on-load-return-chain")
  79. def redirect_yield_chain(self):
  80. self.event_order.append("redirect_yield_chain")
  81. yield rx.redirect("/on-load-yield-chain")
  82. def click_return_int_type(self):
  83. self.event_order.append("click_return_int_type")
  84. return State.event_arg_repr_type(1) # type: ignore
  85. def click_return_dict_type(self):
  86. self.event_order.append("click_return_dict_type")
  87. return State.event_arg_repr_type({"a": 1}) # type: ignore
  88. async def click_yield_interim_value_async(self):
  89. self.interim_value = "interim"
  90. yield
  91. await asyncio.sleep(0.5)
  92. self.interim_value = "final"
  93. def click_yield_interim_value(self):
  94. self.interim_value = "interim"
  95. yield
  96. time.sleep(0.5)
  97. self.interim_value = "final"
  98. app = rx.App(state=rx.State)
  99. token_input = rx.input(
  100. value=State.router.session.client_token, is_read_only=True, id="token"
  101. )
  102. @app.add_page
  103. def index():
  104. return rx.fragment(
  105. token_input,
  106. rx.input(value=State.interim_value, is_read_only=True, id="interim_value"),
  107. rx.button(
  108. "Return Event",
  109. id="return_event",
  110. on_click=State.click_return_event,
  111. ),
  112. rx.button(
  113. "Return Events",
  114. id="return_events",
  115. on_click=State.click_return_events,
  116. ),
  117. rx.button(
  118. "Yield Chain",
  119. id="yield_chain",
  120. on_click=State.click_yield_chain,
  121. ),
  122. rx.button(
  123. "Yield Many events",
  124. id="yield_many_events",
  125. on_click=State.click_yield_many_events,
  126. ),
  127. rx.button(
  128. "Yield Nested",
  129. id="yield_nested",
  130. on_click=State.click_yield_nested,
  131. ),
  132. rx.button(
  133. "Redirect Yield Chain",
  134. id="redirect_yield_chain",
  135. on_click=State.redirect_yield_chain,
  136. ),
  137. rx.button(
  138. "Redirect Return Chain",
  139. id="redirect_return_chain",
  140. on_click=State.redirect_return_chain,
  141. ),
  142. rx.button(
  143. "Click Int Type",
  144. id="click_int_type",
  145. on_click=lambda: State.event_arg_repr_type(1), # type: ignore
  146. ),
  147. rx.button(
  148. "Click Dict Type",
  149. id="click_dict_type",
  150. on_click=lambda: State.event_arg_repr_type({"a": 1}), # type: ignore
  151. ),
  152. rx.button(
  153. "Return Chain Int Type",
  154. id="return_int_type",
  155. on_click=State.click_return_int_type,
  156. ),
  157. rx.button(
  158. "Return Chain Dict Type",
  159. id="return_dict_type",
  160. on_click=State.click_return_dict_type,
  161. ),
  162. rx.button(
  163. "Click Yield Interim Value (Async)",
  164. id="click_yield_interim_value_async",
  165. on_click=State.click_yield_interim_value_async,
  166. ),
  167. rx.button(
  168. "Click Yield Interim Value",
  169. id="click_yield_interim_value",
  170. on_click=State.click_yield_interim_value,
  171. ),
  172. )
  173. def on_load_return_chain():
  174. return rx.fragment(
  175. rx.text("return"),
  176. token_input,
  177. )
  178. def on_load_yield_chain():
  179. return rx.fragment(
  180. rx.text("yield"),
  181. token_input,
  182. )
  183. def on_mount_return_chain():
  184. return rx.fragment(
  185. rx.text(
  186. "return",
  187. on_mount=State.on_load_return_chain,
  188. on_unmount=lambda: State.event_arg("unmount"), # type: ignore
  189. ),
  190. token_input,
  191. rx.button("Unmount", on_click=rx.redirect("/"), id="unmount"),
  192. )
  193. def on_mount_yield_chain():
  194. return rx.fragment(
  195. rx.text(
  196. "yield",
  197. on_mount=[
  198. State.on_load_yield_chain,
  199. lambda: State.event_arg("mount"), # type: ignore
  200. ],
  201. on_unmount=State.event_no_args,
  202. ),
  203. token_input,
  204. rx.button("Unmount", on_click=rx.redirect("/"), id="unmount"),
  205. )
  206. app.add_page(on_load_return_chain, on_load=State.on_load_return_chain) # type: ignore
  207. app.add_page(on_load_yield_chain, on_load=State.on_load_yield_chain) # type: ignore
  208. app.add_page(on_mount_return_chain)
  209. app.add_page(on_mount_yield_chain)
  210. @pytest.fixture(scope="module")
  211. def event_chain(tmp_path_factory) -> Generator[AppHarness, None, None]:
  212. """Start EventChain app at tmp_path via AppHarness.
  213. Args:
  214. tmp_path_factory: pytest tmp_path_factory fixture
  215. Yields:
  216. running AppHarness instance
  217. """
  218. with AppHarness.create(
  219. root=tmp_path_factory.mktemp("event_chain"),
  220. app_source=EventChain, # type: ignore
  221. ) as harness:
  222. yield harness
  223. @pytest.fixture
  224. def driver(event_chain: AppHarness) -> Generator[WebDriver, None, None]:
  225. """Get an instance of the browser open to the event_chain app.
  226. Args:
  227. event_chain: harness for EventChain app
  228. Yields:
  229. WebDriver instance.
  230. """
  231. assert event_chain.app_instance is not None, "app is not running"
  232. driver = event_chain.frontend()
  233. try:
  234. yield driver
  235. finally:
  236. driver.quit()
  237. def assert_token(event_chain: AppHarness, driver: WebDriver) -> str:
  238. """Get the token associated with backend state.
  239. Args:
  240. event_chain: harness for EventChain app.
  241. driver: WebDriver instance.
  242. Returns:
  243. The token visible in the driver browser.
  244. """
  245. assert event_chain.app_instance is not None
  246. token_input = driver.find_element(By.ID, "token")
  247. assert token_input
  248. # wait for the backend connection to send the token
  249. token = event_chain.poll_for_value(token_input)
  250. assert token is not None
  251. state_name = event_chain.get_full_state_name(["_state"])
  252. return f"{token}_{state_name}"
  253. @pytest.mark.parametrize(
  254. ("button_id", "exp_event_order"),
  255. [
  256. ("return_event", ["click_return_event", "event_no_args"]),
  257. (
  258. "return_events",
  259. ["click_return_events", "event_arg:7", "event_arg:8", "event_arg:9"],
  260. ),
  261. (
  262. "yield_chain",
  263. [
  264. "click_yield_chain:0",
  265. "click_yield_chain:1",
  266. "click_yield_chain:2",
  267. "click_yield_chain:3",
  268. "event_arg:10",
  269. "event_arg:11",
  270. "event_arg:12",
  271. ],
  272. ),
  273. (
  274. "yield_many_events",
  275. [
  276. "click_yield_many_events",
  277. "click_yield_many_events_done",
  278. *[f"event_arg:{ix}" for ix in range(MANY_EVENTS)],
  279. ],
  280. ),
  281. (
  282. "yield_nested",
  283. [
  284. "click_yield_nested",
  285. "event_nested_1",
  286. "event_arg:yield_nested",
  287. "event_nested_2",
  288. "event_arg:nested_1",
  289. "event_nested_3",
  290. "event_arg:nested_2",
  291. "event_no_args",
  292. "event_arg:nested_3",
  293. ],
  294. ),
  295. (
  296. "redirect_return_chain",
  297. [
  298. "redirect_return_chain",
  299. "on_load_return_chain",
  300. "event_arg:1",
  301. "event_arg:2",
  302. "event_arg:3",
  303. ],
  304. ),
  305. (
  306. "redirect_yield_chain",
  307. [
  308. "redirect_yield_chain",
  309. "on_load_yield_chain",
  310. "event_arg:4",
  311. "event_arg:5",
  312. "event_arg:6",
  313. ],
  314. ),
  315. (
  316. "click_int_type",
  317. ["event_arg_repr:1_int"],
  318. ),
  319. (
  320. "click_dict_type",
  321. ["event_arg_repr:{'a': 1}_dict"],
  322. ),
  323. (
  324. "return_int_type",
  325. ["click_return_int_type", "event_arg_repr:1_int"],
  326. ),
  327. (
  328. "return_dict_type",
  329. ["click_return_dict_type", "event_arg_repr:{'a': 1}_dict"],
  330. ),
  331. ],
  332. )
  333. @pytest.mark.asyncio
  334. async def test_event_chain_click(
  335. event_chain: AppHarness,
  336. driver: WebDriver,
  337. button_id: str,
  338. exp_event_order: list[str],
  339. ):
  340. """Click the button, assert that the events are handled in the correct order.
  341. Args:
  342. event_chain: AppHarness for the event_chain app
  343. driver: selenium WebDriver open to the app
  344. button_id: the ID of the button to click
  345. exp_event_order: the expected events recorded in the State
  346. """
  347. token = assert_token(event_chain, driver)
  348. state_name = event_chain.get_state_name("_state")
  349. btn = driver.find_element(By.ID, button_id)
  350. btn.click()
  351. async def _has_all_events():
  352. return len(
  353. (await event_chain.get_state(token)).substates[state_name].event_order
  354. ) == len(exp_event_order)
  355. await AppHarness._poll_for_async(_has_all_events)
  356. event_order = (await event_chain.get_state(token)).substates[state_name].event_order
  357. assert event_order == exp_event_order
  358. @pytest.mark.parametrize(
  359. ("uri", "exp_event_order"),
  360. [
  361. (
  362. "/on-load-return-chain",
  363. [
  364. "on_load_return_chain",
  365. "event_arg:1",
  366. "event_arg:2",
  367. "event_arg:3",
  368. ],
  369. ),
  370. (
  371. "/on-load-yield-chain",
  372. [
  373. "on_load_yield_chain",
  374. "event_arg:4",
  375. "event_arg:5",
  376. "event_arg:6",
  377. ],
  378. ),
  379. ],
  380. )
  381. @pytest.mark.asyncio
  382. async def test_event_chain_on_load(
  383. event_chain: AppHarness,
  384. driver: WebDriver,
  385. uri: str,
  386. exp_event_order: list[str],
  387. ):
  388. """Load the URI, assert that the events are handled in the correct order.
  389. Args:
  390. event_chain: AppHarness for the event_chain app
  391. driver: selenium WebDriver open to the app
  392. uri: the page to load
  393. exp_event_order: the expected events recorded in the State
  394. """
  395. assert event_chain.frontend_url is not None
  396. driver.get(event_chain.frontend_url + uri)
  397. token = assert_token(event_chain, driver)
  398. state_name = event_chain.get_state_name("_state")
  399. async def _has_all_events():
  400. return len(
  401. (await event_chain.get_state(token)).substates[state_name].event_order
  402. ) == len(exp_event_order)
  403. await AppHarness._poll_for_async(_has_all_events)
  404. backend_state = (await event_chain.get_state(token)).substates[state_name]
  405. assert backend_state.event_order == exp_event_order
  406. assert backend_state.is_hydrated is True
  407. @pytest.mark.parametrize(
  408. ("uri", "exp_event_order"),
  409. [
  410. (
  411. "/on-mount-return-chain",
  412. [
  413. "on_load_return_chain",
  414. "event_arg:unmount",
  415. "on_load_return_chain",
  416. "event_arg:1",
  417. "event_arg:2",
  418. "event_arg:3",
  419. "event_arg:1",
  420. "event_arg:2",
  421. "event_arg:3",
  422. "event_arg:unmount",
  423. ],
  424. ),
  425. (
  426. "/on-mount-yield-chain",
  427. [
  428. "on_load_yield_chain",
  429. "event_arg:mount",
  430. "event_no_args",
  431. "on_load_yield_chain",
  432. "event_arg:mount",
  433. "event_arg:4",
  434. "event_arg:5",
  435. "event_arg:6",
  436. "event_arg:4",
  437. "event_arg:5",
  438. "event_arg:6",
  439. "event_no_args",
  440. ],
  441. ),
  442. ],
  443. )
  444. @pytest.mark.asyncio
  445. async def test_event_chain_on_mount(
  446. event_chain: AppHarness,
  447. driver: WebDriver,
  448. uri: str,
  449. exp_event_order: list[str],
  450. ):
  451. """Load the URI, assert that the events are handled in the correct order.
  452. These pages use `on_mount` and `on_unmount`, which get fired twice in dev mode
  453. due to react StrictMode being used.
  454. In prod mode, these events are only fired once.
  455. Args:
  456. event_chain: AppHarness for the event_chain app
  457. driver: selenium WebDriver open to the app
  458. uri: the page to load
  459. exp_event_order: the expected events recorded in the State
  460. """
  461. assert event_chain.frontend_url is not None
  462. driver.get(event_chain.frontend_url + uri)
  463. token = assert_token(event_chain, driver)
  464. state_name = event_chain.get_state_name("_state")
  465. unmount_button = driver.find_element(By.ID, "unmount")
  466. assert unmount_button
  467. unmount_button.click()
  468. async def _has_all_events():
  469. return len(
  470. (await event_chain.get_state(token)).substates[state_name].event_order
  471. ) == len(exp_event_order)
  472. await AppHarness._poll_for_async(_has_all_events)
  473. event_order = (await event_chain.get_state(token)).substates[state_name].event_order
  474. assert event_order == exp_event_order
  475. @pytest.mark.parametrize(
  476. ("button_id",),
  477. [
  478. ("click_yield_interim_value_async",),
  479. ("click_yield_interim_value",),
  480. ],
  481. )
  482. def test_yield_state_update(event_chain: AppHarness, driver: WebDriver, button_id: str):
  483. """Click the button, assert that the interim value is set, then final value is set.
  484. Args:
  485. event_chain: AppHarness for the event_chain app
  486. driver: selenium WebDriver open to the app
  487. button_id: the ID of the button to click
  488. """
  489. interim_value_input = driver.find_element(By.ID, "interim_value")
  490. assert_token(event_chain, driver)
  491. btn = driver.find_element(By.ID, button_id)
  492. btn.click()
  493. assert (
  494. event_chain.poll_for_value(interim_value_input, exp_not_equal="") == "interim"
  495. )
  496. assert (
  497. event_chain.poll_for_value(interim_value_input, exp_not_equal="interim")
  498. == "final"
  499. )