main.py 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460
  1. from modelverse_state import status
  2. import sys
  3. from collections import defaultdict
  4. import os
  5. import gzip
  6. import time
  7. import cPickle as pickle
  8. # Work around Python 2 where a 'big integer' automatically becomes a long
  9. if sys.version > '3': # pragma: no cover
  10. integer_types = (int,)
  11. primitive_types = (int, float, str, bool)
  12. else: # pragma: no cover
  13. integer_types = (int, long)
  14. primitive_types = (int, long, float, str, bool, unicode)
  15. complex_primitives = frozenset(["if", "while", "assign", "call", "break", "continue", "return","resolve","access", "constant", "input", "output", "declare", "global"])
  16. def instance_to_string(value):
  17. return value["value"]
  18. def string_to_instance(value):
  19. return {'value': value}
  20. class ModelverseState(object):
  21. def __init__(self, bootfile = None):
  22. self.free_id = 0
  23. self.edges = {}
  24. self.outgoing = {}
  25. self.incoming = {}
  26. self.values = {}
  27. self.nodes = set()
  28. self.GC = True
  29. self.to_delete = set()
  30. self.cache = {}
  31. if bootfile is not None:
  32. self.root = self.parse(bootfile)
  33. else:
  34. self.root, _ = self.create_node()
  35. def dump_modelverse(self):
  36. with open("/tmp/modelverse.out", "w") as f:
  37. f.write("digraph main {\n")
  38. for n in self.nodes:
  39. if n in self.values:
  40. f.write("a_%s [label=\"a_%s (%s)\"];\n" % (n, n, self.values[n]))
  41. else:
  42. f.write("a_%s [label=\"a_%s\"];\n" % (n, n))
  43. for i, e in self.edges.iteritems():
  44. f.write("%s -> %s [label=\"%s\"];\n" % (e[0], e[1], i))
  45. f.write("}")
  46. return (self.root, status.SUCCESS)
  47. def parse(self, filename):
  48. picklefile = filename + ".pickle"
  49. try:
  50. if os.path.getmtime(picklefile) > os.path.getmtime(filename):
  51. # Pickle is more recent than grammarfile, so we can use it
  52. self.root, self.free_id, self.nodes, self.edges, self.values = pickle.load(open(picklefile, 'rb'))
  53. for name in self.edges:
  54. source, destination = self.edges[name]
  55. self.outgoing.setdefault(source, set()).add(name)
  56. self.incoming.setdefault(destination, set()).add(name)
  57. return self.root
  58. else:
  59. raise Exception("Invalid pickle")
  60. except Exception as e:
  61. # We have to parse the file and create the pickle
  62. symbols = {}
  63. def resolve(symb):
  64. try:
  65. return int(symb)
  66. except:
  67. if symb[0] == "?":
  68. derefs = symb[1:].split("/")
  69. v, _ = self.read_dict(symbols["root"], "__hierarchy")
  70. for deref in derefs:
  71. v, _ = self.read_dict(v, deref)
  72. return v
  73. else:
  74. return symbols[symb]
  75. with gzip.open(filename, 'rb') as f:
  76. for line in f:
  77. element_type, constructor = line.split(None, 1)
  78. name, values = constructor.split("(", 1)
  79. values, _ = values.rsplit(")", 1)
  80. if element_type == "Node":
  81. name = name.split()[0]
  82. if values == "":
  83. symbols[name], status = self.create_node()
  84. else:
  85. value = values
  86. if value in complex_primitives:
  87. value = string_to_instance(value)
  88. else:
  89. value = eval(value)
  90. symbols[name], status = self.create_nodevalue(value)
  91. elif element_type == "Edge":
  92. name = name.split()[0]
  93. values = [v.split()[0] for v in values.split(",")]
  94. symbols[name], status = self.create_edge(resolve(values[0]), resolve(values[1]))
  95. elif element_type == "Dict":
  96. values = [v.split()[0] for v in values.split(",")]
  97. if values[1] in complex_primitives:
  98. values[1] = string_to_instance(values[1])
  99. else:
  100. values[1] = eval(values[1])
  101. self.create_dict(resolve(values[0]), values[1], resolve(values[2]))
  102. else:
  103. raise Exception("Unknown element type: %s" % element_type)
  104. if status != 100:
  105. raise Exception("Failed to process line for reason %s: %s" % (status, line))
  106. # Creation successful, now also create a pickle
  107. with open(picklefile, 'wb') as f:
  108. pickle.dump((symbols["root"], self.free_id, self.nodes, self.edges, self.values), f, pickle.HIGHEST_PROTOCOL)
  109. return symbols["root"]
  110. def read_root(self):
  111. return (self.root, status.SUCCESS)
  112. def create_node(self):
  113. self.nodes.add(self.free_id)
  114. self.free_id += 1
  115. return (self.free_id - 1, status.SUCCESS)
  116. def create_edge(self, source, target):
  117. if source not in self.edges and source not in self.nodes:
  118. return (None, status.FAIL_CE_SOURCE)
  119. elif target not in self.edges and target not in self.nodes:
  120. return (None, status.FAIL_CE_TARGET)
  121. else:
  122. self.outgoing.setdefault(source, set()).add(self.free_id)
  123. self.incoming.setdefault(target, set()).add(self.free_id)
  124. self.edges[self.free_id] = (source, target)
  125. self.free_id += 1
  126. return (self.free_id - 1, status.SUCCESS)
  127. def is_valid_datavalue(self, value):
  128. if isinstance(value, dict):
  129. if "value" in value and value["value"] in complex_primitives:
  130. return True
  131. else:
  132. return False
  133. elif not isinstance(value, primitive_types):
  134. return False
  135. elif isinstance(value, integer_types) and not (-2**63 <= value <= 2**64 - 1):
  136. return False
  137. return True
  138. def create_nodevalue(self, value):
  139. if not self.is_valid_datavalue(value):
  140. return (None, status.FAIL_CNV_OOB)
  141. self.values[self.free_id] = value
  142. self.nodes.add(self.free_id)
  143. self.free_id += 1
  144. return (self.free_id - 1, status.SUCCESS)
  145. def create_dict(self, source, data, destination):
  146. if source not in self.nodes and source not in self.edges:
  147. return (None, status.FAIL_CDICT_SOURCE)
  148. if destination not in self.nodes and destination not in self.edges:
  149. return (None, status.FAIL_CDICT_TARGET)
  150. if not self.is_valid_datavalue(data):
  151. return (None, status.FAIL_CDICT_OOB)
  152. n = self.create_nodevalue(data)[0]
  153. e = self.create_edge(source, destination)[0]
  154. self.create_edge(e, n)
  155. self.cache.setdefault(source, {})[data] = e
  156. return (None, status.SUCCESS)
  157. def read_value(self, node):
  158. if node in self.values:
  159. return (self.values[node], status.SUCCESS)
  160. elif node not in self.nodes:
  161. return (None, status.FAIL_RV_UNKNOWN)
  162. else:
  163. return (None, status.FAIL_RV_NO_VALUE)
  164. def read_outgoing(self, elem):
  165. if elem in self.edges or elem in self.nodes:
  166. if elem in self.outgoing:
  167. return (list(self.outgoing[elem]), status.SUCCESS)
  168. else:
  169. return ([], status.SUCCESS)
  170. else:
  171. return (None, status.FAIL_RO_UNKNOWN)
  172. def read_incoming(self, elem):
  173. if elem in self.edges or elem in self.nodes:
  174. if elem in self.incoming:
  175. return (list(self.incoming[elem]), status.SUCCESS)
  176. else:
  177. return ([], status.SUCCESS)
  178. else:
  179. return (None, status.FAIL_RI_UNKNOWN)
  180. def read_edge(self, edge):
  181. if edge in self.edges:
  182. return (list(self.edges[edge]), status.SUCCESS)
  183. else:
  184. return ([None, None], status.FAIL_RE_UNKNOWN)
  185. def read_dict(self, node, value):
  186. try:
  187. first = self.cache[node][value]
  188. # Got hit, so validate
  189. if (self.edges[first][0] == node) and \
  190. (len(self.outgoing[first]) == 1) and \
  191. (self.values[self.edges[list(self.outgoing[first])[0]][1]] == value):
  192. return (self.edges[first][1], status.SUCCESS)
  193. # Hit but invalid now
  194. del self.cache[node][value]
  195. except KeyError:
  196. # Didn't exist
  197. pass
  198. # Get all outgoing links
  199. if node in self.outgoing:
  200. for e1 in self.outgoing[node]:
  201. # For each link, we read the links that might link to a data value
  202. if e1 in self.outgoing:
  203. for e2 in self.outgoing[e1]:
  204. # Now read out the target of the link
  205. target = self.edges[e2][1]
  206. # And access its value
  207. if target in self.values and self.values[target] == value:
  208. # Found a match
  209. # Now get the target of the original link
  210. self.cache.setdefault(node, {})[value] = e1
  211. return (self.edges[e1][1], status.SUCCESS)
  212. if node not in self.nodes and node not in self.edges:
  213. return (None, status.FAIL_RDICT_UNKNOWN)
  214. elif not self.is_valid_datavalue(value):
  215. return (None, status.FAIL_RDICT_OOB)
  216. else:
  217. return (None, status.FAIL_RDICT_NOT_FOUND)
  218. def read_dict_keys(self, node):
  219. if node not in self.nodes and node not in self.edges:
  220. return (None, status.FAIL_RDICTKEYS_UNKNOWN)
  221. result = []
  222. if node in self.outgoing:
  223. for e1 in self.outgoing[node]:
  224. if e1 in self.outgoing:
  225. for e2 in self.outgoing[e1]:
  226. result.append(self.edges[e2][1])
  227. return (result, status.SUCCESS)
  228. def read_dict_edge(self, node, value):
  229. try:
  230. first = self.cache[node][value]
  231. # Got hit, so validate
  232. if (self.edges[first][0] == node) and \
  233. (len(self.outgoing[first]) == 1) and \
  234. (self.values[self.edges[list(self.outgoing[first])[0]][1]] == value):
  235. return (first, status.SUCCESS)
  236. # Hit but invalid now
  237. del self.cache[node][value]
  238. except KeyError:
  239. # Didn't exist
  240. pass
  241. # Get all outgoing links
  242. found = None
  243. if node in self.outgoing:
  244. for e1 in self.outgoing[node]:
  245. # For each link, we read the links that might link to a data value
  246. if e1 in self.outgoing:
  247. for e2 in self.outgoing[e1]:
  248. # Now read out the target of the link
  249. target = self.edges[e2][1]
  250. # And access its value
  251. if target in self.values and self.values[target] == value:
  252. # Found a match
  253. # Now get the target of the original link
  254. if found is not None:
  255. print("Duplicate key on value: %s : %s (%s <-> %s)!" % (self.values[target], type(v), found, e1))
  256. return (None, status.FAIL_RDICTE_AMBIGUOUS)
  257. found = e1
  258. self.cache.setdefault(node, {})[value] = e1
  259. if found is not None:
  260. return (found, status.SUCCESS)
  261. elif node not in self.nodes and node not in self.edges:
  262. return (None, status.FAIL_RDICTE_UNKNOWN)
  263. elif not self.is_valid_datavalue(value):
  264. return (None, status.FAIL_RDICTE_OOB)
  265. else:
  266. return (None, status.FAIL_RDICTE_NOT_FOUND)
  267. def read_dict_node(self, node, value_node):
  268. e, s = self.read_dict_node_edge(node, value_node)
  269. if s != status.SUCCESS:
  270. return (None, {status.FAIL_RDICTNE_UNKNOWN: status.FAIL_RDICTN_UNKNOWN,
  271. status.FAIL_RDICTNE_UNCERTAIN: status.FAIL_RDICTN_UNCERTAIN,
  272. status.FAIL_RDICTNE_AMBIGUOUS: status.FAIL_RDICTN_AMBIGUOUS,
  273. status.FAIL_RDICTNE_OOB: status.FAIL_RDICTN_OOB,
  274. status.FAIL_RDICTNE_NOT_FOUND: status.FAIL_RDICTN_NOT_FOUND}[s])
  275. return (self.edges[e][1], status.SUCCESS)
  276. def read_dict_node_edge(self, node, value_node):
  277. if node not in self.nodes and node not in self.edges:
  278. return (None, status.FAIL_RDICTNE_UNKNOWN)
  279. # Get all outgoing links
  280. found = None
  281. if node in self.outgoing:
  282. for e1 in self.outgoing[node]:
  283. # For each link, we read the links that might link to a data value
  284. if e1 in self.outgoing:
  285. for e2 in self.outgoing[e1]:
  286. # And access its value
  287. if self.edges[e2][1] == value_node:
  288. # Found a match
  289. # Now get the target of the original link
  290. if found is not None:
  291. print("Duplicate key on node: %s (%s <-> %s)!" % (value_node, found, e1))
  292. return (None, status.FAIL_RDICTNE_AMBIGUOUS)
  293. found = e1
  294. if found is not None:
  295. return (found, status.SUCCESS)
  296. else:
  297. return (None, status.FAIL_RDICTNE_NOT_FOUND)
  298. def read_reverse_dict(self, node, value):
  299. if node not in self.nodes and node not in self.edges:
  300. return (None, status.FAIL_RRDICT_UNKNOWN)
  301. elif not self.is_valid_datavalue(value):
  302. return (None, status.FAIL_RRDICT_OOB)
  303. # Get all outgoing links
  304. matches = []
  305. if node in self.incoming:
  306. for e1 in self.incoming[node]:
  307. # For each link, we read the links that might link to a data value
  308. if e1 in self.outgoing:
  309. for e2 in self.outgoing[e1]:
  310. # Now read out the target of the link
  311. target = self.edges[e2][1]
  312. # And access its value
  313. if target in self.values and self.values[target] == value:
  314. # Found a match
  315. if len(self.outgoing[e1]) > 1:
  316. return (None, status.FAIL_RRDICT_UNCERTAIN)
  317. else:
  318. matches.append(e1)
  319. if len(matches) == 0:
  320. return (None, status.FAIL_RRDICT_NOT_FOUND)
  321. else:
  322. return ([self.edges[e][0] for e in matches], status.SUCCESS)
  323. def delete_node(self, node):
  324. if node == self.root:
  325. return (None, status.FAIL_DN_UNKNOWN)
  326. if node not in self.nodes:
  327. return (None, status.FAIL_DN_UNKNOWN)
  328. self.nodes.remove(node)
  329. if node in self.cache:
  330. del self.cache[node]
  331. if node in self.values:
  332. del self.values[node]
  333. s = set()
  334. if node in self.outgoing:
  335. for e in self.outgoing[node]:
  336. s.add(e)
  337. del self.outgoing[node]
  338. if node in self.incoming:
  339. for e in self.incoming[node]:
  340. s.add(e)
  341. del self.incoming[node]
  342. for e in s:
  343. self.delete_edge(e)
  344. if node in self.outgoing:
  345. del self.outgoing[node]
  346. if node in self.incoming:
  347. del self.incoming[node]
  348. return (None, status.SUCCESS)
  349. def delete_edge(self, edge):
  350. if edge not in self.edges:
  351. return (None, status.FAIL_DE_UNKNOWN)
  352. s, t = self.edges[edge]
  353. if t in self.incoming:
  354. self.incoming[t].remove(edge)
  355. if s in self.outgoing:
  356. self.outgoing[s].remove(edge)
  357. del self.edges[edge]
  358. s = set()
  359. if edge in self.outgoing:
  360. for e in self.outgoing[edge]:
  361. s.add(e)
  362. if edge in self.incoming:
  363. for e in self.incoming[edge]:
  364. s.add(e)
  365. for e in s:
  366. self.delete_edge(e)
  367. if edge in self.outgoing:
  368. del self.outgoing[edge]
  369. if edge in self.incoming:
  370. del self.incoming[edge]
  371. if (self.GC) and (t in self.incoming and not self.incoming[t]) and (t not in self.edges):
  372. # Remove this node as well
  373. # Edges aren't deleted like this, as they might have a reachable target and source!
  374. # If they haven't, they will be removed because the source was removed.
  375. self.to_delete.add(t)
  376. return (None, status.SUCCESS)
  377. def garbage_collect(self):
  378. while self.to_delete:
  379. t = self.to_delete.pop()
  380. if t in self.incoming and not self.incoming[t]:
  381. self.delete_node(t)
  382. def purge(self):
  383. self.garbage_collect()
  384. values = set(self.edges)
  385. values.update(self.nodes)
  386. visit_list = [self.root]
  387. while visit_list:
  388. elem = visit_list.pop()
  389. if elem in values:
  390. # Remove it from the leftover values
  391. values.remove(elem)
  392. if elem in self.edges:
  393. visit_list.extend(self.edges[elem])
  394. if elem in self.outgoing:
  395. visit_list.extend(self.outgoing[elem])
  396. if elem in self.incoming:
  397. visit_list.extend(self.incoming[elem])
  398. # All remaining elements are to be purged
  399. if len(values) > 0:
  400. while values:
  401. v = values.pop()
  402. if v in self.nodes:
  403. self.delete_node(v)