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
neza2017 2021-05-24 09:44:49 +08:00 committed by GitHub
parent d189bf4d33
commit e326eaad1e
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
21 changed files with 246 additions and 122 deletions

View File

@ -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)
} }

View File

@ -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{

View File

@ -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
} }

View File

@ -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)

View File

@ -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)

View File

@ -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)

View File

@ -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
} }

View File

@ -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)

View File

@ -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
} }

View File

@ -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)
} }

View File

@ -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)

View File

@ -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 {

View File

@ -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}})

View File

@ -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 {

View File

@ -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)

View File

@ -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():

View File

@ -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

View File

@ -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{

View File

@ -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
} }

View File

@ -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 {

View File

@ -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()