-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathverification_algorithms.py
440 lines (335 loc) · 18.7 KB
/
verification_algorithms.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
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
import spot
from collections import deque, defaultdict
def smaller_than(p, p_prime):
"""
:param p: the first payoff.
:param p_prime: the second payoff.
:return: whether p <= p_prime.
"""
return all(p[i] <= p_prime[i] for i in range(len(p)))
def generate_smaller_payoffs(p):
"""
:param p: a payoff.
:return: the list of payoffs strictly smaller than p by one objective.
"""
payoffs = []
for i in range(len(p)):
if p[i] == 1:
payoffs.append(p[:i] + tuple([0]) + p[i + 1:])
return payoffs
def generate_larger_payoffs(p):
"""
:param p: a payoff.
:return: the list of payoffs strictly larger than p by one objective.
"""
payoffs = []
for i in range(len(p)):
if p[i] == 0:
payoffs.append(p[:i] + tuple([1]) + p[i + 1:])
return payoffs
def add_payoff_to_antichain(pareto_optimal_payoffs, dominating_payoff):
"""
Add a payoff to the antichain of PO payoffs. Removes all dominated payoffs. By construction, the newly added payoff
is strictly larger or incomparable to those of the antichain of PO payoffs and will therefore be in the new list of
PO payoffs.
:param pareto_optimal_payoffs: the list of PO payoffs.
:param dominating_payoff: the new payoff to add.
:return: the new list of PO payoffs.
"""
new_list = [dominating_payoff]
for payoff in pareto_optimal_payoffs:
# check if the payoff is smaller or equal to the dominated payoff (it cannot be equal as the dominating payoff
# is strictly larger or incomparable to every payoff in the set by construction). If it is not, the payoff is
# therefore strictly larger to or incomparable to dominating_payoff and is added to the set.
if not smaller_than(payoff, dominating_payoff):
new_list.append(payoff)
return new_list
def parity_to_acceptance(colors, opposite=False):
"""
Create an acceptance condition corresponding to a parity objective.
:param colors: the SPOT acceptance sets used for the priority function of the parity objective.
:param opposite: whether to create the condition for the opposite objective (satisfied iff the original is not).
:return: the SPOT acceptance condition in string format.
"""
# the number n used in the "parity min even n" is the number of colors used from 0 to n-1
# the << shifts the colors to start from min_color
min_color = min(colors)
if not opposite:
return spot.acc_code("parity min even " + str(len(colors))) << min_color
else:
return spot.acc_code("parity min odd " + str(len(colors))) << min_color
def is_payoff_realizable(p, automaton, colors_map, losing_for_0=False):
"""
Whether the payoff is realizable. In practice, this function checks for the existence of a play with a payoff equal
or larger to p by trying to find a play with a payoff that satisfies at least all the objectives satisfied in p. As
we have shown in the paper, when going through the lattice of payoffs using the antichain optimization algorithm,
this test is sufficient to find the existence of a play with payoff p. In the special case where p does not satisfy
any objective, the existence is decided by trying to find a play satisfying the conjunction of the opposite for all
objectives of Player 1.
:param p: a payoff.
:param automaton: the automaton.
:param colors_map: maps each parity objective to the set of SPOT acceptance sets used to represent its priorities.
:param losing_for_0: whether we check for the extended payoff (0, p).
:return: whether the payoff p can be realized by a run in the automaton, i.e., whether the language of the automaton
is empty given the acceptance condition corresponding to the payoff.
"""
first = True
acc = None
# for each objective satisfied in p, create the acceptance code and add it with a conjunction
for i in range(1, len(p) + 1):
if p[i - 1] == 1:
if first:
acc = parity_to_acceptance(colors_map[i])
first = False
else:
acc &= parity_to_acceptance(colors_map[i])
# if there was no satisfied objective in the payoff, it cannot be encoded properly, and we encode it as the
# conjunction of the opposite of all objectives
if first:
for i in range(1, len(p) + 1):
if first:
acc = parity_to_acceptance(colors_map[i], opposite=True)
first = False
else:
acc &= parity_to_acceptance(colors_map[i], opposite=True)
# if the extended payoff is (0,p) add a conjunction with the opposite of objective 0
if losing_for_0:
acc &= parity_to_acceptance(colors_map[0], opposite=True)
acceptance_condition = spot.acc_cond(acc.used_sets().max_set(), acc)
automaton.set_acceptance(acceptance_condition)
# the payoff is realizable if the language of the automaton is not empty
return not automaton.is_empty()
def antichain_optimization_algorithm(automaton, nbr_objectives, colors_map, realizable):
"""
The algorithm is as presented in the pseudocode in the paper with the following notations: "Q" is written "queue"
and "A" is written "pareto_optimal_payoffs".
:param automaton: the automaton.
:param nbr_objectives: the number t of objectives of Player 1.
:param colors_map: maps each parity objective to the set of SPOT acceptance sets used to represent its priorities.
:param realizable: function which decides if a (extended) payoff is realizable.
:return: whether the PRV problem is satisfied.
"""
maximal_payoff = tuple([1] * nbr_objectives)
pareto_optimal_payoffs = []
queue = deque()
queue.append(maximal_payoff)
visited = defaultdict(int)
while queue:
p = queue.popleft()
if not any(smaller_than(p, p_prime) for p_prime in pareto_optimal_payoffs):
if realizable(p, automaton, colors_map):
pareto_optimal_payoffs.append(p)
if realizable(p, automaton, colors_map, losing_for_0=True):
return False
else:
for p_star in generate_smaller_payoffs(p):
if (not any(smaller_than(p_star, p_prime) for p_prime in pareto_optimal_payoffs)) and \
not visited[p_star]:
queue.append(p_star)
visited[p_star] = 1
return True
def conjunction_of_satisfied_objectives_in_p(nbr_objectives, p, colors_map):
"""
Create the conjunction of acceptance codes for each objective satisfied in the payoff p.
:param nbr_objectives: number of objectives of Player 1.
:param p: a payoff.
:param colors_map: maps each parity objective to the set of SPOT acceptance sets used to represent its priorities.
:return: the required conjunction, and None if there are no satisfied objectives.
"""
# whether we already found a satisfied objective in the payoff
first = True
acc = None
# for each objective satisfied in p, create the acceptance code and add it with a conjunction
for i in range(1, nbr_objectives + 1):
if p[i - 1] == 1:
# if this is the first objective to be satisfied
if first:
acc = parity_to_acceptance(colors_map[i])
first = False
else:
acc &= parity_to_acceptance(colors_map[i])
return acc
def disjunction_of_unsatisfied_objectives_in_p(nbr_objectives, p, colors_map):
"""
Create the disjunction of acceptance codes for each objective unsatisfied in the payoff p.
:param nbr_objectives: number of objectives of Player 1.
:param p: a payoff.
:param colors_map: maps each parity objective to the set of SPOT acceptance sets used to represent its priorities.
:return: the required disjunction, and None if there are no unsatisfied objectives.
"""
# whether we already found an unsatisfied objective in the payoff
first = True
acc = None
# for each objective unsatisfied in p, create the acceptance code and add it with a disjunction
for i in range(1, nbr_objectives + 1):
if p[i - 1] == 0:
# if this is the first objective to be satisfied
if first:
acc = parity_to_acceptance(colors_map[i])
first = False
else:
acc |= parity_to_acceptance(colors_map[i])
return acc
def counter_example_exists(nbr_objectives, automaton, colors_map, pareto_optimal_payoffs):
"""
Check for the existence of a run with a payoff equal or larger to some or incomparable to every payoff in the
current antichain of PO payoffs and that is losing for Player 0.
:param nbr_objectives: number of objectives of Player 1.
:param automaton: the automaton.
:param colors_map: maps each parity objective to the set of SPOT acceptance sets used to represent its priorities.
:param pareto_optimal_payoffs: current antichain of PO payoffs.
:return: whether an accepting run for the condition mentioned above exists.
"""
# acceptance condition should be losing for Player 0
acc = parity_to_acceptance(colors_map[0], opposite=True)
# for each PO payoff in the antichain, we want the conjunction of its satisfied objectives (meaning accepting runs
# whose payoff is equal or larger to this payoff) and add it to a disjunction of its unsatisfied objectives (meaning
# accepting runs whose payoff is incomparable to or strictly larger than this payoff). We combine the two with a
# disjunction since it should satisfy either condition and this is combined as a conjunction for every PO payoff.
for payoff in pareto_optimal_payoffs:
conj_of_sat_obj_in_payoff = conjunction_of_satisfied_objectives_in_p(nbr_objectives, payoff, colors_map)
disj_of_unsat_obj_in_payoff = disjunction_of_unsatisfied_objectives_in_p(nbr_objectives, payoff, colors_map)
# the two should be combined with a disjunction, but it may be the case that either is empty in case of all 0 or
# all 1 in payoffs
# if they are not empty, we do a disjunction between them and add them to the acceptance as a conjunction
if (conj_of_sat_obj_in_payoff is not None) and (disj_of_unsat_obj_in_payoff is not None):
temp = conj_of_sat_obj_in_payoff | disj_of_unsat_obj_in_payoff
acc &= temp
# if the conjunction is not empty but the disjunction is, only add the conjunction to the acceptance
elif (conj_of_sat_obj_in_payoff is not None) and (disj_of_unsat_obj_in_payoff is None):
acc &= conj_of_sat_obj_in_payoff
# if the disjunction is not empty but the conjunction is, only add the disjunction to the acceptance
elif (disj_of_unsat_obj_in_payoff is not None) and (conj_of_sat_obj_in_payoff is None):
acc &= disj_of_unsat_obj_in_payoff
# notice that if there are no payoffs in pareto_optimal_payoffs, we only ask for a losing play for Player 0
acceptance_condition = spot.acc_cond(acc.used_sets().max_set(), acc)
automaton.set_acceptance(acceptance_condition)
# this avoids performing emptiness check and accepting run retrieval in two separate steps, since we need an
# accepting run if the language is not empty for the counterexample algorithm.
accepting_run = automaton.accepting_run()
return accepting_run is not None, accepting_run
def counter_example_dominated(nbr_objectives, automaton, colors_map, counter_example_payoff):
"""
Check for the existence of a run with a payoff strictly larger to counter_example_payoff and that is winning for
Player 0.
:param nbr_objectives: number of objectives of Player 1.
:param automaton: the automaton.
:param colors_map: maps each parity objective to the set of SPOT acceptance sets used to represent its priorities.
:param counter_example_payoff: the counter-example payoff to dominate.
:return: whether an accepting run for the condition mentioned above exists.
"""
# acceptance condition should be winning for Player 0
acc = parity_to_acceptance(colors_map[0])
first = True
disj_for_each_payoffs = None
# for each payoff strictly larger than counter_example_payoff by one objective, we want the conjunction of its
# satisfied objectives (meaning accepting runs whose payoff is equal or larger to this payoff). We combine them with
# a disjunction since we want a run with at least a strictly larger payoff.
for payoff in generate_larger_payoffs(counter_example_payoff):
conj_of_sat_obj_in_payoff = conjunction_of_satisfied_objectives_in_p(nbr_objectives, payoff, colors_map)
# if this is the first conjunction in the disjunction; notice that the conjunction can never be empty as
# there is at least one objective satisfied in payoff (since the smallest payoffs yielded by the call contain
# exactly one satisfied objective).
if first:
disj_for_each_payoffs = conj_of_sat_obj_in_payoff
first = False
else:
disj_for_each_payoffs |= conj_of_sat_obj_in_payoff
# notice that if there are no payoffs larger than the counter example payoff (as it only contains ones), we return
# false as it cannot be dominated. There is no corresponding run.
if disj_for_each_payoffs is None:
return False, None
# if the disjunction is not empty (at least one larger payoff exists), do a conjunction with acc
else:
acc &= disj_for_each_payoffs
acceptance_condition = spot.acc_cond(acc.used_sets().max_set(), acc)
automaton.set_acceptance(acceptance_condition)
# this avoids performing emptiness check and accepting run retrieval in two separate steps, since we need an
# accepting run if the language is not empty for the counterexample algorithm.
accepting_run = automaton.accepting_run()
return accepting_run is not None, accepting_run
def get_payoff_of_accepting_run(nbr_objectives, colors_map, accepting_run):
"""
Compute the payoff of the provided accepting run of the automaton.
:param nbr_objectives: number of objectives of Player 1.
:param colors_map: maps each parity objective to the set of SPOT acceptance sets used to represent its priorities.
:param accepting_run: an accepting run of the automaton for a previously encoded language.
:return: the payoff of accepting_run.
"""
# priorities occurring infinitely often for each priority function
inf_priorities = [[] for _ in range(nbr_objectives + 1)]
"""
when there is only one priority per priority function, in that case there are some transitions with no priorities
for some functions (e.g. a transition has either priority 0 or no priority for the first function and the condition
is Inf(0)). Therefore, the objective is satisfied if 0 is in the list of occurring priorities in the cycle part of
the run.
if all(len(map_for_obj) == 1 for map_for_obj in colors_map.values()):
# maintain a set of occurring priorities in the cycle part of the run
occurring_priorities = set()
# for each transition in the cycle part of the run
for trans in accepting_run.cycle:
# for each acceptance set in the transition (of which there might not be one for each objective)
for acc_set in trans.acc.sets():
# add the acceptance set to the list of occurring priorities
occurring_priorities.add(acc_set)
# payoff is initialized with only zeros
payoff = [0] * nbr_objectives
# for each occurring priority, the corresponding objective is satisfied (we ignore payoff for Player 0)
for prio in occurring_priorities:
if prio > 0:
payoff[prio-1] = 1
return tuple(payoff)
"""
# for each transition in the cycle part of the run
for trans in accepting_run.cycle:
i = 0
# for each acceptance set in the transition (of which there are nbr_objectives + 1 by construction)
for acc_set in trans.acc.sets():
# add the acceptance set to the list of occurring priorities for the correct function
inf_priorities[i].append(acc_set)
i += 1
# minimum priorities occurring infinitely often
min_priorities = []
for i in range(0, nbr_objectives + 1):
min_priorities.append(min(inf_priorities[i]))
# compute the payoff from this minimum priority (acceptance set) occurring infinitely often, omitting Player 0
payoff = []
for i in range(1, nbr_objectives + 1):
# if the minimum priority and minimum acceptance set from colors_map given a parity objective are of the same
# parity then the objective is satisfied.
if (min_priorities[i] % 2) == (min(colors_map[i]) % 2):
payoff.append(1)
else:
payoff.append(0)
# return the payoff (which are tuples)
return tuple(payoff)
def counterexample_based_algorithm(automaton, nbr_objectives, colors_map):
"""
The algorithm is as presented in the pseudocode in the paper with the following notations: "A" is written
"pareto_optimal_payoffs".
:param automaton: the automaton.
:param nbr_objectives: the number t of objectives of Player 1.
:param colors_map: maps each parity objective to the set of SPOT acceptance sets used to represent its priorities.
:return: whether the PRV problem is satisfied.
"""
pareto_optimal_payoffs = []
while True:
a_counter_example_exists, accepting_run_counter_example = \
counter_example_exists(nbr_objectives, automaton, colors_map, pareto_optimal_payoffs)
if a_counter_example_exists:
counter_example_payoff = get_payoff_of_accepting_run(nbr_objectives, colors_map,
accepting_run_counter_example)
is_counter_example_dominated, accepting_run_dominating = counter_example_dominated(nbr_objectives,
automaton, colors_map,
counter_example_payoff)
if is_counter_example_dominated:
dominating_payoff = get_payoff_of_accepting_run(nbr_objectives, colors_map,
accepting_run_dominating)
# update the set of PO payoffs with this new payoff
pareto_optimal_payoffs = add_payoff_to_antichain(pareto_optimal_payoffs, dominating_payoff)
else:
# a counter example is not dominated by another payoff, hence the instance is false
return False
else:
# there are no counter examples, hence the instance is true
return True