-
Notifications
You must be signed in to change notification settings - Fork 3.8k
/
Copy pathdecider.go
395 lines (352 loc) · 14 KB
/
decider.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
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
// Copyright 2019 The Cockroach Authors.
//
// Use of this software is governed by the Business Source License
// included in the file licenses/BSL.txt.
//
// As of the Change Date specified in that file, in accordance with
// the Business Source License, use of this software will be governed
// by the Apache License, Version 2.0, included in the file
// licenses/APL.txt.
// TODO(tbg): rename this package. `lbsplit`?
package split
import (
"context"
"fmt"
"time"
"github.com/cockroachdb/cockroach/pkg/keys"
"github.com/cockroachdb/cockroach/pkg/roachpb"
"github.com/cockroachdb/cockroach/pkg/util/log"
"github.com/cockroachdb/cockroach/pkg/util/metric"
"github.com/cockroachdb/cockroach/pkg/util/syncutil"
"golang.org/x/exp/rand"
)
const minSplitSuggestionInterval = time.Minute
const minNoSplitKeyLoggingMetricsInterval = time.Minute
const minQueriesPerSecondSampleDuration = time.Second
// A Decider collects measurements about the activity (measured in qps) on a
// Replica and, assuming that qps thresholds are exceeded, tries to determine a
// split key that would approximately result in halving the load on each of the
// resultant ranges. Similarly, these measurements are used to determine when a
// range is serving sufficiently little load, such that it should be allowed to
// merge with its left or right hand neighbor.
//
// Operations should call `Record` with a current timestamp. Operation counts
// are aggregated over a second and a QPS is computed.
//
// If the QPS is above a threshold, a split finder is instantiated and the spans
// supplied to Record are sampled for a duration (on the order of ten seconds).
// Assuming that load consistently remains over threshold, and the workload
// touches a diverse enough set of keys to benefit from a split, sampling will
// eventually instruct a caller of Record to carry out a split. When the split
// is initiated, it can obtain the suggested split point from MaybeSplitKey
// (which may have disappeared either due to a drop in qps or a change in the
// workload).
//
// These second-long QPS samples are also aggregated together to track the
// maximum historical QPS over a configurable retention period. This maximum QPS
// measurement, which is accessible through the MaxQPS method, can be used to
// prevent load-based splits from being merged away until the resulting ranges
// have consistently remained below a certain QPS threshold for a sufficiently
// long period of time.
// LoadSplitterMetrics consists of metrics for load-based splitter split key.
type LoadSplitterMetrics struct {
PopularKeyCount *metric.Counter
NoSplitKeyCount *metric.Counter
}
// Decider tracks the latest QPS and if certain conditions are met, records
// incoming requests to find potential split keys and checks if sampled
// candidate split keys satisfy certain requirements.
type Decider struct {
randSource *rand.Rand
qpsThreshold func() float64 // supplied to Init
qpsRetention func() time.Duration // supplied to Init
loadSplitterMetrics *LoadSplitterMetrics // supplied to Init
mu struct {
syncutil.Mutex
// Fields tracking the current qps sample.
lastQPSRollover time.Time // most recent time recorded by requests.
lastQPS float64 // last reqs/s rate as of lastQPSRollover
count int64 // number of requests recorded since last rollover
// Fields tracking historical qps samples.
maxQPS maxQPSTracker
// Fields tracking split key suggestions.
splitFinder *WeightedFinder // populated when engaged or decided
lastSplitSuggestion time.Time // last stipulation to client to carry out split
// Fields tracking logging / metrics around load-based splitter split key.
lastNoSplitKeyLoggingMetrics time.Time
}
}
// Init initializes a Decider (which is assumed to be zero). The signature allows
// embedding the Decider into a larger struct outside of the scope of this package
// without incurring a pointer reference. This is relevant since many Deciders
// may exist in the system at any given point in time.
func Init(
lbs *Decider,
randSource *rand.Rand,
qpsThreshold func() float64,
qpsRetention func() time.Duration,
loadSplitterMetrics *LoadSplitterMetrics,
) {
if randSource == nil {
randSource = rand.New(rand.NewSource(2022))
}
lbs.randSource = randSource
lbs.qpsThreshold = qpsThreshold
lbs.qpsRetention = qpsRetention
lbs.loadSplitterMetrics = loadSplitterMetrics
}
// Record notifies the Decider that 'n' operations are being carried out which
// operate on the span returned by the supplied method. The closure will only
// be called when necessary, that is, when the Decider is considering a split
// and is sampling key spans to determine a suitable split point.
//
// If the returned boolean is true, a split key is available (though it may
// disappear as more keys are sampled) and should be initiated by the caller,
// which can call MaybeSplitKey to retrieve the suggested key.
func (d *Decider) Record(ctx context.Context, now time.Time, n int, span func() roachpb.Span) bool {
d.mu.Lock()
defer d.mu.Unlock()
return d.recordLocked(ctx, now, n, span)
}
func (d *Decider) recordLocked(
ctx context.Context, now time.Time, n int, span func() roachpb.Span,
) bool {
d.mu.count += int64(n)
// First compute requests per second since the last check.
if d.mu.lastQPSRollover.IsZero() {
d.mu.lastQPSRollover = now
}
elapsedSinceLastQPS := now.Sub(d.mu.lastQPSRollover)
if elapsedSinceLastQPS >= minQueriesPerSecondSampleDuration {
// Update the latest QPS and reset the time and request counter.
d.mu.lastQPS = (float64(d.mu.count) / float64(elapsedSinceLastQPS)) * 1e9
d.mu.lastQPSRollover = now
d.mu.count = 0
// Record the latest QPS sample in the historical tracker.
d.mu.maxQPS.record(now, d.qpsRetention(), d.mu.lastQPS)
// If the QPS for the range exceeds the threshold, start actively
// tracking potential for splitting this range based on load.
// This tracking will begin by initiating a splitFinder so it can
// begin to Record requests so it can find a split point. If a
// splitFinder already exists, we check if a split point is ready
// to be used.
if d.mu.lastQPS >= d.qpsThreshold() {
if d.mu.splitFinder == nil {
d.mu.splitFinder = NewWeightedFinder(now, d.randSource)
}
} else {
d.mu.splitFinder = nil
}
}
if d.mu.splitFinder != nil && n != 0 {
s := span()
if s.Key != nil {
d.mu.splitFinder.Record(span(), 1)
}
if d.mu.splitFinder.Ready(now) {
if d.mu.splitFinder.Key() != nil {
if now.Sub(d.mu.lastSplitSuggestion) > minSplitSuggestionInterval {
d.mu.lastSplitSuggestion = now
return true
}
} else {
if now.Sub(d.mu.lastNoSplitKeyLoggingMetrics) > minNoSplitKeyLoggingMetricsInterval {
d.mu.lastNoSplitKeyLoggingMetrics = now
noSplitKeyCauseLogMsg := d.mu.splitFinder.NoSplitKeyCauseLogMsg()
if noSplitKeyCauseLogMsg != "" {
popularKeyFrequency := d.mu.splitFinder.PopularKeyFrequency()
noSplitKeyCauseLogMsg += fmt.Sprintf(", most popular key occurs in %d%% of samples", int(popularKeyFrequency*100))
log.KvDistribution.Infof(ctx, "%s", noSplitKeyCauseLogMsg)
if popularKeyFrequency >= splitKeyThreshold {
d.loadSplitterMetrics.PopularKeyCount.Inc(1)
}
d.loadSplitterMetrics.NoSplitKeyCount.Inc(1)
}
}
}
}
}
return false
}
// RecordMax adds a QPS measurement directly into the Decider's historical QPS
// tracker. The QPS sample is considered to have been captured at the provided
// time.
func (d *Decider) RecordMax(now time.Time, qps float64) {
d.mu.Lock()
defer d.mu.Unlock()
d.mu.maxQPS.record(now, d.qpsRetention(), qps)
}
// LastQPS returns the most recent QPS measurement.
func (d *Decider) LastQPS(ctx context.Context, now time.Time) float64 {
d.mu.Lock()
defer d.mu.Unlock()
d.recordLocked(ctx, now, 0, nil) // force QPS computation
return d.mu.lastQPS
}
// MaxQPS returns the maximum QPS measurement recorded over the retention
// period. If the Decider has not been recording for a full retention period,
// the method returns false.
func (d *Decider) MaxQPS(ctx context.Context, now time.Time) (float64, bool) {
d.mu.Lock()
defer d.mu.Unlock()
d.recordLocked(ctx, now, 0, nil) // force QPS computation
return d.mu.maxQPS.maxQPS(now, d.qpsRetention())
}
// MaybeSplitKey returns a key to perform a split at. The return value will be
// nil if either the Decider hasn't decided that a split should be carried out
// or if it wasn't able to determine a suitable split key.
//
// It is legal to call MaybeSplitKey at any time.
func (d *Decider) MaybeSplitKey(ctx context.Context, now time.Time) roachpb.Key {
var key roachpb.Key
d.mu.Lock()
defer d.mu.Unlock()
d.recordLocked(ctx, now, 0, nil)
if d.mu.splitFinder != nil && d.mu.splitFinder.Ready(now) {
// We've found a key to split at. This key might be in the middle of a
// SQL row. If we fail to rectify that, we'll cause SQL crashes:
//
// https://github.com/cockroachdb/cockroach/pull/42056
//
// While the behavior at the SQL level is arguably bad and should be
// fixed, splitting between column families is also never a good idea
// for performance in general. So, if the split key is, say
//
// /Table/51/52/53/54/55/9/1
//
// then we want to split instead at
//
// /Table/51/52/53/54/55
//
// (see TestDeciderCallsEnsureSafeSplitKey).
//
// The key found here isn't guaranteed to be a valid SQL column family
// key. This is because the keys are sampled from StartKey of requests
// hitting this replica. Ranged operations may well wish to exclude the
// start point by calling .Next() or may span multiple ranges, and so
// such a key may end up being passed to EnsureSafeSplitKey here.
//
// We take the risk that the result may sometimes not be a good split
// point (or even in this range).
//
// Note that we ignore EnsureSafeSplitKey when it returns an error since
// that error only tells us that this key couldn't possibly be a SQL
// key. This is more common than one might think since SQL issues plenty
// of scans over all column families, meaning that we'll frequently find
// a key that has no column family suffix and thus errors out in
// EnsureSafeSplitKey.
key = d.mu.splitFinder.Key()
if safeKey, err := keys.EnsureSafeSplitKey(key); err == nil {
key = safeKey
}
}
return key
}
// Reset deactivates any current attempt at determining a split key. The method
// also discards any historical QPS tracking information.
func (d *Decider) Reset(now time.Time) {
d.mu.Lock()
defer d.mu.Unlock()
d.mu.lastQPSRollover = time.Time{}
d.mu.lastQPS = 0
d.mu.count = 0
d.mu.maxQPS.reset(now, d.qpsRetention())
d.mu.splitFinder = nil
d.mu.lastSplitSuggestion = time.Time{}
d.mu.lastNoSplitKeyLoggingMetrics = time.Time{}
}
// maxQPSTracker collects a series of queries-per-second measurement samples and
// tracks the maximum observed over a period of time.
//
// The tracker internally uses a set of time windows in order to age out old
// partial aggregations without having to do hard resets. The `windows` array is
// a circular buffer of the last N windows of stats. We rotate through the
// circular buffer every so often as determined by `minRetention`.
//
// The tracker can be queried through its `maxQPS` method, which returns the
// maximum of all queries-per-second samples recorded over the retention period.
// If the tracker has not been recording for a full retention period, then the
// method returns false.
//
// The zero-value of a maxQPSTracker can be used immediately.
type maxQPSTracker struct {
windows [6]float64
curIdx int
curStart time.Time
lastReset time.Time
minRetention time.Duration
}
// record adds the qps sample to the tracker.
func (t *maxQPSTracker) record(now time.Time, minRetention time.Duration, qps float64) {
t.maybeReset(now, minRetention)
t.maybeRotate(now)
t.windows[t.curIdx] = max(t.windows[t.curIdx], qps)
}
// reset clears the tracker. maxQPS will begin returning false until a full
// minRetention period has elapsed.
func (t *maxQPSTracker) reset(now time.Time, minRetention time.Duration) {
if minRetention <= 0 {
panic("minRetention must be positive")
}
t.windows = [6]float64{}
t.curIdx = 0
t.curStart = now
t.lastReset = now
t.minRetention = minRetention
}
func (t *maxQPSTracker) maybeReset(now time.Time, minRetention time.Duration) {
// If the retention period changes, simply reset the entire tracker. Merging
// or splitting windows would be a difficult task and could lead to samples
// either not being retained for long-enough, or being retained for too long.
// Resetting indicates to maxQPS that a new retention period needs to be
// measured before accurate results can be returned.
if minRetention != t.minRetention {
t.reset(now, minRetention)
}
}
func (t *maxQPSTracker) maybeRotate(now time.Time) {
sinceLastRotate := now.Sub(t.curStart)
windowWidth := t.windowWidth()
if sinceLastRotate < windowWidth {
// Use the existing window.
return
}
shift := int(sinceLastRotate / windowWidth)
if shift >= len(t.windows) {
// Clear all windows. We had a long period of inactivity.
t.windows = [6]float64{}
t.curIdx = 0
t.curStart = now
return
}
for i := 0; i < shift; i++ {
t.curIdx = (t.curIdx + 1) % len(t.windows)
t.curStart = t.curStart.Add(windowWidth)
t.windows[t.curIdx] = 0
}
}
// maxQPS returns the maximum queries-per-second samples recorded over the last
// retention period. If the tracker has not been recording for a full retention
// period, then the method returns false.
func (t *maxQPSTracker) maxQPS(now time.Time, minRetention time.Duration) (float64, bool) {
t.record(now, minRetention, 0) // expire samples, if necessary
if now.Sub(t.lastReset) < t.minRetention {
// The tracker has not been recording for long enough.
return 0, false
}
qps := 0.0
for _, v := range t.windows {
qps = max(qps, v)
}
return qps, true
}
func (t *maxQPSTracker) windowWidth() time.Duration {
// NB: -1 because during a rotation, only len(t.windows)-1 windows survive.
return t.minRetention / time.Duration(len(t.windows)-1)
}
func max(a, b float64) float64 {
if a > b {
return a
}
return b
}