Skip to content

Commit

Permalink
Test ApplyConfState after restart
Browse files Browse the repository at this point in the history
Signed-off-by: Marek Siarkowicz <siarkowicz@google.com>
  • Loading branch information
serathius committed Nov 26, 2023
1 parent 7fb6b75 commit 30e8316
Showing 1 changed file with 140 additions and 0 deletions.
140 changes: 140 additions & 0 deletions server/etcdserver/server_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -198,6 +198,146 @@ func TestV2SetClusterVersion(t *testing.T) {
}
}

func TestApplyConfStateWithRestart(t *testing.T) {
n := newNodeRecorder()
srv := newServer(t, n)

assert.Equal(t, srv.consistIndex.ConsistentIndex(), uint64(0))

var nodeID uint64 = 1
memberAddContext, err := json.Marshal(&membership.Member{ID: types.ID(nodeID), RaftAttributes: membership.RaftAttributes{PeerURLs: []string{""}}})
if err != nil {
t.Fatal(err)
}

entries := []raftpb.Entry{
{
Term: 1,
Index: 1,
Type: raftpb.EntryConfChange,
Data: pbutil.MustMarshal(&raftpb.ConfChange{
Type: raftpb.ConfChangeAddNode,
NodeID: nodeID,
Context: memberAddContext,
}),
},
{
Term: 1,
Index: 2,
Type: raftpb.EntryConfChange,
Data: pbutil.MustMarshal(&raftpb.ConfChange{
Type: raftpb.ConfChangeRemoveNode,
NodeID: nodeID,
}),
},
{
Term: 1,
Index: 3,
Type: raftpb.EntryConfChange,
Data: pbutil.MustMarshal(&raftpb.ConfChange{
Type: raftpb.ConfChangeAddNode,
NodeID: nodeID,
Context: memberAddContext,
}),
},
}
want := []testutil.Action{
{
Name: "ApplyConfChange",
Params: []any{raftpb.ConfChange{
Type: raftpb.ConfChangeAddNode,
NodeID: nodeID,
Context: memberAddContext,
}},
},
{
Name: "ApplyConfChange",
Params: []any{raftpb.ConfChange{
Type: raftpb.ConfChangeRemoveNode,
NodeID: nodeID,
}},
},
// This action is expected to fail validation, thus NodeID is set to 0
{
Name: "ApplyConfChange",
Params: []any{raftpb.ConfChange{
Type: raftpb.ConfChangeAddNode,
Context: memberAddContext,
NodeID: 0,
}},
},
}

confState := raftpb.ConfState{}

t.Log("Applying entries for the first time")
srv.apply(entries, &confState, nil)
if got, _ := n.Wait(len(want)); !reflect.DeepEqual(got, want) {
t.Errorf("actions don't match\n got %+v\n want %+v", got, want)
}

t.Log("Simulating etcd restart by clearing v2 store")
srv.cluster.SetStore(v2store.New())

t.Log("Reapplying same entries after restart")
srv.apply(entries, &confState, nil)
if got, _ := n.Wait(2 * len(want)); !reflect.DeepEqual(got[len(want):], want) {
t.Errorf("actions don't match\n got %+v\n want %+v", got, want)
}
}

func newServer(t *testing.T, recorder *nodeRecorder) *EtcdServer {
lg := zaptest.NewLogger(t)
be, _ := betesting.NewDefaultTmpBackend(t)
t.Cleanup(func() {
betesting.Close(t, be)
})
srv := &EtcdServer{
lgMu: new(sync.RWMutex),
lg: zaptest.NewLogger(t),
r: *newRaftNode(raftNodeConfig{lg: lg, Node: recorder}),
cluster: membership.NewCluster(lg),
consistIndex: cindex.NewConsistentIndex(be),
}
srv.cluster.SetBackend(schema.NewMembershipBackend(lg, be))
srv.cluster.SetStore(v2store.New())
srv.beHooks = serverstorage.NewBackendHooks(lg, srv.consistIndex)
srv.r.transport = newNopTransporter()
srv.w = mockwait.NewNop()
return srv
}

func memberAddEntry(t *testing.T, nodeID uint64, peerURL string, term, index uint64) raftpb.Entry {
var cc raftpb.ConfChange
cc.Type = raftpb.ConfChangeAddNode
cc.NodeID = nodeID

attr := membership.RaftAttributes{PeerURLs: []string{peerURL}}
var err error
cc.Context, err = json.Marshal(&membership.Member{ID: types.ID(nodeID), RaftAttributes: attr})
if err != nil {
t.Fatal(err)
}
return raftpb.Entry{
Term: term,
Index: index,
Type: raftpb.EntryConfChange,
Data: pbutil.MustMarshal(&cc),
}
}

func memberRemoveEntry(nodeID uint64, term, index uint64) raftpb.Entry {
var cc raftpb.ConfChange
cc.Type = raftpb.ConfChangeRemoveNode
cc.NodeID = nodeID
return raftpb.Entry{
Term: term,
Index: index,
Type: raftpb.EntryConfChange,
Data: pbutil.MustMarshal(&cc),
}
}

func TestApplyConfChangeError(t *testing.T) {
lg := zaptest.NewLogger(t)
be, _ := betesting.NewDefaultTmpBackend(t)
Expand Down

0 comments on commit 30e8316

Please sign in to comment.