primitives.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575
  1. class PrimitiveFinished(Exception):
  2. """Exception to indicate the result value of a primitive, as a return cannot be used."""
  3. def __init__(self, value):
  4. Exception.__init__(self)
  5. self.result = value
  6. class InterpretedFunctionFinished(Exception):
  7. """Exception to indicate the result value of an interpreted function, as a return
  8. cannot be used."""
  9. def __init__(self, value):
  10. Exception.__init__(self)
  11. self.result = value
  12. # Functions annotated with __exception_return use the JIT's calling convention instead of
  13. # the kernel's: returns are handled by throwing a PrimitiveFinished exception; the caller's
  14. # returnvalue is not modified.
  15. #
  16. # ### Rationale for __exception_return
  17. #
  18. # __exception_return is a useful mechanism because it allows us to have a __call_function
  19. # implementation that has O(1) state read overhead. A previous implementation of
  20. # __call_function checked if the caller's frame had been popped whenever
  21. # ModelverseKernel.execute_yield threw a StopIteration exception. However, that incurs O(n) overhead
  22. # _per call,_ where n is the number of StopIteration exceptions that are thrown during the call.
  23. # O(n) is pretty bad, but this actually becomes O(n * m) when m calls to __call_function are
  24. # nested. And that's just not acceptable.
  25. # __exception_return requires kernel support, but I think the complexity gains are well worth it;
  26. # I reckon JIT-to-interpreter switches aren't going to get a whole lot cheaper than this.
  27. EXCEPTION_RETURN_KEY = "__exception_return"
  28. """A dictionary key for functions which request that the kernel throw a InterpretedFunctionFinished
  29. exception with the return value instead of injecting the return value in the caller's frame."""
  30. def integer_subtraction(a, b, **remainder):
  31. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  32. result, = yield [("CNV", [a_value - b_value])]
  33. raise PrimitiveFinished(result)
  34. def integer_addition(a, b, **remainder):
  35. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  36. result, = yield [("CNV", [a_value + b_value])]
  37. raise PrimitiveFinished(result)
  38. def integer_multiplication(a, b, **remainder):
  39. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  40. result, = yield [("CNV", [a_value * b_value])]
  41. raise PrimitiveFinished(result)
  42. def integer_division(a, b, **remainder):
  43. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  44. result, = yield [("CNV", [int(a_value) / b_value])]
  45. raise PrimitiveFinished(result)
  46. def integer_gt(a, b, **remainder):
  47. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  48. result, = yield [("CNV", [a_value > b_value])]
  49. raise PrimitiveFinished(result)
  50. def integer_lt(a, b, **remainder):
  51. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  52. result, = yield [("CNV", [a_value < b_value])]
  53. raise PrimitiveFinished(result)
  54. def integer_neg(a, **remainder):
  55. a_value, = yield [("RV", [a])]
  56. result, = yield [("CNV", [-a_value])]
  57. raise PrimitiveFinished(result)
  58. def bool_and(a, b, **remainder):
  59. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  60. result, = yield [("CNV", [a_value and b_value])]
  61. raise PrimitiveFinished(result)
  62. def bool_or(a, b, **remainder):
  63. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  64. result, = yield [("CNV", [a_value or b_value])]
  65. raise PrimitiveFinished(result)
  66. def bool_not(a, **remainder):
  67. a_value, = yield [("RV", [a])]
  68. result, = yield [("CNV", [not a_value])]
  69. raise PrimitiveFinished(result)
  70. def float_subtraction(a, b, **remainder):
  71. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  72. result, = yield [("CNV", [a_value - b_value])]
  73. raise PrimitiveFinished(result)
  74. def float_addition(a, b, **remainder):
  75. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  76. result, = yield [("CNV", [a_value + b_value])]
  77. raise PrimitiveFinished(result)
  78. def float_multiplication(a, b, **remainder):
  79. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  80. result, = yield [("CNV", [a_value * b_value])]
  81. raise PrimitiveFinished(result)
  82. def float_division(a, b, **remainder):
  83. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  84. result, = yield [("CNV", [float(a_value) / float(b_value)])]
  85. raise PrimitiveFinished(result)
  86. def float_gt(a, b, **remainder):
  87. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  88. result, = yield [("CNV", [a_value > b_value])]
  89. raise PrimitiveFinished(result)
  90. def float_lt(a, b, **remainder):
  91. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  92. result, = yield [("CNV", [a_value < b_value])]
  93. raise PrimitiveFinished(result)
  94. def float_neg(a, **remainder):
  95. a_value, = yield [("RV", [a])]
  96. result, = yield [("CNV", [-a_value])]
  97. raise PrimitiveFinished(result)
  98. def string_join(a, b, **remainder):
  99. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  100. result, = yield [("CNV", [str(a_value) + str(b_value)])]
  101. raise PrimitiveFinished(result)
  102. def string_split(a, b, **remainder):
  103. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  104. result = a_value.split(b_value)
  105. elems = yield [("CN", [])] + [("CNV", [v]) for v in result]
  106. new_val = elems[0]
  107. yield [("CD", [new_val, i, v]) for i, v in enumerate(elems[1:])]
  108. raise PrimitiveFinished(new_val)
  109. def string_get(a, b, **remainder):
  110. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  111. result, = yield [("CNV", [a_value[b_value]])]
  112. raise PrimitiveFinished(result)
  113. def string_len(a, **remainder):
  114. a_value, = yield [("RV", [a])]
  115. result, = yield [("CNV", [len(a_value)])]
  116. raise PrimitiveFinished(result)
  117. def value_eq(a, b, **remainder):
  118. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  119. result, = yield [("CNV", [a_value == b_value])]
  120. raise PrimitiveFinished(result)
  121. def value_neq(a, b, **remainder):
  122. a_value, b_value = yield [("RV", [a]), ("RV", [b])]
  123. result, = yield [("CNV", [a_value != b_value])]
  124. raise PrimitiveFinished(result)
  125. def element_eq(a, b, **remainder):
  126. result, = yield [("CNV", [a == b])]
  127. raise PrimitiveFinished(result)
  128. def element_neq(a, b, **remainder):
  129. result, = yield [("CNV", [a != b])]
  130. raise PrimitiveFinished(result)
  131. def cast_a2s(a, **remainder):
  132. a_value, = yield [("RV", [a])]
  133. result, = yield [("CNV", [str(a_value["value"])])]
  134. raise PrimitiveFinished(result)
  135. def cast_i2f(a, **remainder):
  136. a_value, = yield [("RV", [a])]
  137. result, = yield [("CNV", [float(a_value)])]
  138. raise PrimitiveFinished(result)
  139. def cast_i2s(a, **remainder):
  140. a_value, = yield [("RV", [a])]
  141. result, = yield [("CNV", [str(a_value)])]
  142. raise PrimitiveFinished(result)
  143. def cast_i2b(a, **remainder):
  144. a_value, = yield [("RV", [a])]
  145. result, = yield [("CNV", [bool(a_value)])]
  146. raise PrimitiveFinished(result)
  147. def cast_f2i(a, **remainder):
  148. a_value, = yield [("RV", [a])]
  149. result, = yield [("CNV", [int(a_value)])]
  150. raise PrimitiveFinished(result)
  151. def cast_f2s(a, **remainder):
  152. a_value, = yield [("RV", [a])]
  153. result, = yield [("CNV", [str(a_value)])]
  154. raise PrimitiveFinished(result)
  155. def cast_f2b(a, **remainder):
  156. a_value, = yield [("RV", [a])]
  157. result, = yield [("CNV", [bool(a_value)])]
  158. raise PrimitiveFinished(result)
  159. def cast_s2i(a, **remainder):
  160. a_value, = yield [("RV", [a])]
  161. result, = yield [("CNV", [int(a_value)])]
  162. raise PrimitiveFinished(result)
  163. def cast_s2f(a, **remainder):
  164. a_value, = yield [("RV", [a])]
  165. result, = yield [("CNV", [float(a_value)])]
  166. raise PrimitiveFinished(result)
  167. def cast_s2b(a, **remainder):
  168. a_value, = yield [("RV", [a])]
  169. result, = yield [("CNV", [bool(a_value)])]
  170. raise PrimitiveFinished(result)
  171. def cast_b2i(a, **remainder):
  172. a_value, = yield [("RV", [a])]
  173. result, = yield [("CNV", [int(a_value)])]
  174. raise PrimitiveFinished(result)
  175. def cast_b2f(a, **remainder):
  176. a_value, = yield [("RV", [a])]
  177. result, = yield [("CNV", [float(a_value)])]
  178. raise PrimitiveFinished(result)
  179. def cast_b2s(a, **remainder):
  180. a_value, = yield [("RV", [a])]
  181. result, = yield [("CNV", [str(a_value)])]
  182. raise PrimitiveFinished(result)
  183. def cast_e2s(a, **remainder):
  184. a_value, = yield [("RV", [a])]
  185. result, = yield [("CNV", ["{ID: %s, value: %s}" % (a, a_value)])]
  186. raise PrimitiveFinished(result)
  187. def cast_v2s(a, **remainder):
  188. a_value, = yield [("RV", [a])]
  189. if isinstance(a_value, (str, unicode)):
  190. # String should be encoded to distinguish between 3 and "3"
  191. a_value = '"%s"' % a_value
  192. elif isinstance(a_value, dict):
  193. # Action or type
  194. a_value = a_value["value"]
  195. result, = yield [("CNV", ["%s" % (a_value)])]
  196. raise PrimitiveFinished(result)
  197. def cast_id2s(a, **remainder):
  198. result, = yield [("CNV", ["%s" % (a)])]
  199. raise PrimitiveFinished(result)
  200. def list_append(a, b, **remainder):
  201. a_outgoing, = yield [("RO", [a])]
  202. _ = yield [("CD", [a, len(a_outgoing), b])]
  203. raise PrimitiveFinished(a)
  204. def list_insert(a, b, c, **remainder):
  205. a_outgoing, c_value = yield [("RO", [a]), ("RV", [c])]
  206. links = yield [("RD", [a, i]) for i in range(c_value, len(a_outgoing))] + \
  207. [("RDE", [a, i]) for i in range(c_value, len(a_outgoing))]
  208. values = links[:len(links)/2]
  209. edges = links[len(links)/2:]
  210. yield [("CD", [a, c_value, b])] + \
  211. [("CD", [a, c_value + 1 + index, value]) for index, value in enumerate(values)] + \
  212. [("DE", [i]) for i in edges]
  213. raise PrimitiveFinished(a)
  214. def list_delete(a, b, **remainder):
  215. a_outgoing, b_value = yield [("RO", [a]), ("RV", [b])]
  216. links = yield [("RD", [a, i]) for i in range(b_value, len(a_outgoing))] + \
  217. [("RDE", [a, i]) for i in range(b_value, len(a_outgoing))]
  218. values = links[:len(links)/2]
  219. edges = links[len(links)/2:]
  220. yield [("CD", [a, b_value + index, value]) for index, value in enumerate(values[1:])] + \
  221. [("DE", [i]) for i in edges]
  222. raise PrimitiveFinished(a)
  223. def list_read(a, b, **remainder):
  224. b_value, = yield [("RV", [b])]
  225. result, = yield [("RD", [a, b_value])]
  226. if result is None:
  227. raise Exception("List read out of bounds: %s" % b_value)
  228. raise PrimitiveFinished(result)
  229. def list_len(a, **remainder):
  230. outgoings, = yield [("RO", [a])]
  231. result, = yield [("CNV", [len(outgoings)])]
  232. raise PrimitiveFinished(result)
  233. def dict_add(a, b, c, **remainder):
  234. new_edge, = yield [("CE", [a, c])]
  235. yield [("CE", [new_edge, b])]
  236. raise PrimitiveFinished(a)
  237. def dict_delete(a, b, **remainder):
  238. b_value, = yield [("RV", [b])]
  239. edge, = yield [("RDE", [a, b_value])]
  240. if edge is None:
  241. print("Failed dict_delete for value %s!" % b_value)
  242. keys, = yield [("RDK", [a])]
  243. keys = yield [("RV", [i]) for i in keys]
  244. print("Keys: " + str(keys))
  245. yield [("DE", [edge])]
  246. raise PrimitiveFinished(a)
  247. def dict_delete_node(a, b, **remainder):
  248. edge, = yield [("RDNE", [a, b])]
  249. if edge is None:
  250. print("Failed dict_delete_node!")
  251. yield [("DE", [edge])]
  252. raise PrimitiveFinished(a)
  253. def dict_read(a, b, **remainder):
  254. b_value, = yield [("RV", [b])]
  255. result, = yield [("RD", [a, b_value])]
  256. raise PrimitiveFinished(result)
  257. def dict_read_edge(a, b, **remainder):
  258. b_value, = yield [("RV", [b])]
  259. result, = yield [("RDE", [a, b_value])]
  260. raise PrimitiveFinished(result)
  261. def dict_read_node(a, b, **remainder):
  262. result, = yield [("RDN", [a, b])]
  263. raise PrimitiveFinished(result)
  264. def dict_in(a, b, **remainder):
  265. b_value, = yield [("RV", [b])]
  266. value, = yield [("RD", [a, b_value])]
  267. is_in = value is not None
  268. result, = yield [("CNV", [is_in])]
  269. raise PrimitiveFinished(result)
  270. def dict_in_node(a, b, **remainder):
  271. value, = yield [("RDN", [a, b])]
  272. result, = yield [("CNV", [value is not None])]
  273. raise PrimitiveFinished(result)
  274. def dict_len(a, **remainder):
  275. outgoings, = yield [("RO", [a])]
  276. result, = yield [("CNV", [len(outgoings)])]
  277. raise PrimitiveFinished(result)
  278. def dict_keys(a, **remainder):
  279. keys, result = yield [("RDK", [a]), ("CN", [])]
  280. yield [("CE", [result, v]) for v in keys]
  281. raise PrimitiveFinished(result)
  282. def dict_reverse(a, b, **remainder):
  283. edges, = yield [("RO", [a])]
  284. expanded_edges, = yield [("RE", [i]) for i in edges]
  285. for i, edge in enumerate(expanded_edges):
  286. if b == edge[1]:
  287. # Found our edge: edges[i]
  288. outgoing, = yield [("RO", [edges[i]])]
  289. result, = yield [("RE", [outgoing[0]])]
  290. raise PrimitiveFinished(result[1])
  291. result, = yield [("CNV", ["(unknown: %s)" % b])]
  292. raise PrimitiveFinished(result)
  293. def is_physical_int(a, **remainder):
  294. t, = yield [("RV", [a])]
  295. result, = yield [("CNV", [isinstance(t, int) or isinstance(t, long)])]
  296. raise PrimitiveFinished(result)
  297. def is_physical_string(a, **remainder):
  298. t, = yield [("RV", [a])]
  299. result, = yield [("CNV", [isinstance(t, str) or isinstance(t, unicode)])]
  300. raise PrimitiveFinished(result)
  301. def is_physical_float(a, **remainder):
  302. t, = yield [("RV", [a])]
  303. result, = yield [("CNV", [isinstance(t, float)])]
  304. raise PrimitiveFinished(result)
  305. def is_physical_boolean(a, **remainder):
  306. t, = yield [("RV", [a])]
  307. result, = yield [("CNV", [isinstance(t, bool)])]
  308. raise PrimitiveFinished(result)
  309. def is_physical_action(a, **remainder):
  310. t, = yield [("RV", [a])]
  311. result, = yield [("CNV", [isinstance(t, dict) and t["value"] in ["if", "while", "assign", "call", "break", "continue", "return", "resolve", "access", "constant", "global", "declare"]])]
  312. raise PrimitiveFinished(result)
  313. def create_node(**remainder):
  314. result, = yield [("CN", [])]
  315. raise PrimitiveFinished(result)
  316. def create_edge(a, b, **remainder):
  317. result, = yield [("CE", [a, b])]
  318. raise PrimitiveFinished(result)
  319. def create_value(a, **remainder):
  320. a_value, = yield [("RV", [a])]
  321. result, = yield [("CNV", [a_value])]
  322. raise PrimitiveFinished(result)
  323. def read_nr_out(a, **remainder):
  324. outgoing, = yield [("RO", [a])]
  325. result, = yield [("CNV", [len(outgoing)])]
  326. raise PrimitiveFinished(result)
  327. def read_out(a, b, **remainder):
  328. outgoing, b_value = yield [("RO", [a]), ("RV", [b])]
  329. raise PrimitiveFinished(sorted(outgoing)[b_value])
  330. def read_nr_in(a, **remainder):
  331. incoming, = yield [("RI", [a])]
  332. result, = yield [("CNV", [len(incoming)])]
  333. raise PrimitiveFinished(result)
  334. def read_in(a, b, **remainder):
  335. incoming, b_value = yield [("RI", [a]), ("RV", [b])]
  336. raise PrimitiveFinished(sorted(incoming)[b_value])
  337. def read_edge_src(a, **remainder):
  338. result, = yield [("RE", [a])]
  339. raise PrimitiveFinished(result[0])
  340. def read_edge_dst(a, **remainder):
  341. result, = yield [("RE", [a])]
  342. raise PrimitiveFinished(result[1])
  343. def delete_element(a, **remainder):
  344. edge, = yield [("RE", [a])]
  345. if edge[0] is None:
  346. # Not an edge:
  347. yield [("DN", [a])]
  348. result, = yield [("CNV", [False])]
  349. raise PrimitiveFinished(result)
  350. else:
  351. yield [("DE", [a])]
  352. result, = yield [("CNV", [True])]
  353. raise PrimitiveFinished(result)
  354. def read_root(root, **remainder):
  355. raise PrimitiveFinished(root)
  356. def set_add(a, b, **remainder):
  357. yield [("CE", [a, b])]
  358. raise PrimitiveFinished(a)
  359. def set_pop(a, **remainder):
  360. outgoing, = yield [("RO", [a])]
  361. v, _ = yield [("RE", [outgoing[0]]), ("DE", [outgoing[0]])]
  362. raise PrimitiveFinished(v[1])
  363. def set_remove(a, b, **remainder):
  364. outgoing, b_value = yield [("RO", [a]), ("RV", [b])]
  365. elements = yield [("RE", [i]) for i in outgoing]
  366. values = yield [("RV", [i[1]]) for i in elements]
  367. yield [("DE", [identifier]) for identifier, edge in zip(outgoing, values) if edge == b_value]
  368. raise PrimitiveFinished(a)
  369. def set_remove_node(a, b, **remainder):
  370. outgoing, = yield [("RO", [a])]
  371. elements = yield [("RE", [i]) for i in outgoing]
  372. elements = [elements] if not isinstance(elements[0], list) else elements
  373. yield [("DE", [identifier]) for identifier, edge in zip(outgoing, elements) if edge[1] == b]
  374. raise PrimitiveFinished(a)
  375. def set_in(a, b, **remainder):
  376. outgoing, b_value = yield [("RO", [a]), ("RV", [b])]
  377. if outgoing:
  378. elements = yield [("RE", [i]) for i in outgoing]
  379. values = yield [("RV", [i[1]]) for i in elements]
  380. if b_value in [v for v in values]:
  381. result, = yield [("CNV", [True])]
  382. else:
  383. result, = yield [("CNV", [False])]
  384. else:
  385. result, = yield [("CNV", [False])]
  386. raise PrimitiveFinished(result)
  387. def set_in_node(a, b, **remainder):
  388. outgoing, = yield [("RO", [a])]
  389. if outgoing:
  390. elements = yield [("RE", [i]) for i in outgoing]
  391. if b in [v[1] for v in elements]:
  392. result, = yield [("CNV", [True])]
  393. else:
  394. result, = yield [("CNV", [False])]
  395. else:
  396. result, = yield [("CNV", [False])]
  397. raise PrimitiveFinished(result)
  398. def is_edge(a, **remainder):
  399. edge, = yield [("RE", [a])]
  400. result, = yield [("CNV", [edge[0] is not None])]
  401. raise PrimitiveFinished(result)
  402. #TODO deprecate
  403. def deserialize(a, root, **remainder):
  404. value, = yield [("RV", [a])]
  405. id_mappings = {}
  406. complex_primitives = frozenset(["if", "while", "assign", "call", "break", "continue", "return","resolve","access", "constant", "input", "output", "declare", "global"])
  407. for l in value.split("\n"):
  408. try:
  409. graph_type, constructor = l.split(None, 1)
  410. except:
  411. continue
  412. if graph_type == "N":
  413. # Node
  414. id_mappings[constructor], = yield [("CN", [])]
  415. elif graph_type == "V":
  416. # Node with Value
  417. name, temp = constructor.split("(", 1)
  418. string_value = temp[:-1]
  419. if string_value in complex_primitives:
  420. value = {"value": string_value}
  421. else:
  422. #TODO this is very dangerous!
  423. value = eval(string_value)
  424. id_mappings[name], = yield [("CNV", [value])]
  425. elif graph_type == "E":
  426. # Edge
  427. name, temp = constructor.split("(", 1)
  428. source, target = temp[:-1].split(",", 1)
  429. if target[0] == "?" and target not in id_mappings:
  430. hierarchy = target[1:].split("/")
  431. current, = yield [("RD", [root, "__hierarchy"])]
  432. for i in hierarchy:
  433. current, = yield [("RD", [current, i])]
  434. id_mappings[target] = current
  435. try:
  436. source = int(source)
  437. except:
  438. source = id_mappings[source]
  439. try:
  440. target = int(target)
  441. except:
  442. target = id_mappings[target]
  443. # Both nodes will normally be already present in the matching dictionary, so can just pass them along
  444. id_mappings[name], = yield [("CE", [source, target])]
  445. elif graph_type == "D":
  446. source, value, target = constructor.split(",",3)
  447. if value in complex_primitives:
  448. value = {"value": value}
  449. else:
  450. #TODO this is very dangerous!
  451. value = eval(value)
  452. if target[0] == "?" and target not in id_mappings:
  453. hierarchy = target[1:].split("/")
  454. current, = yield [("RD", [root, "__hierarchy"])]
  455. for i in hierarchy:
  456. current, = yield [("RD", [current, i])]
  457. id_mappings[target] = current
  458. try:
  459. source = int(source)
  460. except:
  461. source = id_mappings[source]
  462. try:
  463. target = int(target)
  464. except:
  465. target = id_mappings[target]
  466. yield [("CD", [source, value, target])]
  467. else:
  468. print("Unknown graph type: " + str(graph_type))
  469. raise PrimitiveFinished(id_mappings["auto_initial_IP"])
  470. def log(a, **remainder):
  471. a_value, = yield [("RV", [a])]
  472. print("== LOG == " + str(a_value))
  473. raise PrimitiveFinished(a)
  474. def read_taskroot(task_root, **remainder):
  475. raise PrimitiveFinished(task_root)
  476. def time(**remainder):
  477. import time
  478. a, = yield [("CNV", [time.time()])]
  479. raise PrimitiveFinished(a)
  480. def hash(a, **remainder):
  481. a_value, = yield [("RV", [a])]
  482. import hashlib
  483. b_value = hashlib.sha512(a_value).hexdigest()
  484. b, = yield [("CNV", [b_value])]
  485. raise PrimitiveFinished(b)