mirror of
https://github.com/etcd-io/etcd.git
synced 2024-09-27 06:25:44 +00:00

Just replaced all go.etcd.io/etcd/raft/v3 with go.etcd.io/raft/v3 under tools/etcd-dump-logs. Signed-off-by: Benjamin Wang <wachao@vmware.com>
279 lines
11 KiB
Go
279 lines
11 KiB
Go
// Copyright 2018 The etcd Authors
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"os"
|
|
"os/exec"
|
|
"path"
|
|
"path/filepath"
|
|
"strings"
|
|
"testing"
|
|
|
|
"go.uber.org/zap/zaptest"
|
|
|
|
"go.etcd.io/etcd/api/v3/authpb"
|
|
"go.etcd.io/etcd/api/v3/etcdserverpb"
|
|
"go.etcd.io/etcd/client/pkg/v3/fileutil"
|
|
"go.etcd.io/etcd/pkg/v3/pbutil"
|
|
"go.etcd.io/etcd/server/v3/storage/wal"
|
|
"go.etcd.io/raft/v3/raftpb"
|
|
)
|
|
|
|
func TestEtcdDumpLogEntryType(t *testing.T) {
|
|
// directory where the command is
|
|
binDir, err := os.Getwd()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
dumpLogsBinary := path.Join(binDir + "/etcd-dump-logs")
|
|
if !fileutil.Exist(dumpLogsBinary) {
|
|
t.Skipf("%q does not exist", dumpLogsBinary)
|
|
}
|
|
|
|
decoderCorrectOutputFormat := filepath.Join(binDir, "/testdecoder/decoder_correctoutputformat.sh")
|
|
decoderWrongOutputFormat := filepath.Join(binDir, "/testdecoder/decoder_wrongoutputformat.sh")
|
|
|
|
p := t.TempDir()
|
|
|
|
memberdir := filepath.Join(p, "member")
|
|
err = os.Mkdir(memberdir, 0744)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
waldir := walDir(p)
|
|
snapdir := snapDir(p)
|
|
|
|
w, err := wal.Create(zaptest.NewLogger(t), waldir, nil)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
err = os.Mkdir(snapdir, 0744)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
|
|
ents := make([]raftpb.Entry, 0)
|
|
|
|
// append entries into wal log
|
|
appendConfigChangeEnts(&ents)
|
|
appendNormalRequestEnts(&ents)
|
|
appendNormalIRREnts(&ents)
|
|
appendUnknownNormalEnts(&ents)
|
|
|
|
// force commit newly appended entries
|
|
err = w.Save(raftpb.HardState{}, ents)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
w.Close()
|
|
|
|
argtests := []struct {
|
|
name string
|
|
args []string
|
|
fileExpected string
|
|
}{
|
|
{"no entry-type", []string{p}, "expectedoutput/listAll.output"},
|
|
{"confchange entry-type", []string{"-entry-type", "ConfigChange", p}, "expectedoutput/listConfigChange.output"},
|
|
{"normal entry-type", []string{"-entry-type", "Normal", p}, "expectedoutput/listNormal.output"},
|
|
{"request entry-type", []string{"-entry-type", "Request", p}, "expectedoutput/listRequest.output"},
|
|
{"internalRaftRequest entry-type", []string{"-entry-type", "InternalRaftRequest", p}, "expectedoutput/listInternalRaftRequest.output"},
|
|
{"range entry-type", []string{"-entry-type", "IRRRange", p}, "expectedoutput/listIRRRange.output"},
|
|
{"put entry-type", []string{"-entry-type", "IRRPut", p}, "expectedoutput/listIRRPut.output"},
|
|
{"del entry-type", []string{"-entry-type", "IRRDeleteRange", p}, "expectedoutput/listIRRDeleteRange.output"},
|
|
{"txn entry-type", []string{"-entry-type", "IRRTxn", p}, "expectedoutput/listIRRTxn.output"},
|
|
{"compaction entry-type", []string{"-entry-type", "IRRCompaction", p}, "expectedoutput/listIRRCompaction.output"},
|
|
{"lease grant entry-type", []string{"-entry-type", "IRRLeaseGrant", p}, "expectedoutput/listIRRLeaseGrant.output"},
|
|
{"lease revoke entry-type", []string{"-entry-type", "IRRLeaseRevoke", p}, "expectedoutput/listIRRLeaseRevoke.output"},
|
|
{"confchange and txn entry-type", []string{"-entry-type", "ConfigChange,IRRCompaction", p}, "expectedoutput/listConfigChangeIRRCompaction.output"},
|
|
{"decoder_correctoutputformat", []string{"-stream-decoder", decoderCorrectOutputFormat, p}, "expectedoutput/decoder_correctoutputformat.output"},
|
|
{"decoder_wrongoutputformat", []string{"-stream-decoder", decoderWrongOutputFormat, p}, "expectedoutput/decoder_wrongoutputformat.output"},
|
|
}
|
|
|
|
for _, argtest := range argtests {
|
|
t.Run(argtest.name, func(t *testing.T) {
|
|
cmd := exec.Command(dumpLogsBinary, argtest.args...)
|
|
actual, err := cmd.CombinedOutput()
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
expected, err := os.ReadFile(path.Join(binDir, argtest.fileExpected))
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !bytes.Equal(actual, expected) {
|
|
t.Errorf(`Got input of length %d, wanted input of length %d
|
|
==== BEGIN RECEIVED FILE ====
|
|
%s
|
|
==== END RECEIVED FILE ====
|
|
==== BEGIN EXPECTED FILE ====
|
|
%s
|
|
==== END EXPECTED FILE ====
|
|
`, len(actual), len(expected), actual, expected)
|
|
}
|
|
})
|
|
}
|
|
|
|
}
|
|
|
|
func appendConfigChangeEnts(ents *[]raftpb.Entry) {
|
|
configChangeData := []raftpb.ConfChange{
|
|
{ID: 1, Type: raftpb.ConfChangeAddNode, NodeID: 2, Context: []byte("")},
|
|
{ID: 2, Type: raftpb.ConfChangeRemoveNode, NodeID: 2, Context: []byte("")},
|
|
{ID: 3, Type: raftpb.ConfChangeUpdateNode, NodeID: 2, Context: []byte("")},
|
|
{ID: 4, Type: raftpb.ConfChangeAddLearnerNode, NodeID: 3, Context: []byte("")},
|
|
}
|
|
configChangeEntries := []raftpb.Entry{
|
|
{Term: 1, Index: 1, Type: raftpb.EntryConfChange, Data: pbutil.MustMarshal(&configChangeData[0])},
|
|
{Term: 2, Index: 2, Type: raftpb.EntryConfChange, Data: pbutil.MustMarshal(&configChangeData[1])},
|
|
{Term: 2, Index: 3, Type: raftpb.EntryConfChange, Data: pbutil.MustMarshal(&configChangeData[2])},
|
|
{Term: 2, Index: 4, Type: raftpb.EntryConfChange, Data: pbutil.MustMarshal(&configChangeData[3])},
|
|
}
|
|
*ents = append(*ents, configChangeEntries...)
|
|
}
|
|
|
|
func appendNormalRequestEnts(ents *[]raftpb.Entry) {
|
|
a := true
|
|
b := false
|
|
|
|
requests := []etcdserverpb.Request{
|
|
{ID: 0, Method: "", Path: "/path0", Val: "{\"hey\":\"ho\",\"hi\":[\"yo\"]}", Dir: true, PrevValue: "", PrevIndex: 0, PrevExist: &b, Expiration: 9, Wait: false, Since: 1, Recursive: false, Sorted: false, Quorum: false, Time: 1, Stream: false, Refresh: &b},
|
|
{ID: 1, Method: "QGET", Path: "/path1", Val: "{\"0\":\"1\",\"2\":[\"3\"]}", Dir: false, PrevValue: "", PrevIndex: 0, PrevExist: &b, Expiration: 9, Wait: false, Since: 1, Recursive: false, Sorted: false, Quorum: false, Time: 1, Stream: false, Refresh: &b},
|
|
{ID: 2, Method: "SYNC", Path: "/path2", Val: "{\"0\":\"1\",\"2\":[\"3\"]}", Dir: false, PrevValue: "", PrevIndex: 0, PrevExist: &b, Expiration: 2, Wait: false, Since: 1, Recursive: false, Sorted: false, Quorum: false, Time: 1, Stream: false, Refresh: &b},
|
|
{ID: 3, Method: "DELETE", Path: "/path3", Val: "{\"hey\":\"ho\",\"hi\":[\"yo\"]}", Dir: false, PrevValue: "", PrevIndex: 0, PrevExist: &a, Expiration: 2, Wait: false, Since: 1, Recursive: false, Sorted: false, Quorum: false, Time: 1, Stream: false, Refresh: &b},
|
|
{ID: 4, Method: "RANDOM", Path: "/path4/superlong" + strings.Repeat("/path", 30), Val: "{\"hey\":\"ho\",\"hi\":[\"yo\"]}", Dir: false, PrevValue: "", PrevIndex: 0, PrevExist: &b, Expiration: 2, Wait: false, Since: 1, Recursive: false, Sorted: false, Quorum: false, Time: 1, Stream: false, Refresh: &b},
|
|
}
|
|
|
|
for i, request := range requests {
|
|
var currentry raftpb.Entry
|
|
currentry.Term = 3
|
|
currentry.Index = uint64(i + 5)
|
|
currentry.Type = raftpb.EntryNormal
|
|
currentry.Data = pbutil.MustMarshal(&request)
|
|
*ents = append(*ents, currentry)
|
|
}
|
|
}
|
|
|
|
func appendNormalIRREnts(ents *[]raftpb.Entry) {
|
|
irrrange := &etcdserverpb.RangeRequest{Key: []byte("1"), RangeEnd: []byte("hi"), Limit: 6, Revision: 1, SortOrder: 1, SortTarget: 0, Serializable: false, KeysOnly: false, CountOnly: false, MinModRevision: 0, MaxModRevision: 20000, MinCreateRevision: 0, MaxCreateRevision: 20000}
|
|
|
|
irrput := &etcdserverpb.PutRequest{Key: []byte("foo1"), Value: []byte("bar1"), Lease: 1, PrevKv: false, IgnoreValue: false, IgnoreLease: true}
|
|
|
|
irrdeleterange := &etcdserverpb.DeleteRangeRequest{Key: []byte("0"), RangeEnd: []byte("9"), PrevKv: true}
|
|
|
|
delInRangeReq := &etcdserverpb.RequestOp{Request: &etcdserverpb.RequestOp_RequestDeleteRange{
|
|
RequestDeleteRange: &etcdserverpb.DeleteRangeRequest{
|
|
Key: []byte("a"), RangeEnd: []byte("b"),
|
|
},
|
|
},
|
|
}
|
|
|
|
irrtxn := &etcdserverpb.TxnRequest{Success: []*etcdserverpb.RequestOp{delInRangeReq}, Failure: []*etcdserverpb.RequestOp{delInRangeReq}}
|
|
|
|
irrcompaction := &etcdserverpb.CompactionRequest{Revision: 0, Physical: true}
|
|
|
|
irrleasegrant := &etcdserverpb.LeaseGrantRequest{TTL: 1, ID: 1}
|
|
|
|
irrleaserevoke := &etcdserverpb.LeaseRevokeRequest{ID: 2}
|
|
|
|
irralarm := &etcdserverpb.AlarmRequest{Action: 3, MemberID: 4, Alarm: 5}
|
|
|
|
irrauthenable := &etcdserverpb.AuthEnableRequest{}
|
|
|
|
irrauthdisable := &etcdserverpb.AuthDisableRequest{}
|
|
|
|
irrauthenticate := &etcdserverpb.InternalAuthenticateRequest{Name: "myname", Password: "password", SimpleToken: "token"}
|
|
|
|
irrauthuseradd := &etcdserverpb.AuthUserAddRequest{Name: "name1", Password: "pass1", Options: &authpb.UserAddOptions{NoPassword: false}}
|
|
|
|
irrauthuserdelete := &etcdserverpb.AuthUserDeleteRequest{Name: "name1"}
|
|
|
|
irrauthuserget := &etcdserverpb.AuthUserGetRequest{Name: "name1"}
|
|
|
|
irrauthuserchangepassword := &etcdserverpb.AuthUserChangePasswordRequest{Name: "name1", Password: "pass2"}
|
|
|
|
irrauthusergrantrole := &etcdserverpb.AuthUserGrantRoleRequest{User: "user1", Role: "role1"}
|
|
|
|
irrauthuserrevokerole := &etcdserverpb.AuthUserRevokeRoleRequest{Name: "user2", Role: "role2"}
|
|
|
|
irrauthuserlist := &etcdserverpb.AuthUserListRequest{}
|
|
|
|
irrauthrolelist := &etcdserverpb.AuthRoleListRequest{}
|
|
|
|
irrauthroleadd := &etcdserverpb.AuthRoleAddRequest{Name: "role2"}
|
|
|
|
irrauthroledelete := &etcdserverpb.AuthRoleDeleteRequest{Role: "role1"}
|
|
|
|
irrauthroleget := &etcdserverpb.AuthRoleGetRequest{Role: "role3"}
|
|
|
|
perm := &authpb.Permission{
|
|
PermType: authpb.WRITE,
|
|
Key: []byte("Keys"),
|
|
RangeEnd: []byte("RangeEnd"),
|
|
}
|
|
|
|
irrauthrolegrantpermission := &etcdserverpb.AuthRoleGrantPermissionRequest{Name: "role3", Perm: perm}
|
|
|
|
irrauthrolerevokepermission := &etcdserverpb.AuthRoleRevokePermissionRequest{Role: "role3", Key: []byte("key"), RangeEnd: []byte("rangeend")}
|
|
|
|
irrs := []etcdserverpb.InternalRaftRequest{
|
|
{ID: 5, Range: irrrange},
|
|
{ID: 6, Put: irrput},
|
|
{ID: 7, DeleteRange: irrdeleterange},
|
|
{ID: 8, Txn: irrtxn},
|
|
{ID: 9, Compaction: irrcompaction},
|
|
{ID: 10, LeaseGrant: irrleasegrant},
|
|
{ID: 11, LeaseRevoke: irrleaserevoke},
|
|
{ID: 12, Alarm: irralarm},
|
|
{ID: 13, AuthEnable: irrauthenable},
|
|
{ID: 14, AuthDisable: irrauthdisable},
|
|
{ID: 15, Authenticate: irrauthenticate},
|
|
{ID: 16, AuthUserAdd: irrauthuseradd},
|
|
{ID: 17, AuthUserDelete: irrauthuserdelete},
|
|
{ID: 18, AuthUserGet: irrauthuserget},
|
|
{ID: 19, AuthUserChangePassword: irrauthuserchangepassword},
|
|
{ID: 20, AuthUserGrantRole: irrauthusergrantrole},
|
|
{ID: 21, AuthUserRevokeRole: irrauthuserrevokerole},
|
|
{ID: 22, AuthUserList: irrauthuserlist},
|
|
{ID: 23, AuthRoleList: irrauthrolelist},
|
|
{ID: 24, AuthRoleAdd: irrauthroleadd},
|
|
{ID: 25, AuthRoleDelete: irrauthroledelete},
|
|
{ID: 26, AuthRoleGet: irrauthroleget},
|
|
{ID: 27, AuthRoleGrantPermission: irrauthrolegrantpermission},
|
|
{ID: 28, AuthRoleRevokePermission: irrauthrolerevokepermission},
|
|
}
|
|
|
|
for i, irr := range irrs {
|
|
var currentry raftpb.Entry
|
|
currentry.Term = uint64(i + 4)
|
|
currentry.Index = uint64(i + 10)
|
|
currentry.Type = raftpb.EntryNormal
|
|
currentry.Data = pbutil.MustMarshal(&irr)
|
|
*ents = append(*ents, currentry)
|
|
}
|
|
}
|
|
|
|
func appendUnknownNormalEnts(ents *[]raftpb.Entry) {
|
|
var currentry raftpb.Entry
|
|
currentry.Term = 27
|
|
currentry.Index = 34
|
|
currentry.Type = raftpb.EntryNormal
|
|
currentry.Data = []byte("?")
|
|
*ents = append(*ents, currentry)
|
|
}
|