From a49e3b9ad4797639e04b241bb0e2ccf686175a25 Mon Sep 17 00:00:00 2001 From: Tobias Schottdorf Date: Wed, 30 Jan 2019 12:36:26 +0100 Subject: [PATCH] storage: fix NPE while printing trivial truncateDecision Fixes #34398. Release note: None --- pkg/storage/raft_log_queue.go | 20 ++++++++++---------- pkg/storage/raft_log_queue_test.go | 16 +++++++++++++--- 2 files changed, 23 insertions(+), 13 deletions(-) diff --git a/pkg/storage/raft_log_queue.go b/pkg/storage/raft_log_queue.go index 31366601b373..ac6bf33559b3 100644 --- a/pkg/storage/raft_log_queue.go +++ b/pkg/storage/raft_log_queue.go @@ -90,9 +90,9 @@ func newRaftLogQueue(store *Store, db *client.DB, gossip *gossip.Gossip) *raftLo } // newTruncateDecision returns a truncateDecision for the given Replica if no -// error occurs. If no truncation can be carried out, a zero decision is -// returned. -func newTruncateDecision(ctx context.Context, r *Replica) (*truncateDecision, error) { +// error occurs. If input data to establish a truncateDecision is missing, a +// zero decision is returned. +func newTruncateDecision(ctx context.Context, r *Replica) (truncateDecision, error) { rangeID := r.RangeID now := timeutil.Now() @@ -123,20 +123,20 @@ func newTruncateDecision(ctx context.Context, r *Replica) (*truncateDecision, er r.mu.Unlock() if err != nil { - return nil, errors.Errorf("error retrieving first index for r%d: %s", rangeID, err) + return truncateDecision{}, errors.Errorf("error retrieving first index for r%d: %s", rangeID, err) } if raftStatus == nil { if log.V(6) { log.Infof(ctx, "the raft group doesn't exist for r%d", rangeID) } - return &truncateDecision{}, nil + return truncateDecision{}, nil } // Is this the raft leader? We only perform log truncation on the raft leader // which has the up to date info on followers. if raftStatus.RaftState != raft.StateLeader { - return &truncateDecision{}, nil + return truncateDecision{}, nil } // For all our followers, overwrite the RecentActive field (which is always @@ -155,7 +155,7 @@ func newTruncateDecision(ctx context.Context, r *Replica) (*truncateDecision, er } input := truncateDecisionInput{ - RaftStatus: raftStatus, + RaftStatus: *raftStatus, LogSize: raftLogSize, MaxLogSize: targetSize, FirstIndex: firstIndex, @@ -164,7 +164,7 @@ func newTruncateDecision(ctx context.Context, r *Replica) (*truncateDecision, er } decision := computeTruncateDecision(input) - return &decision, nil + return decision, nil } func updateRaftProgressFromActivity( @@ -203,7 +203,7 @@ const ( ) type truncateDecisionInput struct { - RaftStatus *raft.Status // never nil + RaftStatus raft.Status LogSize, MaxLogSize int64 FirstIndex, LastIndex uint64 PendingPreemptiveSnapshotIndex uint64 @@ -315,7 +315,7 @@ func (td *truncateDecision) ShouldTruncate() bool { // snapshots. See #8629. func computeTruncateDecision(input truncateDecisionInput) truncateDecision { decision := truncateDecision{Input: input} - decision.QuorumIndex = getQuorumIndex(input.RaftStatus) + decision.QuorumIndex = getQuorumIndex(&input.RaftStatus) decision.NewFirstIndex = decision.QuorumIndex decision.ChosenVia = truncatableIndexChosenViaQuorumIndex diff --git a/pkg/storage/raft_log_queue_test.go b/pkg/storage/raft_log_queue_test.go index a1e99ff31108..9d7b9a8887a5 100644 --- a/pkg/storage/raft_log_queue_test.go +++ b/pkg/storage/raft_log_queue_test.go @@ -201,7 +201,7 @@ func TestComputeTruncateDecision(t *testing.T) { "should truncate: false [truncate 0 entries to first index 2 (chosen via: first index)]", }} for i, c := range testCases { - status := &raft.Status{ + status := raft.Status{ Progress: make(map[uint64]raft.Progress), } for j, v := range c.progress { @@ -243,7 +243,7 @@ func TestComputeTruncateDecisionProgressStatusProbe(t *testing.T) { testutils.RunTrueAndFalse(t, "tooLarge", func(t *testing.T, tooLarge bool) { testutils.RunTrueAndFalse(t, "active", func(t *testing.T, active bool) { - status := &raft.Status{ + status := raft.Status{ Progress: make(map[uint64]raft.Progress), } for j, v := range []uint64{500, 400, 300, 200, 100} { @@ -277,10 +277,20 @@ func TestComputeTruncateDecisionProgressStatusProbe(t *testing.T) { }) } +func TestTruncateDecisionZeroValue(t *testing.T) { + defer leaktest.AfterTest(t)() + + var decision truncateDecision + assert.False(t, decision.ShouldTruncate()) + assert.Zero(t, decision.NumNewRaftSnapshots()) + assert.Zero(t, decision.NumTruncatableIndexes()) + assert.Equal(t, "should truncate: false [truncate 0 entries to first index 0 (chosen via: )]", decision.String()) +} + func TestTruncateDecisionNumSnapshots(t *testing.T) { defer leaktest.AfterTest(t)() - status := &raft.Status{ + status := raft.Status{ Progress: map[uint64]raft.Progress{ // Fully caught up. 5: {State: raft.ProgressStateReplicate, Match: 11, Next: 12},