2022-09-23 01:50:52 +00:00
|
|
|
package proxy
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"testing"
|
|
|
|
|
2023-02-26 03:31:49 +00:00
|
|
|
"github.com/cockroachdb/errors"
|
2023-09-21 01:45:27 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2022-09-23 01:50:52 +00:00
|
|
|
"google.golang.org/grpc"
|
|
|
|
|
2023-04-06 11:14:32 +00:00
|
|
|
"github.com/milvus-io/milvus/pkg/util/paramtable"
|
2022-09-23 01:50:52 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestInitHook(t *testing.T) {
|
2022-12-07 10:01:19 +00:00
|
|
|
paramtable.Get().Save(Params.ProxyCfg.SoPath.Key, "")
|
2022-09-23 01:50:52 +00:00
|
|
|
initHook()
|
|
|
|
assert.IsType(t, defaultHook{}, hoo)
|
|
|
|
|
2022-12-07 10:01:19 +00:00
|
|
|
paramtable.Get().Save(Params.ProxyCfg.SoPath.Key, "/a/b/hook.so")
|
2022-10-14 11:59:24 +00:00
|
|
|
err := initHook()
|
2023-06-08 07:36:36 +00:00
|
|
|
assert.Error(t, err)
|
2022-12-07 10:01:19 +00:00
|
|
|
paramtable.Get().Save(Params.ProxyCfg.SoPath.Key, "")
|
2022-09-23 01:50:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type mockHook struct {
|
|
|
|
defaultHook
|
|
|
|
mockRes interface{}
|
|
|
|
mockErr error
|
|
|
|
}
|
|
|
|
|
2022-09-24 01:56:51 +00:00
|
|
|
func (m mockHook) Mock(ctx context.Context, req interface{}, fullMethod string) (bool, interface{}, error) {
|
2022-09-23 01:50:52 +00:00
|
|
|
return true, m.mockRes, m.mockErr
|
|
|
|
}
|
|
|
|
|
|
|
|
type req struct {
|
|
|
|
method string
|
|
|
|
}
|
|
|
|
|
2022-09-28 05:26:54 +00:00
|
|
|
type BeforeMockCtxKey int
|
|
|
|
|
2022-09-23 01:50:52 +00:00
|
|
|
type beforeMock struct {
|
|
|
|
defaultHook
|
2022-09-28 05:26:54 +00:00
|
|
|
method string
|
|
|
|
ctxKey BeforeMockCtxKey
|
|
|
|
ctxValue string
|
|
|
|
err error
|
2022-09-23 01:50:52 +00:00
|
|
|
}
|
|
|
|
|
2022-09-28 05:26:54 +00:00
|
|
|
func (b beforeMock) Before(ctx context.Context, r interface{}, fullMethod string) (context.Context, error) {
|
2022-09-23 01:50:52 +00:00
|
|
|
re, ok := r.(*req)
|
|
|
|
if !ok {
|
2022-09-28 05:26:54 +00:00
|
|
|
return ctx, errors.New("r is invalid type")
|
2022-09-23 01:50:52 +00:00
|
|
|
}
|
|
|
|
re.method = b.method
|
2022-09-28 05:26:54 +00:00
|
|
|
return context.WithValue(ctx, b.ctxKey, b.ctxValue), b.err
|
2022-09-23 01:50:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type resp struct {
|
|
|
|
method string
|
|
|
|
}
|
|
|
|
|
|
|
|
type afterMock struct {
|
|
|
|
defaultHook
|
|
|
|
method string
|
|
|
|
err error
|
|
|
|
}
|
|
|
|
|
2022-09-24 01:56:51 +00:00
|
|
|
func (a afterMock) After(ctx context.Context, r interface{}, err error, fullMethod string) error {
|
2022-09-23 01:50:52 +00:00
|
|
|
re, ok := r.(*resp)
|
|
|
|
if !ok {
|
|
|
|
return errors.New("r is invalid type")
|
|
|
|
}
|
|
|
|
re.method = a.method
|
|
|
|
return a.err
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestHookInterceptor(t *testing.T) {
|
|
|
|
var (
|
|
|
|
ctx = context.Background()
|
|
|
|
info = &grpc.UnaryServerInfo{
|
|
|
|
FullMethod: "test",
|
|
|
|
}
|
2023-06-25 09:20:43 +00:00
|
|
|
emptyFullMethod = &grpc.UnaryServerInfo{
|
|
|
|
FullMethod: "",
|
|
|
|
}
|
2022-09-23 01:50:52 +00:00
|
|
|
interceptor = UnaryServerHookInterceptor()
|
|
|
|
mockHoo = mockHook{mockRes: "mock", mockErr: errors.New("mock")}
|
|
|
|
r = &req{method: "req"}
|
|
|
|
re = &resp{method: "resp"}
|
2022-09-28 05:26:54 +00:00
|
|
|
beforeHoo = beforeMock{method: "before", ctxKey: 100, ctxValue: "hook", err: errors.New("before")}
|
2022-09-23 01:50:52 +00:00
|
|
|
afterHoo = afterMock{method: "after", err: errors.New("after")}
|
|
|
|
|
|
|
|
res interface{}
|
|
|
|
err error
|
|
|
|
)
|
|
|
|
|
|
|
|
hoo = mockHoo
|
|
|
|
res, err = interceptor(ctx, "request", info, func(ctx context.Context, req interface{}) (interface{}, error) {
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
assert.Equal(t, res, mockHoo.mockRes)
|
|
|
|
assert.Equal(t, err, mockHoo.mockErr)
|
2023-06-25 09:20:43 +00:00
|
|
|
res, err = interceptor(ctx, "request", emptyFullMethod, func(ctx context.Context, req interface{}) (interface{}, error) {
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
assert.Equal(t, res, mockHoo.mockRes)
|
|
|
|
assert.Equal(t, err, mockHoo.mockErr)
|
2022-09-23 01:50:52 +00:00
|
|
|
|
|
|
|
hoo = beforeHoo
|
|
|
|
_, err = interceptor(ctx, r, info, func(ctx context.Context, req interface{}) (interface{}, error) {
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
assert.Equal(t, r.method, beforeHoo.method)
|
|
|
|
assert.Equal(t, err, beforeHoo.err)
|
|
|
|
|
2022-09-28 05:26:54 +00:00
|
|
|
beforeHoo.err = nil
|
|
|
|
hoo = beforeHoo
|
|
|
|
_, err = interceptor(ctx, r, info, func(ctx context.Context, req interface{}) (interface{}, error) {
|
|
|
|
assert.Equal(t, beforeHoo.ctxValue, ctx.Value(beforeHoo.ctxKey))
|
|
|
|
return nil, nil
|
|
|
|
})
|
|
|
|
assert.Equal(t, r.method, beforeHoo.method)
|
|
|
|
assert.Equal(t, err, beforeHoo.err)
|
|
|
|
|
2022-09-23 01:50:52 +00:00
|
|
|
hoo = afterHoo
|
|
|
|
_, err = interceptor(ctx, r, info, func(ctx context.Context, r interface{}) (interface{}, error) {
|
|
|
|
return re, nil
|
|
|
|
})
|
|
|
|
assert.Equal(t, re.method, afterHoo.method)
|
|
|
|
assert.Equal(t, err, afterHoo.err)
|
|
|
|
|
|
|
|
hoo = defaultHook{}
|
|
|
|
res, err = interceptor(ctx, r, info, func(ctx context.Context, r interface{}) (interface{}, error) {
|
|
|
|
return &resp{
|
|
|
|
method: r.(*req).method,
|
|
|
|
}, nil
|
|
|
|
})
|
|
|
|
assert.Equal(t, res.(*resp).method, r.method)
|
|
|
|
assert.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestDefaultHook(t *testing.T) {
|
|
|
|
d := defaultHook{}
|
|
|
|
assert.NoError(t, d.Init(nil))
|
2023-10-17 13:00:11 +00:00
|
|
|
{
|
|
|
|
_, err := d.VerifyAPIKey("key")
|
|
|
|
assert.Error(t, err)
|
|
|
|
}
|
2022-09-23 01:50:52 +00:00
|
|
|
assert.NotPanics(t, func() {
|
|
|
|
d.Release()
|
|
|
|
})
|
|
|
|
}
|
2023-09-26 02:43:25 +00:00
|
|
|
|
|
|
|
func TestUpdateProxyFunctionCallMetric(t *testing.T) {
|
|
|
|
assert.NotPanics(t, func() {
|
|
|
|
updateProxyFunctionCallMetric("/milvus.proto.milvus.MilvusService/Flush")
|
|
|
|
updateProxyFunctionCallMetric("Flush")
|
|
|
|
updateProxyFunctionCallMetric("")
|
|
|
|
})
|
|
|
|
}
|