-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathstructures.py
123 lines (99 loc) · 3.86 KB
/
structures.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
from typing import List, Tuple, Dict
import networkx as nx
from matplotlib import pyplot as plt
# structure representing graph
class Graph:
# just making type hints
def __init__(self, v_labels: Dict[int, str], adjacency_list: Dict[int, List[Tuple[int, str]]]):
self.v_labels = v_labels
self.adjacency_list = adjacency_list
if not self.is_correct():
raise TypeError("graph not implemented correctly")
# checking if adjacency list is correct and coherent with vertex labels
def is_correct(self):
# check if indexes (keys) are matching
if set(self.v_labels.keys()) != set(self.adjacency_list.keys()):
return False
# iterate over keys
for v1 in self.adjacency_list:
# iterate over elements
for v2, label in self.adjacency_list[v1]:
if not (v1, label) in self.adjacency_list[v2]:
return False
return True
def visualize(self):
plt.figure()
edges = {(v1, v2): label for v1 in self.adjacency_list for v2, label in self.adjacency_list[v1]}
G = nx.Graph()
G.add_nodes_from(index for index in self.v_labels)
G.add_edges_from(edges.keys())
# feel free to change
options = {
"node_size": 500,
"node_color": "white",
"edgecolors": "black",
"linewidths": 0.5,
"width": 0.5,
"verticalalignment": "top"
}
pos = nx.spring_layout(G)
nx.draw_networkx(G, pos, **options)
nx.draw_networkx_labels(G, pos, {v: self.v_labels[v] for v in self.adjacency_list}, font_size=10,
verticalalignment="baseline")
nx.draw_networkx_edge_labels(G, pos, edges)
return plt.gcf()
def remove(self, i):
self.v_labels.pop(i)
self.adjacency_list.pop(i)
for v in self.adjacency_list:
newV = []
for a in self.adjacency_list[v]:
if a[0] != i:
newV.append(a)
self.adjacency_list[v] = newV
def remove_edge(self, edges, givenNodes, left_dict):
for vList in self.adjacency_list:
for edge in edges:
if (givenNodes[left_dict.index(edge[0])], edge[1]) in self.adjacency_list[vList]:
self.adjacency_list[vList].remove((givenNodes[left_dict.index(edge[0])], edge[1]))
def add_node(self, label, edges, GR, i, A):
newIndex = max(self.adjacency_list.keys()) + 1
self.v_labels[newIndex] = label
edge_list = []
for edge in edges:
if edge[0] in GR:
edge_list.append((A[GR.index(edge[0])], edge[1]))
self.adjacency_list[A[GR.index(edge[0])]].append((newIndex, edge[1]))
self.adjacency_list[newIndex] = edge_list
GR.append(i)
A.append(newIndex)
return GR
def add_edge(self, edge, node):
self.adjacency_list[node].append(edge)
''' feel free to implement your class methods'''
class Production:
# just making type hints
def __init__(self, left: Graph, connector: Graph, right: Graph):
self.left = left
self.connector = connector
self.right = right
''' feel free to implement your class methods'''
class Grammar:
# just making type hints
def __init__(self, input_graph: Graph, production_list: List[Production]):
self.input_graph = input_graph
self.production_list = production_list
# to be able to go back once
self.previous_graph = None
# example - make sure arguments are correct - otherwise exception is thrown
if __name__ == '__main__':
G = Graph(
{5: "A", 1: "B", 2: "C"},
{
5: [(1, "a"), (2, "b")],
1: [(5, "a"), (2, "c")],
2: [(5, "b"), (1, "c")]
}
)
G.visualize()
plt.show()