mirror of
https://gitee.com/milvus-io/milvus.git
synced 2026-01-07 19:31:51 +08:00
issue: #44358 Implement complete snapshot management system including creation, deletion, listing, description, and restoration capabilities across all system components. Key features: - Create snapshots for entire collections - Drop snapshots by name with proper cleanup - List snapshots with collection filtering - Describe snapshot details and metadata Components added/modified: - Client SDK with full snapshot API support and options - DataCoord snapshot service with metadata management - Proxy layer with task-based snapshot operations - Protocol buffer definitions for snapshot RPCs - Comprehensive unit tests with mockey framework - Integration tests for end-to-end validation Technical implementation: - Snapshot metadata storage in etcd with proper indexing - File-based snapshot data persistence in object storage - Garbage collection integration for snapshot cleanup - Error handling and validation across all operations - Thread-safe operations with proper locking mechanisms <!-- This is an auto-generated comment: release notes by coderabbit.ai --> - Core invariant/assumption: snapshots are immutable point‑in‑time captures identified by (collection, snapshot name/ID); etcd snapshot metadata is authoritative for lifecycle (PENDING → COMMITTED → DELETING) and per‑segment manifests live in object storage (Avro / StorageV2). GC and restore logic must see snapshotRefIndex loaded (snapshotMeta.IsRefIndexLoaded) before reclaiming or relying on segment/index files. - New capability added: full end‑to‑end snapshot subsystem — client SDK APIs (Create/Drop/List/Describe/Restore + restore job queries), DataCoord SnapshotWriter/Reader (Avro + StorageV2 manifests), snapshotMeta in meta, SnapshotManager orchestration (create/drop/describe/list/restore), copy‑segment restore tasks/inspector/checker, proxy & RPC surface, GC integration, and docs/tests — enabling point‑in‑time collection snapshots persisted to object storage and restorations orchestrated across components. - Logic removed/simplified and why: duplicated recursive compaction/delta‑log traversal and ad‑hoc lookup code were consolidated behind two focused APIs/owners (Handler.GetDeltaLogFromCompactTo for delta traversal and SnapshotManager/SnapshotReader for snapshot I/O). MixCoord/coordinator broker paths were converted to thin RPC proxies. This eliminates multiple implementations of the same traversal/lookup, reducing divergence and simplifying responsibility boundaries. - Why this does NOT introduce data loss or regressions: snapshot create/drop use explicit two‑phase semantics (PENDING → COMMIT/DELETING) with SnapshotWriter writing manifests and metadata before commit; GC uses snapshotRefIndex guards and IsRefIndexLoaded/GetSnapshotBySegment/GetSnapshotByIndex checks to avoid removing referenced files; restore flow pre‑allocates job IDs, validates resources (partitions/indexes), performs rollback on failure (rollbackRestoreSnapshot), and converts/updates segment/index metadata only after successful copy tasks. Extensive unit and integration tests exercise pending/deleting/GC/restore/error paths to ensure idempotence and protection against premature deletion. <!-- end of auto-generated comment: release notes by coderabbit.ai --> --------- Signed-off-by: Wei Liu <wei.liu@zilliz.com>
618 lines
16 KiB
Go
618 lines
16 KiB
Go
// Licensed to the LF AI & Data foundation under one
|
|
// or more contributor license agreements. See the NOTICE file
|
|
// distributed with this work for additional information
|
|
// regarding copyright ownership. The ASF licenses this file
|
|
// to you under the Apache License, Version 2.0 (the
|
|
// "License"); you may not use this file except in compliance
|
|
// with the License. You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package proxy
|
|
|
|
import (
|
|
"context"
|
|
"testing"
|
|
|
|
"github.com/bytedance/mockey"
|
|
"github.com/cockroachdb/errors"
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/milvus-io/milvus-proto/go-api/v2/milvuspb"
|
|
"github.com/milvus-io/milvus/pkg/v2/util/merr"
|
|
)
|
|
|
|
func TestProxy_CreateSnapshot_Success(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.CreateSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "test_collection",
|
|
DbName: "default",
|
|
}
|
|
|
|
// Mock successful enqueue
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
// Set task result to simulate successful execution
|
|
if cst, ok := t.(*createSnapshotTask); ok {
|
|
cst.result = merr.Success()
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
// Mock successful task completion
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.CreateSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.True(t, merr.Ok(result))
|
|
}
|
|
|
|
func TestProxy_CreateSnapshot_EnqueueFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.CreateSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "test_collection",
|
|
DbName: "default",
|
|
}
|
|
|
|
expectedError := errors.New("enqueue failed")
|
|
|
|
// Mock enqueue failure
|
|
mock := mockey.Mock((*ddTaskQueue).Enqueue).Return(expectedError).Build()
|
|
defer mock.UnPatch()
|
|
|
|
result, err := proxy.CreateSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err) // API should not return error, but error status
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result))
|
|
assert.Contains(t, result.GetReason(), "enqueue failed")
|
|
}
|
|
|
|
func TestProxy_CreateSnapshot_WaitToFinishFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.CreateSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "test_collection",
|
|
DbName: "default",
|
|
}
|
|
|
|
expectedError := errors.New("task execution failed")
|
|
|
|
// Mock successful enqueue but failed task execution
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).Return(nil).Build()
|
|
defer mock1.UnPatch()
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(expectedError).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.CreateSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err) // API should not return error, but error status
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result))
|
|
assert.Contains(t, result.GetReason(), "task execution failed")
|
|
}
|
|
|
|
func TestProxy_DropSnapshot_Success(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.DropSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
}
|
|
|
|
// Mock successful enqueue and task completion
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
if dst, ok := t.(*dropSnapshotTask); ok {
|
|
dst.result = merr.Success()
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.DropSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.True(t, merr.Ok(result))
|
|
}
|
|
|
|
func TestProxy_DropSnapshot_EnqueueFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.DropSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
}
|
|
|
|
expectedError := errors.New("enqueue failed")
|
|
|
|
mock := mockey.Mock((*ddTaskQueue).Enqueue).Return(expectedError).Build()
|
|
defer mock.UnPatch()
|
|
|
|
result, err := proxy.DropSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result))
|
|
assert.Contains(t, result.GetReason(), "enqueue failed")
|
|
}
|
|
|
|
func TestProxy_DropSnapshot_WaitToFinishFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.DropSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
}
|
|
|
|
expectedError := errors.New("task execution failed")
|
|
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).Return(nil).Build()
|
|
defer mock1.UnPatch()
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(expectedError).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.DropSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result))
|
|
assert.Contains(t, result.GetReason(), "task execution failed")
|
|
}
|
|
|
|
func TestProxy_DescribeSnapshot_Success(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.DescribeSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
}
|
|
|
|
// Mock successful enqueue and task completion
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
if dst, ok := t.(*describeSnapshotTask); ok {
|
|
dst.result = &milvuspb.DescribeSnapshotResponse{
|
|
Status: merr.Success(),
|
|
Name: "test_snapshot",
|
|
CollectionName: "test_collection",
|
|
}
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.DescribeSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.True(t, merr.Ok(result.GetStatus()))
|
|
assert.Equal(t, "test_snapshot", result.GetName())
|
|
assert.Equal(t, "test_collection", result.GetCollectionName())
|
|
}
|
|
|
|
func TestProxy_DescribeSnapshot_EnqueueFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.DescribeSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
}
|
|
|
|
expectedError := errors.New("enqueue failed")
|
|
|
|
mock := mockey.Mock((*ddTaskQueue).Enqueue).Return(expectedError).Build()
|
|
defer mock.UnPatch()
|
|
|
|
result, err := proxy.DescribeSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result.GetStatus()))
|
|
assert.Contains(t, result.GetStatus().GetReason(), "enqueue failed")
|
|
}
|
|
|
|
func TestProxy_DescribeSnapshot_WaitToFinishFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.DescribeSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
}
|
|
|
|
expectedError := errors.New("task execution failed")
|
|
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).Return(nil).Build()
|
|
defer mock1.UnPatch()
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(expectedError).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.DescribeSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result.GetStatus()))
|
|
assert.Contains(t, result.GetStatus().GetReason(), "task execution failed")
|
|
}
|
|
|
|
func TestProxy_ListSnapshots_Success(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.ListSnapshotsRequest{
|
|
CollectionName: "test_collection",
|
|
}
|
|
|
|
// Mock successful enqueue and task completion
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
if lst, ok := t.(*listSnapshotsTask); ok {
|
|
lst.result = &milvuspb.ListSnapshotsResponse{
|
|
Status: merr.Success(),
|
|
Snapshots: []string{
|
|
"snapshot1",
|
|
"snapshot2",
|
|
},
|
|
}
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.ListSnapshots(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.True(t, merr.Ok(result.GetStatus()))
|
|
assert.Len(t, result.GetSnapshots(), 2)
|
|
assert.Equal(t, "snapshot1", result.GetSnapshots()[0])
|
|
assert.Equal(t, "snapshot2", result.GetSnapshots()[1])
|
|
}
|
|
|
|
func TestProxy_ListSnapshots_EnqueueFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.ListSnapshotsRequest{
|
|
CollectionName: "test_collection",
|
|
}
|
|
|
|
expectedError := errors.New("enqueue failed")
|
|
|
|
mock := mockey.Mock((*ddTaskQueue).Enqueue).Return(expectedError).Build()
|
|
defer mock.UnPatch()
|
|
|
|
result, err := proxy.ListSnapshots(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result.GetStatus()))
|
|
assert.Contains(t, result.GetStatus().GetReason(), "enqueue failed")
|
|
}
|
|
|
|
func TestProxy_ListSnapshots_WaitToFinishFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.ListSnapshotsRequest{
|
|
CollectionName: "test_collection",
|
|
}
|
|
|
|
expectedError := errors.New("task execution failed")
|
|
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).Return(nil).Build()
|
|
defer mock1.UnPatch()
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(expectedError).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.ListSnapshots(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result.GetStatus()))
|
|
assert.Contains(t, result.GetStatus().GetReason(), "task execution failed")
|
|
}
|
|
|
|
func TestProxy_RestoreSnapshot_Success(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.RestoreSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "restored_collection",
|
|
DbName: "default",
|
|
}
|
|
|
|
// Mock successful enqueue and task completion
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
if rst, ok := t.(*restoreSnapshotTask); ok {
|
|
rst.result = &milvuspb.RestoreSnapshotResponse{
|
|
Status: merr.Success(),
|
|
}
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.RestoreSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.True(t, merr.Ok(result.GetStatus()))
|
|
}
|
|
|
|
func TestProxy_RestoreSnapshot_EnqueueFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.RestoreSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "restored_collection",
|
|
}
|
|
|
|
expectedError := errors.New("enqueue failed")
|
|
|
|
mock := mockey.Mock((*ddTaskQueue).Enqueue).Return(expectedError).Build()
|
|
defer mock.UnPatch()
|
|
|
|
result, err := proxy.RestoreSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result.GetStatus()))
|
|
assert.Contains(t, result.GetStatus().GetReason(), "enqueue failed")
|
|
}
|
|
|
|
func TestProxy_RestoreSnapshot_WaitToFinishFailure(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.RestoreSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "restored_collection",
|
|
}
|
|
|
|
expectedError := errors.New("task execution failed")
|
|
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).Return(nil).Build()
|
|
defer mock1.UnPatch()
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(expectedError).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.RestoreSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.False(t, merr.Ok(result.GetStatus()))
|
|
assert.Contains(t, result.GetStatus().GetReason(), "task execution failed")
|
|
}
|
|
|
|
// Test task creation and basic properties
|
|
func TestProxy_CreateSnapshot_TaskCreation(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.CreateSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "test_collection",
|
|
DbName: "default",
|
|
}
|
|
|
|
var enqueuedTask *createSnapshotTask
|
|
|
|
// Capture the task that gets enqueued
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
if cst, ok := t.(*createSnapshotTask); ok {
|
|
enqueuedTask = cst
|
|
cst.result = merr.Success()
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
_, err := proxy.CreateSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, enqueuedTask)
|
|
assert.Equal(t, "test_snapshot", enqueuedTask.req.GetName())
|
|
assert.Equal(t, "test_collection", enqueuedTask.req.GetCollectionName())
|
|
assert.Equal(t, "default", enqueuedTask.req.GetDbName())
|
|
assert.Equal(t, CreateSnapshotTaskName, enqueuedTask.Name())
|
|
}
|
|
|
|
// Test edge cases
|
|
func TestProxy_CreateSnapshot_NilRequest(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
// This should cause a panic when accessing req fields
|
|
assert.Panics(t, func() {
|
|
proxy.CreateSnapshot(context.Background(), nil)
|
|
})
|
|
}
|
|
|
|
func TestProxy_DropSnapshot_EmptySnapshotName(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.DropSnapshotRequest{
|
|
Name: "", // Empty snapshot name
|
|
}
|
|
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
if dst, ok := t.(*dropSnapshotTask); ok {
|
|
dst.result = merr.Success()
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.DropSnapshot(context.Background(), req)
|
|
|
|
// Should succeed at API level, actual validation happens in task
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.True(t, merr.Ok(result))
|
|
}
|
|
|
|
// Test metrics and observability
|
|
func TestProxy_CreateSnapshot_MetricsRecording(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.CreateSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "test_collection",
|
|
DbName: "default",
|
|
}
|
|
|
|
// Mock successful execution
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
if cst, ok := t.(*createSnapshotTask); ok {
|
|
cst.result = merr.Success()
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.CreateSnapshot(context.Background(), req)
|
|
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.True(t, merr.Ok(result))
|
|
// In a real test, we would verify metrics were recorded correctly
|
|
// This test demonstrates the API behavior focuses on the business logic
|
|
}
|
|
|
|
// Test context handling
|
|
func TestProxy_CreateSnapshot_ContextCancellation(t *testing.T) {
|
|
proxy := &Proxy{
|
|
sched: &taskScheduler{
|
|
ddQueue: &ddTaskQueue{},
|
|
},
|
|
}
|
|
|
|
req := &milvuspb.CreateSnapshotRequest{
|
|
Name: "test_snapshot",
|
|
CollectionName: "test_collection",
|
|
DbName: "default",
|
|
}
|
|
|
|
// Create a canceled context
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
cancel()
|
|
|
|
// Mock successful enqueue
|
|
mock1 := mockey.Mock((*ddTaskQueue).Enqueue).To(func(t task) error {
|
|
if cst, ok := t.(*createSnapshotTask); ok {
|
|
cst.result = merr.Success()
|
|
}
|
|
return nil
|
|
}).Build()
|
|
defer mock1.UnPatch()
|
|
|
|
mock2 := mockey.Mock((*TaskCondition).WaitToFinish).Return(nil).Build()
|
|
defer mock2.UnPatch()
|
|
|
|
result, err := proxy.CreateSnapshot(ctx, req)
|
|
|
|
// API should still complete successfully as cancellation handling
|
|
// is typically done within the task execution
|
|
assert.NoError(t, err)
|
|
assert.NotNil(t, result)
|
|
assert.True(t, merr.Ok(result))
|
|
}
|