mirror of https://github.com/milvus-io/milvus.git
master client,reconnect if grpc failed (#5360)
master client,reconnect if grpc failed Signed-off-by: yefu.chen <yefu.chen@zilliz.com>pull/5362/head
parent
d189bf4d33
commit
e326eaad1e
|
@ -35,6 +35,7 @@ type UniqueID = typeutil.UniqueID
|
||||||
type IDAllocator struct {
|
type IDAllocator struct {
|
||||||
Allocator
|
Allocator
|
||||||
|
|
||||||
|
etcdAddr []string
|
||||||
masterAddress string
|
masterAddress string
|
||||||
masterClient types.MasterService
|
masterClient types.MasterService
|
||||||
|
|
||||||
|
@ -46,7 +47,7 @@ type IDAllocator struct {
|
||||||
PeerID UniqueID
|
PeerID UniqueID
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewIDAllocator(ctx context.Context, masterAddr string) (*IDAllocator, error) {
|
func NewIDAllocator(ctx context.Context, masterAddr string, etcdAddr []string) (*IDAllocator, error) {
|
||||||
|
|
||||||
ctx1, cancel := context.WithCancel(ctx)
|
ctx1, cancel := context.WithCancel(ctx)
|
||||||
a := &IDAllocator{
|
a := &IDAllocator{
|
||||||
|
@ -56,6 +57,7 @@ func NewIDAllocator(ctx context.Context, masterAddr string) (*IDAllocator, error
|
||||||
Role: "IDAllocator",
|
Role: "IDAllocator",
|
||||||
},
|
},
|
||||||
countPerRPC: IDCountPerRPC,
|
countPerRPC: IDCountPerRPC,
|
||||||
|
etcdAddr: etcdAddr,
|
||||||
masterAddress: masterAddr,
|
masterAddress: masterAddr,
|
||||||
}
|
}
|
||||||
a.TChan = &EmptyTicker{}
|
a.TChan = &EmptyTicker{}
|
||||||
|
@ -69,7 +71,8 @@ func NewIDAllocator(ctx context.Context, masterAddr string) (*IDAllocator, error
|
||||||
|
|
||||||
func (ia *IDAllocator) Start() error {
|
func (ia *IDAllocator) Start() error {
|
||||||
var err error
|
var err error
|
||||||
ia.masterClient, err = msc.NewClient(ia.masterAddress, 20*time.Second)
|
|
||||||
|
ia.masterClient, err = msc.NewClient(ia.masterAddress, ia.etcdAddr, 20*time.Second)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -138,9 +138,8 @@ func (node *DataNode) SetDataServiceInterface(ds types.DataService) error {
|
||||||
func (node *DataNode) Init() error {
|
func (node *DataNode) Init() error {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
|
|
||||||
node.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress}, typeutil.DataNodeRole,
|
node.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress})
|
||||||
Params.IP+":"+strconv.Itoa(Params.Port), false)
|
node.session.Init(typeutil.DataNodeRole, Params.IP+":"+strconv.Itoa(Params.Port), false)
|
||||||
node.session.Init()
|
|
||||||
|
|
||||||
req := &datapb.RegisterNodeRequest{
|
req := &datapb.RegisterNodeRequest{
|
||||||
Base: &commonpb.MsgBase{
|
Base: &commonpb.MsgBase{
|
||||||
|
|
|
@ -108,9 +108,8 @@ func (s *Server) SetMasterClient(masterClient types.MasterService) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Server) Init() error {
|
func (s *Server) Init() error {
|
||||||
s.session = sessionutil.NewSession(s.ctx, []string{Params.EtcdAddress}, typeutil.DataServiceRole,
|
s.session = sessionutil.NewSession(s.ctx, []string{Params.EtcdAddress})
|
||||||
Params.IP, true)
|
s.session.Init(typeutil.DataServiceRole, Params.IP, true)
|
||||||
s.session.Init()
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,7 @@ import (
|
||||||
"github.com/milvus-io/milvus/internal/proto/internalpb"
|
"github.com/milvus-io/milvus/internal/proto/internalpb"
|
||||||
"github.com/milvus-io/milvus/internal/types"
|
"github.com/milvus-io/milvus/internal/types"
|
||||||
"github.com/milvus-io/milvus/internal/util/retry"
|
"github.com/milvus-io/milvus/internal/util/retry"
|
||||||
|
"github.com/milvus-io/milvus/internal/util/sessionutil"
|
||||||
"github.com/milvus-io/milvus/internal/util/typeutil"
|
"github.com/milvus-io/milvus/internal/util/typeutil"
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"go.etcd.io/etcd/clientv3"
|
"go.etcd.io/etcd/clientv3"
|
||||||
|
@ -787,7 +788,7 @@ func newTestServer(t *testing.T) *Server {
|
||||||
|
|
||||||
etcdCli, err := initEtcd(Params.EtcdAddress)
|
etcdCli, err := initEtcd(Params.EtcdAddress)
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
_, err = etcdCli.Delete(context.Background(), "/session", clientv3.WithPrefix())
|
_, err = etcdCli.Delete(context.Background(), sessionutil.DefaultServiceRoot, clientv3.WithPrefix())
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
svr, err := CreateServer(context.TODO(), factory)
|
svr, err := CreateServer(context.TODO(), factory)
|
||||||
|
|
|
@ -70,7 +70,7 @@ func NewServer(ctx context.Context, factory msgstream.Factory) (*Server, error)
|
||||||
msFactory: factory,
|
msFactory: factory,
|
||||||
grpcErrChan: make(chan error),
|
grpcErrChan: make(chan error),
|
||||||
newMasterServiceClient: func(s string) (types.MasterService, error) {
|
newMasterServiceClient: func(s string) (types.MasterService, error) {
|
||||||
return msc.NewClient(s, 20*time.Second)
|
return msc.NewClient(s, []string{dn.Params.EtcdAddress}, 20*time.Second)
|
||||||
},
|
},
|
||||||
newDataServiceClient: func(s string) types.DataService {
|
newDataServiceClient: func(s string) types.DataService {
|
||||||
return dsc.NewClient(Params.DataServiceAddress)
|
return dsc.NewClient(Params.DataServiceAddress)
|
||||||
|
|
|
@ -68,7 +68,7 @@ func NewServer(ctx context.Context, factory msgstream.Factory) (*Server, error)
|
||||||
cancel: cancel,
|
cancel: cancel,
|
||||||
grpcErrChan: make(chan error),
|
grpcErrChan: make(chan error),
|
||||||
newMasterServiceClient: func(s string) (types.MasterService, error) {
|
newMasterServiceClient: func(s string) (types.MasterService, error) {
|
||||||
return msc.NewClient(s, 10*time.Second)
|
return msc.NewClient(s, []string{dataservice.Params.EtcdAddress}, 10*time.Second)
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
s.dataService, err = dataservice.CreateServer(s.ctx, factory)
|
s.dataService, err = dataservice.CreateServer(s.ctx, factory)
|
||||||
|
|
|
@ -13,12 +13,16 @@ package grpcmasterserviceclient
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"fmt"
|
||||||
|
"path"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/milvus-io/milvus/internal/proto/commonpb"
|
"github.com/milvus-io/milvus/internal/proto/commonpb"
|
||||||
"github.com/milvus-io/milvus/internal/proto/internalpb"
|
"github.com/milvus-io/milvus/internal/proto/internalpb"
|
||||||
"github.com/milvus-io/milvus/internal/proto/masterpb"
|
"github.com/milvus-io/milvus/internal/proto/masterpb"
|
||||||
"github.com/milvus-io/milvus/internal/proto/milvuspb"
|
"github.com/milvus-io/milvus/internal/proto/milvuspb"
|
||||||
|
"github.com/milvus-io/milvus/internal/util/sessionutil"
|
||||||
|
"github.com/milvus-io/milvus/internal/util/typeutil"
|
||||||
otgrpc "github.com/opentracing-contrib/go-grpc"
|
otgrpc "github.com/opentracing-contrib/go-grpc"
|
||||||
"github.com/opentracing/opentracing-go"
|
"github.com/opentracing/opentracing-go"
|
||||||
"google.golang.org/grpc"
|
"google.golang.org/grpc"
|
||||||
|
@ -32,28 +36,57 @@ type GrpcClient struct {
|
||||||
//inner member
|
//inner member
|
||||||
addr string
|
addr string
|
||||||
timeout time.Duration
|
timeout time.Duration
|
||||||
grpcTimeout time.Duration
|
reconnTry int
|
||||||
retry int
|
recallTry int
|
||||||
|
|
||||||
|
sess *sessionutil.Session
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewClient(addr string, timeout time.Duration) (*GrpcClient, error) {
|
func getMasterServiceAddr(sess *sessionutil.Session) (string, error) {
|
||||||
|
msess, err := sess.GetSessions(typeutil.MasterServiceRole)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
key := path.Join(sessionutil.DefaultServiceRoot, typeutil.MasterServiceRole)
|
||||||
|
var ms *sessionutil.Session
|
||||||
|
var ok bool
|
||||||
|
if ms, ok = msess[key]; !ok {
|
||||||
|
return "", fmt.Errorf("number of master service is incorrect, %d", len(msess))
|
||||||
|
}
|
||||||
|
return ms.Address, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewClient(addr string, etcdAddr []string, timeout time.Duration) (*GrpcClient, error) {
|
||||||
|
sess := sessionutil.NewSession(context.Background(), etcdAddr)
|
||||||
|
|
||||||
|
if addr == "" {
|
||||||
|
var err error
|
||||||
|
if addr, err = getMasterServiceAddr(sess); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return &GrpcClient{
|
return &GrpcClient{
|
||||||
grpcClient: nil,
|
grpcClient: nil,
|
||||||
conn: nil,
|
conn: nil,
|
||||||
addr: addr,
|
addr: addr,
|
||||||
timeout: timeout,
|
timeout: timeout,
|
||||||
grpcTimeout: time.Second * 5,
|
reconnTry: 300,
|
||||||
retry: 300,
|
recallTry: 3,
|
||||||
|
sess: sess,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) Init() error {
|
func (c *GrpcClient) reconnect() error {
|
||||||
|
addr, err := getMasterServiceAddr(c.sess)
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
tracer := opentracing.GlobalTracer()
|
tracer := opentracing.GlobalTracer()
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), c.timeout)
|
ctx, cancel := context.WithTimeout(context.Background(), c.timeout)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
var err error
|
for i := 0; i < c.reconnTry; i++ {
|
||||||
for i := 0; i < c.retry; i++ {
|
if c.conn, err = grpc.DialContext(ctx, addr, grpc.WithInsecure(), grpc.WithBlock(),
|
||||||
if c.conn, err = grpc.DialContext(ctx, c.addr, grpc.WithInsecure(), grpc.WithBlock(),
|
|
||||||
grpc.WithUnaryInterceptor(
|
grpc.WithUnaryInterceptor(
|
||||||
otgrpc.OpenTracingClientInterceptor(tracer)),
|
otgrpc.OpenTracingClientInterceptor(tracer)),
|
||||||
grpc.WithStreamInterceptor(
|
grpc.WithStreamInterceptor(
|
||||||
|
@ -68,6 +101,27 @@ func (c *GrpcClient) Init() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *GrpcClient) Init() error {
|
||||||
|
tracer := opentracing.GlobalTracer()
|
||||||
|
ctx, cancel := context.WithTimeout(context.Background(), c.timeout)
|
||||||
|
defer cancel()
|
||||||
|
var err error
|
||||||
|
for i := 0; i < c.reconnTry; i++ {
|
||||||
|
if c.conn, err = grpc.DialContext(ctx, c.addr, grpc.WithInsecure(), grpc.WithBlock(),
|
||||||
|
grpc.WithUnaryInterceptor(
|
||||||
|
otgrpc.OpenTracingClientInterceptor(tracer)),
|
||||||
|
grpc.WithStreamInterceptor(
|
||||||
|
otgrpc.OpenTracingStreamClientInterceptor(tracer))); err == nil {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return c.reconnect()
|
||||||
|
}
|
||||||
|
c.grpcClient = masterpb.NewMasterServiceClient(c.conn)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) Start() error {
|
func (c *GrpcClient) Start() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -75,91 +129,171 @@ func (c *GrpcClient) Stop() error {
|
||||||
return c.conn.Close()
|
return c.conn.Close()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (c *GrpcClient) recall(caller func() (interface{}, error)) (interface{}, error) {
|
||||||
|
ret, err := caller()
|
||||||
|
if err == nil {
|
||||||
|
return ret, nil
|
||||||
|
}
|
||||||
|
for i := 0; i < c.recallTry; i++ {
|
||||||
|
err = c.reconnect()
|
||||||
|
if err == nil {
|
||||||
|
ret, err = caller()
|
||||||
|
if err == nil {
|
||||||
|
return ret, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ret, err
|
||||||
|
}
|
||||||
|
|
||||||
// TODO: timeout need to be propagated through ctx
|
// TODO: timeout need to be propagated through ctx
|
||||||
func (c *GrpcClient) GetComponentStates(ctx context.Context) (*internalpb.ComponentStates, error) {
|
func (c *GrpcClient) GetComponentStates(ctx context.Context) (*internalpb.ComponentStates, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.GetComponentStates(ctx, &internalpb.GetComponentStatesRequest{})
|
return c.grpcClient.GetComponentStates(ctx, &internalpb.GetComponentStatesRequest{})
|
||||||
|
})
|
||||||
|
return ret.(*internalpb.ComponentStates), err
|
||||||
}
|
}
|
||||||
func (c *GrpcClient) GetTimeTickChannel(ctx context.Context) (*milvuspb.StringResponse, error) {
|
func (c *GrpcClient) GetTimeTickChannel(ctx context.Context) (*milvuspb.StringResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.GetTimeTickChannel(ctx, &internalpb.GetTimeTickChannelRequest{})
|
return c.grpcClient.GetTimeTickChannel(ctx, &internalpb.GetTimeTickChannelRequest{})
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.StringResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
//just define a channel, not used currently
|
//just define a channel, not used currently
|
||||||
func (c *GrpcClient) GetStatisticsChannel(ctx context.Context) (*milvuspb.StringResponse, error) {
|
func (c *GrpcClient) GetStatisticsChannel(ctx context.Context) (*milvuspb.StringResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.GetStatisticsChannel(ctx, &internalpb.GetStatisticsChannelRequest{})
|
return c.grpcClient.GetStatisticsChannel(ctx, &internalpb.GetStatisticsChannelRequest{})
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.StringResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
//receive ddl from rpc and time tick from proxy service, and put them into this channel
|
//receive ddl from rpc and time tick from proxy service, and put them into this channel
|
||||||
func (c *GrpcClient) GetDdChannel(ctx context.Context) (*milvuspb.StringResponse, error) {
|
func (c *GrpcClient) GetDdChannel(ctx context.Context) (*milvuspb.StringResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.GetDdChannel(ctx, &internalpb.GetDdChannelRequest{})
|
return c.grpcClient.GetDdChannel(ctx, &internalpb.GetDdChannelRequest{})
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.StringResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
//DDL request
|
//DDL request
|
||||||
func (c *GrpcClient) CreateCollection(ctx context.Context, in *milvuspb.CreateCollectionRequest) (*commonpb.Status, error) {
|
func (c *GrpcClient) CreateCollection(ctx context.Context, in *milvuspb.CreateCollectionRequest) (*commonpb.Status, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.CreateCollection(ctx, in)
|
return c.grpcClient.CreateCollection(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*commonpb.Status), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) DropCollection(ctx context.Context, in *milvuspb.DropCollectionRequest) (*commonpb.Status, error) {
|
func (c *GrpcClient) DropCollection(ctx context.Context, in *milvuspb.DropCollectionRequest) (*commonpb.Status, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.DropCollection(ctx, in)
|
return c.grpcClient.DropCollection(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*commonpb.Status), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) HasCollection(ctx context.Context, in *milvuspb.HasCollectionRequest) (*milvuspb.BoolResponse, error) {
|
func (c *GrpcClient) HasCollection(ctx context.Context, in *milvuspb.HasCollectionRequest) (*milvuspb.BoolResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.HasCollection(ctx, in)
|
return c.grpcClient.HasCollection(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.BoolResponse), err
|
||||||
}
|
}
|
||||||
func (c *GrpcClient) DescribeCollection(ctx context.Context, in *milvuspb.DescribeCollectionRequest) (*milvuspb.DescribeCollectionResponse, error) {
|
func (c *GrpcClient) DescribeCollection(ctx context.Context, in *milvuspb.DescribeCollectionRequest) (*milvuspb.DescribeCollectionResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.DescribeCollection(ctx, in)
|
return c.grpcClient.DescribeCollection(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.DescribeCollectionResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) ShowCollections(ctx context.Context, in *milvuspb.ShowCollectionsRequest) (*milvuspb.ShowCollectionsResponse, error) {
|
func (c *GrpcClient) ShowCollections(ctx context.Context, in *milvuspb.ShowCollectionsRequest) (*milvuspb.ShowCollectionsResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.ShowCollections(ctx, in)
|
return c.grpcClient.ShowCollections(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.ShowCollectionsResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) CreatePartition(ctx context.Context, in *milvuspb.CreatePartitionRequest) (*commonpb.Status, error) {
|
func (c *GrpcClient) CreatePartition(ctx context.Context, in *milvuspb.CreatePartitionRequest) (*commonpb.Status, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.CreatePartition(ctx, in)
|
return c.grpcClient.CreatePartition(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*commonpb.Status), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) DropPartition(ctx context.Context, in *milvuspb.DropPartitionRequest) (*commonpb.Status, error) {
|
func (c *GrpcClient) DropPartition(ctx context.Context, in *milvuspb.DropPartitionRequest) (*commonpb.Status, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.DropPartition(ctx, in)
|
return c.grpcClient.DropPartition(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*commonpb.Status), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) HasPartition(ctx context.Context, in *milvuspb.HasPartitionRequest) (*milvuspb.BoolResponse, error) {
|
func (c *GrpcClient) HasPartition(ctx context.Context, in *milvuspb.HasPartitionRequest) (*milvuspb.BoolResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.HasPartition(ctx, in)
|
return c.grpcClient.HasPartition(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.BoolResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) ShowPartitions(ctx context.Context, in *milvuspb.ShowPartitionsRequest) (*milvuspb.ShowPartitionsResponse, error) {
|
func (c *GrpcClient) ShowPartitions(ctx context.Context, in *milvuspb.ShowPartitionsRequest) (*milvuspb.ShowPartitionsResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.ShowPartitions(ctx, in)
|
return c.grpcClient.ShowPartitions(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.ShowPartitionsResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
//index builder service
|
//index builder service
|
||||||
func (c *GrpcClient) CreateIndex(ctx context.Context, in *milvuspb.CreateIndexRequest) (*commonpb.Status, error) {
|
func (c *GrpcClient) CreateIndex(ctx context.Context, in *milvuspb.CreateIndexRequest) (*commonpb.Status, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.CreateIndex(ctx, in)
|
return c.grpcClient.CreateIndex(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*commonpb.Status), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) DropIndex(ctx context.Context, in *milvuspb.DropIndexRequest) (*commonpb.Status, error) {
|
func (c *GrpcClient) DropIndex(ctx context.Context, in *milvuspb.DropIndexRequest) (*commonpb.Status, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.DropIndex(ctx, in)
|
return c.grpcClient.DropIndex(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*commonpb.Status), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) DescribeIndex(ctx context.Context, in *milvuspb.DescribeIndexRequest) (*milvuspb.DescribeIndexResponse, error) {
|
func (c *GrpcClient) DescribeIndex(ctx context.Context, in *milvuspb.DescribeIndexRequest) (*milvuspb.DescribeIndexResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.DescribeIndex(ctx, in)
|
return c.grpcClient.DescribeIndex(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.DescribeIndexResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
//global timestamp allocator
|
//global timestamp allocator
|
||||||
func (c *GrpcClient) AllocTimestamp(ctx context.Context, in *masterpb.AllocTimestampRequest) (*masterpb.AllocTimestampResponse, error) {
|
func (c *GrpcClient) AllocTimestamp(ctx context.Context, in *masterpb.AllocTimestampRequest) (*masterpb.AllocTimestampResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.AllocTimestamp(ctx, in)
|
return c.grpcClient.AllocTimestamp(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*masterpb.AllocTimestampResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) AllocID(ctx context.Context, in *masterpb.AllocIDRequest) (*masterpb.AllocIDResponse, error) {
|
func (c *GrpcClient) AllocID(ctx context.Context, in *masterpb.AllocIDRequest) (*masterpb.AllocIDResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.AllocID(ctx, in)
|
return c.grpcClient.AllocID(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*masterpb.AllocIDResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
// UpdateChannelTimeTick used to handle ChannelTimeTickMsg
|
// UpdateChannelTimeTick used to handle ChannelTimeTickMsg
|
||||||
func (c *GrpcClient) UpdateChannelTimeTick(ctx context.Context, in *internalpb.ChannelTimeTickMsg) (*commonpb.Status, error) {
|
func (c *GrpcClient) UpdateChannelTimeTick(ctx context.Context, in *internalpb.ChannelTimeTickMsg) (*commonpb.Status, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.UpdateChannelTimeTick(ctx, in)
|
return c.grpcClient.UpdateChannelTimeTick(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*commonpb.Status), err
|
||||||
}
|
}
|
||||||
|
|
||||||
//receiver time tick from proxy service, and put it into this channel
|
//receiver time tick from proxy service, and put it into this channel
|
||||||
func (c *GrpcClient) DescribeSegment(ctx context.Context, in *milvuspb.DescribeSegmentRequest) (*milvuspb.DescribeSegmentResponse, error) {
|
func (c *GrpcClient) DescribeSegment(ctx context.Context, in *milvuspb.DescribeSegmentRequest) (*milvuspb.DescribeSegmentResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.DescribeSegment(ctx, in)
|
return c.grpcClient.DescribeSegment(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.DescribeSegmentResponse), err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *GrpcClient) ShowSegments(ctx context.Context, in *milvuspb.ShowSegmentsRequest) (*milvuspb.ShowSegmentsResponse, error) {
|
func (c *GrpcClient) ShowSegments(ctx context.Context, in *milvuspb.ShowSegmentsRequest) (*milvuspb.ShowSegmentsResponse, error) {
|
||||||
|
ret, err := c.recall(func() (interface{}, error) {
|
||||||
return c.grpcClient.ShowSegments(ctx, in)
|
return c.grpcClient.ShowSegments(ctx, in)
|
||||||
|
})
|
||||||
|
return ret.(*milvuspb.ShowSegmentsResponse), err
|
||||||
}
|
}
|
||||||
|
|
|
@ -35,6 +35,7 @@ import (
|
||||||
"github.com/milvus-io/milvus/internal/proto/schemapb"
|
"github.com/milvus-io/milvus/internal/proto/schemapb"
|
||||||
"github.com/milvus-io/milvus/internal/types"
|
"github.com/milvus-io/milvus/internal/types"
|
||||||
"github.com/milvus-io/milvus/internal/util/retry"
|
"github.com/milvus-io/milvus/internal/util/retry"
|
||||||
|
"github.com/milvus-io/milvus/internal/util/sessionutil"
|
||||||
"github.com/milvus-io/milvus/internal/util/typeutil"
|
"github.com/milvus-io/milvus/internal/util/typeutil"
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"go.etcd.io/etcd/clientv3"
|
"go.etcd.io/etcd/clientv3"
|
||||||
|
@ -86,7 +87,7 @@ func TestGrpcService(t *testing.T) {
|
||||||
|
|
||||||
etcdCli, err := initEtcd(cms.Params.EtcdAddress)
|
etcdCli, err := initEtcd(cms.Params.EtcdAddress)
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
_, err = etcdCli.Delete(ctx, "/session", clientv3.WithPrefix())
|
_, err = etcdCli.Delete(ctx, sessionutil.DefaultServiceRoot, clientv3.WithPrefix())
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
err = core.Init()
|
err = core.Init()
|
||||||
|
@ -171,7 +172,7 @@ func TestGrpcService(t *testing.T) {
|
||||||
|
|
||||||
svr.masterService.UpdateStateCode(internalpb.StateCode_Healthy)
|
svr.masterService.UpdateStateCode(internalpb.StateCode_Healthy)
|
||||||
|
|
||||||
cli, err := grpcmasterserviceclient.NewClient(Params.Address, 3*time.Second)
|
cli, err := grpcmasterserviceclient.NewClient(Params.Address, []string{cms.Params.EtcdAddress}, 3*time.Second)
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
err = cli.Init()
|
err = cli.Init()
|
||||||
|
@ -871,7 +872,7 @@ func TestRun(t *testing.T) {
|
||||||
|
|
||||||
etcdCli, err := initEtcd(cms.Params.EtcdAddress)
|
etcdCli, err := initEtcd(cms.Params.EtcdAddress)
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
_, err = etcdCli.Delete(ctx, "/session", clientv3.WithPrefix())
|
_, err = etcdCli.Delete(ctx, sessionutil.DefaultServiceRoot, clientv3.WithPrefix())
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
err = svr.Run()
|
err = svr.Run()
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
|
@ -184,7 +184,7 @@ func (s *Server) init() error {
|
||||||
masterServiceAddr := Params.MasterAddress
|
masterServiceAddr := Params.MasterAddress
|
||||||
log.Debug("proxynode", zap.String("master address", masterServiceAddr))
|
log.Debug("proxynode", zap.String("master address", masterServiceAddr))
|
||||||
timeout := 3 * time.Second
|
timeout := 3 * time.Second
|
||||||
s.masterServiceClient, err = grpcmasterserviceclient.NewClient(masterServiceAddr, timeout)
|
s.masterServiceClient, err = grpcmasterserviceclient.NewClient(masterServiceAddr, []string{proxynode.Params.EtcdAddress}, timeout)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
|
@ -130,7 +130,7 @@ func (s *Server) init() error {
|
||||||
log.Debug("Master service", zap.String("address", addr))
|
log.Debug("Master service", zap.String("address", addr))
|
||||||
log.Debug("Init master service client ...")
|
log.Debug("Init master service client ...")
|
||||||
|
|
||||||
masterService, err := msc.NewClient(addr, 20*time.Second)
|
masterService, err := msc.NewClient(addr, []string{qn.Params.EtcdAddress}, 20*time.Second)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -107,7 +107,7 @@ func (s *Server) init() error {
|
||||||
log.Debug("Master service", zap.String("address", Params.MasterAddress))
|
log.Debug("Master service", zap.String("address", Params.MasterAddress))
|
||||||
log.Debug("Init master service client ...")
|
log.Debug("Init master service client ...")
|
||||||
|
|
||||||
masterService, err := msc.NewClient(Params.MasterAddress, 20*time.Second)
|
masterService, err := msc.NewClient(Params.MasterAddress, []string{qs.Params.EtcdAddress}, 20*time.Second)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
panic(err)
|
panic(err)
|
||||||
|
|
|
@ -79,9 +79,8 @@ func NewIndexNode(ctx context.Context) (*IndexNode, error) {
|
||||||
|
|
||||||
func (i *IndexNode) Init() error {
|
func (i *IndexNode) Init() error {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
i.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress}, typeutil.IndexNodeRole,
|
i.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress})
|
||||||
Params.IP+":"+strconv.Itoa(Params.Port), false)
|
i.session.Init(typeutil.IndexNodeRole, Params.IP+":"+strconv.Itoa(Params.Port), false)
|
||||||
i.session.Init()
|
|
||||||
|
|
||||||
err := funcutil.WaitForComponentHealthy(ctx, i.serviceClient, "IndexService", 1000000, time.Millisecond*200)
|
err := funcutil.WaitForComponentHealthy(ctx, i.serviceClient, "IndexService", 1000000, time.Millisecond*200)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|
|
@ -89,9 +89,8 @@ func (i *IndexService) Init() error {
|
||||||
log.Debug("indexservice", zap.String("etcd address", Params.EtcdAddress))
|
log.Debug("indexservice", zap.String("etcd address", Params.EtcdAddress))
|
||||||
|
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
i.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress}, typeutil.IndexServiceRole,
|
i.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress})
|
||||||
Params.Address, true)
|
i.session.Init(typeutil.IndexServiceRole, Params.Address, true)
|
||||||
i.session.Init()
|
|
||||||
|
|
||||||
connectEtcdFn := func() error {
|
connectEtcdFn := func() error {
|
||||||
etcdClient, err := clientv3.New(clientv3.Config{Endpoints: []string{Params.EtcdAddress}})
|
etcdClient, err := clientv3.New(clientv3.Config{Endpoints: []string{Params.EtcdAddress}})
|
||||||
|
|
|
@ -821,9 +821,8 @@ func (c *Core) BuildIndex(segID typeutil.UniqueID, field *schemapb.FieldSchema,
|
||||||
func (c *Core) Init() error {
|
func (c *Core) Init() error {
|
||||||
var initError error = nil
|
var initError error = nil
|
||||||
c.initOnce.Do(func() {
|
c.initOnce.Do(func() {
|
||||||
c.session = sessionutil.NewSession(c.ctx, []string{Params.EtcdAddress}, typeutil.MasterServiceRole,
|
c.session = sessionutil.NewSession(c.ctx, []string{Params.EtcdAddress})
|
||||||
Params.Address, true)
|
c.session.Init(typeutil.MasterServiceRole, Params.Address, true)
|
||||||
c.session.Init()
|
|
||||||
|
|
||||||
connectEtcdFn := func() error {
|
connectEtcdFn := func() error {
|
||||||
if c.etcdCli, initError = clientv3.New(clientv3.Config{Endpoints: []string{Params.EtcdAddress}, DialTimeout: 5 * time.Second}); initError != nil {
|
if c.etcdCli, initError = clientv3.New(clientv3.Config{Endpoints: []string{Params.EtcdAddress}, DialTimeout: 5 * time.Second}); initError != nil {
|
||||||
|
|
|
@ -16,6 +16,7 @@ import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"math/rand"
|
"math/rand"
|
||||||
|
"path"
|
||||||
"sync"
|
"sync"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
@ -218,10 +219,10 @@ func TestMasterService(t *testing.T) {
|
||||||
|
|
||||||
etcdCli, err := clientv3.New(clientv3.Config{Endpoints: []string{Params.EtcdAddress}, DialTimeout: 5 * time.Second})
|
etcdCli, err := clientv3.New(clientv3.Config{Endpoints: []string{Params.EtcdAddress}, DialTimeout: 5 * time.Second})
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
_, err = etcdCli.Delete(ctx, ProxyNodeSessionPrefix, clientv3.WithPrefix())
|
_, err = etcdCli.Delete(ctx, sessionutil.DefaultServiceRoot, clientv3.WithPrefix())
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
defer func() {
|
defer func() {
|
||||||
_, _ = etcdCli.Delete(ctx, ProxyNodeSessionPrefix, clientv3.WithPrefix())
|
_, _ = etcdCli.Delete(ctx, sessionutil.DefaultServiceRoot, clientv3.WithPrefix())
|
||||||
}()
|
}()
|
||||||
|
|
||||||
pm := &proxyMock{
|
pm := &proxyMock{
|
||||||
|
@ -253,14 +254,6 @@ func TestMasterService(t *testing.T) {
|
||||||
err = core.SetQueryService(qm)
|
err = core.SetQueryService(qm)
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
//TODO initialize master's session manager before core init
|
|
||||||
/*
|
|
||||||
self := sessionutil.NewSession("masterservice", funcutil.GetLocalIP()+":"+strconv.Itoa(53100), true)
|
|
||||||
sm := sessionutil.NewSessionManager(ctx, Params.EtcdAddress, Params.MetaRootPath, self)
|
|
||||||
sm.Init()
|
|
||||||
sessionutil.SetGlobalSessionManager(sm)
|
|
||||||
*/
|
|
||||||
|
|
||||||
err = core.Init()
|
err = core.Init()
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
|
@ -1458,9 +1451,9 @@ func TestMasterService(t *testing.T) {
|
||||||
s2, err := json.Marshal(&p2)
|
s2, err := json.Marshal(&p2)
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
_, err = core.etcdCli.Put(ctx2, ProxyNodeSessionPrefix+"-1", string(s1))
|
_, err = core.etcdCli.Put(ctx2, path.Join(sessionutil.DefaultServiceRoot, typeutil.ProxyNodeRole)+"-1", string(s1))
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
_, err = core.etcdCli.Put(ctx2, ProxyNodeSessionPrefix+"-2", string(s2))
|
_, err = core.etcdCli.Put(ctx2, path.Join(sessionutil.DefaultServiceRoot, typeutil.ProxyNodeRole)+"-2", string(s2))
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
time.Sleep(time.Second)
|
time.Sleep(time.Second)
|
||||||
|
|
||||||
|
|
|
@ -15,6 +15,7 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"path"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"github.com/coreos/etcd/mvcc/mvccpb"
|
"github.com/coreos/etcd/mvcc/mvccpb"
|
||||||
|
@ -38,9 +39,6 @@ type timetickSync struct {
|
||||||
sendChan chan map[typeutil.UniqueID]*internalpb.ChannelTimeTickMsg
|
sendChan chan map[typeutil.UniqueID]*internalpb.ChannelTimeTickMsg
|
||||||
}
|
}
|
||||||
|
|
||||||
// ProxyNodeSessionPrefix used for etcd watch
|
|
||||||
const ProxyNodeSessionPrefix = "session/proxynode"
|
|
||||||
|
|
||||||
func newTimeTickSync(ctx context.Context, factory msgstream.Factory, cli *clientv3.Client) (*timetickSync, error) {
|
func newTimeTickSync(ctx context.Context, factory msgstream.Factory, cli *clientv3.Client) (*timetickSync, error) {
|
||||||
tss := timetickSync{
|
tss := timetickSync{
|
||||||
lock: sync.Mutex{},
|
lock: sync.Mutex{},
|
||||||
|
@ -109,7 +107,8 @@ func (t *timetickSync) UpdateTimeTick(in *internalpb.ChannelTimeTickMsg) error {
|
||||||
|
|
||||||
// StartWatch watch proxy node change and process all channels' timetick msg
|
// StartWatch watch proxy node change and process all channels' timetick msg
|
||||||
func (t *timetickSync) StartWatch() {
|
func (t *timetickSync) StartWatch() {
|
||||||
rch := t.etcdCli.Watch(t.ctx, ProxyNodeSessionPrefix, clientv3.WithPrefix(), clientv3.WithCreatedNotify())
|
proxyNodePrefix := path.Join(sessionutil.DefaultServiceRoot, typeutil.ProxyNodeRole)
|
||||||
|
rch := t.etcdCli.Watch(t.ctx, proxyNodePrefix, clientv3.WithPrefix(), clientv3.WithCreatedNotify())
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
case <-t.ctx.Done():
|
case <-t.ctx.Done():
|
||||||
|
|
|
@ -89,9 +89,8 @@ func (node *ProxyNode) Init() error {
|
||||||
// todo wait for proxyservice state changed to Healthy
|
// todo wait for proxyservice state changed to Healthy
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
|
|
||||||
node.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress}, typeutil.ProxyNodeRole,
|
node.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress})
|
||||||
Params.NetworkAddress, false)
|
node.session.Init(typeutil.ProxyNodeRole, Params.NetworkAddress, false)
|
||||||
node.session.Init()
|
|
||||||
|
|
||||||
err := funcutil.WaitForComponentHealthy(ctx, node.proxyService, "ProxyService", 1000000, time.Millisecond*200)
|
err := funcutil.WaitForComponentHealthy(ctx, node.proxyService, "ProxyService", 1000000, time.Millisecond*200)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -177,7 +176,7 @@ func (node *ProxyNode) Init() error {
|
||||||
log.Debug("create query message stream ...")
|
log.Debug("create query message stream ...")
|
||||||
|
|
||||||
masterAddr := Params.MasterAddress
|
masterAddr := Params.MasterAddress
|
||||||
idAllocator, err := allocator.NewIDAllocator(node.ctx, masterAddr)
|
idAllocator, err := allocator.NewIDAllocator(node.ctx, masterAddr, []string{Params.EtcdAddress})
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
|
|
@ -121,9 +121,8 @@ func NewQueryNodeWithoutID(ctx context.Context, factory msgstream.Factory) *Quer
|
||||||
func (node *QueryNode) Init() error {
|
func (node *QueryNode) Init() error {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
|
|
||||||
node.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress}, typeutil.QueryNodeRole,
|
node.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress})
|
||||||
Params.QueryNodeIP+":"+strconv.FormatInt(Params.QueryNodePort, 10), false)
|
node.session.Init(typeutil.QueryNodeRole, Params.QueryNodeIP+":"+strconv.FormatInt(Params.QueryNodePort, 10), false)
|
||||||
node.session.Init()
|
|
||||||
|
|
||||||
C.SegcoreInit()
|
C.SegcoreInit()
|
||||||
registerReq := &queryPb.RegisterNodeRequest{
|
registerReq := &queryPb.RegisterNodeRequest{
|
||||||
|
|
|
@ -59,9 +59,8 @@ type QueryService struct {
|
||||||
func (qs *QueryService) Init() error {
|
func (qs *QueryService) Init() error {
|
||||||
ctx := context.Background()
|
ctx := context.Background()
|
||||||
|
|
||||||
qs.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress}, typeutil.QueryServiceRole,
|
qs.session = sessionutil.NewSession(ctx, []string{Params.EtcdAddress})
|
||||||
Params.Address, true)
|
qs.session.Init(typeutil.QueryServiceRole, Params.Address, true)
|
||||||
qs.session.Init()
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -16,10 +16,10 @@ import (
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
)
|
)
|
||||||
|
|
||||||
const defaultServiceRoot = "/session/"
|
const DefaultServiceRoot = "/session/"
|
||||||
const defaultIDKey = "id"
|
const DefaultIDKey = "id"
|
||||||
const defaultRetryTimes = 30
|
const DefaultRetryTimes = 30
|
||||||
const defaultTTL = 10
|
const DefaultTTL = 10
|
||||||
|
|
||||||
// Session is a struct to store service's session, including ServerID, ServerName,
|
// Session is a struct to store service's session, including ServerID, ServerName,
|
||||||
// Address.
|
// Address.
|
||||||
|
@ -38,13 +38,10 @@ type Session struct {
|
||||||
|
|
||||||
// NewSession is a helper to build Session object.LeaseID will be assigned after
|
// NewSession is a helper to build Session object.LeaseID will be assigned after
|
||||||
// registeration.
|
// registeration.
|
||||||
func NewSession(ctx context.Context, etcdAddress []string, serverName, address string, exclusive bool) *Session {
|
func NewSession(ctx context.Context, etcdAddress []string) *Session {
|
||||||
ctx, cancel := context.WithCancel(ctx)
|
ctx, cancel := context.WithCancel(ctx)
|
||||||
session := &Session{
|
session := &Session{
|
||||||
ctx: ctx,
|
ctx: ctx,
|
||||||
ServerName: serverName,
|
|
||||||
Address: address,
|
|
||||||
Exclusive: exclusive,
|
|
||||||
cancel: cancel,
|
cancel: cancel,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -65,7 +62,10 @@ func NewSession(ctx context.Context, etcdAddress []string, serverName, address s
|
||||||
|
|
||||||
// Init will initialize base struct in the SessionManager, including getServerID,
|
// Init will initialize base struct in the SessionManager, including getServerID,
|
||||||
// and process keepAliveResponse
|
// and process keepAliveResponse
|
||||||
func (s *Session) Init() {
|
func (s *Session) Init(serverName, address string, exclusive bool) {
|
||||||
|
s.ServerName = serverName
|
||||||
|
s.Address = address
|
||||||
|
s.Exclusive = exclusive
|
||||||
s.checkIDExist()
|
s.checkIDExist()
|
||||||
serverID, err := s.getServerID()
|
serverID, err := s.getServerID()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -82,23 +82,23 @@ func (s *Session) Init() {
|
||||||
// GetServerID gets id from etcd with key: metaRootPath + "/services/id"
|
// GetServerID gets id from etcd with key: metaRootPath + "/services/id"
|
||||||
// Each server get ServerID and add one to id.
|
// Each server get ServerID and add one to id.
|
||||||
func (s *Session) getServerID() (int64, error) {
|
func (s *Session) getServerID() (int64, error) {
|
||||||
return s.getServerIDWithKey(defaultIDKey, defaultRetryTimes)
|
return s.getServerIDWithKey(DefaultIDKey, DefaultRetryTimes)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Session) checkIDExist() {
|
func (s *Session) checkIDExist() {
|
||||||
s.etcdCli.Txn(s.ctx).If(
|
s.etcdCli.Txn(s.ctx).If(
|
||||||
clientv3.Compare(
|
clientv3.Compare(
|
||||||
clientv3.Version(path.Join(defaultServiceRoot, defaultIDKey)),
|
clientv3.Version(path.Join(DefaultServiceRoot, DefaultIDKey)),
|
||||||
"=",
|
"=",
|
||||||
0)).
|
0)).
|
||||||
Then(clientv3.OpPut(path.Join(defaultServiceRoot, defaultIDKey), "1")).Commit()
|
Then(clientv3.OpPut(path.Join(DefaultServiceRoot, DefaultIDKey), "1")).Commit()
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Session) getServerIDWithKey(key string, retryTimes int) (int64, error) {
|
func (s *Session) getServerIDWithKey(key string, retryTimes int) (int64, error) {
|
||||||
res := int64(0)
|
res := int64(0)
|
||||||
getServerIDWithKeyFn := func() error {
|
getServerIDWithKeyFn := func() error {
|
||||||
getResp, err := s.etcdCli.Get(s.ctx, path.Join(defaultServiceRoot, key))
|
getResp, err := s.etcdCli.Get(s.ctx, path.Join(DefaultServiceRoot, key))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -113,10 +113,10 @@ func (s *Session) getServerIDWithKey(key string, retryTimes int) (int64, error)
|
||||||
}
|
}
|
||||||
txnResp, err := s.etcdCli.Txn(s.ctx).If(
|
txnResp, err := s.etcdCli.Txn(s.ctx).If(
|
||||||
clientv3.Compare(
|
clientv3.Compare(
|
||||||
clientv3.Value(path.Join(defaultServiceRoot, defaultIDKey)),
|
clientv3.Value(path.Join(DefaultServiceRoot, DefaultIDKey)),
|
||||||
"=",
|
"=",
|
||||||
value)).
|
value)).
|
||||||
Then(clientv3.OpPut(path.Join(defaultServiceRoot, defaultIDKey), strconv.FormatInt(valueInt+1, 10))).Commit()
|
Then(clientv3.OpPut(path.Join(DefaultServiceRoot, DefaultIDKey), strconv.FormatInt(valueInt+1, 10))).Commit()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -149,7 +149,7 @@ func (s *Session) getServerIDWithKey(key string, retryTimes int) (int64, error)
|
||||||
func (s *Session) registerService() (<-chan *clientv3.LeaseKeepAliveResponse, error) {
|
func (s *Session) registerService() (<-chan *clientv3.LeaseKeepAliveResponse, error) {
|
||||||
var ch <-chan *clientv3.LeaseKeepAliveResponse
|
var ch <-chan *clientv3.LeaseKeepAliveResponse
|
||||||
registerFn := func() error {
|
registerFn := func() error {
|
||||||
resp, err := s.etcdCli.Grant(s.ctx, defaultTTL)
|
resp, err := s.etcdCli.Grant(s.ctx, DefaultTTL)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("register service", zap.Error(err))
|
log.Error("register service", zap.Error(err))
|
||||||
return err
|
return err
|
||||||
|
@ -167,10 +167,10 @@ func (s *Session) registerService() (<-chan *clientv3.LeaseKeepAliveResponse, er
|
||||||
}
|
}
|
||||||
txnResp, err := s.etcdCli.Txn(s.ctx).If(
|
txnResp, err := s.etcdCli.Txn(s.ctx).If(
|
||||||
clientv3.Compare(
|
clientv3.Compare(
|
||||||
clientv3.Version(path.Join(defaultServiceRoot, key)),
|
clientv3.Version(path.Join(DefaultServiceRoot, key)),
|
||||||
"=",
|
"=",
|
||||||
0)).
|
0)).
|
||||||
Then(clientv3.OpPut(path.Join(defaultServiceRoot, key), string(sessionJSON), clientv3.WithLease(resp.ID))).Commit()
|
Then(clientv3.OpPut(path.Join(DefaultServiceRoot, key), string(sessionJSON), clientv3.WithLease(resp.ID))).Commit()
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Printf("compare and swap error %s\n. maybe the key has registered", err)
|
fmt.Printf("compare and swap error %s\n. maybe the key has registered", err)
|
||||||
|
@ -188,7 +188,7 @@ func (s *Session) registerService() (<-chan *clientv3.LeaseKeepAliveResponse, er
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
err := retry.Retry(defaultRetryTimes, time.Millisecond*200, registerFn)
|
err := retry.Retry(DefaultRetryTimes, time.Millisecond*200, registerFn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return ch, nil
|
return ch, nil
|
||||||
}
|
}
|
||||||
|
@ -221,7 +221,7 @@ func (s *Session) processKeepAliveResponse(ch <-chan *clientv3.LeaseKeepAliveRes
|
||||||
// GetSessions will get all sessions registered in etcd.
|
// GetSessions will get all sessions registered in etcd.
|
||||||
func (s *Session) GetSessions(prefix string) (map[string]*Session, error) {
|
func (s *Session) GetSessions(prefix string) (map[string]*Session, error) {
|
||||||
res := make(map[string]*Session)
|
res := make(map[string]*Session)
|
||||||
key := path.Join(defaultServiceRoot, prefix)
|
key := path.Join(DefaultServiceRoot, prefix)
|
||||||
resp, err := s.etcdCli.Get(s.ctx, key, clientv3.WithPrefix(),
|
resp, err := s.etcdCli.Get(s.ctx, key, clientv3.WithPrefix(),
|
||||||
clientv3.WithSort(clientv3.SortByKey, clientv3.SortAscend))
|
clientv3.WithSort(clientv3.SortByKey, clientv3.SortAscend))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -245,7 +245,7 @@ func (s *Session) GetSessions(prefix string) (map[string]*Session, error) {
|
||||||
func (s *Session) WatchServices(prefix string) (addChannel <-chan *Session, delChannel <-chan *Session) {
|
func (s *Session) WatchServices(prefix string) (addChannel <-chan *Session, delChannel <-chan *Session) {
|
||||||
addCh := make(chan *Session, 10)
|
addCh := make(chan *Session, 10)
|
||||||
delCh := make(chan *Session, 10)
|
delCh := make(chan *Session, 10)
|
||||||
rch := s.etcdCli.Watch(s.ctx, path.Join(defaultServiceRoot, prefix), clientv3.WithPrefix(), clientv3.WithPrevKV())
|
rch := s.etcdCli.Watch(s.ctx, path.Join(DefaultServiceRoot, prefix), clientv3.WithPrefix(), clientv3.WithPrevKV())
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
select {
|
select {
|
||||||
|
|
|
@ -26,7 +26,7 @@ func TestGetServerIDConcurrently(t *testing.T) {
|
||||||
cli, err := clientv3.New(clientv3.Config{Endpoints: []string{etcdAddr}})
|
cli, err := clientv3.New(clientv3.Config{Endpoints: []string{etcdAddr}})
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
etcdKV := etcdkv.NewEtcdKV(cli, "")
|
etcdKV := etcdkv.NewEtcdKV(cli, "")
|
||||||
_, err = cli.Delete(ctx, "/session", clientv3.WithPrefix())
|
_, err = cli.Delete(ctx, DefaultServiceRoot, clientv3.WithPrefix())
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
defer etcdKV.Close()
|
defer etcdKV.Close()
|
||||||
|
@ -35,7 +35,7 @@ func TestGetServerIDConcurrently(t *testing.T) {
|
||||||
var wg sync.WaitGroup
|
var wg sync.WaitGroup
|
||||||
var muList sync.Mutex = sync.Mutex{}
|
var muList sync.Mutex = sync.Mutex{}
|
||||||
|
|
||||||
s := NewSession(ctx, []string{etcdAddr}, "test", "testAddr", false)
|
s := NewSession(ctx, []string{etcdAddr})
|
||||||
res := make([]int64, 0)
|
res := make([]int64, 0)
|
||||||
|
|
||||||
getIDFunc := func() {
|
getIDFunc := func() {
|
||||||
|
@ -71,15 +71,16 @@ func TestInit(t *testing.T) {
|
||||||
cli, err := clientv3.New(clientv3.Config{Endpoints: []string{etcdAddr}})
|
cli, err := clientv3.New(clientv3.Config{Endpoints: []string{etcdAddr}})
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
etcdKV := etcdkv.NewEtcdKV(cli, "")
|
etcdKV := etcdkv.NewEtcdKV(cli, "")
|
||||||
_, err = cli.Delete(ctx, "/session", clientv3.WithPrefix())
|
_, err = cli.Delete(ctx, DefaultServiceRoot, clientv3.WithPrefix())
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
defer etcdKV.Close()
|
defer etcdKV.Close()
|
||||||
defer etcdKV.RemoveWithPrefix("")
|
defer etcdKV.RemoveWithPrefix("")
|
||||||
|
|
||||||
s := NewSession(ctx, []string{etcdAddr}, "test", "testAddr", false)
|
s := NewSession(ctx, []string{etcdAddr})
|
||||||
assert.NotEqual(t, 0, s.leaseID)
|
s.Init("test", "testAddr", false)
|
||||||
assert.NotEqual(t, 0, s.ServerID)
|
assert.NotEqual(t, int64(0), s.leaseID)
|
||||||
|
assert.NotEqual(t, int64(0), s.ServerID)
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestUpdateSessions(t *testing.T) {
|
func TestUpdateSessions(t *testing.T) {
|
||||||
|
@ -94,7 +95,7 @@ func TestUpdateSessions(t *testing.T) {
|
||||||
cli, err := clientv3.New(clientv3.Config{Endpoints: []string{etcdAddr}})
|
cli, err := clientv3.New(clientv3.Config{Endpoints: []string{etcdAddr}})
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
etcdKV := etcdkv.NewEtcdKV(cli, "")
|
etcdKV := etcdkv.NewEtcdKV(cli, "")
|
||||||
_, err = cli.Delete(ctx, "/session", clientv3.WithPrefix())
|
_, err = cli.Delete(ctx, DefaultServiceRoot, clientv3.WithPrefix())
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
|
||||||
defer etcdKV.Close()
|
defer etcdKV.Close()
|
||||||
|
@ -103,7 +104,7 @@ func TestUpdateSessions(t *testing.T) {
|
||||||
var wg sync.WaitGroup
|
var wg sync.WaitGroup
|
||||||
var muList sync.Mutex = sync.Mutex{}
|
var muList sync.Mutex = sync.Mutex{}
|
||||||
|
|
||||||
s := NewSession(ctx, []string{etcdAddr}, "test", "testAddr", false)
|
s := NewSession(ctx, []string{etcdAddr})
|
||||||
|
|
||||||
sessions, err := s.GetSessions("test")
|
sessions, err := s.GetSessions("test")
|
||||||
assert.Nil(t, err)
|
assert.Nil(t, err)
|
||||||
|
@ -113,8 +114,8 @@ func TestUpdateSessions(t *testing.T) {
|
||||||
sList := []*Session{}
|
sList := []*Session{}
|
||||||
|
|
||||||
getIDFunc := func() {
|
getIDFunc := func() {
|
||||||
singleS := NewSession(ctx, []string{etcdAddr}, "test", "testAddr", false)
|
singleS := NewSession(ctx, []string{etcdAddr})
|
||||||
singleS.Init()
|
singleS.Init("test", "testAddr", false)
|
||||||
muList.Lock()
|
muList.Lock()
|
||||||
sList = append(sList, singleS)
|
sList = append(sList, singleS)
|
||||||
muList.Unlock()
|
muList.Unlock()
|
||||||
|
|
Loading…
Reference in New Issue