-
Notifications
You must be signed in to change notification settings - Fork 178
/
chunkDataPackRequest.go
44 lines (36 loc) · 2.02 KB
/
chunkDataPackRequest.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
package verification
import (
"github.com/onflow/flow-go/model/flow"
"github.com/onflow/flow-go/model/flow/filter"
)
// ChunkDataPackRequest is an internal data structure in fetcher engine that is passed between the engine
// and requester module. It conveys required information for requesting a chunk data pack.
type ChunkDataPackRequest struct {
ChunkID flow.Identifier
Height uint64 // block height of execution result of the chunk, used to drop chunk requests of sealed heights.
Agrees flow.IdentifierList // execution node ids that generated the result of chunk.
Disagrees flow.IdentifierList // execution node ids that generated a conflicting result with result of chunk.
Targets flow.IdentityList // list of all execution nodes identity at the block height of this chunk (including non-responders).
}
func (c ChunkDataPackRequest) ID() flow.Identifier {
return c.ChunkID
}
func (c ChunkDataPackRequest) Checksum() flow.Identifier {
return c.ChunkID
}
type ChunkDataPackRequestList []*ChunkDataPackRequest
// SampleTargets returns identifier of execution nodes that can be asked for the chunk data pack, based on
// the agree and disagree execution nodes of the chunk data pack request.
func (c ChunkDataPackRequest) SampleTargets(count int) flow.IdentifierList {
// if there are enough receipts produced the same result (agrees), we sample from them.
if len(c.Agrees) >= count {
return c.Targets.Filter(filter.HasNodeID(c.Agrees...)).Sample(uint(count)).NodeIDs()
}
// since there is at least one agree, then usually, we just need `count - 1` extra nodes as backup.
// We pick these extra nodes randomly from the rest nodes who we haven't received its receipt.
// In the case where all other execution nodes has produced different results, then we will only
// fetch from the one produced the same result (the only agree)
need := uint(count - len(c.Agrees))
nonResponders := c.Targets.Filter(filter.Not(filter.HasNodeID(c.Disagrees...))).Sample(need).NodeIDs()
return append(c.Agrees, nonResponders...)
}