milvus/internal/util/idalloc/basic_allocator_test.go

155 lines
4.3 KiB
Go

package idalloc
import (
"context"
"testing"
"github.com/cockroachdb/errors"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
"go.uber.org/atomic"
"google.golang.org/grpc"
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
"github.com/milvus-io/milvus/internal/mocks"
"github.com/milvus-io/milvus/internal/types"
"github.com/milvus-io/milvus/pkg/proto/rootcoordpb"
"github.com/milvus-io/milvus/pkg/util/paramtable"
"github.com/milvus-io/milvus/pkg/util/syncutil"
)
func TestLocalAllocator(t *testing.T) {
allocator := newLocalAllocator()
ts, err := allocator.allocateOne()
assert.Error(t, err)
assert.Zero(t, ts)
allocator.update(1, 100)
counter := atomic.NewUint64(0)
for i := 0; i < 100; i++ {
ts, err := allocator.allocateOne()
assert.NoError(t, err)
assert.NotZero(t, ts)
counter.Add(ts)
}
assert.Equal(t, uint64(5050), counter.Load())
// allocator exhausted.
ts, err = allocator.allocateOne()
assert.Error(t, err)
assert.Zero(t, ts)
// allocator can not be rollback.
allocator.update(90, 100)
ts, err = allocator.allocateOne()
assert.Error(t, err)
assert.Zero(t, ts)
// allocator can be only increasing.
allocator.update(101, 100)
ts, err = allocator.allocateOne()
assert.NoError(t, err)
assert.Equal(t, ts, uint64(101))
// allocator can be exhausted.
allocator.exhausted()
ts, err = allocator.allocateOne()
assert.Error(t, err)
assert.Zero(t, ts)
}
func TestRemoteTSOAllocator(t *testing.T) {
paramtable.Init()
paramtable.SetNodeID(1)
client := NewMockRootCoordClient(t)
f := syncutil.NewFuture[types.RootCoordClient]()
f.Set(client)
allocator := newTSOAllocator(f)
ts, count, err := allocator.batchAllocate(context.Background(), 100)
assert.NoError(t, err)
assert.NotZero(t, ts)
assert.Equal(t, count, 100)
// Test error.
client = mocks.NewMockRootCoordClient(t)
client.EXPECT().AllocTimestamp(mock.Anything, mock.Anything).RunAndReturn(
func(ctx context.Context, atr *rootcoordpb.AllocTimestampRequest, co ...grpc.CallOption) (*rootcoordpb.AllocTimestampResponse, error) {
return nil, errors.New("test")
},
)
f = syncutil.NewFuture[types.RootCoordClient]()
f.Set(client)
allocator = newTSOAllocator(f)
_, _, err = allocator.batchAllocate(context.Background(), 100)
assert.Error(t, err)
client.EXPECT().AllocTimestamp(mock.Anything, mock.Anything).Unset()
client.EXPECT().AllocTimestamp(mock.Anything, mock.Anything).RunAndReturn(
func(ctx context.Context, atr *rootcoordpb.AllocTimestampRequest, co ...grpc.CallOption) (*rootcoordpb.AllocTimestampResponse, error) {
return &rootcoordpb.AllocTimestampResponse{
Status: &commonpb.Status{
ErrorCode: commonpb.ErrorCode_ForceDeny,
},
}, nil
},
)
f = syncutil.NewFuture[types.RootCoordClient]()
f.Set(client)
allocator = newTSOAllocator(f)
_, _, err = allocator.batchAllocate(context.Background(), 100)
assert.Error(t, err)
}
func TestRemoteIDAllocator(t *testing.T) {
paramtable.Init()
paramtable.SetNodeID(1)
client := NewMockRootCoordClient(t)
f := syncutil.NewFuture[types.RootCoordClient]()
f.Set(client)
allocator := newIDAllocator(f)
ts, count, err := allocator.batchAllocate(context.Background(), 100)
assert.NoError(t, err)
assert.NotZero(t, ts)
assert.Equal(t, count, 100)
// Test error.
client = mocks.NewMockRootCoordClient(t)
client.EXPECT().AllocID(mock.Anything, mock.Anything).RunAndReturn(
func(ctx context.Context, atr *rootcoordpb.AllocIDRequest, co ...grpc.CallOption) (*rootcoordpb.AllocIDResponse, error) {
return nil, errors.New("test")
},
)
f = syncutil.NewFuture[types.RootCoordClient]()
f.Set(client)
allocator = newIDAllocator(f)
_, _, err = allocator.batchAllocate(context.Background(), 100)
assert.Error(t, err)
client.EXPECT().AllocID(mock.Anything, mock.Anything).Unset()
client.EXPECT().AllocID(mock.Anything, mock.Anything).RunAndReturn(
func(ctx context.Context, atr *rootcoordpb.AllocIDRequest, co ...grpc.CallOption) (*rootcoordpb.AllocIDResponse, error) {
return &rootcoordpb.AllocIDResponse{
Status: &commonpb.Status{
ErrorCode: commonpb.ErrorCode_ForceDeny,
},
}, nil
},
)
f = syncutil.NewFuture[types.RootCoordClient]()
f.Set(client)
allocator = newIDAllocator(f)
_, _, err = allocator.batchAllocate(context.Background(), 100)
assert.Error(t, err)
}