forked from tlaplus/Examples
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Paxos.tla
395 lines (375 loc) · 25.7 KB
/
Paxos.tla
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
-------------------------------- MODULE Paxos -------------------------------
(***************************************************************************)
(* This is a high-level specification of the Paxos consensus algorithm. *)
(* It refines the spec in module Voting, which you should read before *)
(* reading this module. In the Paxos consensus algorithm, acceptors *)
(* communicate by sending messages. There are additional processes called *)
(* leaders. The specification here essentially considers there to be a *)
(* separate leader for each ballot number. We can consider "leader" to be *)
(* a role, where in an implementation there will be a finite number of *)
(* leader processes each of which plays infinitely many of these leader *)
(* roles. *)
(* *)
(* Note: The algorithm described here is the Paxos consensus algorithm. *)
(* It is the crucial component of the Paxos algorithm, which implements a *)
(* fault-tolerant state machine using a sequence of instances of the *)
(* consensus algorithm. The Paxos algorithm is sometimes called *)
(* MultiPaxos, with the Paxos consensus algorithm being incorrectly called *)
(* the Paxos algorithm. I'm afraid I have contributed to this confusion *)
(* by being lazy and calling the module "Paxos" instead of *)
(* "PaxosConsensus". But incarnations of this module have already *)
(* appeared, so I'm reluctant to change its name now. *)
(***************************************************************************)
EXTENDS Integers
(***************************************************************************)
(* The constants and the assumptions about them are the same as for the *)
(* Voting algorithm. However, the second conjunct of the assumption, *)
(* which asserts that any two quorums have a non-empty intersection, is *)
(* not needed for the Paxos consensus algorithm to implement the Voting *)
(* algorithm. The Voting algorithm, and it, do not satisfy consensus *)
(* without that assumption. *)
(***************************************************************************)
CONSTANTS Value, Acceptor, Quorum
ASSUME /\ \A Q \in Quorum : Q \subseteq Acceptor
/\ \A Q1, Q2 \in Quorum : Q1 \cap Q2 /= {}
Ballot == Nat
None == CHOOSE v : v \notin Ballot
(*************************************************************************)
(* This defines None to be an unspecified value that is not a ballot *)
(* number. *)
(*************************************************************************)
(***************************************************************************)
(* We now define Message to be the set of all possible messages that can *)
(* be sent in the algorithm. In TLA+, the expression *)
(* *)
(* (1) [type |-> "1a", bal |-> b] *)
(* *)
(* is a record r with two components, a `type'component, r.type, that *)
(* equals "1a" and whose bal component, r.bal, that equals b. The *)
(* expression *)
(* *)
(* (2) [type : {"1a"}, bal : Ballot] *)
(* *)
(* is the set of all records r with a components `type' and bal such that *)
(* r.type is an element of {"1a"} and r.bal is an element of Ballot. *)
(* Since "1a" is the only element of {"1a"}, formula (2) is the set of all *)
(* elements (1) such that b \in Ballot. *)
(* *)
(* The function of each type of message in the set Message is explained *)
(* below with the action that can send it. *)
(***************************************************************************)
Message ==
[type : {"1a"}, bal : Ballot]
\cup [type : {"1b"}, acc : Acceptor, bal : Ballot,
mbal : Ballot \cup {-1}, mval : Value \cup {None}]
\cup [type : {"2a"}, bal : Ballot, val : Value]
\cup [type : {"2b"}, acc : Acceptor, bal : Ballot, val : Value]
-----------------------------------------------------------------------------
(***************************************************************************)
(* We now declare the following variables: *)
(* *)
(* maxBal - Is the same as the variable of that name in the Voting *)
(* algorithm. *)
(* *)
(* maxVBal *)
(* maxVal - As in the Voting algorithm, a vote is a <<ballot, value>> *)
(* pair. The pair <<maxVBal[a], maxVal[a]>> is the vote with *)
(* the largest ballot number cast by acceptor a . It equals *)
(* <<-1, None>> if a has not cast any vote. *)
(* *)
(* msgs - The set of all messages that have been sent. *)
(* *)
(* Messages are added to msgs when they are sent and are never removed. *)
(* An operation that is performed upon receipt of a message is represented *)
(* by an action that is enabled when the message is in msgs. This *)
(* simplifies the spec in the following ways: *)
(* *)
(* - A message can be broadcast to multiple recipients by just adding *)
(* (a single copy of) it to msgs. *)
(* *)
(* - Never removing the message automatically allows the possibility of *)
(* the same message being received twice. *)
(* *)
(* Since we are considering only safety, there is no need to explicitly *)
(* model message loss. The safety part of the spec says only what *)
(* messages may be received and does not assert that any message actually *)
(* is received. Thus, there is no difference between a lost message and *)
(* one that is never received. *)
(***************************************************************************)
VARIABLES maxBal, maxVBal, maxVal, msgs
vars == <<maxBal, maxVBal, maxVal, msgs>>
(*************************************************************************)
(* It's convenient to name the tuple of all variables in a spec. *)
(*************************************************************************)
(***************************************************************************)
(* The invariant that describes the "types" of the variables. *)
(***************************************************************************)
TypeOK == /\ maxBal \in [Acceptor -> Ballot \cup {-1}]
/\ maxVBal \in [Acceptor -> Ballot \cup {-1}]
/\ maxVal \in [Acceptor -> Value \cup {None}]
/\ msgs \subseteq Message
(***************************************************************************)
(* The initial predicate should be obvious from the descriptions of the *)
(* variables given above. *)
(***************************************************************************)
Init == /\ maxBal = [a \in Acceptor |-> -1]
/\ maxVBal = [a \in Acceptor |-> -1]
/\ maxVal = [a \in Acceptor |-> None]
/\ msgs = {}
----------------------------------------------------------------------------
(***************************************************************************)
(* We now define the subactions of the next-state actions. We begin by *)
(* defining an action that will be used in those subactions. The action *)
(* Send(m) asserts that message m is added to the set msgs. *)
(***************************************************************************)
Send(m) == msgs' = msgs \cup {m}
(***************************************************************************)
(* The ballot b leader can perform actions Phase1a(b) and Phase2a(b). In *)
(* the Phase1a(b) action, it sends to all acceptors a phase 1a message (a *)
(* message m with m.type = "1a") that begins ballot b. Remember that the *)
(* same process can perform the role of leader for many different ballot *)
(* numbers b. In practice, it will stop playing the role of leader of *)
(* ballot b when it begins a higher-numbered ballot. (Remember the *)
(* definition of [type |-> "1a", bal |-> b] from the comment preceding the *)
(* definition of Message.) *)
(***************************************************************************)
Phase1a(b) == /\ Send([type |-> "1a", bal |-> b])
/\ UNCHANGED <<maxBal, maxVBal, maxVal>>
(************************************************************************)
(* Note that there is no enabling condition to prevent sending the *)
(* phase 1a message a second time. Since messages are never removed *)
(* from msg, performing the action a second time leaves msg and all the *)
(* other spec variables unchanged, so it's a stuttering step. Since *)
(* stuttering steps are always allowed, there's no reason to try to *)
(* prevent them. *)
(************************************************************************)
(***************************************************************************)
(* Upon receipt of a ballot b phase 1a message, acceptor a can perform a *)
(* Phase1b(a) action only if b > maxBal[a]. The action sets maxBal[a] to *)
(* b and sends a phase 1b message to the leader containing the values of *)
(* maxVBal[a] and maxVal[a]. This action implements the *)
(* IncreaseMaxBal(a,b) action of the Voting algorithm for b = m.bal. *)
(***************************************************************************)
Phase1b(a) ==
/\ \E m \in msgs :
/\ m.type = "1a"
/\ m.bal > maxBal[a]
/\ maxBal' = [maxBal EXCEPT ![a] = m.bal]
/\ Send([type |-> "1b", acc |-> a, bal |-> m.bal,
mbal |-> maxVBal[a], mval |-> maxVal[a]])
/\ UNCHANGED <<maxVBal, maxVal>>
(***************************************************************************)
(* In the Phase2a(b, v) action, the ballot b leader sends a type "2a" *)
(* message asking the acceptors to vote for v in ballot number b. The *)
(* enabling conditions of the action--its first two conjuncts--ensure that *)
(* three of the four enabling conditions of action VoteFor(a, b, v) in *)
(* module Voting will be true when acceptor a receives that message. *)
(* Those three enabling conditions are the second through fourth conjuncts *)
(* of that action. *)
(* *)
(* The first conjunct of Phase2a(b, v) asserts that at most one phase 2a *)
(* message is ever sent for ballot b. Since an acceptor will vote for a *)
(* value in ballot b only when it receives the appropriate phase 2a *)
(* message, the phase 2a message sent by this action this ensures that *)
(* these two enabling conjuncts of VoteFor(a,b,v) will be true forever: *)
(* *)
(* /\ \A vt \in votes[a] : vt[1] /= b *)
(* /\ \A c \in Acceptor \ {a} : *)
(* \A vt \in votes[c] : (vt[1] = b) => (vt[2] = v) *)
(* *)
(* The second conjunct of the Phase2a(b, v) action is the heart of the *)
(* Paxos consensus algorithm. It's a bit complicated, but I've tried a *)
(* number of times to write it in English, and it's much easier to *)
(* understand when written in mathematics. The LET/IN construct locally *)
(* defines Q1 to be the set of phase 1b messages sent in ballot number b *)
(* by acceptors in quorum Q; and it defines Q1bv to be the subset of those *)
(* messages indicating that the sender had voted in some ballot (which *)
(* must have been numbered less than b). You should study the IN clause *)
(* to convince yourself that it equals ShowsSafeAt(Q, b, v), defined in *)
(* module Voting, using the values of maxBal[a], maxVBal[a], and maxVal[a] *)
(* `a' sent in its phase 1b message to describe what votes it had cast *)
(* when it sent that message. Moreover, since `a' will no longer cast any *)
(* votes in ballots numbered less than b, the IN clause implies that *)
(* ShowsSafeAt(Q, b, v) is still true and will remain true forever. *)
(* Hence, this conjunct of Phase2a(b, v) checks the enabling condition *)
(* *)
(* /\ \E Q \in Quorum : ShowsSafeAt(Q, b, v) *)
(* *)
(* of module Voting's VoteFor(a, b, v) action. *)
(* *)
(* The type "2a" message sent by this action therefore tells every *)
(* acceptor `a' that, when it receives the message, all the enabling *)
(* conditions of VoteFor(a, b, v) but the first, maxBal[a] =< b, are *)
(* satisfied. *)
(***************************************************************************)
Phase2a(b, v) ==
/\ ~ \E m \in msgs : m.type = "2a" /\ m.bal = b
/\ \E Q \in Quorum :
LET Q1b == {m \in msgs : /\ m.type = "1b"
/\ m.acc \in Q
/\ m.bal = b}
Q1bv == {m \in Q1b : m.mbal >= 0}
IN /\ \A a \in Q : \E m \in Q1b : m.acc = a
/\ \/ Q1bv = {}
\/ \E m \in Q1bv :
/\ m.mval = v
/\ \A mm \in Q1bv : m.mbal >= mm.mbal
/\ Send([type |-> "2a", bal |-> b, val |-> v])
/\ UNCHANGED <<maxBal, maxVBal, maxVal>>
(***************************************************************************)
(* The Phase2b(a) action describes what acceptor `a' does when it receives *)
(* a phase 2a message m, which is sent by the leader of ballot m.bal *)
(* asking acceptors to vote for m.val in that ballot. Acceptor `a' acts *)
(* on that request, voting for m.val in ballot number m.bal, iff m.bal >= *)
(* maxBal[a], which means that `a' has not participated in any ballot *)
(* numbered greater than m.bal. Thus, this enabling condition of the *)
(* Phase2b(a) action together with the receipt of the phase 2a message m *)
(* implies that the VoteFor(a, m.bal, m.val) action of module Voting is *)
(* enabled and can be executed. The Phase2b(a) message updates maxBal[a], *)
(* maxVBal[a], and maxVal[a] so their values mean what they were claimed *)
(* to mean in the comments preceding the variable declarations. *)
(***************************************************************************)
Phase2b(a) ==
\E m \in msgs :
/\ m.type = "2a"
/\ m.bal >= maxBal[a]
/\ maxBal' = [maxBal EXCEPT ![a] = m.bal]
/\ maxVBal' = [maxVBal EXCEPT ![a] = m.bal]
/\ maxVal' = [maxVal EXCEPT ![a] = m.val]
/\ Send([type |-> "2b", acc |-> a,
bal |-> m.bal, val |-> m.val])
(***************************************************************************)
(* The definitions of Next and Spec are what we expect them to be. *)
(***************************************************************************)
Next == \/ \E b \in Ballot : \/ Phase1a(b)
\/ \E v \in Value : Phase2a(b, v)
\/ \E a \in Acceptor : Phase1b(a) \/ Phase2b(a)
Spec == Init /\ [][Next]_vars
----------------------------------------------------------------------------
(***************************************************************************)
(* We define `votes' to be the function such that votes[a] is the set of *)
(* pairs <<b, v>> such that acceptor `a' has voted for v in ballot number *)
(* b by sending executing the Phase2b(a) action to send the appropriate *)
(* type "2b" message. The Paxos consensus algorithm implements the Voting *)
(* algorithm by implementing the variable `votes' of module Voting with *)
(* the expression `votes' of the current module, and implementing the *)
(* variable maxBal of module Voting with the variable of the same name of *)
(* the current module. *)
(***************************************************************************)
votes ==
[a \in Acceptor |->
{<<m.bal, m.val>> : m \in {mm \in msgs: /\ mm.type = "2b"
/\ mm.acc = a }}]
(***************************************************************************)
(* The following INSTANCE statement omits the redundant clause *)
(* *)
(* WITH votes <- votes, maxBal <- maxBal, *)
(* Value <- Value, Acceptor <- Acceptor, Quorum <- Quorum *)
(***************************************************************************)
V == INSTANCE Voting
(***************************************************************************)
(* The inductive invariant Inv explains why the Paxos consensus algorithm *)
(* implements the Voting algorithm. It is defined after the INSTANCE *)
(* statement because it uses the operator V!ShowsSafeAt imported by that *)
(* statement. *)
(***************************************************************************)
Inv ==
/\ TypeOK
/\ \A a \in Acceptor : maxBal[a] >= maxVBal[a]
/\ \A a \in Acceptor : IF maxVBal[a] = -1
THEN maxVal[a] = None
ELSE <<maxVBal[a], maxVal[a]>> \in votes[a]
/\ \A m \in msgs :
/\ (m.type = "1b") => /\ maxBal[m.acc] >= m.bal
/\ (m.mbal >= 0) =>
<<m.mbal, m.mval>> \in votes[m.acc]
/\ (m.type = "2a") => /\ \E Q \in Quorum :
V!ShowsSafeAt(Q, m.bal, m.val)
/\ \A mm \in msgs : /\ mm.type ="2a"
/\ mm.bal = m.bal
=> mm.val = m.val
/\ (m.type = "2b") => /\ maxVBal[m.acc] >= m.bal
/\ \E mm \in msgs : /\ mm.type = "2a"
/\ mm.bal = m.bal
/\ mm.val = m.val
(***************************************************************************)
(* The following two theorems assert that Inv is an invariant of *)
(* the Paxos cconsensus algorithm and that this algorithm implements *)
(* the Voting algorithm with the declared variables and constants *)
(* of that algorithm implemented by the correspondingly-named expressions *)
(* in the current module. *)
(***************************************************************************)
THEOREM Invariance == Spec => []Inv
THEOREM Implementation == Spec => V!Spec
(***************************************************************************)
(* The ASSUME statement of this module trivial implies trivially implies *)
(* the instantiated version of the ASSUME statement of module Voting. *)
(* (Because the INSTANCE statement substitutes the constants of the *)
(* current module for the constants of the same name in module Voting, the *)
(* imported assumption is the same as the assumption of the current *)
(* module.) Hence, this theorem imported from module Voting is true in the *)
(* current module *)
(* *)
(* THEOREM V!Implementation == V!Spec => V!C!Spec *)
(* *)
(* Theorems Implementation and V!Implementation imply *)
(* *)
(* THEOREM Spec => V!C!Spec *)
(* *)
(* This theorem asserts that the Paxos consensus algorithm implements the *)
(* Consensus specification by substituting for the variable `chosen' of *)
(* the Consensus specification the value V!chosen of the current module. *)
(* The expression V!chosen is obtained by substituting the expression *)
(* `votes' of the current module for the variable `votes' of module Voting *)
(* in the expression `chosen' of module Voting. *)
(* *)
(* In other words, as we should expect, "implements" is a transitive *)
(* relation--under a suitable understanding of what transitivity means in *)
(* this situation. *)
(***************************************************************************)
(***************************************************************************)
(* This current module is distributed with two models, TinyModel and *)
(* SmallModel. SmallModel is the same as the model by that name for the *)
(* Voting specification. TinyModel is the same except it defines Ballot *)
(* to contain only two elements. Run TLC on these models. You should *)
(* find that it takes a couple of seconds to run TinyModel and two or *)
(* three minutes to run SmallModel. *)
(* *)
(* Next, try the same thing you did with the Voting algorithm: Modify the *)
(* models so the assumption that any pair of quorums has an element in *)
(* common is no longer true. (Again, it's best to modify clones of the *)
(* models.) This time, running TLC will not find an error. The *)
(* correctness of theorems Invariance and Implementation does not depend *)
(* on that assumption. The Paxos consensus algorithm still correctly *)
(* implements the Voting algorithm; but the Voting algorithm is incorrect *)
(* if the assumption does not hold. *)
(* *)
(* Now go back to the original SmallModel, in which the quorum assumption *)
(* holds. The sets Acceptor and Value are symmetry sets for the spec. *)
(* (See the "Model Values and Symmetry" help page to find out what that *)
(* means.) Try editing the values substituted for Acceptor and/or Value by *)
(* selecting the "Symmetry set" option and comparing the number of *)
(* reachable states TLC found and the time it took. (Remember to use *)
(* cloned models.) *)
(* *)
(* When you have other things to do while TLC is running, try increasing *)
(* the size of the model a very little bit at a time and see how the *)
(* running time increases. You'll find that it increases exponentially *)
(* with the numbers of acceptors, values, and ballots. *)
(* *)
(* Fortunately, exhaustively checking a small model is very effective at *)
(* finding errors. Since the Paxos consensus algorithm has been proved *)
(* correct, and that proof has been read by many people, I'm sure that the *)
(* basic algorithm is correct. Checking this spec on SmallModel makes me *)
(* quite confident that there are no "coding errors" in this TLA+ *)
(* specification of the algorithm. *)
(* *)
(* For checking safety properties, TLC can obtain close to linear speedup *)
(* using dozens of processors. After designing a new distributed *)
(* algorithm, you will have plenty of time to run TLC while the algorithm *)
(* is being implemented and the implementation tested. Use that time to *)
(* run it for as long as you can on the largest machine(s) that you can. *)
(* Testing the implementation is unlikely to find subtle errors in the *)
(* algorithm. *)
(***************************************************************************)
============================================================================