forked from vitessio/vitess
-
Notifications
You must be signed in to change notification settings - Fork 1
/
result.go
160 lines (136 loc) · 3.96 KB
/
result.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
package throttler
import (
"bytes"
"fmt"
"sync"
"text/template"
"time"
"github.com/youtube/vitess/go/vt/topo/topoproto"
)
type rateChange string
const (
increasedRate rateChange = "increased"
decreasedRate = "decreased"
unchangedRate = "not changed"
)
type goodOrBadRate string
const (
goodRate = "good"
badRate = "bad"
ignoredRate = "ignored"
)
var resultStringTemplate = template.Must(template.New("result.String()").Parse(
`rate was: {{.RateChange}} from: {{.OldRate}} to: {{.NewRate}}
alias: {{.Alias}} lag: {{.LagRecordNow.Stats.SecondsBehindMaster}}s
last change: {{.TimeSinceLastRateChange}} rate: {{.CurrentRate}} good/bad? {{.GoodOrBad}} skipped b/c: {{.MemorySkipReason}} good/bad: {{.HighestGood}}/{{.LowestBad}}
state (old/tested/new): {{.OldState}}/{{.TestedState}}/{{.NewState}}
lag before: {{.LagBefore}} ({{.AgeOfBeforeLag}} ago) rates (master/slave): {{.MasterRate}}/{{.GuessedSlaveRate}} backlog (old/new): {{.GuessedSlaveBacklogOld}}/{{.GuessedSlaveBacklogNew}}
reason: {{.Reason}}`))
// result is generated by the MaxReplicationLag module for each processed
// "replicationLagRecord".
// It captures the details and the decision of the processing.
type result struct {
Now time.Time
RateChange rateChange
lastRateChange time.Time
OldState state
TestedState state
NewState state
OldRate int64
NewRate int64
Reason string
CurrentRate int64
GoodOrBad goodOrBadRate
MemorySkipReason string
HighestGood int64
LowestBad int64
LagRecordNow replicationLagRecord
LagRecordBefore replicationLagRecord
MasterRate int64
GuessedSlaveRate int64
GuessedSlaveBacklogOld int
GuessedSlaveBacklogNew int
}
func (r result) String() string {
var b bytes.Buffer
if err := resultStringTemplate.Execute(&b, r); err != nil {
panic(fmt.Sprintf("failed to Execute() template: %v", err))
}
return b.String()
}
func (r result) Alias() string {
return topoproto.TabletAliasString(r.LagRecordNow.Tablet.Alias)
}
func (r result) TimeSinceLastRateChange() string {
if r.lastRateChange.IsZero() {
return "n/a"
}
return fmt.Sprintf("%.1fs", r.Now.Sub(r.lastRateChange).Seconds())
}
func (r result) LagBefore() string {
if r.LagRecordBefore.isZero() {
return "n/a"
}
return fmt.Sprintf("%ds", r.LagRecordBefore.Stats.SecondsBehindMaster)
}
func (r result) AgeOfBeforeLag() string {
if r.LagRecordBefore.isZero() {
return "n/a"
}
return fmt.Sprintf("%.1fs", r.LagRecordNow.time.Sub(r.LagRecordBefore.time).Seconds())
}
// resultRing implements a ring buffer for "result" instances.
type resultRing struct {
// mu guards the fields below.
mu sync.Mutex
// position holds the index of the *next* result in the ring.
position int
// wrapped becomes true when the ring buffer "wrapped" at least once and we
// started reusing entries.
wrapped bool
// values is the underlying ring buffer.
values []result
}
// newResultRing creates a new resultRing.
func newResultRing(capacity int) *resultRing {
return &resultRing{
values: make([]result, capacity),
}
}
// add inserts a new result into the ring buffer.
func (rr *resultRing) add(r result) {
rr.mu.Lock()
defer rr.mu.Unlock()
rr.values[rr.position] = r
rr.position++
if rr.position == len(rr.values) {
rr.position = 0
rr.wrapped = true
}
}
// latestValues returns all values of the buffer. Entries are sorted in reverse
// chronological order i.e. newer items come first.
func (rr *resultRing) latestValues() []result {
rr.mu.Lock()
defer rr.mu.Unlock()
start := rr.position - 1
if start == -1 {
// Current position is at the end.
start = len(rr.values) - 1
}
count := len(rr.values)
if !rr.wrapped {
count = rr.position
}
results := make([]result, count)
for i := 0; i < count; i++ {
pos := start - i
if pos < 0 {
// We started in the middle of the array and need to wrap around at the
// beginning of it.
pos += count
}
results[i] = rr.values[pos%count]
}
return results
}