-
Notifications
You must be signed in to change notification settings - Fork 58
/
proposer.go
135 lines (116 loc) · 3.12 KB
/
proposer.go
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
package paxos
import (
"log"
"time"
)
type proposer struct {
id int
// stable
lastSeq int
value string
valueN int
acceptors map[int]promise
nt network
}
func newProposer(id int, value string, nt network, acceptors ...int) *proposer {
p := &proposer{id: id, nt: nt, lastSeq: 0, value: value, acceptors: make(map[int]promise)}
for _, a := range acceptors {
p.acceptors[a] = message{}
}
return p
}
func (p *proposer) run() {
var ok bool
var m message
// stage 1: do prepare until reach the majority
for !p.majorityReached() {
if !ok {
ms := p.prepare()
for i := range ms {
p.nt.send(ms[i])
}
}
m, ok = p.nt.recv(time.Second)
if !ok {
// the previous prepare is failed
// continue to do another prepare
continue
}
switch m.typ {
case Promise:
p.receivePromise(m)
default:
log.Panicf("proposer: %d unexpected message type: %v", p.id, m.typ)
}
}
log.Printf("proposer: %d promise %d reached majority %d", p.id, p.n(), p.majority())
// stage 2: do propose
log.Printf("proposer: %d starts to propose [%d: %s]", p.id, p.n(), p.value)
ms := p.propose()
for i := range ms {
p.nt.send(ms[i])
}
}
// If the proposer receives the requested responses from a majority of
// the acceptors, then it can issue a proposal with number n and value
// v, where v is the value of the highest-numbered proposal among the
// responses, or is any value selected by the proposer if the responders
// reported no proposals.
func (p *proposer) propose() []message {
ms := make([]message, p.majority())
i := 0
for to, promise := range p.acceptors {
if promise.number() == p.n() {
ms[i] = message{from: p.id, to: to, typ: Propose, n: p.n(), value: p.value}
i++
}
if i == p.majority() {
break
}
}
return ms
}
// A proposer chooses a new proposal number n and sends a request to
// each member of some set of acceptors, asking it to respond with:
// (a) A promise never again to accept a proposal numbered less than n, and
// (b) The proposal with the highest number less than n that it has accepted, if any.
func (p *proposer) prepare() []message {
p.lastSeq++
ms := make([]message, p.majority())
i := 0
for to := range p.acceptors {
ms[i] = message{from: p.id, to: to, typ: Prepare, n: p.n()}
i++
if i == p.majority() {
break
}
}
return ms
}
func (p *proposer) receivePromise(promise message) {
prevPromise := p.acceptors[promise.from]
if prevPromise.number() < promise.number() {
log.Printf("proposer: %d received a new promise %+v", p.id, promise)
p.acceptors[promise.from] = promise
//update value to the value with a larger N
if promise.proposalNumber() > p.valueN {
log.Printf("proposer: %d updated the value [%s] to %s", p.id, p.value, promise.proposalValue())
p.valueN = promise.proposalNumber()
p.value = promise.proposalValue()
}
}
}
func (p *proposer) majority() int { return len(p.acceptors)/2 + 1 }
func (p *proposer) majorityReached() bool {
m := 0
for _, promise := range p.acceptors {
if promise.number() == p.n() {
m++
}
}
if m >= p.majority() {
return true
}
return false
}
func (p *proposer) n() int { return p.lastSeq<<16 | p.id }