mvkcontroller.xml 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. <class name="MvKController">
  2. <relationships>
  3. <association name="to_mvi" class="Server" min="1" max="1"/>
  4. </relationships>
  5. <constructor>
  6. <parameter name="params"/>
  7. <body>
  8. <![CDATA[
  9. self.mvs = ModelverseState("../bootstrap/bootstrap.m")
  10. self.root = self.mvs.read_root()[0]
  11. self.mvk = ModelverseKernel(self.root)
  12. self.all_failed = False
  13. self.timeout = False
  14. self.users = set()
  15. self.input_queue = defaultdict(list)
  16. self.output_queue = defaultdict(list)
  17. self.source = None
  18. self.mvs_operations = {
  19. "CN": self.mvs.create_node,
  20. "CE": self.mvs.create_edge,
  21. "CNV": self.mvs.create_nodevalue,
  22. "CD": self.mvs.create_dict,
  23. "RV": self.mvs.read_value,
  24. "RO": self.mvs.read_outgoing,
  25. "RI": self.mvs.read_incoming,
  26. "RE": self.mvs.read_edge,
  27. "RD": self.mvs.read_dict,
  28. "RDN": self.mvs.read_dict_node,
  29. "RDNE": self.mvs.read_dict_node_edge,
  30. "RDE": self.mvs.read_dict_edge,
  31. "RRD": self.mvs.read_reverse_dict,
  32. "RR": self.mvs.read_root,
  33. "RDK": self.mvs.read_dict_keys,
  34. "DE": self.mvs.delete_edge,
  35. "DN": self.mvs.delete_node,
  36. }
  37. self.execute_modelverse("", "load_primitives", [])
  38. ]]>
  39. </body>
  40. </constructor>
  41. <method name="execute_modelverse">
  42. <parameter name="username"/>
  43. <parameter name="operation"/>
  44. <parameter name="params"/>
  45. <body>
  46. <![CDATA[
  47. reply = None
  48. commands = []
  49. while 1:
  50. commands = self.mvk.execute_yields(username, operation, params, reply)
  51. if commands is None:
  52. break
  53. reply = [self.mvs_operations[command[0]](*(command[1]))[0] for command in commands]
  54. if len(reply) == 1:
  55. reply = reply[0]
  56. ]]>
  57. </body>
  58. </method>
  59. <scxml initial="init_server">
  60. <state id="init_server">
  61. <onentry>
  62. <raise scope="cd" event="create_instance">
  63. <parameter expr="'to_mvi'"/>
  64. <parameter expr="'Server'"/>
  65. <parameter expr="''"/>
  66. <parameter expr="8001"/>
  67. </raise>
  68. </onentry>
  69. <transition event="instance_created" target="../running">
  70. <parameter name="instancename"/>
  71. <raise scope="cd" event="start_instance">
  72. <parameter expr="instancename"/>
  73. </raise>
  74. </transition>
  75. </state>
  76. <parallel id="running">
  77. <state id="wait_for_requests">
  78. <state id="wait">
  79. <transition event="from_mvi" target=".">
  80. <parameter name="source"/>
  81. <parameter name="data"/>
  82. <script>
  83. # No JSON encoding necessary, as it is not complex
  84. try:
  85. if data["op"] == "set_input":
  86. if "element_type" in data:
  87. if data["element_type"] == "V":
  88. value = json.loads(data["value"])
  89. else:
  90. value = data["value"]
  91. args = [data["element_type"], value]
  92. self.input_queue[data["username"]].append((source, [args]))
  93. else:
  94. d = []
  95. for t, v in json.loads(data["data"]):
  96. if t == "V":
  97. v = json.loads(v)
  98. d.append([t, v])
  99. self.input_queue[data["username"]].append((source, d))
  100. elif data["op"] == "get_output":
  101. self.output_queue[data["username"]].append(source)
  102. else:
  103. print("DROPPING unknown operation: " + str(data["op"]))
  104. except:
  105. print("DROPPING deserialization error")
  106. </script>
  107. </transition>
  108. </state>
  109. </state>
  110. <state id="execution">
  111. <state id="execution">
  112. <onentry>
  113. <script>
  114. self.timeout = False
  115. self.destination = None
  116. if self.users:
  117. user = self.users.pop()
  118. # Check if there are values to input
  119. if self.input_queue[user]:
  120. source, args = self.input_queue[user].pop(0)
  121. for args_entry in args:
  122. self.execute_modelverse(user, "set_input", args_entry)
  123. self.destination = source
  124. self.value = {"id": 200, "value": "OK"}
  125. self.all_failed = False
  126. # Now process for some steps, or until we are again blocked for input
  127. for x in xrange(200):
  128. self.execute_modelverse(user, "execute_rule", [])
  129. if not self.mvk.success:
  130. # Blocking or broken, so quit already to stop wasting CPU
  131. break
  132. self.all_failed = False
  133. # Check that we don't have anything to output yet, otherwise we wait
  134. if self.destination is None:
  135. # Perform output if there is anything
  136. if self.output_queue[user]:
  137. self.execute_modelverse(user, "get_output", [])
  138. if self.mvk.success:
  139. self.destination = self.output_queue[user].pop(0)
  140. self.value = self.mvk.returnvalue
  141. self.all_failed = False
  142. else:
  143. out = self.mvs.read_outgoing(self.root)[0]
  144. for m in out:
  145. src, user = self.mvs.read_edge(m)[0]
  146. outgoing = self.mvs.read_outgoing(m)[0]
  147. first = self.mvs.read_edge(outgoing[0])[0]
  148. dest = first[1]
  149. name = self.mvs.read_value(dest)[0]
  150. if name.startswith("__"):
  151. continue
  152. self.users.add(name)
  153. self.timeout = self.all_failed
  154. self.all_failed = True
  155. </script>
  156. </onentry>
  157. <transition cond="self.destination is not None" after="0" target=".">
  158. <raise event="HTTP_input" scope="narrow" target="'to_mvi/%s' % self.destination">
  159. <parameter expr="self.value"/>
  160. </raise>
  161. </transition>
  162. <transition cond="self.timeout and self.destination is None" after="0.1" target="."/>
  163. <transition cond="not self.timeout and self.destination is None" after="0" target="."/>
  164. </state>
  165. </state>
  166. <state id="remove_sockets">
  167. <state id="remove_sockets">
  168. <transition event="delete_socket" target=".">
  169. <parameter name="socket"/>
  170. <script>
  171. for user in self.output_queue.keys():
  172. self.output_queue[user] = [s for s in self.output_queue[user] if s != socket]
  173. </script>
  174. </transition>
  175. </state>
  176. </state>
  177. </parallel>
  178. </scxml>
  179. </class>