test_app.py 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. from typing import List, Tuple, Type
  2. import pytest
  3. from pynecone.app import App, DefaultState
  4. from pynecone.components import Box
  5. from pynecone.event import Event
  6. from pynecone.middleware import HydrateMiddleware
  7. from pynecone.state import State
  8. from pynecone.style import Style
  9. @pytest.fixture
  10. def app() -> App:
  11. """A base app.
  12. Returns:
  13. The app.
  14. """
  15. return App()
  16. @pytest.fixture
  17. def index_page():
  18. """An index page.
  19. Returns:
  20. The index page.
  21. """
  22. def index():
  23. return Box.create("Index")
  24. return index
  25. @pytest.fixture
  26. def about_page():
  27. """An about page.
  28. Returns:
  29. The about page.
  30. """
  31. def about():
  32. return Box.create("About")
  33. return about
  34. @pytest.fixture()
  35. def TestState() -> Type[State]:
  36. """A default state.
  37. Returns:
  38. A default state.
  39. """
  40. class TestState(State):
  41. var: int
  42. return TestState
  43. def test_default_app(app: App):
  44. """Test creating an app with no args.
  45. Args:
  46. app: The app to test.
  47. """
  48. assert app.state() == DefaultState()
  49. assert app.middleware == [HydrateMiddleware()]
  50. assert app.style == Style()
  51. def test_add_page_default_route(app: App, index_page, about_page):
  52. """Test adding a page to an app.
  53. Args:
  54. app: The app to test.
  55. index_page: The index page.
  56. about_page: The about page.
  57. """
  58. assert app.pages == {}
  59. app.add_page(index_page)
  60. assert set(app.pages.keys()) == {"index"}
  61. app.add_page(about_page)
  62. assert set(app.pages.keys()) == {"index", "about"}
  63. def test_add_page_set_route(app: App, index_page):
  64. """Test adding a page to an app.
  65. Args:
  66. app: The app to test.
  67. index_page: The index page.
  68. """
  69. assert app.pages == {}
  70. app.add_page(index_page, route="/test")
  71. assert set(app.pages.keys()) == {"test"}
  72. def test_add_page_set_route_nested(app: App, index_page):
  73. """Test adding a page to an app.
  74. Args:
  75. app: The app to test.
  76. index_page: The index page.
  77. """
  78. assert app.pages == {}
  79. app.add_page(index_page, route="/test/nested")
  80. assert set(app.pages.keys()) == {"test/nested"}
  81. def test_initialize_with_state(TestState: Type[State]):
  82. """Test setting the state of an app.
  83. Args:
  84. TestState: The default state.
  85. """
  86. app = App(state=TestState)
  87. assert app.state == TestState
  88. # Get a state for a given token.
  89. token = "token"
  90. state = app.state_manager.get_state(token)
  91. assert isinstance(state, TestState)
  92. assert state.var == 0 # type: ignore
  93. def test_set_and_get_state(TestState: Type[State]):
  94. """Test setting and getting the state of an app with different tokens.
  95. Args:
  96. TestState: The default state.
  97. """
  98. app = App(state=TestState)
  99. # Create two tokens.
  100. token1 = "token1"
  101. token2 = "token2"
  102. # Get the default state for each token.
  103. state1 = app.state_manager.get_state(token1)
  104. state2 = app.state_manager.get_state(token2)
  105. assert state1.var == 0 # type: ignore
  106. assert state2.var == 0 # type: ignore
  107. # Set the vars to different values.
  108. state1.var = 1
  109. state2.var = 2
  110. app.state_manager.set_state(token1, state1)
  111. app.state_manager.set_state(token2, state2)
  112. # Get the states again and check the values.
  113. state1 = app.state_manager.get_state(token1)
  114. state2 = app.state_manager.get_state(token2)
  115. assert state1.var == 1 # type: ignore
  116. assert state2.var == 2 # type: ignore
  117. @pytest.fixture
  118. def list_mutation_state():
  119. """A fixture to create a state with list mutation features.
  120. Returns:
  121. A state with list mutation features.
  122. """
  123. class TestState(State):
  124. """The test state."""
  125. # plain list
  126. plain_friends = ["Tommy"]
  127. def make_friend(self):
  128. self.plain_friends.append("another-fd")
  129. def change_first_friend(self):
  130. self.plain_friends[0] = "Jenny"
  131. def unfriend_all_friends(self):
  132. self.plain_friends.clear()
  133. def unfriend_first_friend(self):
  134. del self.plain_friends[0]
  135. def remove_last_friend(self):
  136. self.plain_friends.pop()
  137. def make_friends_with_colleagues(self):
  138. colleagues = ["Peter", "Jimmy"]
  139. self.plain_friends.extend(colleagues)
  140. def remove_tommy(self):
  141. self.plain_friends.remove("Tommy")
  142. # list in dict
  143. friends_in_dict = {"Tommy": ["Jenny"]}
  144. def remove_jenny_from_tommy(self):
  145. self.friends_in_dict["Tommy"].remove("Jenny")
  146. def add_jimmy_to_tommy_friends(self):
  147. self.friends_in_dict["Tommy"].append("Jimmy")
  148. def tommy_has_no_fds(self):
  149. self.friends_in_dict["Tommy"].clear()
  150. # nested list
  151. friends_in_nested_list = [["Tommy"], ["Jenny"]]
  152. def remove_first_group(self):
  153. self.friends_in_nested_list.pop(0)
  154. def remove_first_person_from_first_group(self):
  155. self.friends_in_nested_list[0].pop(0)
  156. def add_jimmy_to_second_group(self):
  157. self.friends_in_nested_list[1].append("Jimmy")
  158. return TestState()
  159. @pytest.mark.asyncio
  160. @pytest.mark.parametrize(
  161. "event_tuples",
  162. [
  163. pytest.param(
  164. [
  165. (
  166. "test_state.make_friend",
  167. {"test_state": {"plain_friends": ["Tommy", "another-fd"]}},
  168. ),
  169. (
  170. "test_state.change_first_friend",
  171. {"test_state": {"plain_friends": ["Jenny", "another-fd"]}},
  172. ),
  173. ],
  174. id="append then __setitem__",
  175. ),
  176. pytest.param(
  177. [
  178. (
  179. "test_state.unfriend_first_friend",
  180. {"test_state": {"plain_friends": []}},
  181. ),
  182. (
  183. "test_state.make_friend",
  184. {"test_state": {"plain_friends": ["another-fd"]}},
  185. ),
  186. ],
  187. id="delitem then append",
  188. ),
  189. pytest.param(
  190. [
  191. (
  192. "test_state.make_friends_with_colleagues",
  193. {"test_state": {"plain_friends": ["Tommy", "Peter", "Jimmy"]}},
  194. ),
  195. (
  196. "test_state.remove_tommy",
  197. {"test_state": {"plain_friends": ["Peter", "Jimmy"]}},
  198. ),
  199. (
  200. "test_state.remove_last_friend",
  201. {"test_state": {"plain_friends": ["Peter"]}},
  202. ),
  203. (
  204. "test_state.unfriend_all_friends",
  205. {"test_state": {"plain_friends": []}},
  206. ),
  207. ],
  208. id="extend, remove, pop, clear",
  209. ),
  210. pytest.param(
  211. [
  212. (
  213. "test_state.add_jimmy_to_second_group",
  214. {
  215. "test_state": {
  216. "friends_in_nested_list": [["Tommy"], ["Jenny", "Jimmy"]]
  217. }
  218. },
  219. ),
  220. (
  221. "test_state.remove_first_person_from_first_group",
  222. {
  223. "test_state": {
  224. "friends_in_nested_list": [[], ["Jenny", "Jimmy"]]
  225. }
  226. },
  227. ),
  228. (
  229. "test_state.remove_first_group",
  230. {"test_state": {"friends_in_nested_list": [["Jenny", "Jimmy"]]}},
  231. ),
  232. ],
  233. id="nested list",
  234. ),
  235. pytest.param(
  236. [
  237. (
  238. "test_state.add_jimmy_to_tommy_friends",
  239. {"test_state": {"friends_in_dict": {"Tommy": ["Jenny", "Jimmy"]}}},
  240. ),
  241. (
  242. "test_state.remove_jenny_from_tommy",
  243. {"test_state": {"friends_in_dict": {"Tommy": ["Jimmy"]}}},
  244. ),
  245. (
  246. "test_state.tommy_has_no_fds",
  247. {"test_state": {"friends_in_dict": {"Tommy": []}}},
  248. ),
  249. ],
  250. id="list in dict",
  251. ),
  252. ],
  253. )
  254. async def test_list_mutation_detection__plain_list(
  255. event_tuples: List[Tuple[str, List[str]]], list_mutation_state: State
  256. ):
  257. """Test list mutation detection
  258. when reassignment is not explicitly included in the logic.
  259. Args:
  260. event_tuples: From parametrization.
  261. list_mutation_state: A state with list mutation features.
  262. """
  263. for event_name, expected_delta in event_tuples:
  264. result = await list_mutation_state.process(
  265. Event(
  266. token="fake-token",
  267. name=event_name,
  268. router_data={"pathname": "/", "query": {}},
  269. payload={},
  270. )
  271. )
  272. assert result.delta == expected_delta