-
Notifications
You must be signed in to change notification settings - Fork 178
/
blocktimer.go
77 lines (66 loc) · 2.82 KB
/
blocktimer.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
package blocktimer
import (
"fmt"
"time"
"github.com/onflow/flow-go/state/protocol"
)
// Is a functor that generates a current timestamp, usually it's just time.Now().
// Used to make testing easier
type timestampGenerator = func() time.Time
// BlockTimestamp is a helper structure that performs building and validation of valid
// timestamp for blocks that are generated by block builder and checked by hotstuff event loop.
// Let τ be the time stamp of the parent block and t be the current clock time of the proposer that is building the child block
// An honest proposer sets the Timestamp of its proposal according to the following rule:
// if t is within the interval [τ + minInterval, τ + maxInterval], then the proposer sets Timestamp := t
// otherwise, the proposer chooses the time stamp from the interval that is closest to its current time t, i.e.
// if t < τ + minInterval, the proposer sets Timestamp := τ + minInterval
// if τ + maxInterval < t, the proposer sets Timestamp := τ + maxInterval
type BlockTimestamp struct {
minInterval time.Duration
maxInterval time.Duration
generator timestampGenerator
}
var DefaultBlockTimer = NewNoopBlockTimer()
// NewBlockTimer creates new block timer with specific intervals and time.Now as generator
func NewBlockTimer(minInterval, maxInterval time.Duration) (*BlockTimestamp, error) {
if minInterval >= maxInterval {
return nil, fmt.Errorf("invariant minInterval < maxInterval is not satisfied, %d >= %d", minInterval, maxInterval)
}
if minInterval <= 0 {
return nil, fmt.Errorf("invariant minInterval > 0 it not satisifed")
}
return &BlockTimestamp{
minInterval: minInterval,
maxInterval: maxInterval,
generator: func() time.Time { return time.Now().UTC() },
}, nil
}
// Build generates a timestamp based on definition of valid timestamp.
func (b BlockTimestamp) Build(parentTimestamp time.Time) time.Time {
// calculate the timestamp and cutoffs
timestamp := b.generator()
from := parentTimestamp.Add(b.minInterval)
to := parentTimestamp.Add(b.maxInterval)
// adjust timestamp if outside of cutoffs
if timestamp.Before(from) {
timestamp = from
}
if timestamp.After(to) {
timestamp = to
}
return timestamp
}
// Validate accepts parent and current timestamps and checks if current timestamp satisfies
// definition of valid timestamp.
// Timestamp is valid if: Timestamp ∈ [τ + minInterval, τ + maxInterval]
// Returns:
// * model.ErrInvalidBlockTimestamp - timestamp is invalid
// * nil - success
func (b BlockTimestamp) Validate(parentTimestamp, currentTimestamp time.Time) error {
from := parentTimestamp.Add(b.minInterval)
to := parentTimestamp.Add(b.maxInterval)
if currentTimestamp.Before(from) || currentTimestamp.After(to) {
return protocol.NewInvalidBlockTimestamp("timestamp %v is not within interval [%v; %v]", currentTimestamp, from, to)
}
return nil
}