From 30e831625aad9cd37de5ac93094e92856e10341f Mon Sep 17 00:00:00 2001 From: Marek Siarkowicz Date: Sun, 26 Nov 2023 14:41:23 +0100 Subject: [PATCH] Test ApplyConfState after restart Signed-off-by: Marek Siarkowicz --- server/etcdserver/server_test.go | 140 +++++++++++++++++++++++++++++++ 1 file changed, 140 insertions(+) diff --git a/server/etcdserver/server_test.go b/server/etcdserver/server_test.go index 5a6a8299ad94..2fc5e11c8cf2 100644 --- a/server/etcdserver/server_test.go +++ b/server/etcdserver/server_test.go @@ -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)