-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathprogramGraph.py
60 lines (49 loc) · 1.93 KB
/
programGraph.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
from API import *
from utilities import *
class ProgramGraph:
"""A program graph is a state in the search space"""
def __init__(self, nodes):
self.nodes = nodes if isinstance(nodes, tuple) else tuple(nodes)
@staticmethod
def fromRoot(r, oneParent=False):
if not oneParent:
ns = set()
def reachable(n):
if n in ns: return
ns.add(n)
for c in n.children():
reachable(c)
reachable(r)
return ProgramGraph(ns)
else:
ns = []
def visit(n):
ns.append(n)
for c in n.children(): visit(c)
visit(r)
return ProgramGraph(ns)
def __len__(self): return len(self.nodes)
def prettyPrint(self):
variableOfNode = [None for _ in self.nodes]
nameOfNode = [None for _ in self.nodes] # pp of node
lines = []
def getIndex(p):
for i, pp in enumerate(self.nodes):
if p is pp: return i
assert False
def pp(j):
if variableOfNode[j] is not None: return variableOfNode[j]
serialization = [t if not isinstance(t,Program) else pp(getIndex(t))
for t in self.nodes[j].serialize()]
expression = f"({' '.join(map(str, serialization))})"
variableOfNode[j] = f"${len(lines)}"
lines.append(f"{variableOfNode[j]} <- {expression}")
return variableOfNode[j]
for j in range(len(self.nodes)):
pp(j)
return "\n".join(lines)
def extend(self, newNode):
return ProgramGraph([newNode] + list(self.nodes))
def objects(self, oneParent=False):
return [o for o in self.nodes
if not oneParent or not any( any( c is o for c in op.children() ) for op in self.nodes )]