-
Notifications
You must be signed in to change notification settings - Fork 3.8k
/
replica_proposal_buf.go
1262 lines (1162 loc) · 52.3 KB
/
replica_proposal_buf.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
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
// 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.
package kvserver
import (
"context"
"github.com/cockroachdb/cockroach/pkg/clusterversion"
"sync"
"sync/atomic"
"github.com/cockroachdb/cockroach/pkg/kv/kvserver/closedts/tracker"
"github.com/cockroachdb/cockroach/pkg/kv/kvserver/kvserverpb"
"github.com/cockroachdb/cockroach/pkg/kv/kvserver/raftutil"
"github.com/cockroachdb/cockroach/pkg/roachpb"
"github.com/cockroachdb/cockroach/pkg/settings/cluster"
"github.com/cockroachdb/cockroach/pkg/util/errorutil"
"github.com/cockroachdb/cockroach/pkg/util/hlc"
"github.com/cockroachdb/cockroach/pkg/util/log"
"github.com/cockroachdb/cockroach/pkg/util/protoutil"
"github.com/cockroachdb/errors"
"go.etcd.io/etcd/raft/v3"
"go.etcd.io/etcd/raft/v3/raftpb"
)
// propBuf is a multi-producer, single-consumer buffer for Raft proposals on a
// range. The buffer supports concurrent insertion of proposals.
//
// The proposal buffer also handles the assignment of maximum lease indexes for
// commands. Picking the maximum lease index for commands is done atomically
// with determining the order in which they are inserted into the buffer to
// ensure that lease indexes are not assigned in a different order from that in
// which commands are proposed (and thus likely applied). If this order was to
// get out of sync then some commands would necessarily be rejected beneath Raft
// during application (see checkForcedErr).
//
// The proposal buffer also is in charge of advancing the respective range's
// closed timestamp by assigning closed timestamp to proposals. For this
// purpose, new requests starting evaluation needs to synchronize with the
// proposal buffer (see TrackEvaluatingRequest).
//
// Proposals enter the buffer via Insert() or ReinsertLocked(). They are moved
// into Raft via FlushLockedWithRaftGroup() when the buffer fills up, or during
// the next handleRaftReady iteration, whichever happens earlier. This
// introduces no additional latency into the replication pipeline compared to
// moving them into Raft directly because Raft would not begin replicating the
// proposals until the next handleRaftReady iteration anyway.
//
// propBuf inherits the locking of the proposer that it is bound to during
// initialization. Methods called "...Locked" and "...RLocked" expect the
// corresponding locker() and rlocker() to be held.
type propBuf struct {
p proposer
clock *hlc.Clock
settings *cluster.Settings
// evalTracker tracks currently-evaluating requests, making sure that
// proposals coming out of the propBuf don't carry closed timestamps below
// currently-evaluating requests.
evalTracker tracker.Tracker
full sync.Cond
// arr contains the buffered proposals.
arr propBufArray
// allocatedIdx is the next index into propBufArray to allocate. Accessed
// atomically.
allocatedIdx int64
// assignedLAI represents the highest LAI that was assigned to a proposal.
// This is set at the same time as assignedClosedTimestamp.
assignedLAI uint64
// assignedClosedTimestamp is the largest "closed timestamp" - i.e. the
// largest timestamp that was communicated to other replicas as closed,
// representing a promise that this leaseholder will not evaluate writes with
// timestamp <= assignedClosedTimestamp anymore. It is set when proposals are
// flushed from the buffer, and also by the side-transport which closes
// timestamps out of band.
//
// Note that this field is not used by the local replica (or by anybody)
// directly to decide whether follower reads can be served. See
// ReplicaState.closed_timestamp.
//
// This field can be read under the proposer's read lock, and written to under
// the write lock.
assignedClosedTimestamp hlc.Timestamp
// Buffer used to avoid allocations.
scratchFooter kvserverpb.RaftCommandFooter
testing struct {
// leaseIndexFilter can be used by tests to override the max lease index
// assigned to a proposal by returning a non-zero lease index.
leaseIndexFilter func(*ProposalData) (indexOverride uint64)
// insertFilter allows tests to inject errors at Insert() time.
insertFilter func(*ProposalData) error
// submitProposalFilter can be used by tests to observe and optionally
// drop Raft proposals before they are handed to etcd/raft to begin the
// process of replication. Dropped proposals are still eligible to be
// reproposed due to ticks.
submitProposalFilter func(*ProposalData) (drop bool, err error)
// allowLeaseProposalWhenNotLeader, if set, makes the proposal buffer allow
// lease request proposals even when the replica inserting that proposal is
// not the Raft leader. This can be used in tests to allow a replica to
// acquire a lease without first moving the Raft leadership to it (e.g. it
// allows tests to expire leases by stopping the old leaseholder's liveness
// heartbeats and then expect other replicas to take the lease without
// worrying about Raft).
allowLeaseProposalWhenNotLeader bool
// allowLeaseTransfersWhenTargetMayNeedSnapshot, if set, makes the proposal
// buffer allow lease request proposals even when the proposer cannot prove
// that the lease transfer target does not need a Raft snapshot.
allowLeaseTransfersWhenTargetMayNeedSnapshot bool
// dontCloseTimestamps inhibits the closing of timestamps.
dontCloseTimestamps bool
}
}
type rangeLeaderInfo struct {
// leaderKnown is set if the local Raft machinery knows who the leader is. If
// not set, all other fields are empty.
leaderKnown bool
// leader represents the Raft group's leader. Not set if leaderKnown is not
// set.
leader roachpb.ReplicaID
// iAmTheLeader is set if the local replica is the leader.
iAmTheLeader bool
// leaderEligibleForLease is set if the leader is known and its type of
// replica allows it to acquire a lease.
leaderEligibleForLease bool
}
// A proposer is an object that uses a propBuf to coordinate Raft proposals.
type proposer interface {
locker() sync.Locker
rlocker() sync.Locker
// The following require the proposer to hold (at least) a shared lock.
getReplicaID() roachpb.ReplicaID
destroyed() destroyStatus
firstIndex() uint64
leaseAppliedIndex() uint64
enqueueUpdateCheck()
closedTimestampTarget() hlc.Timestamp
// The following require the proposer to hold an exclusive lock.
withGroupLocked(func(proposerRaft) error) error
registerProposalLocked(*ProposalData)
leaderStatusRLocked(ctx context.Context, raftGroup proposerRaft) rangeLeaderInfo
ownsValidLeaseRLocked(ctx context.Context, now hlc.ClockTimestamp) bool
// rejectProposalWithRedirectLocked rejects a proposal and redirects the
// proposer to try it on another node. This is used to sometimes reject lease
// acquisitions when another replica is the leader; the intended consequence
// of the rejection is that the request that caused the lease acquisition
// attempt is tried on the leader, at which point it should result in a lease
// acquisition attempt by that node (or, perhaps by then the leader will have
// already gotten a lease and the request can be serviced directly).
rejectProposalWithRedirectLocked(
ctx context.Context,
prop *ProposalData,
redirectTo roachpb.ReplicaID,
)
// rejectProposalWithLeaseTransferRejectedLocked rejects a proposal for a
// lease transfer when the transfer is deemed to be unsafe. The intended
// consequence of the rejection is that the lease transfer attempt will be
// rejected. Higher levels that decide whether or not to attempt a lease
// transfer have weaker versions of the same check, so we don't expect to see
// repeated lease transfer rejections.
rejectProposalWithLeaseTransferRejectedLocked(
ctx context.Context,
prop *ProposalData,
lease *roachpb.Lease,
reason raftutil.ReplicaNeedsSnapshotStatus,
)
// leaseDebugRLocked returns info on the current lease.
leaseDebugRLocked() string
}
// proposerRaft abstracts the propBuf's dependency on *raft.RawNode, to help
// testing.
type proposerRaft interface {
Step(raftpb.Message) error
Status() raft.Status
ProposeConfChange(raftpb.ConfChangeI) error
}
// Init initializes the proposal buffer and binds it to the provided proposer.
func (b *propBuf) Init(
p proposer, tracker tracker.Tracker, clock *hlc.Clock, settings *cluster.Settings,
) {
b.p = p
b.full.L = p.rlocker()
b.clock = clock
b.evalTracker = tracker
b.settings = settings
b.assignedLAI = p.leaseAppliedIndex()
}
// AllocatedIdx returns the highest index that was allocated. This generally
// corresponds to the size of the buffer but, if the buffer is full, the
// allocated index can temporarily be in advance of the size.
func (b *propBuf) AllocatedIdx() int {
return int(atomic.LoadInt64(&b.allocatedIdx))
}
// clearAllocatedIdx resets the allocated index, emptying the buffer. Returns
// the number of elements that were in the buffer.
func (b *propBuf) clearAllocatedIdx() int {
return int(atomic.SwapInt64(&b.allocatedIdx, 0))
}
// incAllocatedIdx allocates a slot into the the buffer that a new proposal can
// be written to. Returns the index of the slot.
func (b *propBuf) incAllocatedIdx() int {
return int(atomic.AddInt64(&b.allocatedIdx, 1)) - 1 // -1 since the index is 0-based
}
// Insert inserts a new command into the proposal buffer to be proposed to the
// proposer's Raft group. The method accepts the Raft command as part of the
// ProposalData struct. ProposalData.encodedCommand is expected to contain a
// partial encoding of the command. That byte slice is expected to contain
// marshaled information for all of the command's fields except for
// MaxLeaseIndex, and ClosedTimestamp. These fields will be assigned later, when
// the buffer is flushed (after the command is sequenced in the buffer). It is
// also expected that the byte slice has sufficient capacity to marshal these
// fields into it.
//
// Insert takes ownership of the supplied token; the caller should tok.Move() it
// into this method. It will be used to untrack the request once it comes out of the
// proposal buffer.
func (b *propBuf) Insert(ctx context.Context, p *ProposalData, tok TrackedRequestToken) error {
defer tok.DoneIfNotMoved(ctx)
// Hold the read lock while inserting into the proposal buffer. Other
// insertion attempts will also grab the read lock, so they can insert
// concurrently. Consumers of the proposal buffer will grab the write lock,
// so they must wait for concurrent insertion attempts to finish.
b.p.rlocker().Lock()
defer b.p.rlocker().Unlock()
if filter := b.testing.insertFilter; filter != nil {
if err := filter(p); err != nil {
return err
}
}
// Update the proposal buffer counter and determine which index we should
// insert at.
idx, err := b.allocateIndex(ctx, false /* wLocked */)
if err != nil {
return err
}
if log.V(4) {
log.Infof(p.ctx, "submitting proposal %x", p.idKey)
}
// Insert the proposal into the buffer's array. The buffer now takes ownership
// of the token.
p.tok = tok.Move(ctx)
b.insertIntoArray(p, idx)
return nil
}
// ReinsertLocked inserts a command that has already passed through the proposal
// buffer back into the buffer to be reproposed at a new Raft log index. Unlike
// Insert, it does not modify the command.
func (b *propBuf) ReinsertLocked(ctx context.Context, p *ProposalData) error {
// Update the proposal buffer counter and determine which index we should
// insert at.
idx, err := b.allocateIndex(ctx, true /* wLocked */)
if err != nil {
return err
}
// Insert the proposal into the buffer's array.
b.insertIntoArray(p, idx)
return nil
}
// allocateIndex allocates a buffer index to be used for storing a proposal. The
// method will repeat the atomic update operation until it is able to
// successfully reserve an index in the array. If an attempt finds that the
// array is full then it may flush the array before trying again.
//
// The method expects that either the proposer's read lock or write lock is
// held. It does not mandate which, but expects the caller to specify using
// the wLocked argument.
func (b *propBuf) allocateIndex(ctx context.Context, wLocked bool) (int, error) {
// Repeatedly attempt to find an open index in the buffer's array.
for {
// NB: We need to check whether the proposer is destroyed before each
// iteration in case the proposer has been destroyed between the initial
// check and the current acquisition of the read lock. Failure to do so
// will leave pending proposals that never get cleared.
if status := b.p.destroyed(); !status.IsAlive() {
return 0, status.err
}
idx := b.incAllocatedIdx()
if idx < b.arr.len() {
// The buffer is not full. Our slot in the array is reserved.
return idx, nil
} else if wLocked {
// The buffer is full and we're holding the exclusive lock. Flush
// the buffer before trying again.
if err := b.flushLocked(ctx); err != nil {
return 0, err
}
} else if idx == b.arr.len() {
// The buffer is full and we were the first request to notice out of
// potentially many requests holding the shared lock and trying to
// insert concurrently. Eagerly attempt to flush the buffer before
// trying again.
if err := b.flushRLocked(ctx); err != nil {
return 0, err
}
} else {
// The buffer is full and we were not the first request to notice
// out of potentially many requests holding the shared lock and
// trying to insert concurrently. Wait for the buffer to be flushed
// by someone else before trying again.
b.full.Wait()
}
}
}
// insertIntoArray inserts the proposal into the proposal buffer's array at the
// specified index. It also schedules a Raft update check if necessary.
func (b *propBuf) insertIntoArray(p *ProposalData, idx int) {
b.arr.asSlice()[idx] = p
if idx == 0 {
// If this is the first proposal in the buffer, schedule a Raft update
// check to inform Raft processing about the new proposal. Everyone else
// can rely on the request that added the first proposal to the buffer
// having already scheduled a Raft update check.
b.p.enqueueUpdateCheck()
}
}
func (b *propBuf) flushRLocked(ctx context.Context) error {
// Upgrade the shared lock to an exclusive lock. After doing so, check again
// whether the proposer has been destroyed. If so, wake up other goroutines
// waiting for the flush.
b.p.rlocker().Unlock()
defer b.p.rlocker().Lock()
b.p.locker().Lock()
defer b.p.locker().Unlock()
if status := b.p.destroyed(); !status.IsAlive() {
b.full.Broadcast()
return status.err
}
return b.flushLocked(ctx)
}
func (b *propBuf) flushLocked(ctx context.Context) error {
return b.p.withGroupLocked(func(raftGroup proposerRaft) error {
_, err := b.FlushLockedWithRaftGroup(ctx, raftGroup)
return err
})
}
// FlushLockedWithRaftGroup flushes the commands from the proposal buffer and
// resets the buffer back to an empty state. Each command is handed off to the
// Raft proposals map, at which point they are owned by the Raft processor.
//
// If raftGroup is non-nil (the common case) then the commands will also be
// proposed to the RawNode. This initiates Raft replication of the commands.
//
// Returns the number of proposals flushed from the proposal buffer, counting
// proposals even if they were dropped and never handed to the RawNode. This
// second part is important, because it ensures that even if we drop a lease
// request by calling rejectProposalWithRedirectLocked, we still inform the
// caller of its presence. This ensures that callers like handleRaftReady
// consider unquiescing and waking the Raft leader, which may be necessary to
// notice the failure of the leader and allow a future lease request through.
func (b *propBuf) FlushLockedWithRaftGroup(
ctx context.Context, raftGroup proposerRaft,
) (int, error) {
// We hold the write lock while reading from and flushing the proposal
// buffer. This ensures that we synchronize with all producers and other
// consumers.
used := b.clearAllocatedIdx()
// Before returning, consider resizing the proposal buffer's array,
// depending on how much of it was used before the current flush.
defer b.arr.adjustSize(used)
if used == 0 {
// The buffer is empty. Nothing to do.
return 0, nil
} else if used > b.arr.len() {
// The buffer is full and at least one writer has tried to allocate
// on top of the full buffer, so notify them to try again.
used = b.arr.len()
defer b.full.Broadcast()
}
// Iterate through the proposals in the buffer and propose them to Raft.
// While doing so, build up batches of entries and submit them to Raft all
// at once. Building up batches of entries and proposing them with a single
// Step can dramatically reduce the number of messages required to commit
// and apply them.
buf := b.arr.asSlice()[:used]
ents := make([]raftpb.Entry, 0, used)
// Compute the closed timestamp target, which will be used to assign a closed
// timestamp to all proposals in this batch.
closedTSTarget := b.p.closedTimestampTarget()
// Remember the first error that we see when proposing the batch. We don't
// immediately return this error because we want to finish clearing out the
// buffer and registering each of the proposals with the proposer, but we
// stop trying to propose commands to raftGroup.
var firstErr error
for i, p := range buf {
if p == nil {
log.Fatalf(ctx, "unexpected nil proposal in buffer")
return 0, nil // unreachable, for linter
}
buf[i] = nil // clear buffer
reproposal := !p.tok.stillTracked()
// Conditionally reject the proposal based on the state of the raft group.
if b.maybeRejectUnsafeProposalLocked(ctx, raftGroup, p) {
p.tok.doneIfNotMovedLocked(ctx)
continue
}
// Raft processing bookkeeping.
b.p.registerProposalLocked(p)
// Exit the tracker.
if !reproposal && p.Request.AppliesTimestampCache() {
// Sanity check that the request is tracked by the evaluation tracker at
// this point. It's supposed to be tracked until the
// doneIfNotMovedLocked() call below.
wts := p.Request.WriteTimestamp()
lb := b.evalTracker.LowerBound(ctx)
if wts.Less(lb) {
wts, lb := wts, lb // copies escape to heap
log.Fatalf(ctx, "%v", errorutil.UnexpectedWithIssueErrorf(72428,
"request writing below tracked lower bound: wts: %s < lb: %s; ba: %s; lease: %s.",
wts, lb, p.Request, b.p.leaseDebugRLocked()))
}
}
p.tok.doneIfNotMovedLocked(ctx)
// If we don't have a raft group or if the raft group has rejected one
// of the proposals, we don't try to propose any more proposals. The
// rest of the proposals will still be registered with the proposer, so
// they will eventually be reproposed.
if raftGroup == nil || firstErr != nil {
continue
}
// Figure out what closed timestamp this command will carry.
//
// If this is a reproposal, we don't reassign the LAI. We also don't
// reassign the closed timestamp: we could, in principle, but we'd have to
// make a copy of the encoded command as to not modify the copy that's
// already stored in the local replica's raft entry cache.
if !reproposal {
lai, closedTimestamp, err := b.allocateLAIAndClosedTimestampLocked(ctx, p, closedTSTarget)
if err != nil {
firstErr = err
continue
}
err = b.marshallLAIAndClosedTimestampToProposalLocked(ctx, p, lai, closedTimestamp)
if err != nil {
firstErr = err
continue
}
}
// Potentially drop the proposal before passing it to etcd/raft, but
// only after performing necessary bookkeeping.
if filter := b.testing.submitProposalFilter; filter != nil {
if drop, err := filter(p); drop || err != nil {
firstErr = err
continue
}
}
// Coordinate proposing the command to etcd/raft.
if crt := p.command.ReplicatedEvalResult.ChangeReplicas; crt != nil {
// Flush any previously batched (non-conf change) proposals to
// preserve the correct ordering or proposals. Later proposals
// will start a new batch.
if err := proposeBatch(raftGroup, b.p.getReplicaID(), ents); err != nil {
firstErr = err
continue
}
ents = ents[len(ents):]
confChangeCtx := kvserverpb.ConfChangeContext{
CommandID: string(p.idKey),
Payload: p.encodedCommand,
}
encodedCtx, err := protoutil.Marshal(&confChangeCtx)
if err != nil {
firstErr = err
continue
}
cc, err := crt.ConfChange(encodedCtx)
if err != nil {
firstErr = err
continue
}
if err := raftGroup.ProposeConfChange(
cc,
); err != nil && !errors.Is(err, raft.ErrProposalDropped) {
// Silently ignore dropped proposals (they were always silently
// ignored prior to the introduction of ErrProposalDropped).
// TODO(bdarnell): Handle ErrProposalDropped better.
// https://github.com/cockroachdb/cockroach/issues/21849
firstErr = err
continue
}
} else {
// Add to the batch of entries that will soon be proposed. It is
// possible that this batching can cause the batched MsgProp to grow
// past the size limit where etcd/raft will drop the entire thing
// (see raft.Config.MaxUncommittedEntriesSize), but that's not a
// concern. This setting is configured to twice the maximum quota in
// the proposal quota pool, so for batching to cause a message to be
// dropped the uncommitted portion of the Raft log would already
// need to be at least as large as the proposal quota size, assuming
// that all in-flight proposals are reproposed in a single batch.
ents = append(ents, raftpb.Entry{
Data: p.encodedCommand,
})
}
}
if firstErr != nil {
return 0, firstErr
}
return used, proposeBatch(raftGroup, b.p.getReplicaID(), ents)
}
// maybeRejectUnsafeProposalLocked conditionally rejects proposals that are
// deemed unsafe, given the current state of the raft group. Requests that may
// be deemed unsafe and rejected at this level are those whose safety has some
// dependency on raft leadership, follower progress, leadership term, commit
// index, or other properties of raft. By rejecting these requests on the
// "flushing" side of the proposal buffer (i.e. while holding the raftMu), we
// can perform the safety checks without risk of the state of the raft group
// changing before the proposal is passed to etcd/raft.
//
// Currently, the request types which may be rejected by this function are:
// - RequestLease when the proposer is not the raft leader (with caveats).
// - TransferLease when the proposer cannot guarantee that the lease transfer
// target does not currently need a Raft snapshot to catch up on its Raft log.
// In such cases, the proposer cannot guarantee that the lease transfer target
// will not need a Raft snapshot to catch up to and apply the lease transfer.
// This requires that the proposer is the raft leader.
//
// The function returns true if the proposal was rejected, and false if not.
// If the proposal was rejected and true is returned, it will have been cleaned
// up (passed to Replica.cleanupFailedProposalLocked) and finished
// (ProposalData.finishApplication called).
func (b *propBuf) maybeRejectUnsafeProposalLocked(
ctx context.Context, raftGroup proposerRaft, p *ProposalData,
) (rejected bool) {
if raftGroup == nil {
// If we do not have a raft group, we won't try to propose this proposal.
// Instead, we will register the proposal so that it can be reproposed later
// with a raft group. Wait until that point to determine whether to reject
// the proposal or not.
return false
}
switch {
case p.Request.IsSingleRequestLeaseRequest():
// Handle an edge case about lease acquisitions: we don't want to forward
// lease acquisitions to another node (which is what happens when we're not
// the leader) because:
// a) if there is a different leader, that leader should acquire the lease
// itself and thus avoid a change of leadership caused by the leaseholder
// and leader being different (Raft leadership follows the lease), and
// b) being a follower, it's possible that this replica is behind in
// applying the log. Thus, there might be another lease in place that this
// follower doesn't know about, in which case the lease we're proposing here
// would be rejected. Not only would proposing such a lease be wasted work,
// but we're trying to protect against pathological cases where it takes a
// long time for this follower to catch up (for example because it's waiting
// for a snapshot, and the snapshot is queued behind many other snapshots).
// In such a case, we don't want all requests arriving at this node to be
// blocked on this lease acquisition (which is very likely to eventually
// fail anyway).
//
// Thus, we do one of two things:
// - if the leader is known, we reject this proposal and make sure the
// request that needed the lease is redirected to the leaseholder;
// - if the leader is not known, we don't do anything special here to
// terminate the proposal, but we know that Raft will reject it with a
// ErrProposalDropped. We'll eventually re-propose it once a leader is
// known, at which point it will either go through or be rejected based on
// whether or not it is this replica that became the leader.
//
// A special case is when the leader is known, but is ineligible to get the
// lease. In that case, we have no choice but to continue with the proposal.
//
// Lease extensions for a currently held lease always go through, to
// keep the lease alive until the normal lease transfer mechanism can
// colocate it with the leader.
li := b.leaderStatusRLocked(ctx, raftGroup)
if li.iAmTheLeader {
return false
}
leaderKnownAndEligible := li.leaderKnown && li.leaderEligibleForLease
ownsCurrentLease := b.p.ownsValidLeaseRLocked(ctx, b.clock.NowAsClockTimestamp())
if leaderKnownAndEligible && !ownsCurrentLease && !b.testing.allowLeaseProposalWhenNotLeader {
log.VEventf(ctx, 2, "not proposing lease acquisition because we're not the leader; replica %d is",
li.leader)
b.p.rejectProposalWithRedirectLocked(ctx, p, li.leader)
return true
}
// If the leader is not known, or if it is known but it's ineligible
// for the lease, continue with the proposal as explained above. We
// also send lease extensions for an existing leaseholder.
if ownsCurrentLease {
log.VEventf(ctx, 2, "proposing lease extension even though we're not the leader; we hold the current lease")
} else if !li.leaderKnown {
log.VEventf(ctx, 2, "proposing lease acquisition even though we're not the leader; the leader is unknown")
} else {
log.VEventf(ctx, 2, "proposing lease acquisition even though we're not the leader; the leader is ineligible")
}
return false
case p.Request.IsSingleTransferLeaseRequest():
// When performing a lease transfer, the outgoing leaseholder revokes its
// lease before proposing the lease transfer request, meaning that it
// promises to stop using the previous lease to serve reads or writes. The
// lease transfer request is then proposed and committed to the Raft log, at
// which point the new lease officially becomes active. However, this new
// lease is not usable until the incoming leaseholder applies the Raft entry
// that contains the lease transfer and notices that it is now the
// leaseholder for the range.
//
// The effect of this handoff is that there exists a "power vacuum" time
// period when the outgoing leaseholder has revoked its previous lease but
// the incoming leaseholder has not yet applied its new lease. During this
// time period, a range is effectively unavailable for strong reads and
// writes, because no replica will act as the leaseholder. Instead, requests
// that require the lease will be redirected back and forth between the
// outgoing leaseholder and the incoming leaseholder (the client backs off).
// To minimize the disruption caused by lease transfers, we need to minimize
// this time period.
//
// We assume that if a lease transfer target is sufficiently caught up on
// its log such that it will be able to apply the lease transfer through log
// entry application then this unavailability window will be acceptable.
// This may be a faulty assumption in cases with severe replication lag, but
// we must balance any heuristics here that attempts to determine "too much
// lag" with the possibility of starvation of lease transfers under
// sustained write load and a resulting sustained replication lag. See
// #38065 and #42379, which removed such a heuristic. For now, we don't try
// to make such a determination.
//
// However, we draw a distinction between lease transfer targets that will
// be able to apply the lease transfer through log entry application and
// those that will require a Raft snapshot to catch up and apply the lease
// transfer. Raft snapshots are more expensive than Raft entry replication.
// They are also significantly more likely to be delayed due to queueing
// behind other snapshot traffic in the system. This potential for delay
// makes transferring a lease to a replica that needs a snapshot very risky,
// as doing so has the effect of inducing range unavailability until the
// snapshot completes, which could take seconds, minutes, or hours.
//
// In the future, we will likely get better at prioritizing snapshots to
// improve the responsiveness of snapshots that are needed to recover
// availability. However, even in this world, it is not worth inducing
// unavailability that can only be recovered through a Raft snapshot. It is
// better to catch the desired lease target up on the log first and then
// initiate the lease transfer once its log is connected to the leader's.
//
// For this reason, unless we can guarantee that the lease transfer target
// does not need a Raft snapshot, we don't let it through. This same check
// lives at higher levels in the stack as well (i.e. in the allocator). The
// higher level checks avoid wasted work and respond more gracefully to
// invalid targets (e.g. they pick the next best target). However, this is
// the only place where the protection is airtight against race conditions
// because the check is performed:
// 1. by the current Raft leader, else the proposal will fail
// 2. while holding latches that prevent interleaving log truncation
//
// If an error is thrown here, the outgoing leaseholder still won't be able
// to use its revoked lease. However, it will be able to immediately request
// a new lease. This may be disruptive, which is why we try to avoid hitting
// this airtight protection as much as possible by detecting the failure
// scenario before revoking the outgoing lease.
status := raftGroup.Status()
firstIndex := b.p.firstIndex()
newLease := p.command.ReplicatedEvalResult.State.Lease
newLeaseTarget := newLease.Replica.ReplicaID
snapStatus := raftutil.ReplicaMayNeedSnapshot(&status, firstIndex, newLeaseTarget)
if snapStatus != raftutil.NoSnapshotNeeded && !b.testing.allowLeaseTransfersWhenTargetMayNeedSnapshot {
b.p.rejectProposalWithLeaseTransferRejectedLocked(ctx, p, newLease, snapStatus)
return true
}
return false
default:
return false
}
}
// leaderStatusRLocked returns the rangeLeaderInfo for the provided raft group,
// or an empty rangeLeaderInfo if the raftGroup is nil.
func (b *propBuf) leaderStatusRLocked(ctx context.Context, raftGroup proposerRaft) rangeLeaderInfo {
leaderInfo := b.p.leaderStatusRLocked(ctx, raftGroup)
// Sanity check.
if leaderInfo.leaderKnown && leaderInfo.leader == b.p.getReplicaID() &&
!leaderInfo.iAmTheLeader {
log.Fatalf(ctx,
"inconsistent Raft state: state %s while the current replica is also the lead: %d",
raftGroup.Status().RaftState, leaderInfo.leader)
}
return leaderInfo
}
// allocateLAIAndClosedTimestampLocked computes a LAI and closed timestamp to be
// carried by an outgoing proposal.
//
// closedTSTarget is the timestamp that should be closed for this range
// according to the range's closing policy. This function will look at the
// particularities of the range and of the proposal and decide to close a
// different timestamp.
//
// This shouldn't be called for reproposals; we don't want to update the closed
// timestamp and LAI they carry (we could, in principle, but we'd have to make a
// copy of the encoded command as to not modify the copy that's already stored
// in the local replica's raft entry cache).
func (b *propBuf) allocateLAIAndClosedTimestampLocked(
ctx context.Context, p *ProposalData, closedTSTarget hlc.Timestamp,
) (uint64, hlc.Timestamp, error) {
// Assign a LeaseAppliedIndex (see checkForcedErr). These provide replay
// protection.
//
// Proposals coming from lease requests (not transfers) have their own replay
// protection, via the lease sequence and the previous lease's proposal
// timestamp; this is necessary as lease requests are proposed while not
// holding the lease (and so the proposed does not know a valid LAI to use).
// They will not check the lease applied index proposed from followers). While
// it would be legal to still assign a LAI to lease requests, historically it
// has been mildly inconvenient in testing, and might belie the fact that
// LAI-related concepts just don't apply. Instead, we assign a zero LAI to
// lease proposals, with a condition that matches that used in
// checkForcedError to identify lease requests. Note that lease *transfers*
// are only ever proposed by leaseholders, and they use the LAI to prevent
// replays (though they could in principle also be handled like lease
// requests).
var lai uint64
if !p.Request.IsSingleRequestLeaseRequest() {
b.assignedLAI++
lai = b.assignedLAI
}
if filter := b.testing.leaseIndexFilter; filter != nil {
if override := filter(p); override != 0 {
lai = override
}
}
if b.testing.dontCloseTimestamps {
return lai, hlc.Timestamp{}, nil
}
// Lease requests don't carry closed timestamps. The reason for this differ
// between lease extensions and brand new leases:
// - A lease extension cannot carry a closed timestamp assigned in the same
// way as we do for regular proposal because they're proposed without a MLAI,
// and so two lease extensions might commute and both apply, which would
// result in a closed timestamp regression when the reordered extension
// applies. The command application side doesn't bother protecting against
// such regressions. Besides, the considerations for brand new leases below
// also apply.
// - For a brand new lease, one might think that the lease start time can be
// considered a closed timestamp(*) since, if this replica gets the lease, it
// will not evaluate writes at lower timestamps. Unfortunately, there's a
// problem: while it's true that this replica, and this range in general, will
// not permit writes at timestamps below this lease's start time, it might
// happen that the range is in the process of merging with its left neighbor.
// If this range has already been Subsumed as the RHS of a merge then, after
// merge, the joint range will allow writes to the former RHS's key space at
// timestamps above the RHS's freeze start (which is below the start time of
// this lease). Thus, if this lease were to close its start timestamp while
// subsumed, then it'd be possible for follower reads to be served before the
// merge finalizes at timestamps that would become un-closed after the merge.
// Since this scenario is about subsumed ranges, we could make a distinction
// between brand new leases for subsumed ranges versus other brand new leases,
// and let the former category close the lease start time. But, for
// simplicity, we don't close timestamps on any lease requests.
//
// As opposed to lease requests, lease transfers behave like regular
// proposals: they get a closed timestamp based on closedTSTarget. Note that
// transfers carry a summary of the timestamp cache, so the new leaseholder
// will be aware of all the reads performed by the previous leaseholder.
//
// (*) If we were to close the lease start time, we'd have to forward the
// lease start time to b.assignedClosedTimestamp. We surprisingly might have
// previously closed a timestamp above the lease start time - when we close
// timestamps in the future, then attempt to transfer our lease away (and thus
// proscribe it) but the transfer fails and we're now acquiring a new lease to
// replace the proscribed one. Also, if we ever re-introduce closed
// timestamps carried by lease requests, make sure to resurrect the old
// TestRejectedLeaseDoesntDictateClosedTimestamp and protect against that
// scenario.
if p.Request.IsSingleRequestLeaseRequest() {
return lai, hlc.Timestamp{}, nil
}
// Note that under a steady lease, for requests that leave intents we must
// have WriteTimestamp.Less(b.assignedClosedTimestamp) and we used to assert
// that here. However, this does not have to be true for proposals that
// evaluated under an old lease and which are only entering the proposal
// buffer after the lease has returned and in the process of doing so
// incremented b.assignedClosedTimestamp. These proposals have no effect (as
// they apply as a no-op) but the proposal tracker has no knowledge of the
// lease changes and would therefore witness what looks like a violation of
// the invariant above. We have an authoritative assertion in
// (*replicaAppBatch).assertNoWriteBelowClosedTimestamp that is not
// susceptible to the above false positive.
//
// See https://github.com/cockroachdb/cockroach/issues/72428#issuecomment-976428551.
lb := b.evalTracker.LowerBound(ctx)
if !lb.IsEmpty() {
// If the tracker told us that requests are currently evaluating at
// timestamps >= lb, then we can close up to lb.Prev(). We use FloorPrev()
// to get rid of the logical ticks; we try to not publish closed ts with
// logical ticks when there's no good reason for them.
closedTSTarget.Backward(lb.FloorPrev())
}
// We can't close timestamps above the current lease's expiration.
closedTSTarget.Backward(p.leaseStatus.ClosedTimestampUpperBound())
// We're about to close closedTSTarget. The propBuf needs to remember that
// in order for incoming requests to be bumped above it (through
// TrackEvaluatingRequest).
if !b.forwardClosedTimestampLocked(closedTSTarget) {
closedTSTarget = b.assignedClosedTimestamp
}
return lai, closedTSTarget, nil
}
// marshallLAIAndClosedTimestampToProposalLocked modifies p.encodedCommand,
// adding the LAI and closed timestamp.
func (b *propBuf) marshallLAIAndClosedTimestampToProposalLocked(
ctx context.Context, p *ProposalData, lai uint64, closedTimestamp hlc.Timestamp,
) error {
buf := &b.scratchFooter
buf.MaxLeaseIndex = lai
// Also assign MaxLeaseIndex to the in-memory copy. The in-memory copy is
// checked for sanity at application time, on the proposing replica.
p.command.MaxLeaseIndex = lai
// Fill in the closed ts in the proposal.
buf.ClosedTimestamp = closedTimestamp
// NOTE(andrei): We don't assigned to the in-memory command
// (p.command.ClosedTimestamp) because that would cause an allocation (see
// comments on the proto field about why it needs to be nullable). It'd be
// nice to assign to it, for consistency, but nobody needs it.
if log.ExpensiveLogEnabled(ctx, 4) {
log.VEventf(ctx, 4, "attaching closed timestamp %s to proposal %x",
closedTimestamp, p.idKey)
}
// Here we rely on p.encodedCommand to have been allocated with enough
// capacity for this footer.
preLen := len(p.encodedCommand)
p.encodedCommand = p.encodedCommand[:preLen+buf.Size()]
_, err := protoutil.MarshalTo(buf, p.encodedCommand[preLen:])
return err
}
func (b *propBuf) forwardAssignedLAILocked(v uint64) {
if b.assignedLAI < v {
b.assignedLAI = v
}
}
// forwardClosedTimestamp forwards the closed timestamp tracked by the propBuf.
func (b *propBuf) forwardClosedTimestampLocked(closedTS hlc.Timestamp) bool {
return b.assignedClosedTimestamp.Forward(closedTS)
}
func proposeBatch(raftGroup proposerRaft, replID roachpb.ReplicaID, ents []raftpb.Entry) error {
if len(ents) == 0 {
return nil
}
if err := raftGroup.Step(raftpb.Message{
Type: raftpb.MsgProp,
From: uint64(replID),
Entries: ents,
}); errors.Is(err, raft.ErrProposalDropped) {
// Silently ignore dropped proposals (they were always silently
// ignored prior to the introduction of ErrProposalDropped).
// TODO(bdarnell): Handle ErrProposalDropped better.
// https://github.com/cockroachdb/cockroach/issues/21849
return nil
} else if err != nil {
return err
}
return nil
}
// FlushLockedWithoutProposing is like FlushLockedWithRaftGroup but it does not
// attempt to propose any of the commands that it is flushing. Instead, it is
// used exclusively to flush all entries in the buffer into the proposals map.
//
// The intended usage of this method is to flush all proposals in the buffer
// into the proposals map so that they can all be manipulated in a single place.
// The representative example of this is a caller that wants to flush the buffer
// into the proposals map before canceling all proposals.
func (b *propBuf) FlushLockedWithoutProposing(ctx context.Context) {
if _, err := b.FlushLockedWithRaftGroup(ctx, nil /* raftGroup */); err != nil {
log.Fatalf(ctx, "unexpected error: %+v", err)
}
}
// OnLeaseChangeLocked is called when a new lease is applied to this range.
// appliedClosedTS is the range's closed timestamp after the new lease was
// applied; the closed timestamp tracked by the propBuf is updated accordingly.
// Similarly, appliedLAI is the highest LAI of an applied command; the propBuf
// will propose commands with higher LAIs.
func (b *propBuf) OnLeaseChangeLocked(
leaseOwned bool, appliedClosedTS hlc.Timestamp, appliedLAI uint64,
) {
if leaseOwned {
b.forwardClosedTimestampLocked(appliedClosedTS)
b.forwardAssignedLAILocked(appliedLAI)
} else {
// Zero out to avoid any confusion.
b.assignedClosedTimestamp = hlc.Timestamp{}
b.assignedLAI = 0
}
}
// EvaluatingRequestsCount returns the count of requests currently tracked by
// the propBuf.
func (b *propBuf) EvaluatingRequestsCount() int {
b.p.rlocker().Lock()
defer b.p.rlocker().Unlock()
return b.evalTracker.Count()
}
// TrackedRequestToken represents the result of propBuf.TrackEvaluatingRequest:
// a token to be later used for untracking the respective request.
//
// This token tries to make it easy to pass responsibility for untracking. The
// intended pattern is:
// tok := propbBuf.TrackEvaluatingRequest()
// defer tok.DoneIfNotMoved()
// fn(tok.Move())
//
// A zero value TrackedRequestToken acts as a no-op: calling DoneIfNotMoved() on
// it will not interact with the tracker at all, but will cause stillTracked()
// to switch from true->false.
type TrackedRequestToken struct {
done bool
tok tracker.RemovalToken
b *propBuf
}
// DoneIfNotMoved untracks the request if Move had not been called on the token
// previously. If Move had been called, this is a no-op.
//
// Note that if this ends up actually destroying the token (i.e. if Move() had
// not been called previously) this takes r.mu, so it's pretty expensive. On
// happy paths, the token is expected to have been Move()d, and a batch of
// tokens are expected to be destroyed at once by the propBuf (which calls
// doneLocked).
func (t *TrackedRequestToken) DoneIfNotMoved(ctx context.Context) {
if t.done {
return
}
if t.b != nil {
t.b.p.locker().Lock()
defer t.b.p.locker().Unlock()
}
t.doneIfNotMovedLocked(ctx)
}
// doneIfNotMovedLocked untrackes the request. It is idempotent; in particular,
// this is used when wanting to untrack a proposal that might, in fact, be a
// reproposal.
func (t *TrackedRequestToken) doneIfNotMovedLocked(ctx context.Context) {
if t.done {
return
}