test_sequence.py 32 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738
  1. # Copyright 2021-2024 Avaiga Private Limited
  2. #
  3. # Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with
  4. # the License. You may obtain a copy of the License at
  5. #
  6. # http://www.apache.org/licenses/LICENSE-2.0
  7. #
  8. # Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on
  9. # an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the
  10. # specific language governing permissions and limitations under the License.
  11. from unittest import mock
  12. import pytest
  13. from taipy.config import Config
  14. from taipy.config.common.scope import Scope
  15. from taipy.core.common._utils import _Subscriber
  16. from taipy.core.data._data_manager_factory import _DataManagerFactory
  17. from taipy.core.data.data_node import DataNode
  18. from taipy.core.data.in_memory import InMemoryDataNode
  19. from taipy.core.data.pickle import PickleDataNode
  20. from taipy.core.scenario._scenario_manager import _ScenarioManager
  21. from taipy.core.scenario.scenario import Scenario
  22. from taipy.core.sequence._sequence_manager import _SequenceManager
  23. from taipy.core.sequence.sequence import Sequence
  24. from taipy.core.sequence.sequence_id import SequenceId
  25. from taipy.core.task._task_manager import _TaskManager
  26. from taipy.core.task.task import Task, TaskId
  27. def test_sequence_equals():
  28. task_config = Config.configure_task("mult_by_3", print, [], None)
  29. scenario_config = Config.configure_scenario("scenario", [task_config])
  30. scenario = _ScenarioManager._create(scenario_config)
  31. scenario.add_sequences({"print": list(scenario.tasks.values())})
  32. sequence_1 = scenario.sequences["print"]
  33. sequence_id = sequence_1.id
  34. assert sequence_1.name == "print"
  35. sequence_2 = _SequenceManager._get(sequence_id)
  36. # To test if instance is same type
  37. task = Task("task", {}, print, [], [], sequence_id)
  38. assert sequence_1 == sequence_2
  39. assert sequence_1 != sequence_id
  40. assert sequence_1 != task
  41. def test_create_sequence():
  42. input = InMemoryDataNode("foo", Scope.SCENARIO)
  43. output = InMemoryDataNode("bar", Scope.SCENARIO)
  44. task = Task("baz", {}, print, [input], [output], TaskId("task_id"))
  45. sequence = Sequence({"description": "description"}, [task], sequence_id=SequenceId("name_1"))
  46. assert sequence.id == "name_1"
  47. assert sequence.owner_id is None
  48. assert sequence.description == "description"
  49. assert sequence.foo == input
  50. assert sequence.bar == output
  51. assert sequence.baz.id == task.id
  52. assert sequence.tasks == {task.config_id: task}
  53. assert sequence.data_nodes == {"foo": input, "bar": output}
  54. assert sequence.parent_ids == set()
  55. with pytest.raises(AttributeError):
  56. _ = sequence.qux
  57. assert sequence.get_label() == sequence.id
  58. assert sequence.get_simple_label() == sequence.id
  59. input_1 = InMemoryDataNode("input", Scope.SCENARIO)
  60. output_1 = InMemoryDataNode("output", Scope.SCENARIO)
  61. task_1 = Task("task_1", {}, print, [input_1], [output_1], TaskId("task_id_1"))
  62. sequence_1 = Sequence(
  63. {"description": "description"},
  64. [task_1],
  65. owner_id="owner_id",
  66. parent_ids={"scenario_id"},
  67. sequence_id=SequenceId("name_1"),
  68. )
  69. assert sequence_1.id == "name_1"
  70. assert sequence_1.owner_id == "owner_id"
  71. assert sequence_1.description == "description"
  72. assert sequence_1.input == input_1
  73. assert sequence_1.output == output_1
  74. assert sequence_1.task_1 == task_1
  75. assert sequence_1.tasks == {task_1.config_id: task_1}
  76. assert sequence_1.data_nodes == {"input": input_1, "output": output_1}
  77. assert sequence_1.parent_ids == {"scenario_id"}
  78. assert sequence_1.id is not None
  79. with mock.patch("taipy.core.get") as get_mck:
  80. class MockOwner:
  81. label = "owner_label"
  82. def get_label(self):
  83. return self.label
  84. get_mck.return_value = MockOwner()
  85. assert sequence_1.get_label() == "owner_label > " + sequence_1.id
  86. assert sequence_1.get_simple_label() == sequence_1.id
  87. sequence_2 = Sequence(
  88. {"description": "description", "name": "Name"},
  89. [task, task_1],
  90. owner_id="owner_id",
  91. parent_ids={"parent_id_1", "parent_id_2"},
  92. sequence_id=SequenceId("name_2"),
  93. )
  94. assert sequence_2.owner_id == "owner_id"
  95. assert sequence_2.id == "name_2"
  96. assert sequence_2.description == "description"
  97. assert sequence_2.tasks == {task.config_id: task, task_1.config_id: task_1}
  98. assert sequence_2.data_nodes == {"foo": input, "bar": output, "input": input_1, "output": output_1}
  99. assert sequence_2.parent_ids == {"parent_id_1", "parent_id_2"}
  100. with mock.patch("taipy.core.get") as get_mck:
  101. class MockOwner:
  102. label = "owner_label"
  103. def get_label(self):
  104. return self.label
  105. get_mck.return_value = MockOwner()
  106. assert sequence_2.get_label() == "owner_label > " + sequence_2.name
  107. assert sequence_2.get_simple_label() == sequence_2.name
  108. def test_check_consistency():
  109. sequence_1 = Sequence({}, [], "name_1")
  110. assert sequence_1._is_consistent()
  111. input_2 = InMemoryDataNode("foo", Scope.SCENARIO)
  112. output_2 = InMemoryDataNode("bar", Scope.SCENARIO)
  113. task_2 = Task("tfoo", {}, print, [input_2], [output_2], TaskId("task_id_2"))
  114. sequence_2 = Sequence({}, [task_2], "name_2")
  115. assert sequence_2._is_consistent()
  116. data_node_3 = InMemoryDataNode("foo", Scope.SCENARIO)
  117. task_3 = Task("tfoo", {}, print, [data_node_3], [data_node_3], TaskId("task_id_3"))
  118. sequence_3 = Sequence({}, [task_3], "name_3")
  119. assert not sequence_3._is_consistent() # Not a dag
  120. input_4 = InMemoryDataNode("foo", Scope.SCENARIO)
  121. output_4 = InMemoryDataNode("bar", Scope.SCENARIO)
  122. task_4_1 = Task("tfoo", {}, print, [input_4], [output_4], TaskId("task_id_4_1"))
  123. task_4_2 = Task("tbar", {}, print, [output_4], [input_4], TaskId("task_id_4_2"))
  124. sequence_4 = Sequence({}, [task_4_1, task_4_2], "name_4")
  125. assert not sequence_4._is_consistent() # Not a Dag
  126. class FakeDataNode:
  127. config_id = "config_id_of_a_fake_dn"
  128. input_6 = DataNode("foo", Scope.SCENARIO, "input_id_5")
  129. output_6 = DataNode("bar", Scope.SCENARIO, "output_id_5")
  130. task_6_1 = Task("tfoo", {}, print, [input_6], [output_6], TaskId("task_id_5_1"))
  131. task_6_2 = Task("tbar", {}, print, [output_6], [FakeDataNode()], TaskId("task_id_5_2"))
  132. sequence_6 = Sequence({}, [task_6_1, task_6_2], "name_5")
  133. assert not sequence_6._is_consistent() # Not a DataNode
  134. intermediate_7 = DataNode("foo", Scope.SCENARIO, "intermediate_id_7")
  135. output_7 = DataNode("bar", Scope.SCENARIO, "output_id_7")
  136. task_7_1 = Task("tfoo", {}, print, [], [intermediate_7], TaskId("task_id_7_1"))
  137. task_7_2 = Task("tbar", {}, print, [intermediate_7], [output_7], TaskId("task_id_7_2"))
  138. sequence_7 = Sequence({}, [task_7_1, task_7_2], "name_7")
  139. assert sequence_7._is_consistent()
  140. input_8 = DataNode("foo", Scope.SCENARIO, "output_id_8")
  141. intermediate_8 = DataNode("bar", Scope.SCENARIO, "intermediate_id_8")
  142. task_8_1 = Task("tfoo", {}, print, [input_8], [intermediate_8], TaskId("task_id_8_1"))
  143. task_8_2 = Task("tbar", {}, print, [intermediate_8], [], TaskId("task_id_8_2"))
  144. sequence_8 = Sequence({}, [task_8_1, task_8_2], "name_8")
  145. assert sequence_8._is_consistent()
  146. input_9_1 = DataNode("foo", Scope.SCENARIO, "input_id_9_1")
  147. output_9_1 = DataNode("bar", Scope.SCENARIO, "output_id_9_1")
  148. input_9_2 = DataNode("baz", Scope.SCENARIO, "input_id_9_2")
  149. output_9_2 = DataNode("qux", Scope.SCENARIO, "output_id_9_2")
  150. task_9_1 = Task("tfoo", {}, print, [input_9_1], [output_9_1], TaskId("task_id_9_1"))
  151. task_9_2 = Task("tbar", {}, print, [input_9_2], [output_9_2], TaskId("task_id_9_2"))
  152. sequence_9 = Sequence({}, [task_9_1, task_9_2], "name_9")
  153. assert not sequence_9._is_consistent() # Not connected
  154. input_10_1 = DataNode("foo", Scope.SCENARIO, "output_id_10_1")
  155. intermediate_10_1 = DataNode("bar", Scope.SCENARIO, "intermediate_id_10_1")
  156. intermediate_10_2 = DataNode("baz", Scope.SCENARIO, "intermediate_id_10_2")
  157. output_10 = DataNode("qux", Scope.SCENARIO, "output_id_10")
  158. post_10 = DataNode("quux", Scope.SCENARIO, "post_id_10")
  159. task_10_1 = Task("tfoo", {}, print, [input_10_1], [intermediate_10_1], TaskId("task_id_10_1"))
  160. task_10_2 = Task("tbar", {}, print, [], [intermediate_10_2], TaskId("task_id_10_2"))
  161. task_10_3 = Task("tbaz", {}, print, [intermediate_10_1, intermediate_10_2], [output_10], TaskId("task_id_10_3"))
  162. task_10_4 = Task("tqux", {}, print, [output_10], [post_10], TaskId("task_id_10_4"))
  163. task_10_5 = Task("tquux", {}, print, [output_10], [], TaskId("task_id_10_5"))
  164. sequence_10 = Sequence({}, [task_10_1, task_10_2, task_10_3, task_10_4, task_10_5], "name_10")
  165. assert sequence_10._is_consistent()
  166. def test_get_sorted_tasks():
  167. def assert_equal(tasks_a, tasks_b) -> bool:
  168. if len(tasks_a) != len(tasks_b):
  169. return False
  170. for i in range(len(tasks_a)):
  171. task_a, task_b = tasks_a[i], tasks_b[i]
  172. if isinstance(task_a, list) and isinstance(task_b, list):
  173. if not assert_equal(task_a, task_b):
  174. return False
  175. elif isinstance(task_a, list) or isinstance(task_b, list):
  176. return False
  177. else:
  178. index_task_b = tasks_b.index(task_a)
  179. if any(isinstance(task_b, list) for task_b in tasks_b[i : index_task_b + 1]):
  180. return False
  181. return True
  182. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  183. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  184. data_node_3 = DataNode("baz", Scope.SCENARIO, "s3")
  185. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  186. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  187. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  188. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  189. task_1 = Task(
  190. "grault",
  191. {},
  192. print,
  193. [data_node_1, data_node_2],
  194. [data_node_3, data_node_4],
  195. TaskId("t1"),
  196. )
  197. task_2 = Task("garply", {}, print, [data_node_3], [data_node_5], TaskId("t2"))
  198. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], [data_node_6], TaskId("t3"))
  199. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  200. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  201. # s1 --- ---> s3 ---> t2 ---> s5 ----
  202. # | | |
  203. # |---> t1 ---| -------------------------> t3 ---> s6
  204. # | | |
  205. # s2 --- ---> s4 ---> t4 ---> s7
  206. assert assert_equal(sequence._get_sorted_tasks(), [[task_1], [task_2, task_4], [task_3]])
  207. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  208. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  209. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  210. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  211. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  212. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  213. task_1 = Task(
  214. "grault",
  215. {},
  216. print,
  217. [data_node_1, data_node_2],
  218. [data_node_4],
  219. TaskId("t1"),
  220. )
  221. task_2 = Task("garply", {}, print, None, [data_node_5], TaskId("t2"))
  222. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], [data_node_6], TaskId("t3"))
  223. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  224. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  225. # s1 --- t2 ---> s5 ------
  226. # | |
  227. # |---> t1 ---| -----> t3 ---> s6
  228. # | | |
  229. # s2 --- ---> s4 ---> t4 ---> s7
  230. assert assert_equal(sequence._get_sorted_tasks(), [[task_2, task_1], [task_4, task_3]])
  231. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  232. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  233. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  234. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  235. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  236. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  237. task_1 = Task(
  238. "grault",
  239. {},
  240. print,
  241. [data_node_1, data_node_2],
  242. [data_node_4],
  243. TaskId("t1"),
  244. )
  245. task_2 = Task("garply", {}, print, [data_node_6], [data_node_5], TaskId("t2"))
  246. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], id=TaskId("t3"))
  247. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  248. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  249. # s1 --- s6 ---> t2 ---> s5
  250. # | |
  251. # |---> t1 ---| -----> t3
  252. # | | |
  253. # s2 --- ---> s4 ---> t4 ---> s7
  254. assert assert_equal(sequence._get_sorted_tasks(), [[task_2, task_1], [task_4, task_3]])
  255. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  256. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  257. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  258. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  259. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  260. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  261. task_1 = Task(
  262. "grault",
  263. {},
  264. print,
  265. [data_node_1, data_node_2],
  266. [data_node_4],
  267. TaskId("t1"),
  268. )
  269. task_2 = Task("garply", {}, print, output=[data_node_5], id=TaskId("t2"))
  270. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], None, id=TaskId("t3"))
  271. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  272. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  273. # s1 --- t2 ---> s5
  274. # | |
  275. # |---> t1 ---| -----> t3
  276. # | | |
  277. # s2 --- ---> s4 ---> t4 ---> s7
  278. assert assert_equal(sequence._get_sorted_tasks(), [[task_2, task_1], [task_4, task_3]])
  279. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  280. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  281. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  282. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  283. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  284. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  285. data_node_8 = DataNode("hugh", Scope.SCENARIO, "s8")
  286. task_1 = Task(
  287. "grault",
  288. {},
  289. print,
  290. [data_node_1, data_node_2],
  291. [data_node_4],
  292. TaskId("t1"),
  293. )
  294. task_2 = Task("garply", {}, print, output=[data_node_5], id=TaskId("t2"))
  295. task_3 = Task("waldo", {}, print, [data_node_4], None, id=TaskId("t3"))
  296. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  297. task_5 = Task("bob", {}, print, [data_node_8], None, TaskId("t5"))
  298. sequence = Sequence({}, [task_5, task_4, task_2, task_1, task_3], SequenceId("p1"))
  299. # s1 ---
  300. # |
  301. # |---> t1 ---| -----> t3
  302. # | | |
  303. # s2 --- ---> s4 ---> t4 ---> s7
  304. # t2 ---> s5
  305. # s8 ---> t5
  306. assert assert_equal(sequence._get_sorted_tasks(), [[task_5, task_2, task_1], [task_4, task_3]])
  307. def test_get_inputs():
  308. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  309. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  310. data_node_3 = DataNode("baz", Scope.SCENARIO, "s3")
  311. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  312. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  313. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  314. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  315. task_1 = Task("grault", {}, print, [data_node_1, data_node_2], [data_node_3, data_node_4], TaskId("t1"))
  316. task_2 = Task("garply", {}, print, [data_node_3], [data_node_5], TaskId("t2"))
  317. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], [data_node_6], TaskId("t3"))
  318. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  319. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  320. # s1 --- ---> s3 ---> t2 ---> s5 ----
  321. # | | |
  322. # |---> t1 ---| -------------------------> t3 ---> s6
  323. # | | |
  324. # s2 --- ---> s4 ---> t4 ---> s7
  325. assert sequence.get_inputs() == {data_node_1, data_node_2}
  326. assert sequence.get_outputs() == {data_node_6, data_node_7}
  327. assert sequence.get_intermediate() == {data_node_3, data_node_4, data_node_5}
  328. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  329. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  330. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  331. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  332. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  333. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  334. task_1 = Task("grault", {}, print, [data_node_1, data_node_2], [data_node_4], TaskId("t1"))
  335. task_2 = Task("garply", {}, print, None, [data_node_5], TaskId("t2"))
  336. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], [data_node_6], TaskId("t3"))
  337. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  338. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  339. # s1 --- t2 ---> s5 ------
  340. # | |
  341. # |---> t1 ---| -----> t3 ---> s6
  342. # | | |
  343. # s2 --- ---> s4 ---> t4 ---> s7
  344. assert sequence.get_inputs() == {data_node_1, data_node_2}
  345. assert sequence.get_outputs() == {data_node_6, data_node_7}
  346. assert sequence.get_intermediate() == {data_node_4, data_node_5}
  347. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  348. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  349. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  350. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  351. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  352. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  353. task_1 = Task("grault", {}, print, [data_node_1, data_node_2], [data_node_4], TaskId("t1"))
  354. task_2 = Task("garply", {}, print, [data_node_6], [data_node_5], TaskId("t2"))
  355. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], id=TaskId("t3"))
  356. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  357. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  358. # s1 --- s6 ---> t2 ---> s5
  359. # | |
  360. # |---> t1 ---| -----> t3
  361. # | | |
  362. # s2 --- ---> s4 ---> t4 ---> s7
  363. assert sequence.get_inputs() == {data_node_1, data_node_2, data_node_6}
  364. assert sequence.get_outputs() == {data_node_7}
  365. assert sequence.get_intermediate() == {data_node_4, data_node_5}
  366. data_node_1 = DataNode("foo", Scope.SCENARIO, "s1")
  367. data_node_2 = DataNode("bar", Scope.SCENARIO, "s2")
  368. data_node_4 = DataNode("qux", Scope.SCENARIO, "s4")
  369. data_node_5 = DataNode("quux", Scope.SCENARIO, "s5")
  370. data_node_6 = DataNode("quuz", Scope.SCENARIO, "s6")
  371. data_node_7 = DataNode("corge", Scope.SCENARIO, "s7")
  372. data_node_8 = DataNode("hugh", Scope.SCENARIO, "s8")
  373. task_1 = Task("grault", {}, print, [data_node_1, data_node_2], [data_node_4], TaskId("t1"))
  374. task_2 = Task("garply", {}, print, output=[data_node_5], id=TaskId("t2"))
  375. task_3 = Task("waldo", {}, print, [data_node_4], None, id=TaskId("t3"))
  376. task_4 = Task("fred", {}, print, [data_node_4, data_node_6], [data_node_7], TaskId("t4"))
  377. task_5 = Task("bob", {}, print, [data_node_8], None, TaskId("t5"))
  378. sequence = Sequence({}, [task_5, task_4, task_2, task_1, task_3], SequenceId("p1"))
  379. # s1 ---
  380. # |
  381. # |---> t1 ---| -----> t3
  382. # | | |
  383. # s2 --- ---> s4 ---> t4 ---> s7
  384. # t2 ---> s5 |
  385. # s8 ---> t5 s6 --|
  386. assert sequence.get_inputs() == {data_node_1, data_node_2, data_node_8, data_node_6}
  387. assert sequence.get_outputs() == {data_node_5, data_node_7}
  388. assert sequence.get_intermediate() == {data_node_4}
  389. def test_is_ready_to_run():
  390. data_node_1 = PickleDataNode("foo", Scope.SCENARIO, "s1", properties={"default_data": 1})
  391. data_node_2 = PickleDataNode("bar", Scope.SCENARIO, "s2", properties={"default_data": 2})
  392. data_node_4 = PickleDataNode("qux", Scope.SCENARIO, "s4", properties={"default_data": 4})
  393. data_node_5 = PickleDataNode("quux", Scope.SCENARIO, "s5", properties={"default_data": 5})
  394. data_node_6 = PickleDataNode("quuz", Scope.SCENARIO, "s6", properties={"default_data": 6})
  395. data_node_7 = PickleDataNode("corge", Scope.SCENARIO, "s7", properties={"default_data": 7})
  396. task_1 = Task("grault", {}, print, [data_node_1, data_node_2], [data_node_4], TaskId("t1"))
  397. task_2 = Task("garply", {}, print, [data_node_6], [data_node_5], TaskId("t2"))
  398. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], id=TaskId("t3"))
  399. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  400. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  401. # s1 --- s6 ---> t2 ---> s5
  402. # | |
  403. # |---> t1 ---| -----> t3
  404. # | | |
  405. # s2 --- ---> s4 ---> t4 ---> s7
  406. data_manager = _DataManagerFactory._build_manager()
  407. for dn in [data_node_1, data_node_2, data_node_4, data_node_5, data_node_6, data_node_7]:
  408. data_manager._set(dn)
  409. assert sequence.is_ready_to_run()
  410. data_node_1.edit_in_progress = True
  411. assert not sequence.is_ready_to_run()
  412. data_node_2.edit_in_progress = True
  413. data_node_6.edit_in_progress = True
  414. assert not sequence.is_ready_to_run()
  415. data_node_1.edit_in_progress = False
  416. data_node_2.edit_in_progress = False
  417. data_node_6.edit_in_progress = False
  418. assert sequence.is_ready_to_run()
  419. def test_data_nodes_being_edited():
  420. data_node_1 = PickleDataNode("foo", Scope.SCENARIO, "s1", properties={"default_data": 1})
  421. data_node_2 = PickleDataNode("bar", Scope.SCENARIO, "s2", properties={"default_data": 2})
  422. data_node_4 = PickleDataNode("qux", Scope.SCENARIO, "s4", properties={"default_data": 4})
  423. data_node_5 = PickleDataNode("quux", Scope.SCENARIO, "s5", properties={"default_data": 5})
  424. data_node_6 = PickleDataNode("quuz", Scope.SCENARIO, "s6", properties={"default_data": 6})
  425. data_node_7 = PickleDataNode("corge", Scope.SCENARIO, "s7", properties={"default_data": 7})
  426. task_1 = Task("grault", {}, print, [data_node_1, data_node_2], [data_node_4], TaskId("t1"))
  427. task_2 = Task("garply", {}, print, [data_node_6], [data_node_5], TaskId("t2"))
  428. task_3 = Task("waldo", {}, print, [data_node_5, data_node_4], id=TaskId("t3"))
  429. task_4 = Task("fred", {}, print, [data_node_4], [data_node_7], TaskId("t4"))
  430. sequence = Sequence({}, [task_4, task_2, task_1, task_3], SequenceId("p1"))
  431. # s1 --- s6 ---> t2 ---> s5
  432. # | |
  433. # |---> t1 ---| -----> t3
  434. # | | |
  435. # s2 --- ---> s4 ---> t4 ---> s7
  436. data_manager = _DataManagerFactory._build_manager()
  437. for dn in [data_node_1, data_node_2, data_node_4, data_node_5, data_node_6, data_node_7]:
  438. data_manager._set(dn)
  439. assert len(sequence.data_nodes_being_edited()) == 0
  440. assert sequence.data_nodes_being_edited() == set()
  441. data_node_1.edit_in_progress = True
  442. assert len(sequence.data_nodes_being_edited()) == 1
  443. assert sequence.data_nodes_being_edited() == {data_node_1}
  444. data_node_2.edit_in_progress = True
  445. data_node_6.edit_in_progress = True
  446. assert len(sequence.data_nodes_being_edited()) == 3
  447. assert sequence.data_nodes_being_edited() == {data_node_1, data_node_2, data_node_6}
  448. data_node_4.edit_in_progress = True
  449. data_node_5.edit_in_progress = True
  450. assert len(sequence.data_nodes_being_edited()) == 5
  451. assert sequence.data_nodes_being_edited() == {data_node_1, data_node_2, data_node_4, data_node_5, data_node_6}
  452. data_node_1.edit_in_progress = False
  453. data_node_2.edit_in_progress = False
  454. data_node_6.edit_in_progress = False
  455. assert len(sequence.data_nodes_being_edited()) == 2
  456. assert sequence.data_nodes_being_edited() == {data_node_4, data_node_5}
  457. data_node_4.edit_in_progress = False
  458. data_node_5.edit_in_progress = False
  459. data_node_7.edit_in_progress = True
  460. assert len(sequence.data_nodes_being_edited()) == 1
  461. assert sequence.data_nodes_being_edited() == {data_node_7}
  462. data_node_7.edit_in_progress = False
  463. assert len(sequence.data_nodes_being_edited()) == 0
  464. assert sequence.data_nodes_being_edited() == set()
  465. def test_get_tasks():
  466. task_1 = Task("grault", {}, print, id=TaskId("t1"))
  467. task_2 = Task("garply", {}, print, id=TaskId("t2"))
  468. task_3 = Task("waldo", {}, print, id=TaskId("t3"))
  469. sequence_1 = Sequence({}, [task_1, task_2, task_3], SequenceId("p1"))
  470. assert sequence_1.tasks == {"grault": task_1, "garply": task_2, "waldo": task_3}
  471. def test_get_set_of_tasks():
  472. task_1 = Task("grault", {}, print, id=TaskId("t1"))
  473. task_2 = Task("garply", {}, print, id=TaskId("t2"))
  474. task_3 = Task("waldo", {}, print, id=TaskId("t3"))
  475. sequence_1 = Sequence({}, [task_1, task_2, task_3], SequenceId("p1"))
  476. assert sequence_1._get_set_of_tasks() == {task_1, task_2, task_3}
  477. def test_auto_set_and_reload(task):
  478. tmp_task = Task("tmp_task_config_id", {}, print, list(task.output.values()), [], TaskId("tmp_task_id"))
  479. scenario = Scenario("scenario", [task, tmp_task], {}, sequences={"foo": {}})
  480. _TaskManager._set(task)
  481. _TaskManager._set(tmp_task)
  482. _ScenarioManager._set(scenario)
  483. sequence_1 = scenario.sequences["foo"]
  484. sequence_2 = _SequenceManager._get(sequence_1)
  485. # auto set & reload on tasks attribute
  486. assert len(sequence_1.tasks) == 0
  487. assert len(sequence_2.tasks) == 0
  488. sequence_1.tasks = [tmp_task]
  489. assert len(sequence_1.tasks) == 1
  490. assert sequence_1.tasks[tmp_task.config_id].id == tmp_task.id
  491. assert len(sequence_2.tasks) == 1
  492. assert sequence_2.tasks[tmp_task.config_id].id == tmp_task.id
  493. sequence_2.tasks = [task]
  494. assert len(sequence_1.tasks) == 1
  495. assert sequence_1.tasks[task.config_id].id == task.id
  496. assert len(sequence_2.tasks) == 1
  497. assert sequence_2.tasks[task.config_id].id == task.id
  498. assert sequence_1.owner_id == scenario.id
  499. assert sequence_2.owner_id == scenario.id
  500. # auto set & reload on subscribers attribute
  501. assert len(sequence_1.subscribers) == 0
  502. assert len(sequence_2.subscribers) == 0
  503. sequence_1.subscribers.append(print)
  504. assert len(sequence_1.subscribers) == 1
  505. assert len(sequence_2.subscribers) == 1
  506. sequence_2.subscribers.append(print)
  507. assert len(sequence_1.subscribers) == 2
  508. assert len(sequence_2.subscribers) == 2
  509. sequence_1.subscribers.clear()
  510. assert len(sequence_1.subscribers) == 0
  511. assert len(sequence_2.subscribers) == 0
  512. sequence_1.subscribers.extend([print, map])
  513. assert len(sequence_1.subscribers) == 2
  514. assert len(sequence_2.subscribers) == 2
  515. sequence_1.subscribers.remove(_Subscriber(print, []))
  516. assert len(sequence_1.subscribers) == 1
  517. assert len(sequence_2.subscribers) == 1
  518. sequence_2.subscribers.clear()
  519. assert len(sequence_1.subscribers) == 0
  520. assert len(sequence_2.subscribers) == 0
  521. sequence_1.subscribers + print + len
  522. assert len(sequence_1.subscribers) == 2
  523. assert len(sequence_2.subscribers) == 2
  524. sequence_1.subscribers = []
  525. assert len(sequence_1.subscribers) == 0
  526. assert len(sequence_2.subscribers) == 0
  527. with sequence_1 as sequence:
  528. assert len(sequence.tasks) == 1
  529. assert sequence.tasks[task.config_id].id == task.id
  530. assert len(sequence.subscribers) == 0
  531. assert sequence._is_in_context
  532. sequence.tasks = []
  533. sequence.subscribers = [print]
  534. assert len(sequence.tasks) == 1
  535. assert sequence.tasks[task.config_id].id == task.id
  536. assert len(sequence.subscribers) == 0
  537. assert sequence._is_in_context
  538. assert len(sequence_1.tasks) == 0
  539. assert len(sequence_1.subscribers) == 1
  540. assert not sequence_1._is_in_context
  541. def test_auto_set_and_reload_properties():
  542. scenario = Scenario("scenario", [], {}, sequences={"foo": {}})
  543. _ScenarioManager._set(scenario)
  544. sequence_1 = scenario.sequences["foo"]
  545. sequence_2 = _SequenceManager._get(sequence_1)
  546. # auto set & reload on properties attribute
  547. assert sequence_1.properties == {"name": "foo"}
  548. assert sequence_2.properties == {"name": "foo"}
  549. sequence_1.properties["qux"] = 4
  550. assert sequence_1.properties["qux"] == 4
  551. assert sequence_2.properties["qux"] == 4
  552. sequence_2.properties["qux"] = 5
  553. assert sequence_1.properties["qux"] == 5
  554. assert sequence_2.properties["qux"] == 5
  555. sequence_1.properties["temp_key_1"] = "temp_value_1"
  556. sequence_1.properties["temp_key_2"] = "temp_value_2"
  557. assert sequence_1.properties == {
  558. "qux": 5,
  559. "name": "foo",
  560. "temp_key_1": "temp_value_1",
  561. "temp_key_2": "temp_value_2",
  562. }
  563. assert sequence_2.properties == {
  564. "qux": 5,
  565. "name": "foo",
  566. "temp_key_1": "temp_value_1",
  567. "temp_key_2": "temp_value_2",
  568. }
  569. sequence_1.properties.pop("temp_key_1")
  570. assert "temp_key_1" not in sequence_1.properties.keys()
  571. assert "temp_key_1" not in sequence_1.properties.keys()
  572. assert sequence_1.properties == {
  573. "qux": 5,
  574. "name": "foo",
  575. "temp_key_2": "temp_value_2",
  576. }
  577. assert sequence_2.properties == {
  578. "qux": 5,
  579. "name": "foo",
  580. "temp_key_2": "temp_value_2",
  581. }
  582. sequence_2.properties.pop("temp_key_2")
  583. assert sequence_1.properties == {"name": "foo", "qux": 5}
  584. assert sequence_2.properties == {"name": "foo", "qux": 5}
  585. assert "temp_key_2" not in sequence_1.properties.keys()
  586. assert "temp_key_2" not in sequence_2.properties.keys()
  587. sequence_1.properties["temp_key_3"] = 0
  588. assert sequence_1.properties == {"name": "foo", "qux": 5, "temp_key_3": 0}
  589. assert sequence_2.properties == {"name": "foo", "qux": 5, "temp_key_3": 0}
  590. sequence_1.properties.update({"temp_key_3": 1})
  591. assert sequence_1.properties == {"name": "foo", "qux": 5, "temp_key_3": 1}
  592. assert sequence_2.properties == {"name": "foo", "qux": 5, "temp_key_3": 1}
  593. sequence_1.properties.update({})
  594. assert sequence_1.properties == {"name": "foo", "qux": 5, "temp_key_3": 1}
  595. assert sequence_2.properties == {"name": "foo", "qux": 5, "temp_key_3": 1}
  596. sequence_1.properties["temp_key_4"] = 0
  597. sequence_1.properties["temp_key_5"] = 0
  598. with sequence_1 as sequence:
  599. assert sequence._is_in_context
  600. assert sequence.properties["qux"] == 5
  601. assert sequence.properties["temp_key_3"] == 1
  602. assert sequence.properties["temp_key_4"] == 0
  603. assert sequence.properties["temp_key_5"] == 0
  604. sequence.properties["qux"] = 9
  605. sequence.properties.pop("temp_key_3")
  606. sequence.properties.pop("temp_key_4")
  607. sequence.properties.update({"temp_key_4": 1})
  608. sequence.properties.update({"temp_key_5": 2})
  609. sequence.properties.pop("temp_key_5")
  610. sequence.properties.update({})
  611. assert sequence._is_in_context
  612. assert sequence.properties["qux"] == 5
  613. assert sequence.properties["temp_key_3"] == 1
  614. assert sequence.properties["temp_key_4"] == 0
  615. assert sequence.properties["temp_key_5"] == 0
  616. assert not sequence_1._is_in_context
  617. assert sequence_1.properties["qux"] == 9
  618. assert "temp_key_3" not in sequence_1.properties.keys()
  619. assert sequence_1.properties["temp_key_4"] == 1
  620. assert "temp_key_5" not in sequence_1.properties.keys()
  621. def test_get_parents(sequence):
  622. with mock.patch("taipy.core.get_parents") as mck:
  623. sequence.get_parents()
  624. mck.assert_called_once_with(sequence)
  625. def test_subscribe_sequence():
  626. with mock.patch("taipy.core.subscribe_sequence") as mck:
  627. sequence = Sequence({}, [], "id")
  628. sequence.subscribe(None)
  629. mck.assert_called_once_with(None, None, sequence)
  630. def test_unsubscribe_sequence():
  631. with mock.patch("taipy.core.unsubscribe_sequence") as mck:
  632. sequence = Sequence({}, [], "id")
  633. sequence.unsubscribe(None)
  634. mck.assert_called_once_with(None, None, sequence)
  635. def test_submit_sequence():
  636. with mock.patch("taipy.core.sequence._sequence_manager._SequenceManager._submit") as mck:
  637. sequence = Sequence({}, [], "id")
  638. sequence.submit(None, False)
  639. mck.assert_called_once_with(sequence, None, False, False, None)