functional-tester/tester: use "clientv3" for key stresser

Signed-off-by: Gyuho Lee <gyuhox@gmail.com>
This commit is contained in:
Gyuho Lee 2018-04-03 13:23:04 -07:00
parent 7cc0d689b8
commit 00ed41d175

View File

@ -22,9 +22,9 @@ import (
"sync/atomic" "sync/atomic"
"time" "time"
"github.com/coreos/etcd/clientv3"
"github.com/coreos/etcd/etcdserver" "github.com/coreos/etcd/etcdserver"
"github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes" "github.com/coreos/etcd/etcdserver/api/v3rpc/rpctypes"
pb "github.com/coreos/etcd/etcdserver/etcdserverpb"
"github.com/coreos/etcd/tools/functional-tester/rpcpb" "github.com/coreos/etcd/tools/functional-tester/rpcpb"
"go.uber.org/zap" "go.uber.org/zap"
@ -51,7 +51,7 @@ type keyStresser struct {
wg sync.WaitGroup wg sync.WaitGroup
cancel func() cancel func()
conn *grpc.ClientConn cli *clientv3.Client
// atomicModifiedKeys records the number of keys created and deleted by the stresser. // atomicModifiedKeys records the number of keys created and deleted by the stresser.
atomicModifiedKeys int64 atomicModifiedKeys int64
@ -60,35 +60,33 @@ type keyStresser struct {
func (s *keyStresser) Stress() error { func (s *keyStresser) Stress() error {
// TODO: add backoff option // TODO: add backoff option
conn, err := s.m.DialEtcdGRPCServer() cli, err := s.m.CreateEtcdClient()
if err != nil { if err != nil {
return fmt.Errorf("%v (%q)", err, s.m.EtcdClientEndpoint) return fmt.Errorf("%v (%q)", err, s.m.EtcdClientEndpoint)
} }
ctx, cancel := context.WithCancel(context.Background()) ctx, cancel := context.WithCancel(context.Background())
s.wg.Add(s.N) s.wg.Add(s.N)
s.conn = conn s.cli = cli
s.cancel = cancel s.cancel = cancel
kvc := pb.NewKVClient(conn)
var stressEntries = []stressEntry{ var stressEntries = []stressEntry{
{weight: 0.7, f: newStressPut(kvc, s.keySuffixRange, s.keySize)}, {weight: 0.7, f: newStressPut(cli, s.keySuffixRange, s.keySize)},
{ {
weight: 0.7 * float32(s.keySize) / float32(s.keyLargeSize), weight: 0.7 * float32(s.keySize) / float32(s.keyLargeSize),
f: newStressPut(kvc, s.keySuffixRange, s.keyLargeSize), f: newStressPut(cli, s.keySuffixRange, s.keyLargeSize),
}, },
{weight: 0.07, f: newStressRange(kvc, s.keySuffixRange)}, {weight: 0.07, f: newStressRange(cli, s.keySuffixRange)},
{weight: 0.07, f: newStressRangeInterval(kvc, s.keySuffixRange)}, {weight: 0.07, f: newStressRangeInterval(cli, s.keySuffixRange)},
{weight: 0.07, f: newStressDelete(kvc, s.keySuffixRange)}, {weight: 0.07, f: newStressDelete(cli, s.keySuffixRange)},
{weight: 0.07, f: newStressDeleteInterval(kvc, s.keySuffixRange)}, {weight: 0.07, f: newStressDeleteInterval(cli, s.keySuffixRange)},
} }
if s.keyTxnSuffixRange > 0 { if s.keyTxnSuffixRange > 0 {
// adjust to make up ±70% of workloads with writes // adjust to make up ±70% of workloads with writes
stressEntries[0].weight = 0.35 stressEntries[0].weight = 0.35
stressEntries = append(stressEntries, stressEntry{ stressEntries = append(stressEntries, stressEntry{
weight: 0.35, weight: 0.35,
f: newStressTxn(kvc, s.keyTxnSuffixRange, s.keyTxnOps), f: newStressTxn(cli, s.keyTxnSuffixRange, s.keyTxnOps),
}) })
} }
s.stressTable = createStressTable(stressEntries) s.stressTable = createStressTable(stressEntries)
@ -167,7 +165,7 @@ func (s *keyStresser) Pause() {
func (s *keyStresser) Close() { func (s *keyStresser) Close() {
s.cancel() s.cancel()
s.conn.Close() s.cli.Close()
s.wg.Wait() s.wg.Wait()
s.lg.Info( s.lg.Info(
@ -216,25 +214,26 @@ func (st *stressTable) choose() stressFunc {
return st.entries[idx].f return st.entries[idx].f
} }
func newStressPut(kvc pb.KVClient, keySuffixRange, keySize int) stressFunc { func newStressPut(cli *clientv3.Client, keySuffixRange, keySize int) stressFunc {
return func(ctx context.Context) (error, int64) { return func(ctx context.Context) (error, int64) {
_, err := kvc.Put(ctx, &pb.PutRequest{ _, err := cli.Put(
Key: []byte(fmt.Sprintf("foo%016x", rand.Intn(keySuffixRange))), ctx,
Value: randBytes(keySize), fmt.Sprintf("foo%016x", rand.Intn(keySuffixRange)),
}, grpc.FailFast(false)) string(randBytes(keySize)),
)
return err, 1 return err, 1
} }
} }
func newStressTxn(kvc pb.KVClient, keyTxnSuffixRange, txnOps int) stressFunc { func newStressTxn(cli *clientv3.Client, keyTxnSuffixRange, txnOps int) stressFunc {
keys := make([]string, keyTxnSuffixRange) keys := make([]string, keyTxnSuffixRange)
for i := range keys { for i := range keys {
keys[i] = fmt.Sprintf("/k%03d", i) keys[i] = fmt.Sprintf("/k%03d", i)
} }
return writeTxn(kvc, keys, txnOps) return writeTxn(cli, keys, txnOps)
} }
func writeTxn(kvc pb.KVClient, keys []string, txnOps int) stressFunc { func writeTxn(cli *clientv3.Client, keys []string, txnOps int) stressFunc {
return func(ctx context.Context) (error, int64) { return func(ctx context.Context) (error, int64) {
ks := make(map[string]struct{}, txnOps) ks := make(map[string]struct{}, txnOps)
for len(ks) != txnOps { for len(ks) != txnOps {
@ -244,99 +243,75 @@ func writeTxn(kvc pb.KVClient, keys []string, txnOps int) stressFunc {
for k := range ks { for k := range ks {
selected = append(selected, k) selected = append(selected, k)
} }
com, delOp, putOp := getTxnReqs(selected[0], "bar00") com, delOp, putOp := getTxnOps(selected[0], "bar00")
txnReq := &pb.TxnRequest{ thenOps := []clientv3.Op{delOp}
Compare: []*pb.Compare{com}, elseOps := []clientv3.Op{putOp}
Success: []*pb.RequestOp{delOp}, for i := 1; i < txnOps; i++ { // nested txns
Failure: []*pb.RequestOp{putOp},
}
// add nested txns if any
for i := 1; i < txnOps; i++ {
k, v := selected[i], fmt.Sprintf("bar%02d", i) k, v := selected[i], fmt.Sprintf("bar%02d", i)
com, delOp, putOp = getTxnReqs(k, v) com, delOp, putOp = getTxnOps(k, v)
nested := &pb.RequestOp{ txnOp := clientv3.OpTxn(
Request: &pb.RequestOp_RequestTxn{ []clientv3.Cmp{com},
RequestTxn: &pb.TxnRequest{ []clientv3.Op{delOp},
Compare: []*pb.Compare{com}, []clientv3.Op{putOp},
Success: []*pb.RequestOp{delOp}, )
Failure: []*pb.RequestOp{putOp}, thenOps = append(thenOps, txnOp)
}, elseOps = append(elseOps, txnOp)
},
}
txnReq.Success = append(txnReq.Success, nested)
txnReq.Failure = append(txnReq.Failure, nested)
} }
_, err := cli.Txn(ctx).
_, err := kvc.Txn(ctx, txnReq, grpc.FailFast(false)) If(com).
Else(elseOps...).
Then(thenOps...).
Commit()
return err, int64(txnOps) return err, int64(txnOps)
} }
} }
func getTxnReqs(key, val string) (com *pb.Compare, delOp *pb.RequestOp, putOp *pb.RequestOp) { func getTxnOps(k, v string) (
cmp clientv3.Cmp,
dop clientv3.Op,
pop clientv3.Op) {
// if key exists (version > 0) // if key exists (version > 0)
com = &pb.Compare{ cmp = clientv3.Compare(clientv3.Version(k), ">", 0)
Key: []byte(key), dop = clientv3.OpDelete(k)
Target: pb.Compare_VERSION, pop = clientv3.OpPut(k, v)
Result: pb.Compare_GREATER, return cmp, dop, pop
TargetUnion: &pb.Compare_Version{Version: 0},
}
delOp = &pb.RequestOp{
Request: &pb.RequestOp_RequestDeleteRange{
RequestDeleteRange: &pb.DeleteRangeRequest{
Key: []byte(key),
},
},
}
putOp = &pb.RequestOp{
Request: &pb.RequestOp_RequestPut{
RequestPut: &pb.PutRequest{
Key: []byte(key),
Value: []byte(val),
},
},
}
return com, delOp, putOp
} }
func newStressRange(kvc pb.KVClient, keySuffixRange int) stressFunc { func newStressRange(cli *clientv3.Client, keySuffixRange int) stressFunc {
return func(ctx context.Context) (error, int64) { return func(ctx context.Context) (error, int64) {
_, err := kvc.Range(ctx, &pb.RangeRequest{ _, err := cli.Get(ctx, fmt.Sprintf("foo%016x", rand.Intn(keySuffixRange)))
Key: []byte(fmt.Sprintf("foo%016x", rand.Intn(keySuffixRange))),
}, grpc.FailFast(false))
return err, 0 return err, 0
} }
} }
func newStressRangeInterval(kvc pb.KVClient, keySuffixRange int) stressFunc { func newStressRangeInterval(cli *clientv3.Client, keySuffixRange int) stressFunc {
return func(ctx context.Context) (error, int64) { return func(ctx context.Context) (error, int64) {
start := rand.Intn(keySuffixRange) start := rand.Intn(keySuffixRange)
end := start + 500 end := start + 500
_, err := kvc.Range(ctx, &pb.RangeRequest{ _, err := cli.Get(
Key: []byte(fmt.Sprintf("foo%016x", start)), ctx,
RangeEnd: []byte(fmt.Sprintf("foo%016x", end)), fmt.Sprintf("foo%016x", start),
}, grpc.FailFast(false)) clientv3.WithRange(fmt.Sprintf("foo%016x", end)),
)
return err, 0 return err, 0
} }
} }
func newStressDelete(kvc pb.KVClient, keySuffixRange int) stressFunc { func newStressDelete(cli *clientv3.Client, keySuffixRange int) stressFunc {
return func(ctx context.Context) (error, int64) { return func(ctx context.Context) (error, int64) {
_, err := kvc.DeleteRange(ctx, &pb.DeleteRangeRequest{ _, err := cli.Delete(ctx, fmt.Sprintf("foo%016x", rand.Intn(keySuffixRange)))
Key: []byte(fmt.Sprintf("foo%016x", rand.Intn(keySuffixRange))),
}, grpc.FailFast(false))
return err, 1 return err, 1
} }
} }
func newStressDeleteInterval(kvc pb.KVClient, keySuffixRange int) stressFunc { func newStressDeleteInterval(cli *clientv3.Client, keySuffixRange int) stressFunc {
return func(ctx context.Context) (error, int64) { return func(ctx context.Context) (error, int64) {
start := rand.Intn(keySuffixRange) start := rand.Intn(keySuffixRange)
end := start + 500 end := start + 500
resp, err := kvc.DeleteRange(ctx, &pb.DeleteRangeRequest{ resp, err := cli.Delete(ctx,
Key: []byte(fmt.Sprintf("foo%016x", start)), fmt.Sprintf("foo%016x", start),
RangeEnd: []byte(fmt.Sprintf("foo%016x", end)), clientv3.WithRange(fmt.Sprintf("foo%016x", end)),
}, grpc.FailFast(false)) )
if err == nil { if err == nil {
return nil, resp.Deleted return nil, resp.Deleted
} }