This repository has been archived by the owner on Dec 23, 2019. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathabstract_graph.py
159 lines (141 loc) · 4.34 KB
/
abstract_graph.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
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
class Abstract_Vertex(object):
def __init__(self, vid):
self.inn = []
self.out = []
self.vid = vid
def __hash__(self):
return self.vid
class Abstract_Edge(object):
def __init__(self, v1, v2, eid):
self.v1, self.v2 = v1, v2
v1.out.append(self)
v2.inn.append(self)
self.eid = eid
def __hash__(self):
return self.eid
def length(self):
pass
class Abstract_Graph(object):
def __init__(self):
self.vs = {} # key -> Vertex
self.es = {} # key -> Edges
def get_edge(self, eid):
return self.es.get(eid, None)
def find_all_loops(self, edge, threshold, L):
NO_LOOPS = None
v1 = edge.v2
lst = [(v1,0)]
long_end = None
visited_vs = set()
visited_es = set()
while len(lst) != 0:
(v, deep) = lst.pop(0)
if deep > threshold:
continue
for e1 in v.out:
if e1.length() > L:
if long_end and long_end.eid != e1.eid:
return NO_LOOPS
long_end = e1
else:
visited_vs.add(v)
lst.append((e1.v2, deep + 1))
if not long_end:
return NO_LOOPS
visited_es.add(long_end.eid)
visited_es.add(edge.eid)
good_vertex = [edge.v1.vid, edge.v2.vid, long_end.v1.vid, long_end.v2.vid]
for v in visited_vs:
for e in v.out:
visited_es.add(e.eid)
for e in v.inn:
visited_es.add(e.eid)
for v2 in [e.v2 for e in v.out]:
if v2.vid not in good_vertex and v2 not in visited_vs:
return NO_LOOPS
for v_begin in [e.v1 for e in v.inn]:
if v_begin.vid not in good_vertex and v_begin not in visited_vs:
return NO_LOOPS
if len(visited_es) == 2:
return NO_LOOPS
for v in visited_vs:
if not self.is_connected(v, long_end, threshold):
return NO_LOOPS
paths = self.get_paths(edge.v1, long_end, 2*(threshold + 1), False)
best_path = paths[0]
best_len = self.path_len(best_path)
for path in paths:
if self.path_len(path) > best_len:
if path[0] != edge:
continue
long_edges = set()
false_path = False
for e in path:
if e not in visited_vs:
false_path = True
break
if e.length() > 600 and e.eid in long_edges:
false_path = True
break
if e.length() > 600:
long_edges.add(e.eid)
if false_path:
continue
best_path = self.path_len(path)
best_path = path
return (edge.eid, long_end.eid, visited_es, best_path, visited_vs)
def path_len(self, path):
path_len = 0
for e in path:
path_len += e.length()
return path_len
def is_connected(self, vertex, edge, threshold):
return len(self.get_paths(vertex, edge, threshold, True)) > 0
def get_paths(self, vertex, edge, threshold, one_path = False):
paths = []
lst = [(vertex,0, [])]
while len(lst) != 0:
(v, deep, path) = lst.pop(0)
if deep > threshold:
continue
for e in v.out:
if e.eid == edge.eid:
paths.append(path)
if one_path:
return paths
new_path = list(path)
new_path.append(e)
lst.append((e.v2, deep + 1, new_path))
return paths
def number_of_pathes(v1, v2, limit):
limit += 1 # limit inclusive
ls = [{} for _ in xrange(limit)]
ls[0][v1] = 1
for pos in xrange(limit):
for v, cnt in ls[pos].iteritems:
if v == v2: return cnt
for e in v.out:
pos2 = pos + e.len
if pos2 < limit:
v2 = e.v2
ls[pos2][v2] = ls[pos2].get(v2, 0) + cnt
return 0
def find_paths(v1, v2, e1, len_threshold, deep_threshold):
return __find_all_paths(v1, v2, e1, [], 0, [], len_threshold, deep_threshold)
def __find_all_paths(v1, v2, e1, path, path_len, paths, len_threshold, deep_threshold):
for e in v1.out:
if (len(path) == 0) and e != e1:
continue
if e.len > len_threshold:
return paths
new_path = list(path)
new_path.append(e)
new_path_len = path_len + e.len
if e.v2 == v2:
paths.append((new_path, new_path_len))
return paths
else:
if (len(path) > deep_threshold):
return paths
__find_all_paths(e.v2, v2, e1, new_path, new_path_len, paths, len_threshold - e.len, deep_threshold)
return paths