Skip to content

Commit

Permalink
Merge pull request #100611 from cockroachdb/blathers/backport-release…
Browse files Browse the repository at this point in the history
…-23.1-100401

release-23.1: raftlog: correctly decode empty commands
  • Loading branch information
erikgrinaker authored Apr 9, 2023
2 parents de239a7 + 7e2d441 commit cfbd751
Show file tree
Hide file tree
Showing 3 changed files with 53 additions and 19 deletions.
10 changes: 5 additions & 5 deletions pkg/kv/kvserver/kvserverbase/forced_error.go
Original file line number Diff line number Diff line change
Expand Up @@ -95,11 +95,11 @@ func CheckForcedErr(
requestedLease = *raftCmd.ReplicatedEvalResult.State.Lease
}
if idKey == "" {
// This is an empty Raft command (which is sent by Raft after elections
// to trigger reproposals or during concurrent configuration changes).
// Nothing to do here except making sure that the corresponding batch
// (which is bogus) doesn't get executed (for it is empty and so
// properties like key range are undefined).
// This is an empty Raft command, which is sent by Raft after elections to
// trigger reproposals, during concurrent configuration changes, or to
// unquiesce the Raft group. Nothing to do here except making sure that the
// corresponding batch (which is bogus) doesn't get executed (for it is
// empty and so properties like key range are undefined).
return ForcedErrResult{
LeaseIndex: leaseIndex,
Rejection: ProposalRejectionPermanent,
Expand Down
9 changes: 7 additions & 2 deletions pkg/kv/kvserver/raftlog/entry.go
Original file line number Diff line number Diff line change
Expand Up @@ -183,10 +183,15 @@ func (e *Entry) load() error {
}
e.ID = kvserverbase.CmdIDKey(e.ConfChangeContext.CommandID)
raftCmdBytes = e.ConfChangeContext.Payload
} else if len(raftCmdBytes) == 0 {
// Empty commands may be proposed to wake the leader, e.g. during
// unquiescence. Ignore them during application by clearing the command ID
// and other fields (see CheckForcedErr), similarly to those submitted by
// Raft on leader changes (see EntryEncodingEmpty).
*e = Entry{Entry: e.Entry}
return nil
}

// TODO(tbg): can len(payload)==0 if we propose an empty command to wake up leader?
// If so, is that a problem here?
return errors.Wrap(protoutil.Unmarshal(raftCmdBytes, &e.Cmd), "unmarshalling RaftCommand")
}

Expand Down
53 changes: 41 additions & 12 deletions pkg/kv/kvserver/raftlog/entry_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,17 +18,46 @@ import (
"go.etcd.io/raft/v3/raftpb"
)

func TestLoadInvalidEntry(t *testing.T) {
invalidEnt := raftpb.Entry{
Term: 1,
Index: 1,
Data: EncodeRaftCommand(
// It would be nice to have an "even more invalid" command here but it
// turns out that DecodeRaftCommand "handles" errors via panic().
EntryEncodingStandardWithAC, "foobarzz", []byte("definitely not a protobuf"),
),
func TestNewEntry(t *testing.T) {
// TODO(replication): Add more cases.
testcases := map[string]struct {
data []byte
expectEmpty bool
expectErr bool
}{
// Proposed by Raft on leader change.
"empty entry": {data: nil, expectEmpty: true},
// Proposed by CRDB on unquiescence.
"empty payload": {
data: EncodeRaftCommand(EntryEncodingStandardWithoutAC, "00000000", nil),
expectEmpty: true,
},
"invalid": {
data: EncodeRaftCommand(EntryEncodingStandardWithAC, "00000000", []byte("not a protobuf")),
expectErr: true,
},
}
for name, tc := range testcases {
t.Run(name, func(t *testing.T) {
ent, err := NewEntry(raftpb.Entry{
Term: 1,
Index: 1,
Data: tc.data,
})
if tc.expectErr {
require.Error(t, err)
return
}
require.NoError(t, err)

// Clear out the passed Raft entry, and only assert on the decoded entry.
require.NotNil(t, ent)
ent.Entry = raftpb.Entry{}
if tc.expectEmpty {
require.Zero(t, *ent)
} else {
require.NotZero(t, *ent)
}
})
}
ent, err := NewEntry(invalidEnt)
require.Error(t, err) // specific error doesn't matter
require.Zero(t, ent)
}

0 comments on commit cfbd751

Please sign in to comment.