-
Notifications
You must be signed in to change notification settings - Fork 209
Expand file tree
/
Copy pathterminals.go
More file actions
89 lines (75 loc) · 3.56 KB
/
terminals.go
File metadata and controls
89 lines (75 loc) · 3.56 KB
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
package fork
import (
"fmt"
"github.com/onflow/flow-go/model/flow"
"github.com/onflow/flow-go/storage"
)
// IncludingBlock returns a Terminal implementation where we explicitly
// specify the ID of the lowest block that should be visited
type IncludingBlock flow.Identifier
// LowestHeightToVisit computes the height of the lowest block that should be visited
func (t IncludingBlock) LowestHeightToVisit(headers storage.Headers) (uint64, error) {
terminalHeader, err := headers.ByBlockID(flow.Identifier(t))
if err != nil {
return 0, fmt.Errorf("failed to retrieve header of terminal block %x: %w", flow.Identifier(t), err)
}
return terminalHeader.Height, nil
}
// ConfirmTerminalReached is a self-consistency check that the lowest visited block is
// in fact the expected terminal.
func (t IncludingBlock) ConfirmTerminalReached(headers storage.Headers, lowestVisitedBlock *flow.Header) error {
if lowestVisitedBlock.ID() != flow.Identifier(t) {
return fmt.Errorf("last visited block has ID %x but expecting %x", lowestVisitedBlock.ID(), flow.Identifier(t))
}
return nil
}
// ExcludingBlock returns a Terminal implementation where we explicitly
// specify the ID of the lowest block that should _not_ be visited anymore
type ExcludingBlock flow.Identifier
// LowestHeightToVisit computes the height of the lowest block that should be visited
func (t ExcludingBlock) LowestHeightToVisit(headers storage.Headers) (uint64, error) {
id := flow.Identifier(t)
terminalHeader, err := headers.ByBlockID(id)
if err != nil {
return 0, fmt.Errorf("failed to retrieve header of terminal block %x: %w", id, err)
}
return terminalHeader.Height + 1, nil
}
// ConfirmTerminalReached is a self-consistency check that the lowest visited block is
// in fact the expected terminal.
func (t ExcludingBlock) ConfirmTerminalReached(headers storage.Headers, lowestVisitedBlock *flow.Header) error {
if lowestVisitedBlock.ParentID != flow.Identifier(t) {
return fmt.Errorf("parent of last visited block has ID %x but expecting %x", lowestVisitedBlock.ParentID, flow.Identifier(t))
}
return nil
}
// IncludingHeight returns a Terminal implementation where we
// specify the height of the lowest block that should be visited
type IncludingHeight uint64
// LowestHeightToVisit computes the height of the lowest block that should be visited
func (t IncludingHeight) LowestHeightToVisit(storage.Headers) (uint64, error) {
return uint64(t), nil
}
// ConfirmTerminalReached is a self-consistency check that the lowest visited block is
// in fact the expected terminal.
func (t IncludingHeight) ConfirmTerminalReached(headers storage.Headers, lowestVisitedBlock *flow.Header) error {
if lowestVisitedBlock.Height != uint64(t) {
return fmt.Errorf("expecting terminal block with height %d but got %d", uint64(t), lowestVisitedBlock.Height)
}
return nil
}
// ExcludingHeight returns a Terminal implementation where we
// specify the Height of the lowest block that should _not_ be visited anymore
type ExcludingHeight uint64
// LowestHeightToVisit computes the height of the lowest block that should be visited
func (t ExcludingHeight) LowestHeightToVisit(storage.Headers) (uint64, error) {
return uint64(t) + 1, nil
}
// ConfirmTerminalReached is a self-consistency check that the lowest visited block is
// in fact the expected terminal.
func (t ExcludingHeight) ConfirmTerminalReached(headers storage.Headers, lowestVisitedBlock *flow.Header) error {
if lowestVisitedBlock.Height != uint64(t)+1 {
return fmt.Errorf("expecting terminal block with height %d but got %d", uint64(t)+1, lowestVisitedBlock.Height)
}
return nil
}