milvus/internal/distributed/streaming/wal_test.go

192 lines
5.8 KiB
Go

package streaming
import (
"context"
"sync"
"testing"
"time"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
"github.com/milvus-io/milvus-proto/go-api/v2/msgpb"
"github.com/milvus-io/milvus/internal/distributed/streaming/internal/producer"
"github.com/milvus-io/milvus/internal/mocks/streamingcoord/mock_client"
"github.com/milvus-io/milvus/internal/mocks/streamingnode/client/handler/mock_producer"
"github.com/milvus-io/milvus/internal/mocks/streamingnode/client/mock_handler"
"github.com/milvus-io/milvus/pkg/v2/streaming/util/message"
"github.com/milvus-io/milvus/pkg/v2/streaming/util/types"
"github.com/milvus-io/milvus/pkg/v2/streaming/walimpls/impls/walimplstest"
"github.com/milvus-io/milvus/pkg/v2/util/conc"
"github.com/milvus-io/milvus/pkg/v2/util/typeutil"
)
const (
vChannel1 = "by-dev-rootcoord-dml_1"
vChannel2 = "by-dev-rootcoord-dml_2"
vChannel3 = "by-dev-rootcoord-dml_3"
)
func TestWAL(t *testing.T) {
coordClient := mock_client.NewMockClient(t)
coordClient.EXPECT().Close().Return()
broadcastServce := mock_client.NewMockBroadcastService(t)
broadcastServce.EXPECT().Broadcast(mock.Anything, mock.Anything).RunAndReturn(
func(ctx context.Context, bmm message.BroadcastMutableMessage) (*types.BroadcastAppendResult, error) {
bmm = bmm.WithBroadcastID(1)
result := make(map[string]*types.AppendResult)
for idx, msg := range bmm.SplitIntoMutableMessage() {
result[msg.VChannel()] = &types.AppendResult{
MessageID: walimplstest.NewTestMessageID(int64(idx)),
TimeTick: uint64(time.Now().UnixMilli()),
}
}
return &types.BroadcastAppendResult{
AppendResults: result,
}, nil
})
broadcastServce.EXPECT().Ack(mock.Anything, mock.Anything).Return(nil)
broadcastServce.EXPECT().BlockUntilEvent(mock.Anything, mock.Anything).Return(nil)
coordClient.EXPECT().Broadcast().Return(broadcastServce)
handler := mock_handler.NewMockHandlerClient(t)
handler.EXPECT().Close().Return()
w := &walAccesserImpl{
lifetime: typeutil.NewLifetime(),
streamingCoordClient: coordClient,
handlerClient: handler,
producerMutex: sync.Mutex{},
producers: make(map[string]*producer.ResumableProducer),
appendExecutionPool: conc.NewPool[struct{}](10),
dispatchExecutionPool: conc.NewPool[struct{}](10),
}
ctx := context.Background()
available := make(chan struct{})
p := mock_producer.NewMockProducer(t)
p.EXPECT().IsAvailable().RunAndReturn(func() bool {
select {
case <-available:
return false
default:
return true
}
})
p.EXPECT().Append(mock.Anything, mock.Anything).Return(&types.AppendResult{
MessageID: walimplstest.NewTestMessageID(1),
TimeTick: 10,
TxnCtx: &message.TxnContext{
TxnID: 1,
Keepalive: 10 * time.Second,
},
}, nil)
p.EXPECT().Available().Return(available)
p.EXPECT().Close().Return()
handler.EXPECT().CreateProducer(mock.Anything, mock.Anything).Return(p, nil)
result, err := w.RawAppend(ctx, newInsertMessage(vChannel1))
assert.NoError(t, err)
assert.NotNil(t, result)
// Test committed txn.
txn, err := w.Txn(ctx, TxnOption{
VChannel: vChannel1,
Keepalive: 10 * time.Second,
})
assert.NoError(t, err)
assert.NotNil(t, txn)
err = txn.Append(ctx, newInsertMessage(vChannel1))
assert.NoError(t, err)
err = txn.Append(ctx, newInsertMessage(vChannel1))
assert.NoError(t, err)
result, err = txn.Commit(ctx)
assert.NoError(t, err)
assert.NotNil(t, result)
// Test rollback txn.
txn, err = w.Txn(ctx, TxnOption{
VChannel: vChannel1,
Keepalive: 10 * time.Second,
})
assert.NoError(t, err)
assert.NotNil(t, txn)
err = txn.Append(ctx, newInsertMessage(vChannel1))
assert.NoError(t, err)
err = txn.Append(ctx, newInsertMessage(vChannel1))
assert.NoError(t, err)
err = txn.Rollback(ctx)
assert.NoError(t, err)
resp := w.AppendMessages(ctx,
newInsertMessage(vChannel1),
newInsertMessage(vChannel2),
newInsertMessage(vChannel2),
newInsertMessage(vChannel3),
newInsertMessage(vChannel3),
newInsertMessage(vChannel3),
)
assert.NoError(t, resp.UnwrapFirstError())
r, err := w.Broadcast().Append(ctx, newBroadcastMessage([]string{vChannel1, vChannel2, vChannel3}))
assert.NoError(t, err)
assert.Len(t, r.AppendResults, 3)
err = w.Broadcast().Ack(ctx, types.BroadcastAckRequest{BroadcastID: 1, VChannel: vChannel1})
assert.NoError(t, err)
err = w.Broadcast().BlockUntilResourceKeyAckAll(ctx, message.NewCollectionNameResourceKey("r1"))
assert.NoError(t, err)
err = w.Broadcast().BlockUntilResourceKeyAckOnce(ctx, message.NewCollectionNameResourceKey("r2"))
assert.NoError(t, err)
w.Close()
w.GetLatestMVCCTimestampIfLocal(ctx, vChannel1)
resp = w.AppendMessages(ctx, newInsertMessage(vChannel1))
assert.Error(t, resp.UnwrapFirstError())
r, err = w.Broadcast().Append(ctx, newBroadcastMessage([]string{vChannel1, vChannel2, vChannel3}))
assert.Error(t, err)
assert.Nil(t, r)
err = w.Broadcast().Ack(ctx, types.BroadcastAckRequest{BroadcastID: 1, VChannel: vChannel1})
assert.Error(t, err)
err = w.Broadcast().BlockUntilResourceKeyAckAll(ctx, message.NewCollectionNameResourceKey("r1"))
assert.Error(t, err)
err = w.Broadcast().BlockUntilResourceKeyAckOnce(ctx, message.NewCollectionNameResourceKey("r2"))
assert.Error(t, err)
}
func newInsertMessage(vChannel string) message.MutableMessage {
msg, err := message.NewInsertMessageBuilderV1().
WithVChannel(vChannel).
WithHeader(&message.InsertMessageHeader{}).
WithBody(&msgpb.InsertRequest{}).
BuildMutable()
if err != nil {
panic(err)
}
return msg
}
func newBroadcastMessage(vchannels []string) message.BroadcastMutableMessage {
msg, err := message.NewDropCollectionMessageBuilderV1().
WithBroadcast(vchannels).
WithHeader(&message.DropCollectionMessageHeader{}).
WithBody(&msgpb.DropCollectionRequest{}).
BuildBroadcast()
if err != nil {
panic(err)
}
return msg
}