-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtypes.go
executable file
·522 lines (437 loc) · 12.8 KB
/
types.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
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
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
package hotstuff
import (
"bytes"
"crypto"
"encoding/base64"
"encoding/binary"
"encoding/json"
"fmt"
"io"
"strconv"
"strings"
)
// IDSet implements a set of replica IDs. It is used to show which replicas participated in some event.
type IDSet interface {
// Add adds an ID to the set.
Add(id ID)
// Contains returns true if the set contains the ID.
Contains(id ID) bool
// ForEach calls f for each ID in the set.
ForEach(f func(ID))
// RangeWhile calls f for each ID in the set until f returns false.
RangeWhile(f func(ID) bool)
// Len returns the number of entries in the set.
Len() int
}
// idSetMap implements IDSet using a map.
type idSetMap map[ID]struct{}
// NewIDSet returns a new IDSet using the default implementation.
func NewIDSet() IDSet {
return make(idSetMap)
}
// Add adds an ID to the set.
func (s idSetMap) Add(id ID) {
s[id] = struct{}{}
}
// Contains returns true if the set contains the given ID.
func (s idSetMap) Contains(id ID) bool {
_, ok := s[id]
return ok
}
// ForEach calls f for each ID in the set.
func (s idSetMap) ForEach(f func(ID)) {
for id := range s {
f(id)
}
}
// RangeWhile calls f for each ID in the set until f returns false.
func (s idSetMap) RangeWhile(f func(ID) bool) {
for id := range s {
if !f(id) {
break
}
}
}
// Len returns the number of entries in the set.
func (s idSetMap) Len() int {
return len(s)
}
func (s idSetMap) String() string {
return IDSetToString(s)
}
// IDSetToString formats an IDSet as a string.
func IDSetToString(set IDSet) string {
var sb strings.Builder
sb.WriteString("[ ")
set.ForEach(func(i ID) {
sb.WriteString(strconv.Itoa(int(i)))
sb.WriteString(" ")
})
sb.WriteString("]")
return sb.String()
}
// View is a number that uniquely identifies a view.
type View uint64
// ToBytes returns the view as bytes.
func (v View) ToBytes() []byte {
var viewBytes [8]byte
binary.LittleEndian.PutUint64(viewBytes[:], uint64(v))
return viewBytes[:]
}
// Hash is a SHA256 hash
type Hash [32]byte
func (h Hash) String() string {
return base64.StdEncoding.EncodeToString(h[:])
}
// Command is a client request to be executed by the consensus protocol.
//
// The string type is used because it is immutable and can hold arbitrary bytes of any length.
type Command string
// ToBytes is an object that can be converted into bytes for the purposes of hashing, etc.
type ToBytes interface {
// ToBytes returns the object as bytes.
ToBytes() []byte
}
// PublicKey is the public part of a replica's key pair.
type PublicKey = crypto.PublicKey
// PrivateKey is the private part of a replica's key pair.
type PrivateKey interface {
// Public returns the public key associated with this private key.
Public() PublicKey
}
// QuorumSignature is a signature that is only valid when it contains the signatures of a quorum of replicas.
type QuorumSignature interface {
ToBytes
// Participants returns the IDs of replicas who participated in the threshold signature.
Participants() IDSet
}
// ThresholdSignature is a signature that is only valid when it contains the signatures of a quorum of replicas.
//
// Deprecated: renamed to QuorumSignature
type ThresholdSignature = QuorumSignature
// PartialCert is a signed block hash.
type PartialCert struct {
// shortcut to the signer of the signature
signer ID
signature QuorumSignature
blockHash Hash
}
// NewPartialCert returns a new partial certificate.
func NewPartialCert(signature QuorumSignature, blockHash Hash) PartialCert {
var signer ID
signature.Participants().RangeWhile(func(i ID) bool {
signer = i
return false
})
return PartialCert{signer, signature, blockHash}
}
// Signer returns the ID of the replica that created the certificate.
func (pc PartialCert) Signer() ID {
return pc.signer
}
// Signature returns the signature.
func (pc PartialCert) Signature() QuorumSignature {
return pc.signature
}
// BlockHash returns the hash of the block that was signed.
func (pc PartialCert) BlockHash() Hash {
return pc.blockHash
}
// ToBytes returns a byte representation of the partial certificate.
func (pc PartialCert) ToBytes() []byte {
return append(pc.blockHash[:], pc.signature.ToBytes()...)
}
// SyncInfo holds the highest known QC or TC. // SyncInfo存储最高的已知QC
// Generally, if highQC.View > highTC.View, there is no need to include highTC in the SyncInfo.
// However, if highQC.View < highTC.View, we should still include highQC.
// This can also hold an AggregateQC for Fast-Hotstuff.
type SyncInfo struct {
qc *QuorumCert
tc *TimeoutCert
aggQC *AggregateQC // 这个参数其实是为了fast-hotstuff准备的,先不用管
}
// NewSyncInfo returns a new SyncInfo struct.
func NewSyncInfo() SyncInfo {
return SyncInfo{}
}
// WithQC returns a copy of the SyncInfo struct with the given QC.
func (si SyncInfo) WithQC(qc QuorumCert) SyncInfo {
si.qc = new(QuorumCert)
*si.qc = qc
return si
}
// WithTC returns a copy of the SyncInfo struct with the given TC.
func (si SyncInfo) WithTC(tc TimeoutCert) SyncInfo {
si.tc = new(TimeoutCert)
*si.tc = tc
return si
}
// WithAggQC returns a copy of the SyncInfo struct with the given AggregateQC.
func (si SyncInfo) WithAggQC(aggQC AggregateQC) SyncInfo {
si.aggQC = new(AggregateQC)
*si.aggQC = aggQC
return si
}
// QC returns the quorum certificate, if present.
func (si SyncInfo) QC() (_ QuorumCert, _ bool) {
if si.qc != nil {
return *si.qc, true
}
return
}
// TC returns the timeout certificate, if present.
func (si SyncInfo) TC() (_ TimeoutCert, _ bool) {
if si.tc != nil {
return *si.tc, true
}
return
}
// AggQC returns the AggregateQC, if present.
func (si SyncInfo) AggQC() (_ AggregateQC, _ bool) {
if si.aggQC != nil {
return *si.aggQC, true
}
return
}
func (si SyncInfo) String() string {
var sb strings.Builder
sb.WriteString("{ ")
if si.tc != nil {
fmt.Fprintf(&sb, "%s ", si.tc)
}
if si.qc != nil {
fmt.Fprintf(&sb, "%s ", si.qc)
}
if si.aggQC != nil {
fmt.Fprintf(&sb, "%s ", si.aggQC)
}
sb.WriteRune('}')
return sb.String()
}
// QuorumCert (QC) is a certificate for a Block created by a quorum of partial certificates.
type QuorumCert struct {
signature QuorumSignature
view View
hash Hash
}
// NewQuorumCert creates a new quorum cert from the given values.
func NewQuorumCert(signature QuorumSignature, view View, hash Hash) QuorumCert {
return QuorumCert{signature, view, hash}
}
// ToBytes returns a byte representation of the quorum certificate.
func (qc QuorumCert) ToBytes() []byte {
b := qc.view.ToBytes()
b = append(b, qc.hash[:]...)
if qc.signature != nil {
b = append(b, qc.signature.ToBytes()...)
}
return b
}
// Signature returns the threshold signature.
func (qc QuorumCert) Signature() QuorumSignature {
return qc.signature
}
// BlockHash returns the hash of the block that was signed.
func (qc QuorumCert) BlockHash() Hash {
return qc.hash
}
// View returns the view in which the QC was created.
func (qc QuorumCert) View() View {
return qc.view
}
// Equals returns true if the other QC equals this QC.
func (qc QuorumCert) Equals(other QuorumCert) bool {
if qc.view != other.view {
return false
}
if qc.hash != other.hash {
return false
}
if qc.signature == nil || other.signature == nil {
return qc.signature == other.signature
}
return bytes.Equal(qc.signature.ToBytes(), other.signature.ToBytes())
}
func (qc QuorumCert) String() string {
var sb strings.Builder
if qc.signature != nil {
_ = writeParticipants(&sb, qc.Signature().Participants())
}
return fmt.Sprintf("QC{ hash: %.6s, IDs: [ %s] }", qc.hash, &sb)
}
// TimeoutCert (TC) is a certificate created by a quorum of timeout messages.
type TimeoutCert struct {
signature QuorumSignature
view View
}
// NewTimeoutCert returns a new timeout certificate.
func NewTimeoutCert(signature QuorumSignature, view View) TimeoutCert {
return TimeoutCert{signature, view}
}
// ToBytes returns a byte representation of the timeout certificate.
func (tc TimeoutCert) ToBytes() []byte {
var viewBytes [8]byte
binary.LittleEndian.PutUint64(viewBytes[:], uint64(tc.view))
return append(viewBytes[:], tc.signature.ToBytes()...)
}
// Signature returns the threshold signature.
func (tc TimeoutCert) Signature() QuorumSignature {
return tc.signature
}
// View returns the view in which the timeouts occurred.
func (tc TimeoutCert) View() View {
return tc.view
}
func (tc TimeoutCert) String() string {
var sb strings.Builder
if tc.signature != nil {
_ = writeParticipants(&sb, tc.Signature().Participants())
}
return fmt.Sprintf("TC{ view: %d, IDs: [ %s] }", tc.view, &sb)
}
// AggregateQC is a set of QCs extracted from timeout messages and an aggregate signature of the timeout signatures.
//
// This is used by the Fast-HotStuff consensus protocol.
type AggregateQC struct {
qcs map[ID]QuorumCert
sig QuorumSignature
view View
}
// NewAggregateQC returns a new AggregateQC from the QC map and the threshold signature.
func NewAggregateQC(qcs map[ID]QuorumCert, sig QuorumSignature, view View) AggregateQC {
return AggregateQC{qcs, sig, view}
}
// QCs returns the quorum certificates in the AggregateQC.
func (aggQC AggregateQC) QCs() map[ID]QuorumCert {
return aggQC.qcs
}
// Sig returns the threshold signature in the AggregateQC.
func (aggQC AggregateQC) Sig() QuorumSignature {
return aggQC.sig
}
// View returns the view in which the AggregateQC was created.
func (aggQC AggregateQC) View() View {
return aggQC.view
}
func (aggQC AggregateQC) String() string {
var sb strings.Builder
if aggQC.sig != nil {
_ = writeParticipants(&sb, aggQC.sig.Participants())
}
return fmt.Sprintf("AggQC{ view: %d, IDs: [ %s] }", aggQC.view, &sb)
}
func writeParticipants(wr io.Writer, participants IDSet) (err error) {
participants.RangeWhile(func(id ID) bool {
_, err = fmt.Fprintf(wr, "%d ", id)
return err == nil
})
return err
}
// RapidFair: baseline 在types里增加CollectTxSeq类型,并增加方法
type CollectTxSeq struct {
view View
txSeq Command
syncInfo SyncInfo // 记录qc同步消息
}
func NewCollectTxSeq(v View, txSeq Command, syncInfo SyncInfo) CollectTxSeq {
return CollectTxSeq{v, txSeq, syncInfo}
}
func (col CollectTxSeq) View() View {
return col.view
}
func (col CollectTxSeq) TxSeq() Command {
return col.txSeq
}
func (col CollectTxSeq) SyncInfo() SyncInfo {
return col.syncInfo
}
// RapidFair: baseline 增加ReadyCollect类型的消息
type ReadyCollect struct {
view View
syncInfo SyncInfo
}
func NewReadyCollect(v View, syncInfo SyncInfo) ReadyCollect {
return ReadyCollect{v, syncInfo}
}
func (rc ReadyCollect) View() View {
return rc.view
}
func (rc ReadyCollect) SyncInfo() SyncInfo {
return rc.syncInfo
}
// TxId is a unique identifier for a Tx
type TxID string
// 定义创建TxID的方法,使用clientID和sequenceNum唯一确定交易
// "tx"+clientID+","+SequenceNumber作为交易的唯一id
func NewTxID(clientID uint32, sequenceNum uint64) TxID {
return TxID("tx" + strconv.Itoa(int(clientID)) + "+" + strconv.Itoa(int(sequenceNum)))
}
// RapidFair: 新增virtual view类型,使得optimistic fair order部分有自己的view编号
type VirView uint64
// ToBytes returns the virtual view as bytes.
func (v VirView) ToBytes() []byte {
var viewBytes [8]byte
binary.LittleEndian.PutUint64(viewBytes[:], uint64(v))
return viewBytes[:]
}
// 设计数据结构记录交易序列的哈希Map<ID, Hash(Command)> (Command=txSeq)
type TxSeqHash map[ID]Hash
func (txh TxSeqHash) ToBytes() []byte {
txhB, err := json.Marshal(txh)
if err != nil {
fmt.Printf("[TXList.ToBytes()]: TxSeq serialization error: err=%v", err)
}
return txhB
}
// RapidFair: baseline 实现PreNotify数据类型
type PreNotify struct {
virview View
txSeqHash TxSeqHash
}
func NewPreNofity(v View, tm TxSeqHash) PreNotify {
return PreNotify{v, tm}
}
func (pn PreNotify) VirView() View {
return pn.virview
}
func (pn PreNotify) TxSeqHash() TxSeqHash {
return pn.txSeqHash
}
// 创建TXList类型,存储交易序列类型的数据,包括replica发送的txSeq, updateSeq
// Command反序列化后的类型为,注意RapidFair中, TXList的Data=Hash(txData)
/* message Command {
uint32 ClientID = 1;
uint64 SequenceNumber = 2;
bytes Data = 3;
}*/
type TXList map[ID]Command
func NewTXList() TXList {
return make(TXList)
}
// 使用json的方法序列化交易列表为[]byte
func (txl TXList) ToBytes() []byte {
txlB, err := json.Marshal(txl)
if err != nil {
fmt.Printf("[TXList.ToBytes()]: TxSeq serialization error: err=%v", err)
}
return txlB
}
// 构建MultiCollect类型
type MultiCollect struct {
virView View
txSeq Command
pc PartialCert
}
func NewMultiCollect(v View, txSeq Command, pc PartialCert) MultiCollect {
return MultiCollect{v, txSeq, pc}
}
func (mc MultiCollect) VirView() View {
return mc.virView
}
func (mc MultiCollect) TxSeq() Command {
return mc.txSeq
}
func (mc MultiCollect) PartialCert() PartialCert {
return mc.pc
}
// RapidFair END