-
Notifications
You must be signed in to change notification settings - Fork 180
/
Copy pathblock_signer_decoder.go
71 lines (60 loc) · 2.76 KB
/
block_signer_decoder.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
package signature
import (
"errors"
"fmt"
"github.com/onflow/flow-go/consensus/hotstuff"
"github.com/onflow/flow-go/consensus/hotstuff/model"
"github.com/onflow/flow-go/model/flow"
"github.com/onflow/flow-go/module/signature"
)
// BlockSignerDecoder is a wrapper around the `hotstuff.DynamicCommittee`, which implements
// the auxilluary logic for de-coding signer indices of a block (header) to full node IDs
type BlockSignerDecoder struct {
hotstuff.DynamicCommittee
}
func NewBlockSignerDecoder(committee hotstuff.DynamicCommittee) *BlockSignerDecoder {
return &BlockSignerDecoder{committee}
}
var _ hotstuff.BlockSignerDecoder = (*BlockSignerDecoder)(nil)
// DecodeSignerIDs decodes the signer indices from the given block header into full node IDs.
// Note: A block header contains a quorum certificate for its parent, which proves that the
// consensus committee has reached agreement on validity of parent block. Consequently, the
// returned IdentifierList contains the consensus participants that signed the parent block.
// Expected Error returns during normal operations:
// - signature.InvalidSignerIndicesError if signer indices included in the header do
// not encode a valid subset of the consensus committee
func (b *BlockSignerDecoder) DecodeSignerIDs(header *flow.Header) (flow.IdentifierList, error) {
// root block does not have signer indices
if header.ParentVoterIndices == nil && header.View == 0 {
return []flow.Identifier{}, nil
}
// we will use IdentitiesByEpoch since it's a faster call and avoids DB lookup
members, err := b.IdentitiesByEpoch(header.ParentView)
if err != nil {
if errors.Is(err, model.ErrViewForUnknownEpoch) {
// possibly, we request epoch which is far behind in the past, in this case we won't have it in cache.
// try asking by parent ID
members, err = b.IdentitiesByBlock(header.ParentID)
if err != nil {
return nil, fmt.Errorf("could not retrieve identities for block %x with QC view %d for parent %x: %w",
header.ID(), header.ParentView, header.ParentID, err)
}
} else {
return nil, fmt.Errorf("unexpected error retrieving identities for block %v: %w", header.ID(), err)
}
}
signerIDs, err := signature.DecodeSignerIndicesToIdentifiers(members.NodeIDs(), header.ParentVoterIndices)
if err != nil {
return nil, fmt.Errorf("could not decode signer indices for block %v: %w", header.ID(), err)
}
return signerIDs, nil
}
// NoopBlockSignerDecoder does not decode any signer indices and consistently returns
// nil for the signing node IDs (auxiliary data)
type NoopBlockSignerDecoder struct{}
func NewNoopBlockSignerDecoder() *NoopBlockSignerDecoder {
return &NoopBlockSignerDecoder{}
}
func (b *NoopBlockSignerDecoder) DecodeSignerIDs(_ *flow.Header) (flow.IdentifierList, error) {
return nil, nil
}