test_var_operations.py 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999
  1. """Integration tests for var operations."""
  2. from collections.abc import Generator
  3. import pytest
  4. from selenium.webdriver.common.by import By
  5. from reflex.testing import AppHarness
  6. def VarOperations():
  7. """App with var operations."""
  8. from typing import TypedDict
  9. import reflex as rx
  10. from reflex.vars.base import LiteralVar
  11. from reflex.vars.sequence import ArrayVar
  12. class Object(rx.Base):
  13. name: str = "hello"
  14. optional_none: str | None = None
  15. optional_str: str | None = "hello"
  16. class Person(TypedDict):
  17. name: str
  18. age: int
  19. class VarOperationState(rx.State):
  20. int_var1: rx.Field[int] = rx.field(10)
  21. int_var2: rx.Field[int] = rx.field(5)
  22. int_var3: rx.Field[int] = rx.field(7)
  23. float_var1: rx.Field[float] = rx.field(10.5)
  24. float_var2: rx.Field[float] = rx.field(5.5)
  25. long_float: rx.Field[float] = rx.field(13212312312.1231231)
  26. list1: rx.Field[list] = rx.field([1, 2])
  27. list2: rx.Field[list] = rx.field([3, 4])
  28. list3: rx.Field[list] = rx.field(["first", "second", "third"])
  29. list4: rx.Field[list] = rx.field([Object(name="obj_1"), Object(name="obj_2")])
  30. optional_list: rx.Field[list | None] = rx.field(None)
  31. optional_dict: rx.Field[dict[str, str] | None] = rx.field(None)
  32. optional_list_value: rx.Field[list[str] | None] = rx.field(["red", "yellow"])
  33. optional_dict_value: rx.Field[dict[str, str] | None] = rx.field({"name": "red"})
  34. str_var1: rx.Field[str] = rx.field("first")
  35. str_var2: rx.Field[str] = rx.field("second")
  36. str_var3: rx.Field[str] = rx.field("ThIrD")
  37. str_var4: rx.Field[str] = rx.field("a long string")
  38. dict1: rx.Field[dict[int, int]] = rx.field({1: 2})
  39. dict2: rx.Field[dict[int, int]] = rx.field({3: 4})
  40. html_str: rx.Field[str] = rx.field("<div>hello</div>")
  41. people: rx.Field[list[Person]] = rx.field(
  42. [{"name": "Alice", "age": 30}, {"name": "Bob", "age": 25}]
  43. )
  44. obj: rx.Field[Object] = rx.field(Object())
  45. app = rx.App()
  46. @rx.memo
  47. def memo_comp(list1: list[int], int_var1: int, id: str):
  48. return rx.text(list1, int_var1, id=id)
  49. @rx.memo
  50. def memo_comp_nested(int_var2: int, id: str):
  51. return memo_comp(list1=[3, 4], int_var1=int_var2, id=id)
  52. @app.add_page
  53. def index():
  54. return rx.vstack(
  55. None, # Testing that None doesn't break everything
  56. rx.el.input(
  57. id="token",
  58. value=VarOperationState.router.session.client_token,
  59. is_read_only=True,
  60. ),
  61. # INT INT
  62. rx.text(
  63. VarOperationState.int_var1 + VarOperationState.int_var2,
  64. id="int_add_int",
  65. ),
  66. rx.text(
  67. VarOperationState.int_var1 * VarOperationState.int_var2,
  68. id="int_mult_int",
  69. ),
  70. rx.text(
  71. VarOperationState.int_var1 - VarOperationState.int_var2,
  72. id="int_sub_int",
  73. ),
  74. rx.text(
  75. VarOperationState.int_var1**VarOperationState.int_var2,
  76. id="int_exp_int",
  77. ),
  78. rx.text(
  79. VarOperationState.int_var1 / VarOperationState.int_var2,
  80. id="int_div_int",
  81. ),
  82. rx.text(
  83. VarOperationState.int_var1 // VarOperationState.int_var3,
  84. id="int_floor_int",
  85. ),
  86. rx.text(
  87. VarOperationState.int_var1 % VarOperationState.int_var2,
  88. id="int_mod_int",
  89. ),
  90. rx.text(
  91. VarOperationState.int_var1 | VarOperationState.int_var2,
  92. id="int_or_int",
  93. ),
  94. rx.text(
  95. (VarOperationState.int_var1 > VarOperationState.int_var2).to_string(),
  96. id="int_gt_int",
  97. ),
  98. rx.text(
  99. (VarOperationState.int_var1 < VarOperationState.int_var2).to_string(),
  100. id="int_lt_int",
  101. ),
  102. rx.text(
  103. (VarOperationState.int_var1 >= VarOperationState.int_var2).to_string(),
  104. id="int_gte_int",
  105. ),
  106. rx.text(
  107. (VarOperationState.int_var1 <= VarOperationState.int_var2).to_string(),
  108. id="int_lte_int",
  109. ),
  110. rx.text(
  111. VarOperationState.int_var1 & VarOperationState.int_var2,
  112. id="int_and_int",
  113. ),
  114. rx.text(
  115. (VarOperationState.int_var1 | VarOperationState.int_var2).to_string(),
  116. id="int_or_int",
  117. ),
  118. rx.text(
  119. (VarOperationState.int_var1 == VarOperationState.int_var2).to_string(),
  120. id="int_eq_int",
  121. ),
  122. rx.text(
  123. (VarOperationState.int_var1 != VarOperationState.int_var2).to_string(),
  124. id="int_neq_int",
  125. ),
  126. # INT FLOAT OR FLOAT INT
  127. rx.text(
  128. VarOperationState.float_var1 + VarOperationState.int_var2,
  129. id="float_add_int",
  130. ),
  131. rx.text(
  132. VarOperationState.float_var1 * VarOperationState.int_var2,
  133. id="float_mult_int",
  134. ),
  135. rx.text(
  136. VarOperationState.float_var1 - VarOperationState.int_var2,
  137. id="float_sub_int",
  138. ),
  139. rx.text(
  140. VarOperationState.float_var1**VarOperationState.int_var2,
  141. id="float_exp_int",
  142. ),
  143. rx.text(
  144. VarOperationState.float_var1 / VarOperationState.int_var2,
  145. id="float_div_int",
  146. ),
  147. rx.text(
  148. VarOperationState.float_var1 // VarOperationState.int_var3,
  149. id="float_floor_int",
  150. ),
  151. rx.text(
  152. VarOperationState.float_var1 % VarOperationState.int_var2,
  153. id="float_mod_int",
  154. ),
  155. rx.text(
  156. (VarOperationState.float_var1 > VarOperationState.int_var2).to_string(),
  157. id="float_gt_int",
  158. ),
  159. rx.text(
  160. (VarOperationState.float_var1 < VarOperationState.int_var2).to_string(),
  161. id="float_lt_int",
  162. ),
  163. rx.text(
  164. (
  165. VarOperationState.float_var1 >= VarOperationState.int_var2
  166. ).to_string(),
  167. id="float_gte_int",
  168. ),
  169. rx.text(
  170. (
  171. VarOperationState.float_var1 <= VarOperationState.int_var2
  172. ).to_string(),
  173. id="float_lte_int",
  174. ),
  175. rx.text(
  176. (
  177. VarOperationState.float_var1 == VarOperationState.int_var2
  178. ).to_string(),
  179. id="float_eq_int",
  180. ),
  181. rx.text(
  182. (
  183. VarOperationState.float_var1 != VarOperationState.int_var2
  184. ).to_string(),
  185. id="float_neq_int",
  186. ),
  187. rx.text(
  188. (VarOperationState.float_var1 & VarOperationState.int_var2).to_string(),
  189. id="float_and_int",
  190. ),
  191. rx.text(
  192. (VarOperationState.float_var1 | VarOperationState.int_var2).to_string(),
  193. id="float_or_int",
  194. ),
  195. # INT, DICT
  196. rx.text(
  197. (VarOperationState.int_var1 | VarOperationState.dict1).to_string(),
  198. id="int_or_dict",
  199. ),
  200. rx.text(
  201. (VarOperationState.int_var1 & VarOperationState.dict1).to_string(),
  202. id="int_and_dict",
  203. ),
  204. rx.text(
  205. (VarOperationState.int_var1 == VarOperationState.dict1).to_string(),
  206. id="int_eq_dict",
  207. ),
  208. rx.text(
  209. (VarOperationState.int_var1 != VarOperationState.dict1).to_string(),
  210. id="int_neq_dict",
  211. ),
  212. # FLOAT FLOAT
  213. rx.text(
  214. VarOperationState.float_var1 + VarOperationState.float_var2,
  215. id="float_add_float",
  216. ),
  217. rx.text(
  218. VarOperationState.float_var1 * VarOperationState.float_var2,
  219. id="float_mult_float",
  220. ),
  221. rx.text(
  222. VarOperationState.float_var1 - VarOperationState.float_var2,
  223. id="float_sub_float",
  224. ),
  225. rx.text(
  226. VarOperationState.float_var1**VarOperationState.float_var2,
  227. id="float_exp_float",
  228. ),
  229. rx.text(
  230. VarOperationState.float_var1 / VarOperationState.float_var2,
  231. id="float_div_float",
  232. ),
  233. rx.text(
  234. VarOperationState.float_var1 // VarOperationState.float_var2,
  235. id="float_floor_float",
  236. ),
  237. rx.text(
  238. VarOperationState.float_var1 % VarOperationState.float_var2,
  239. id="float_mod_float",
  240. ),
  241. rx.text(
  242. (
  243. VarOperationState.float_var1 > VarOperationState.float_var2
  244. ).to_string(),
  245. id="float_gt_float",
  246. ),
  247. rx.text(
  248. (
  249. VarOperationState.float_var1 < VarOperationState.float_var2
  250. ).to_string(),
  251. id="float_lt_float",
  252. ),
  253. rx.text(
  254. (
  255. VarOperationState.float_var1 >= VarOperationState.float_var2
  256. ).to_string(),
  257. id="float_gte_float",
  258. ),
  259. rx.text(
  260. (
  261. VarOperationState.float_var1 <= VarOperationState.float_var2
  262. ).to_string(),
  263. id="float_lte_float",
  264. ),
  265. rx.text(
  266. (
  267. VarOperationState.float_var1 == VarOperationState.float_var2
  268. ).to_string(),
  269. id="float_eq_float",
  270. ),
  271. rx.text(
  272. (
  273. VarOperationState.float_var1 != VarOperationState.float_var2
  274. ).to_string(),
  275. id="float_neq_float",
  276. ),
  277. rx.text(
  278. (
  279. VarOperationState.float_var1 & VarOperationState.float_var2
  280. ).to_string(),
  281. id="float_and_float",
  282. ),
  283. rx.text(
  284. (
  285. VarOperationState.float_var1 | VarOperationState.float_var2
  286. ).to_string(),
  287. id="float_or_float",
  288. ),
  289. # FLOAT STR
  290. rx.text(
  291. VarOperationState.float_var1 | VarOperationState.str_var1,
  292. id="float_or_str",
  293. ),
  294. rx.text(
  295. VarOperationState.float_var1 & VarOperationState.str_var1,
  296. id="float_and_str",
  297. ),
  298. rx.text(
  299. (
  300. VarOperationState.float_var1 == VarOperationState.str_var1
  301. ).to_string(),
  302. id="float_eq_str",
  303. ),
  304. rx.text(
  305. (
  306. VarOperationState.float_var1 != VarOperationState.str_var1
  307. ).to_string(),
  308. id="float_neq_str",
  309. ),
  310. # FLOAT LIST
  311. rx.text(
  312. (VarOperationState.float_var1 | VarOperationState.list1).to_string(),
  313. id="float_or_list",
  314. ),
  315. rx.text(
  316. (VarOperationState.float_var1 & VarOperationState.list1).to_string(),
  317. id="float_and_list",
  318. ),
  319. rx.text(
  320. (VarOperationState.float_var1 == VarOperationState.list1).to_string(),
  321. id="float_eq_list",
  322. ),
  323. rx.text(
  324. (VarOperationState.float_var1 != VarOperationState.list1).to_string(),
  325. id="float_neq_list",
  326. ),
  327. # FLOAT DICT
  328. rx.text(
  329. (VarOperationState.float_var1 | VarOperationState.dict1).to_string(),
  330. id="float_or_dict",
  331. ),
  332. rx.text(
  333. (VarOperationState.float_var1 & VarOperationState.dict1).to_string(),
  334. id="float_and_dict",
  335. ),
  336. rx.text(
  337. (VarOperationState.float_var1 == VarOperationState.dict1).to_string(),
  338. id="float_eq_dict",
  339. ),
  340. rx.text(
  341. (VarOperationState.float_var1 != VarOperationState.dict1).to_string(),
  342. id="float_neq_dict",
  343. ),
  344. # STR STR
  345. rx.text(
  346. VarOperationState.str_var1 + VarOperationState.str_var2,
  347. id="str_add_str",
  348. ),
  349. rx.text(
  350. (VarOperationState.str_var1 > VarOperationState.str_var2).to_string(),
  351. id="str_gt_str",
  352. ),
  353. rx.text(
  354. (VarOperationState.str_var1 < VarOperationState.str_var2).to_string(),
  355. id="str_lt_str",
  356. ),
  357. rx.text(
  358. (VarOperationState.str_var1 >= VarOperationState.str_var2).to_string(),
  359. id="str_gte_str",
  360. ),
  361. rx.text(
  362. (VarOperationState.str_var1 <= VarOperationState.str_var2).to_string(),
  363. id="str_lte_str",
  364. ),
  365. rx.text(
  366. (
  367. VarOperationState.float_var1 == VarOperationState.float_var2
  368. ).to_string(),
  369. id="str_eq_str",
  370. ),
  371. rx.text(
  372. (
  373. VarOperationState.float_var1 != VarOperationState.float_var2
  374. ).to_string(),
  375. id="str_neq_str",
  376. ),
  377. rx.text(
  378. VarOperationState.str_var1.contains("fir").to_string(),
  379. id="str_contains",
  380. ),
  381. rx.text(
  382. VarOperationState.str_var1 | VarOperationState.str_var1,
  383. id="str_or_str",
  384. ),
  385. rx.text(
  386. VarOperationState.str_var1 & VarOperationState.str_var2,
  387. id="str_and_str",
  388. ),
  389. # STR, INT
  390. rx.text(
  391. VarOperationState.str_var1 * VarOperationState.int_var2,
  392. id="str_mult_int",
  393. ),
  394. rx.text(
  395. VarOperationState.str_var1 & VarOperationState.int_var2,
  396. id="str_and_int",
  397. ),
  398. rx.text(
  399. VarOperationState.str_var1 | VarOperationState.int_var2,
  400. id="str_or_int",
  401. ),
  402. rx.text(
  403. (VarOperationState.str_var1 == VarOperationState.int_var1).to_string(),
  404. id="str_eq_int",
  405. ),
  406. rx.text(
  407. (VarOperationState.str_var1 != VarOperationState.int_var1).to_string(),
  408. id="str_neq_int",
  409. ),
  410. # STR, LIST
  411. rx.text(
  412. VarOperationState.str_var1 | VarOperationState.list1,
  413. id="str_or_list",
  414. ),
  415. rx.text(
  416. (VarOperationState.str_var1 & VarOperationState.list1).to_string(),
  417. id="str_and_list",
  418. ),
  419. rx.text(
  420. (VarOperationState.str_var1 == VarOperationState.list1).to_string(),
  421. id="str_eq_list",
  422. ),
  423. rx.text(
  424. (VarOperationState.str_var1 != VarOperationState.list1).to_string(),
  425. id="str_neq_list",
  426. ),
  427. # STR, DICT
  428. rx.text(
  429. VarOperationState.str_var1 | VarOperationState.dict1,
  430. id="str_or_dict",
  431. ),
  432. rx.text(
  433. (VarOperationState.str_var1 & VarOperationState.dict1).to_string(),
  434. id="str_and_dict",
  435. ),
  436. rx.text(
  437. (VarOperationState.str_var1 == VarOperationState.dict1).to_string(),
  438. id="str_eq_dict",
  439. ),
  440. rx.text(
  441. (VarOperationState.str_var1 != VarOperationState.dict1).to_string(),
  442. id="str_neq_dict",
  443. ),
  444. # LIST, LIST
  445. rx.text(
  446. (VarOperationState.list1 + VarOperationState.list2).to_string(),
  447. id="list_add_list",
  448. ),
  449. rx.text(
  450. (VarOperationState.list1 & VarOperationState.list2).to_string(),
  451. id="list_and_list",
  452. ),
  453. rx.text(
  454. (VarOperationState.list1 | VarOperationState.list2).to_string(),
  455. id="list_or_list",
  456. ),
  457. rx.text(
  458. (VarOperationState.list1 > VarOperationState.list2).to_string(),
  459. id="list_gt_list",
  460. ),
  461. rx.text(
  462. (VarOperationState.list1 < VarOperationState.list2).to_string(),
  463. id="list_lt_list",
  464. ),
  465. rx.text(
  466. (VarOperationState.list1 >= VarOperationState.list2).to_string(),
  467. id="list_gte_list",
  468. ),
  469. rx.text(
  470. (VarOperationState.list1 <= VarOperationState.list2).to_string(),
  471. id="list_lte_list",
  472. ),
  473. rx.text(
  474. (VarOperationState.list1 == VarOperationState.list2).to_string(),
  475. id="list_eq_list",
  476. ),
  477. rx.text(
  478. (VarOperationState.list1 != VarOperationState.list2).to_string(),
  479. id="list_neq_list",
  480. ),
  481. rx.text(
  482. VarOperationState.list1.contains(1).to_string(),
  483. id="list_contains",
  484. ),
  485. rx.text(VarOperationState.list4.pluck("name").to_string(), id="list_pluck"),
  486. rx.text(VarOperationState.list1.reverse().to_string(), id="list_reverse"),
  487. # LIST, INT
  488. rx.text(
  489. (VarOperationState.list1 * VarOperationState.int_var2).to_string(),
  490. id="list_mult_int",
  491. ),
  492. rx.text(
  493. (VarOperationState.list1 | VarOperationState.int_var1).to_string(),
  494. id="list_or_int",
  495. ),
  496. rx.text(
  497. (VarOperationState.list1 & VarOperationState.int_var1).to_string(),
  498. id="list_and_int",
  499. ),
  500. rx.text(
  501. (VarOperationState.list1 == VarOperationState.int_var1).to_string(),
  502. id="list_eq_int",
  503. ),
  504. rx.text(
  505. (VarOperationState.list1 != VarOperationState.int_var1).to_string(),
  506. id="list_neq_int",
  507. ),
  508. # LIST, DICT
  509. rx.text(
  510. (VarOperationState.list1 | VarOperationState.dict1).to_string(),
  511. id="list_or_dict",
  512. ),
  513. rx.text(
  514. (VarOperationState.list1 & VarOperationState.dict1).to_string(),
  515. id="list_and_dict",
  516. ),
  517. rx.text(
  518. (VarOperationState.list1 == VarOperationState.dict1).to_string(),
  519. id="list_eq_dict",
  520. ),
  521. rx.text(
  522. (VarOperationState.list1 != VarOperationState.dict1).to_string(),
  523. id="list_neq_dict",
  524. ),
  525. # DICT, DICT
  526. rx.text(
  527. (VarOperationState.dict1 | VarOperationState.dict2).to_string(),
  528. id="dict_or_dict",
  529. ),
  530. rx.text(
  531. (VarOperationState.dict1 & VarOperationState.dict2).to_string(),
  532. id="dict_and_dict",
  533. ),
  534. rx.text(
  535. (VarOperationState.dict1 == VarOperationState.dict2).to_string(),
  536. id="dict_eq_dict",
  537. ),
  538. rx.text(
  539. (VarOperationState.dict1 != VarOperationState.dict2).to_string(),
  540. id="dict_neq_dict",
  541. ),
  542. rx.text(
  543. VarOperationState.dict1.contains(1).to_string(),
  544. id="dict_contains",
  545. ),
  546. rx.text(VarOperationState.str_var3.lower(), id="str_lower"),
  547. rx.text(VarOperationState.str_var3.upper(), id="str_upper"),
  548. rx.text(VarOperationState.str_var4.split(" ").to_string(), id="str_split"),
  549. rx.text(VarOperationState.list3.join(""), id="list_join"),
  550. rx.text(VarOperationState.list3.join(","), id="list_join_comma"),
  551. # Index from an op var
  552. rx.text(
  553. VarOperationState.list3[VarOperationState.int_var1 % 3],
  554. id="list_index_mod",
  555. ),
  556. rx.html(
  557. VarOperationState.html_str,
  558. id="html_str",
  559. ),
  560. rx.el.mark("second"),
  561. rx.text(ArrayVar.range(2, 5).join(","), id="list_join_range1"),
  562. rx.text(ArrayVar.range(2, 10, 2).join(","), id="list_join_range2"),
  563. rx.text(ArrayVar.range(5, 0, -1).join(","), id="list_join_range3"),
  564. rx.text(ArrayVar.range(0, 3).join(","), id="list_join_range4"),
  565. rx.box(
  566. rx.foreach(
  567. ArrayVar.range(0, 2),
  568. lambda x: rx.text(VarOperationState.list1[x], as_="p"),
  569. ),
  570. id="foreach_list_arg",
  571. ),
  572. rx.box(
  573. rx.foreach(
  574. ArrayVar.range(0, 2),
  575. lambda x, ix: rx.text(VarOperationState.list1[ix], as_="p"),
  576. ),
  577. id="foreach_list_ix",
  578. ),
  579. rx.box(
  580. rx.foreach(
  581. LiteralVar.create(list(range(0, 3))).to(ArrayVar, list[int]),
  582. lambda x: rx.foreach(
  583. ArrayVar.range(x),
  584. lambda y: rx.text(VarOperationState.list1[y], as_="p"),
  585. ),
  586. ),
  587. id="foreach_list_nested",
  588. ),
  589. memo_comp(
  590. list1=VarOperationState.list1,
  591. int_var1=VarOperationState.int_var1,
  592. id="memo_comp",
  593. ),
  594. memo_comp_nested(
  595. int_var2=VarOperationState.int_var2,
  596. id="memo_comp_nested",
  597. ),
  598. # length
  599. rx.box(
  600. rx.text(VarOperationState.list3.length()),
  601. id="list_length",
  602. ),
  603. rx.box(
  604. rx.text(VarOperationState.obj.length()),
  605. id="obj_length",
  606. ),
  607. # foreach in a match
  608. rx.box(
  609. rx.match(
  610. VarOperationState.list3.length(),
  611. (0, rx.text("No choices")),
  612. (1, rx.text("One choice")),
  613. rx.foreach(VarOperationState.list3, lambda choice: rx.text(choice)),
  614. ),
  615. id="foreach_in_match",
  616. ),
  617. # Literal range var in a foreach
  618. rx.box(rx.foreach(range(42, 80, 27), rx.text.span), id="range_in_foreach1"),
  619. rx.box(rx.foreach(range(42, 80, 3), rx.text.span), id="range_in_foreach2"),
  620. rx.box(rx.foreach(range(42, 20, -6), rx.text.span), id="range_in_foreach3"),
  621. rx.box(rx.foreach(range(42, 43, 5), rx.text.span), id="range_in_foreach4"),
  622. # Literal dict in a foreach
  623. rx.box(rx.foreach({"a": 1, "b": 2}, rx.text.span), id="dict_in_foreach1"),
  624. # State Var dict in a foreach
  625. rx.box(
  626. rx.foreach(VarOperationState.dict1, rx.text.span),
  627. id="dict_in_foreach2",
  628. ),
  629. rx.box(
  630. rx.foreach(
  631. VarOperationState.dict1.merge(VarOperationState.dict2),
  632. rx.text.span,
  633. ),
  634. id="dict_in_foreach3",
  635. ),
  636. rx.box(
  637. rx.foreach("abcdef", lambda x: rx.text.span(x + " ")),
  638. id="str_in_foreach",
  639. ),
  640. rx.box(
  641. rx.foreach(VarOperationState.str_var1, lambda x: rx.text.span(x + " ")),
  642. id="str_var_in_foreach",
  643. ),
  644. rx.box(
  645. rx.foreach(
  646. VarOperationState.people,
  647. lambda person: rx.text.span(
  648. "Hello " + person["name"], person["age"] + 3
  649. ),
  650. ),
  651. id="typed_dict_in_foreach",
  652. ),
  653. rx.box(
  654. rx.foreach(VarOperationState.optional_list, rx.text.span),
  655. id="optional_list",
  656. ),
  657. rx.box(
  658. rx.foreach(VarOperationState.optional_dict, rx.text.span),
  659. id="optional_dict",
  660. ),
  661. rx.box(
  662. rx.foreach(VarOperationState.optional_list_value, rx.text.span),
  663. id="optional_list_value",
  664. ),
  665. rx.box(
  666. rx.foreach(VarOperationState.optional_dict_value, rx.text.span),
  667. id="optional_dict_value",
  668. ),
  669. rx.box(
  670. rx.text.span(f"{rx.Var.create(13212312312.1231231)}"),
  671. id="float_format",
  672. ),
  673. rx.box(
  674. rx.text.span(f"{rx.Var.create(13212312312.1231231):.0f}"),
  675. id="float_format_0f",
  676. ),
  677. rx.box(
  678. rx.text.span(f"{rx.Var.create(13212312312.1231231):.1f}"),
  679. id="float_format_1f",
  680. ),
  681. rx.box(
  682. rx.text.span(f"{rx.Var.create(13212312312.1231231):.2f}"),
  683. id="float_format_2f",
  684. ),
  685. rx.box(
  686. rx.text.span(f"{rx.Var.create(13212312312.1231231):,}"),
  687. id="float_format_comma",
  688. ),
  689. rx.box(
  690. rx.text.span(f"{rx.Var.create(13212312312.1231231):_}"),
  691. id="float_format_underscore",
  692. ),
  693. rx.box(
  694. rx.text.span(f"{rx.Var.create(13212312312.1231231):,.0f}"),
  695. id="float_format_comma_0f",
  696. ),
  697. rx.box(
  698. rx.text.span(f"{rx.Var.create(13212312312.1231231):,.1f}"),
  699. id="float_format_comma_1f",
  700. ),
  701. rx.box(
  702. rx.text.span(f"{rx.Var.create(13212312312.1231231):,.2f}"),
  703. id="float_format_comma_2f",
  704. ),
  705. rx.box(
  706. rx.text.span(f"{rx.Var.create(13212312312.1231231):_.0f}"),
  707. id="float_format_underscore_0f",
  708. ),
  709. rx.box(
  710. rx.text.span(f"{rx.Var.create(13212312312.1231231):_.1f}"),
  711. id="float_format_underscore_1f",
  712. ),
  713. rx.box(
  714. rx.text.span(f"{rx.Var.create(13212312312.1231231):_.2f}"),
  715. id="float_format_underscore_2f",
  716. ),
  717. # ObjectVar
  718. rx.box(
  719. rx.text.span(VarOperationState.obj.name),
  720. id="obj_name",
  721. ),
  722. rx.box(
  723. rx.text.span(VarOperationState.obj.optional_none),
  724. id="obj_optional_none",
  725. ),
  726. rx.box(
  727. rx.text.span(VarOperationState.obj.optional_str),
  728. id="obj_optional_str",
  729. ),
  730. rx.box(
  731. rx.text.span(VarOperationState.obj.get("optional_none")),
  732. id="obj_optional_none_get_none",
  733. ),
  734. rx.box(
  735. rx.text.span(VarOperationState.obj.get("optional_none", "foo")),
  736. id="obj_optional_none_get_foo",
  737. ),
  738. rx.box(
  739. rx.text.span(round(VarOperationState.long_float)),
  740. id="float_round",
  741. ),
  742. rx.box(
  743. rx.text.span(round(VarOperationState.long_float, 2)),
  744. id="float_round_2",
  745. ),
  746. )
  747. @pytest.fixture(scope="module")
  748. def var_operations(tmp_path_factory) -> Generator[AppHarness, None, None]:
  749. """Start VarOperations app at tmp_path via AppHarness.
  750. Args:
  751. tmp_path_factory: pytest tmp_path_factory fixture
  752. Yields:
  753. running AppHarness instance
  754. """
  755. with AppHarness.create(
  756. root=tmp_path_factory.mktemp("var_operations"),
  757. app_source=VarOperations,
  758. ) as harness:
  759. assert harness.app_instance is not None, "app is not running"
  760. yield harness
  761. @pytest.fixture
  762. def driver(var_operations: AppHarness):
  763. """Get an instance of the browser open to the var operations app.
  764. Args:
  765. var_operations: harness for VarOperations app
  766. Yields:
  767. WebDriver instance.
  768. """
  769. driver = var_operations.frontend()
  770. try:
  771. token_input = driver.find_element(By.ID, "token")
  772. assert token_input
  773. # wait for the backend connection to send the token
  774. token = var_operations.poll_for_value(token_input)
  775. assert token is not None
  776. yield driver
  777. finally:
  778. driver.quit()
  779. def test_var_operations(driver, var_operations: AppHarness):
  780. """Test that the var operations produce the right results.
  781. Args:
  782. driver: selenium WebDriver open to the app
  783. var_operations: AppHarness for the var operations app
  784. """
  785. tests = [
  786. # int, int
  787. ("int_add_int", "15"),
  788. ("int_mult_int", "50"),
  789. ("int_sub_int", "5"),
  790. ("int_exp_int", "100000"),
  791. ("int_div_int", "2"),
  792. ("int_floor_int", "1"),
  793. ("int_mod_int", "0"),
  794. ("int_gt_int", "true"),
  795. ("int_lt_int", "false"),
  796. ("int_gte_int", "true"),
  797. ("int_lte_int", "false"),
  798. ("int_and_int", "5"),
  799. ("int_or_int", "10"),
  800. ("int_eq_int", "false"),
  801. ("int_neq_int", "true"),
  802. # int, float
  803. ("float_add_int", "15.5"),
  804. ("float_mult_int", "52.5"),
  805. ("float_sub_int", "5.5"),
  806. ("float_exp_int", "127628.15625"),
  807. ("float_div_int", "2.1"),
  808. ("float_floor_int", "1"),
  809. ("float_mod_int", "0.5"),
  810. ("float_gt_int", "true"),
  811. ("float_lt_int", "false"),
  812. ("float_gte_int", "true"),
  813. ("float_lte_int", "false"),
  814. ("float_eq_int", "false"),
  815. ("float_neq_int", "true"),
  816. ("float_and_int", "5"),
  817. ("float_or_int", "10.5"),
  818. # int, dict
  819. ("int_or_dict", "10"),
  820. ("int_and_dict", '{"1":2}'),
  821. ("int_eq_dict", "false"),
  822. ("int_neq_dict", "true"),
  823. # float, float
  824. ("float_add_float", "16"),
  825. ("float_mult_float", "57.75"),
  826. ("float_sub_float", "5"),
  827. ("float_exp_float", "413562.49323606625"),
  828. ("float_div_float", "1.9090909090909092"),
  829. ("float_floor_float", "1"),
  830. ("float_mod_float", "5"),
  831. ("float_gt_float", "true"),
  832. ("float_lt_float", "false"),
  833. ("float_gte_float", "true"),
  834. ("float_lte_float", "false"),
  835. ("float_eq_float", "false"),
  836. ("float_neq_float", "true"),
  837. ("float_and_float", "5.5"),
  838. ("float_or_float", "10.5"),
  839. # float, str
  840. ("float_or_str", "10.5"),
  841. ("float_and_str", "first"),
  842. ("float_eq_str", "false"),
  843. ("float_neq_str", "true"),
  844. # float, list
  845. ("float_or_list", "10.5"),
  846. ("float_and_list", "[1,2]"),
  847. ("float_eq_list", "false"),
  848. ("float_neq_list", "true"),
  849. # float, dict
  850. ("float_or_dict", "10.5"),
  851. ("float_and_dict", '{"1":2}'),
  852. ("float_eq_dict", "false"),
  853. ("float_neq_dict", "true"),
  854. # str, str
  855. ("str_add_str", "firstsecond"),
  856. ("str_gt_str", "false"),
  857. ("str_lt_str", "true"),
  858. ("str_gte_str", "false"),
  859. ("str_lte_str", "true"),
  860. ("str_eq_str", "false"),
  861. ("str_neq_str", "true"),
  862. ("str_and_str", "second"),
  863. ("str_or_str", "first"),
  864. ("str_contains", "true"),
  865. ("str_lower", "third"),
  866. ("str_upper", "THIRD"),
  867. ("str_split", '["a","long","string"]'),
  868. # str, int
  869. ("str_mult_int", "firstfirstfirstfirstfirst"),
  870. ("str_and_int", "5"),
  871. ("str_or_int", "first"),
  872. ("str_eq_int", "false"),
  873. ("str_neq_int", "true"),
  874. # str, list
  875. ("str_and_list", "[1,2]"),
  876. ("str_or_list", "first"),
  877. ("str_eq_list", "false"),
  878. ("str_neq_list", "true"),
  879. # str, dict
  880. ("str_or_dict", "first"),
  881. ("str_and_dict", '{"1":2}'),
  882. ("str_eq_dict", "false"),
  883. ("str_neq_dict", "true"),
  884. # list, list
  885. ("list_add_list", "[1,2,3,4]"),
  886. ("list_gt_list", "false"),
  887. ("list_lt_list", "true"),
  888. ("list_gte_list", "false"),
  889. ("list_lte_list", "true"),
  890. ("list_eq_list", "false"),
  891. ("list_neq_list", "true"),
  892. ("list_and_list", "[3,4]"),
  893. ("list_or_list", "[1,2]"),
  894. ("list_contains", "true"),
  895. ("list_pluck", '["obj_1","obj_2"]'),
  896. ("list_reverse", "[2,1]"),
  897. ("list_join", "firstsecondthird"),
  898. ("list_join_comma", "first,second,third"),
  899. ("list_join_range1", "2,3,4"),
  900. ("list_join_range2", "2,4,6,8"),
  901. ("list_join_range3", "5,4,3,2,1"),
  902. ("list_join_range4", "0,1,2"),
  903. # list, int
  904. ("list_mult_int", "[1,2,1,2,1,2,1,2,1,2]"),
  905. ("list_or_int", "[1,2]"),
  906. ("list_and_int", "10"),
  907. ("list_eq_int", "false"),
  908. ("list_neq_int", "true"),
  909. # list, dict
  910. ("list_and_dict", '{"1":2}'),
  911. ("list_or_dict", "[1,2]"),
  912. ("list_eq_dict", "false"),
  913. ("list_neq_dict", "true"),
  914. # dict, dict
  915. ("dict_or_dict", '{"1":2}'),
  916. ("dict_and_dict", '{"3":4}'),
  917. ("dict_eq_dict", "false"),
  918. ("dict_neq_dict", "true"),
  919. ("dict_contains", "true"),
  920. # index from an op var
  921. ("list_index_mod", "second"),
  922. # html component with var
  923. ("html_str", "hello"),
  924. # index into list with foreach
  925. ("foreach_list_arg", "1\n2"),
  926. ("foreach_list_ix", "1\n2"),
  927. ("foreach_list_nested", "1\n1\n2"),
  928. # rx.memo component with state
  929. ("memo_comp", "1210"),
  930. ("memo_comp_nested", "345"),
  931. # length
  932. ("list_length", "3"),
  933. ("obj_length", "3"),
  934. # foreach in a match
  935. ("foreach_in_match", "first\nsecond\nthird"),
  936. # literal range in a foreach
  937. ("range_in_foreach1", "4269"),
  938. ("range_in_foreach2", "42454851545760636669727578"),
  939. ("range_in_foreach3", "42363024"),
  940. ("range_in_foreach4", "42"),
  941. ("dict_in_foreach1", "a1b2"),
  942. ("dict_in_foreach2", "12"),
  943. ("dict_in_foreach3", "1234"),
  944. ("str_in_foreach", "a b c d e f"),
  945. ("str_var_in_foreach", "f i r s t"),
  946. ("typed_dict_in_foreach", "Hello Alice33Hello Bob28"),
  947. # fstring operations
  948. ("float_format", "13212312312.123123"),
  949. ("float_format_0f", "13212312312"),
  950. ("float_format_1f", "13212312312.1"),
  951. ("float_format_2f", "13212312312.12"),
  952. ("float_format_comma", "13,212,312,312.123"),
  953. ("float_format_underscore", "13_212_312_312.123"),
  954. ("float_format_comma_0f", "13,212,312,312"),
  955. ("float_format_comma_1f", "13,212,312,312.1"),
  956. ("float_format_comma_2f", "13,212,312,312.12"),
  957. ("float_format_underscore_0f", "13_212_312_312"),
  958. ("float_format_underscore_1f", "13_212_312_312.1"),
  959. ("float_format_underscore_2f", "13_212_312_312.12"),
  960. ("obj_name", "hello"),
  961. ("obj_optional_none", ""),
  962. ("obj_optional_str", "hello"),
  963. ("obj_optional_none_get_none", ""),
  964. ("obj_optional_none_get_foo", "foo"),
  965. ("float_round", "13212312312"),
  966. ("float_round_2", "13212312312.12"),
  967. ]
  968. for tag, expected in tests:
  969. print(tag)
  970. assert driver.find_element(By.ID, tag).text == expected
  971. # Highlight component with var query (does not plumb ID)
  972. assert driver.find_element(By.TAG_NAME, "mark").text == "second"