test_event_chain.py 17 KB

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