mirror of
https://gitee.com/milvus-io/milvus.git
synced 2024-12-02 11:59:00 +08:00
6eff45989e
Signed-off-by: xige-16 <xi.ge@zilliz.com>
149 lines
3.7 KiB
Go
149 lines
3.7 KiB
Go
// Copyright 2016 TiKV Project 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,
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package kv
|
|
|
|
import (
|
|
"context"
|
|
"path"
|
|
"time"
|
|
"github.com/czs007/suvlim/util/etcdutil"
|
|
"github.com/czs007/suvlim/errors"
|
|
"github.com/pingcap/log"
|
|
"go.etcd.io/etcd/clientv3"
|
|
"go.uber.org/zap"
|
|
)
|
|
|
|
const (
|
|
requestTimeout = 10 * time.Second
|
|
slowRequestTime = 1 * time.Second
|
|
)
|
|
|
|
var (
|
|
errTxnFailed = errors.New("failed to commit transaction")
|
|
)
|
|
|
|
type etcdKVBase struct {
|
|
client *clientv3.Client
|
|
rootPath string
|
|
}
|
|
|
|
// NewEtcdKVBase creates a new etcd kv.
|
|
func NewEtcdKVBase(client *clientv3.Client, rootPath string) *etcdKVBase {
|
|
return &etcdKVBase{
|
|
client: client,
|
|
rootPath: rootPath,
|
|
}
|
|
}
|
|
|
|
func (kv *etcdKVBase) Load(key string) (string, error) {
|
|
key = path.Join(kv.rootPath, key)
|
|
|
|
resp, err := etcdutil.EtcdKVGet(kv.client, key)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
if n := len(resp.Kvs); n == 0 {
|
|
return "", nil
|
|
} else if n > 1 {
|
|
return "", errors.Errorf("load more than one kvs: key %v kvs %v", key, n)
|
|
}
|
|
return string(resp.Kvs[0].Value), nil
|
|
}
|
|
|
|
func (kv *etcdKVBase) Save(key, value string) error {
|
|
key = path.Join(kv.rootPath, key)
|
|
|
|
txn := NewSlowLogTxn(kv.client)
|
|
resp, err := txn.Then(clientv3.OpPut(key, value)).Commit()
|
|
if err != nil {
|
|
log.Error("save to etcd meet error", zap.String("key", key), zap.String("value", value))
|
|
return errors.WithStack(err)
|
|
}
|
|
if !resp.Succeeded {
|
|
return errors.WithStack(errTxnFailed)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (kv *etcdKVBase) Remove(key string) error {
|
|
key = path.Join(kv.rootPath, key)
|
|
|
|
txn := NewSlowLogTxn(kv.client)
|
|
resp, err := txn.Then(clientv3.OpDelete(key)).Commit()
|
|
if err != nil {
|
|
log.Error("remove from etcd meet error", zap.String("key", key))
|
|
return errors.WithStack(err)
|
|
}
|
|
if !resp.Succeeded {
|
|
return errors.WithStack(errTxnFailed)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// SlowLogTxn wraps etcd transaction and log slow one.
|
|
type SlowLogTxn struct {
|
|
clientv3.Txn
|
|
cancel context.CancelFunc
|
|
}
|
|
|
|
// NewSlowLogTxn create a SlowLogTxn.
|
|
func NewSlowLogTxn(client *clientv3.Client) clientv3.Txn {
|
|
ctx, cancel := context.WithTimeout(client.Ctx(), requestTimeout)
|
|
return &SlowLogTxn{
|
|
Txn: client.Txn(ctx),
|
|
cancel: cancel,
|
|
}
|
|
}
|
|
|
|
// If takes a list of comparison. If all comparisons passed in succeed,
|
|
// the operations passed into Then() will be executed. Or the operations
|
|
// passed into Else() will be executed.
|
|
func (t *SlowLogTxn) If(cs ...clientv3.Cmp) clientv3.Txn {
|
|
return &SlowLogTxn{
|
|
Txn: t.Txn.If(cs...),
|
|
cancel: t.cancel,
|
|
}
|
|
}
|
|
|
|
// Then takes a list of operations. The Ops list will be executed, if the
|
|
// comparisons passed in If() succeed.
|
|
func (t *SlowLogTxn) Then(ops ...clientv3.Op) clientv3.Txn {
|
|
return &SlowLogTxn{
|
|
Txn: t.Txn.Then(ops...),
|
|
cancel: t.cancel,
|
|
}
|
|
}
|
|
|
|
// Commit implements Txn Commit interface.
|
|
func (t *SlowLogTxn) Commit() (*clientv3.TxnResponse, error) {
|
|
start := time.Now()
|
|
resp, err := t.Txn.Commit()
|
|
t.cancel()
|
|
|
|
cost := time.Since(start)
|
|
if cost > slowRequestTime {
|
|
log.Warn("txn runs too slow",
|
|
zap.Error(err),
|
|
zap.Reflect("response", resp),
|
|
zap.Duration("cost", cost))
|
|
}
|
|
//label := "success"
|
|
//if err != nil {
|
|
// label = "failed"
|
|
//}
|
|
//txnCounter.WithLabelValues(label).Inc()
|
|
//txnDuration.WithLabelValues(label).Observe(cost.Seconds())
|
|
return resp, errors.WithStack(err)
|
|
}
|