mvkcontroller.xml 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. <class name="MvKController">
  2. <relationships>
  3. <association name="to_mvi" class="Server" min="1" max="1"/>
  4. <association name="tasks" class="Task"/>
  5. <association name="services" class="Service"/>
  6. </relationships>
  7. <constructor>
  8. <parameter name="params"/>
  9. <body>
  10. <![CDATA[
  11. self.mvs = ModelverseState("../bootstrap/bootstrap.m.gz")
  12. # Enable Garbage Collection
  13. self.mvs.GC = False
  14. self.root = self.mvs.read_root()
  15. # Instantiate the kernel.
  16. self.mvk = ModelverseKernel(self.root)
  17. # Parse kernel-related options
  18. default_kernel_type = 'adaptive-jit'
  19. kernel_opts = [default_kernel_type]
  20. for parameter in params:
  21. if parameter.startswith('--kernel='):
  22. kernel_opts = parameter[len('--kernel='):].split(',')
  23. self.port = int(params[0])
  24. self.sc_map = {}
  25. self.forward = None
  26. self.HTTP_reply = None
  27. self.input_queue = defaultdict(list)
  28. self.output_queue = defaultdict(list)
  29. self.mvs_operations = {
  30. "CN": self.mvs.create_node,
  31. "CE": self.mvs.create_edge,
  32. "CNV": self.mvs.create_nodevalue,
  33. "CD": self.mvs.create_dict,
  34. "RV": self.mvs.read_value,
  35. "RO": self.mvs.read_outgoing,
  36. "RI": self.mvs.read_incoming,
  37. "RE": self.mvs.read_edge,
  38. "RD": self.mvs.read_dict,
  39. "RDN": self.mvs.read_dict_node,
  40. "RDNE": self.mvs.read_dict_node_edge,
  41. "RDE": self.mvs.read_dict_edge,
  42. "RRD": self.mvs.read_reverse_dict,
  43. "RR": self.mvs.read_root,
  44. "RDK": self.mvs.read_dict_keys,
  45. "DE": self.mvs.delete_edge,
  46. "DN": self.mvs.delete_node,
  47. }
  48. self.execute_modelverse("", "load_primitives", [])
  49. ]]>
  50. </body>
  51. </constructor>
  52. <method name="execute_modelverse">
  53. <parameter name="taskname"/>
  54. <parameter name="operation"/>
  55. <parameter name="params"/>
  56. <body>
  57. <![CDATA[
  58. reply = None
  59. commands = []
  60. mvk = self.mvk
  61. mvs_operations = self.mvs_operations
  62. try:
  63. while 1:
  64. commands = mvk.execute_yields(taskname, operation, params, reply)
  65. if commands is None:
  66. break
  67. reply = [mvs_operations[command[0]](*(command[1])) for command in commands]
  68. except:
  69. print("ERROR: " + str(self.mvk.debug_info.get(taskname, "Unknown taskname")))
  70. raise
  71. ]]>
  72. </body>
  73. </method>
  74. <scxml initial="init_server">
  75. <state id="init_server">
  76. <onentry>
  77. <raise scope="cd" event="create_instance">
  78. <parameter expr="'to_mvi'"/>
  79. <parameter expr="'Server'"/>
  80. <parameter expr="''"/>
  81. <parameter expr="self.port"/>
  82. </raise>
  83. </onentry>
  84. <transition event="instance_created" target="../running">
  85. <parameter name="instancename"/>
  86. <raise scope="cd" event="start_instance">
  87. <parameter expr="instancename"/>
  88. </raise>
  89. </transition>
  90. </state>
  91. <parallel id="running">
  92. <state id="forward_inputs">
  93. <state id="forward">
  94. <transition cond="set(self.input_queue) &amp; set(self.sc_map)" target=".">
  95. <script>
  96. self.input_task = (set(self.input_queue) &amp; set(self.sc_map)).pop()
  97. value = self.input_queue[self.input_task].pop(0)
  98. if not self.input_queue[self.input_task]:
  99. del self.input_queue[self.input_task]
  100. </script>
  101. <raise event="input" scope="narrow" target="self.sc_map[self.input_task]">
  102. <parameter expr="value"/>
  103. </raise>
  104. </transition>
  105. </state>
  106. </state>
  107. <state id="forward_outputs">
  108. <state id="forward">
  109. <transition cond="set(self.output_queue) &amp; set(self.sc_map)" target=".">
  110. <script>
  111. self.output_task = (set(self.output_queue) &amp; set(self.sc_map)).pop()
  112. value = self.output_queue[self.output_task].pop(0)
  113. if not self.output_queue[self.output_task]:
  114. del self.output_queue[self.output_task]
  115. </script>
  116. <raise event="output" scope="narrow" target="self.sc_map[self.output_task]">
  117. <parameter expr="value"/>
  118. </raise>
  119. </transition>
  120. </state>
  121. </state>
  122. <state id="wait_for_requests">
  123. <state id="wait">
  124. <transition event="from_mvi" target=".">
  125. <parameter name="source"/>
  126. <parameter name="data"/>
  127. <script>
  128. self.source = source
  129. # No JSON encoding necessary, as it is not complex
  130. try:
  131. if data["op"] == "set_input":
  132. if "value" in data:
  133. value = [json.loads(data["value"])]
  134. else:
  135. value = json.loads(data["data"])
  136. self.forward = (data["taskname"], "input", value)
  137. self.HTTP_reply = "OK"
  138. elif data["op"] == "get_output":
  139. self.forward = (data["taskname"], "output", self.source)
  140. self.HTTP_reply = None
  141. elif data["op"] == "pause":
  142. #TODO
  143. self.HTTP_reply = "NotImplemented"
  144. elif data["op"] == "resume":
  145. #TODO
  146. self.HTTP_reply = "NotImplemented"
  147. else:
  148. self.HTTP_reply = "Unknown command: %s" % data["op"]
  149. except ValueError as e:
  150. self.HTTP_reply = "Error when deserializing request: %s" % traceback.format_exc()
  151. except Exception as e:
  152. self.HTTP_reply = "Unknown Modelverse Error: %s" % traceback.format_exc()
  153. </script>
  154. </transition>
  155. <transition cond="not self.forward and self.HTTP_reply is not None" target=".">
  156. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % self.source">
  157. <parameter expr="json.dumps(self.HTTP_reply)"/>
  158. </raise>
  159. <script>
  160. self.HTTP_reply = None
  161. self.forward = None
  162. </script>
  163. </transition>
  164. <transition cond="self.forward and self.forward[1] == 'input' and self.forward[0] in self.sc_map" target=".">
  165. <raise event="input" scope="narrow" target="self.sc_map[self.forward[0]]">
  166. <parameter expr="self.forward[2]"/>
  167. </raise>
  168. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % self.source">
  169. <parameter expr="json.dumps(self.HTTP_reply)"/>
  170. </raise>
  171. <script>
  172. self.HTTP_reply = None
  173. self.forward = None
  174. </script>
  175. </transition>
  176. <transition cond="self.forward and self.forward[1] == 'input' and self.forward[0] not in self.sc_map" target=".">
  177. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % self.source">
  178. <parameter expr="json.dumps(self.HTTP_reply)"/>
  179. </raise>
  180. <script>
  181. self.input_queue[self.forward[0]].append(self.forward[2])
  182. self.HTTP_reply = None
  183. self.forward = None
  184. </script>
  185. </transition>
  186. <transition cond="self.forward and self.forward[1] == 'output' and self.forward[0] in self.sc_map" target=".">
  187. <raise event="output" scope="narrow" target="self.sc_map[self.forward[0]]">
  188. <parameter expr="self.forward[2]"/>
  189. </raise>
  190. <script>
  191. self.HTTP_reply = None
  192. self.forward = None
  193. </script>
  194. </transition>
  195. <transition cond="self.forward and self.forward[1] == 'output' and self.forward[0] not in self.sc_map" target=".">
  196. <script>
  197. self.output_queue[self.forward[0]].append(self.forward[2])
  198. self.HTTP_reply = None
  199. self.forward = None
  200. </script>
  201. </transition>
  202. </state>
  203. </state>
  204. <parallel id="execution">
  205. <state id="refresh_tasks" initial="refresh_tasks">
  206. <state id="refresh_tasks">
  207. <onentry>
  208. <script>
  209. out = self.mvs.read_outgoing(self.root)
  210. tasks = set()
  211. for m in out:
  212. src, task = self.mvs.read_edge(m)
  213. outgoing = self.mvs.read_outgoing(m)
  214. first = self.mvs.read_edge(outgoing[0])
  215. dest = first[1]
  216. name = self.mvs.read_value(dest)
  217. if name == "__hierarchy":
  218. # Skip the special case, as this is neither data, nor a task
  219. continue
  220. tasks.add(name)
  221. self.new_tasks = set([i for i in tasks if i not in self.sc_map])
  222. self.old_tasks = set([i for i in self.sc_map if i not in tasks])
  223. </script>
  224. </onentry>
  225. <transition cond="self.new_tasks" target="../create_SC"/>
  226. <transition cond="not self.new_tasks and self.old_tasks" target="../remove_SC"/>
  227. <transition after="1.0" target="."/>
  228. </state>
  229. <state id="create_SC" initial="creating">
  230. <state id="creating">
  231. <onentry>
  232. <script>
  233. self.task = self.new_tasks.pop()
  234. </script>
  235. <raise event="create_instance" scope="cd">
  236. <parameter expr="'services' if self.task.startswith('__') else 'tasks'"/>
  237. <parameter expr="'Service' if self.task.startswith('__') else 'Task'"/>
  238. <parameter expr="self.task"/>
  239. <parameter expr="self.mvs_operations"/>
  240. <parameter expr="self.mvk"/>
  241. </raise>
  242. </onentry>
  243. <transition event="instance_created" target="../created">
  244. <parameter name="instancename"/>
  245. <raise scope="cd" event="start_instance">
  246. <parameter expr="instancename" />
  247. </raise>
  248. <script>
  249. self.sc_map[self.task] = instancename
  250. </script>
  251. </transition>
  252. </state>
  253. <state id="created">
  254. <transition cond="self.new_tasks" target="../creating"/>
  255. <transition cond="not self.new_tasks and self.old_tasks" target="../../remove_SC"/>
  256. <transition cond="not self.new_tasks and not self.old_tasks" target="../../refresh_tasks"/>
  257. </state>
  258. </state>
  259. <state id="remove_SC">
  260. <transition cond="self.old_tasks" target=".">
  261. <script>
  262. task = self.old_tasks.pop()
  263. </script>
  264. <raise scope="cd" event="delete_instance">
  265. <parameter expr="self.sc_map[task]"/>
  266. </raise>
  267. <script>
  268. del self.sc_map[task]
  269. </script>
  270. </transition>
  271. <transition cond="not self.old_tasks" target="../refresh_tasks"/>
  272. </state>
  273. </state>
  274. <state id="mvs_GC" initial="suspend_tasks">
  275. <state id="suspend_tasks">
  276. <onentry>
  277. <raise scope="broad" event="pause_task"/>
  278. </onentry>
  279. <transition target="../mvs_GC"/>
  280. </state>
  281. <state id="mvs_GC">
  282. <onentry>
  283. <script>
  284. self.execute_modelverse("", "protect_temporary_variables", [])
  285. self.mvs.purge()
  286. self.execute_modelverse("", "unprotect_temporary_variables", [])
  287. # Clean up all used memory now, as this is rather much due to the copy of the MvS
  288. import gc
  289. gc.collect()
  290. </script>
  291. <raise scope="broad" event="resume_task"/>
  292. </onentry>
  293. <transition after="self.sccd_yield() + 10" target="."/>
  294. </state>
  295. </state>
  296. </parallel>
  297. <state id="remove_sockets">
  298. <state id="remove_sockets">
  299. <transition event="delete_socket" target=".">
  300. <parameter name="socket"/>
  301. <script>
  302. for task in self.output_queue.keys():
  303. self.output_queue[task] = [s for s in self.output_queue[task] if s != socket]
  304. </script>
  305. </transition>
  306. </state>
  307. </state>
  308. </parallel>
  309. </scxml>
  310. </class>