-
Notifications
You must be signed in to change notification settings - Fork 0
/
expression_parser.py
214 lines (179 loc) · 5.78 KB
/
expression_parser.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
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
# coding: utf-8
class treeNode(object):
def __init__(self):
self.tipo = None # Operador (Op) ou Propriedade (Pr)
self.conteudo = None
self.oper = None
self.rotulo = 0 #precisa??
self.left = None
self.right = None
class parserCTL():
def __init__(self):
#self.leftExp = None
#self.rightExp = None
self.cont = 0 # Controle de abertura e fechamento de parenteses
self.listaNos = []
self.expressao = None
def parse(self, exp):
pos = self.identificaExpressao(exp)
no = treeNode()
# Cria nó do tipo Propriedade e
# interrompe a recursão
if (pos == -1):
no.tipo = 'Pr'
no.conteudo = exp[1:(int(len(exp))-1)]
no.oper = None
no.left = None
no.right = None
self.listaNos.append(no)
self.expressao = exp
return no
# Le próximo operador
operador = ""
i = pos
while (exp[i] != '('):
operador += exp[i]
i += 1
# Le os operadores e analisa a expressao sobre a
# qual atuam. Operadores AX, EF, AG, EG, AU, ->
# e <-> são substituidos por seus equivalentes, de
# forma que estes operadores não estarão presentes
# na expressão final. O índice de posição passa a
# apontar para o restante da expressão.
if (operador == "AX"):
pos +=2
#self.leftExp = lePropriedade(exp, pos)
exp = "(!(EX(!" + self.lePropriedade(exp, pos) + ")))"
operador = "!" # Novo último operador
pos = 1 # Índice aponta para o restante da expressão
elif (operador == "EF"):
pos += 2
exp = "(EU((TRUE)," + self.lePropriedade(exp, pos) + "))"
operador = "EU"
pos = 1
elif (operador == "AG"):
pos += 2
exp = "(!(EU((TRUE),(!" + self.lePropriedade(exp, pos) + "))))"
operador = "!"
pos = 1
elif (operador == "EG"):
pos += 2
exp = "(!(AF(!" + self.lePropriedade(exp, pos) + ")))"
operador = "!"
pos = 1
# Há controle de posição do início da expressão e de
# onde ocorre a vírgula que separa as duas expressões
# resultantes. Nas funções AU, -> e <->, precisa-se
# armazenar as expressões à esquerda e à direita da
# vírgula/operador para realizar a substituição pela
# expressão equivalente.
elif (operador == "AU"):
pos += 2
temp = self.lePropriedade(exp, pos)
virgula = self.identificaExpressao(temp)
expEsquerda = self.lePropriedade(temp, 1)
virgula += 1
expDireita = self.lePropriedade(temp, virgula)
exp = "((AF" + expDireita + ")&(!(EU((!" + expDireita + "),((!" + expEsquerda + ")&(!" + expDireita + "))))))"
operador = "&"
pos = self.identificaExpressao(exp)
elif (operador == "->"):
pos += 2
expEsquerda = self.lePropriedade(exp, 1)
expDireita = self.lePropriedade(exp, pos)
exp = "((!" + expEsquerda + ")|" + expDireita + ")"
operador = "|"
pos = self.identificaExpressao(exp)
elif (operador == "<->"):
pos += 3
expEsquerda = self.lePropriedade(exp, 1)
expDireita = self.lePropriedade(exp, pos)
exp = "(((!" + expEsquerda + ")|" + expDireita + ")&((!" + expDireita + ")|" + expEsquerda + "))"
operador = "&"
pos = self.identificaExpressao(exp)
# Após feitas as devidas substituições por
# expressões equivalentes, analisa-se os opera-
# dores finais para encerrar o mapeamento da
# expressão lida em árvore sintática. Quando
# um nó da árvore possui apenas um filho,
# adota-se que este é seu filho esquerdo.
no = treeNode()
no.tipo = 'Op'
no.conteudo = exp
no.oper = operador
if(operador == "EX"):
pos += 2
expEsquerda = self.lePropriedade(exp, pos)
no.left = self.parse(expEsquerda)
elif(operador == "AF"):
pos += 2
expEsquerda = self.lePropriedade(exp, pos)
no.left = self.parse(expEsquerda)
elif(operador == "EU"):
pos += 2
temp = self.lePropriedade(exp, pos)
virgula = self.identificaExpressao(temp)
expEsquerda = self.lePropriedade(temp, 1)
virgula += 1
expDireita = self.lePropriedade(temp, virgula)
no.left = self.parse(expEsquerda)
no.right = self.parse(expDireita)
elif(operador == "&"):
expEsquerda = self.lePropriedade(exp, 1)
pos += 1
expDireita = self.lePropriedade(exp, pos)
no.left = self.parse(expEsquerda)
no.right = self.parse(expDireita)
elif(operador == "|"):
expEsquerda = self.lePropriedade(exp, 1)
pos += 1
expDireita = self.lePropriedade(exp, pos)
no.left = self.parse(expEsquerda)
no.right = self.parse(expDireita)
elif(operador == "!"):
pos += 1
expEsquerda = self.lePropriedade(exp, pos)
no.left = self.parse(expEsquerda)
else:
print("Operador Invalido.")
exit(1)
self.listaNos.append(no)
self.expressao = exp
return no
# Retorna posição do próximo operador da expressão. Caso
# essa expresão seja uma propriedade (nó folha), retorna -1
def identificaExpressao(self, exp):
cont = 0
isProp = False
for i in range(0, int(len(exp))):
# Leitura de operadores básicos
if (exp[i] == '('):
cont += 1
elif (exp[i] == ')'):
cont -= 1
# Se encontrou algum parâmetro, verifica se
# é um operador (próximo parenteses abre) ou
# propriedade (próximo parenteses fecha)
elif (cont == 1):
j = i
while(exp[j] != '('):
if (exp[j] == ')'):
isProp = True
break
j += 1
if not isProp:
return i
return -1
# Encontra propriedade a partir de uma posição
def lePropriedade(self, exp, pos):
propriedade = "("
contador = 1
pos += 1
while(contador != 0):
propriedade += exp[pos]
if(exp[pos] == '('):
contador += 1
elif(exp[pos] == ')'):
contador -= 1
pos += 1
return propriedade