forked from TonyZhangND/paxos_proof
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathGraph.dfy
More file actions
109 lines (93 loc) · 2.82 KB
/
Graph.dfy
File metadata and controls
109 lines (93 loc) · 2.82 KB
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
type value = nat
datatype vertex = V(id:value)
datatype edge = E(v1:vertex, v2:vertex)
datatype graph = G(vertices:set<vertex>, edges:set<edge>)
predicate isValidGraph(g:graph) {
&& |g.vertices| > 0
&& forall e | e in g.edges ::
&& e.v1 in g.vertices
&& e.v2 in g.vertices
}
predicate isValidPath(p:seq<edge>)
decreases p
{
if |p| <= 1 then true
else p[0].v2 == p[1].v1 && isValidPath(p[1..])
}
predicate pathLeadsToTarget(source:vertex, target:value, p:seq<edge>)
decreases p
requires isValidPath(p)
{
if |p| == 0 then source.id == target
else
var e := p[0];
pathLeadsToTarget(e.v2, target, p[1..])
}
// Main correct path predicate
predicate isCorrectPath(g:graph, source:vertex, target:value, p:seq<edge>)
requires isValidGraph(g)
{
&& (forall e | e in p :: e in g.edges)
&& isValidPath(p)
&& pathLeadsToTarget(source, target, p)
}
method neighborsSeq(edges:set<edge>, n:vertex) returns (nbrs:seq<vertex>)
decreases edges;
{
if |edges| == 0 {
return [];
} else {
var e :| e in edges;
var nbrs := neighborsSeq(edges-{e}, n);
if e.v1 == n {
return [e.v2] + nbrs;
} else {
return nbrs;
}
}
}
method hasPathBfsImp(g:graph, source:vertex, target:value) returns (b:bool)
requires isValidGraph(g);
ensures (exists p :: isCorrectPath(g, source, target, p)) <==> b
{
var visited := {source};
var q := [source];
ghost var processed := {};
while |q| > 0
invariant forall i | 0 <= i < |q| :: q[i] !in processed
invariant forall v | v in g.vertices :: v in processed ==> v in visited
decreases g.vertices - processed
{
var curr := q[0];
assert curr !in processed;
assert |processed + {curr}| > |processed|;
processed := processed + {curr};
q := q[1..];
if curr.id == target {
b := true;
return;
} else {
var nbrs := neighborsSeq(g.edges, curr);
var i := 0;
while i < |nbrs|
decreases |nbrs| - i
invariant forall k | 0 <= k < |q| :: q[k] !in processed
{
var n := nbrs[i];
if n !in visited {
visited := visited + {n};
assert n !in processed;
q := q + [n];
}
i := i + 1;
}
}
}
b := false;
}
/*****************************************************************************************
* Proof Code *
*****************************************************************************************/
// function reachable(g:graph, source:vertex) : set<vertex>
// {
// }