milvus/pkg/util/replicateutil/config_validator_test.go
yihao.dai 20411e5218
fix: Fix replicator cannot stop and enhance replicate config validator (#44531)
1. Fix replicator cannot stop if error occurs on replicate stream RPC.
2. Simplify replicate stream client.
3. Enhance replicate config validator: 
1. Compare the incoming replicate config, cluster attributes must not be
changed.
  2. Cluster URI must be unique.
  3. Remove the check of pchannel prefix.
 
issue: https://github.com/milvus-io/milvus/issues/44123

---------

Signed-off-by: bigsheeper <yihao.dai@zilliz.com>
Co-authored-by: chyezh <chyezh@outlook.com>
2025-09-24 11:54:03 +08:00

934 lines
26 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 replicateutil
import (
"testing"
"github.com/stretchr/testify/assert"
"github.com/milvus-io/milvus-proto/go-api/v2/commonpb"
)
// createValidValidatorConfig creates a valid ReplicateConfiguration for testing
func createValidValidatorConfig() *commonpb.ReplicateConfiguration {
return &commonpb.ReplicateConfiguration{
Clusters: []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
{
ClusterId: "cluster-2",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19531",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
},
CrossClusterTopology: []*commonpb.CrossClusterTopology{
{
SourceClusterId: "cluster-1",
TargetClusterId: "cluster-2",
},
},
}
}
// createStarTopologyConfig creates a valid star topology configuration
func createStarTopologyConfig() *commonpb.ReplicateConfiguration {
return &commonpb.ReplicateConfiguration{
Clusters: []*commonpb.MilvusCluster{
{
ClusterId: "center-cluster",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
{
ClusterId: "leaf-cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19531",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
{
ClusterId: "leaf-cluster-2",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19532",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
},
CrossClusterTopology: []*commonpb.CrossClusterTopology{
{
SourceClusterId: "center-cluster",
TargetClusterId: "leaf-cluster-1",
},
{
SourceClusterId: "center-cluster",
TargetClusterId: "leaf-cluster-2",
},
},
}
}
func TestNewReplicateConfigValidator(t *testing.T) {
config := createValidValidatorConfig()
currentPChannels := []string{"channel-1", "channel-2"}
t.Run("success - create validator without current config", func(t *testing.T) {
validator := NewReplicateConfigValidator(config, nil, "cluster-1", currentPChannels)
assert.NotNil(t, validator)
assert.Equal(t, config, validator.incomingConfig)
assert.Equal(t, currentPChannels, validator.currentPChannels)
assert.NotNil(t, validator.clusterMap)
assert.Equal(t, 0, len(validator.clusterMap)) // clusterMap is built during validation
assert.Nil(t, validator.currentConfig)
})
t.Run("success - create validator with current config", func(t *testing.T) {
currentConfig := createValidValidatorConfig()
validator := NewReplicateConfigValidator(config, currentConfig, "cluster-1", currentPChannels)
assert.NotNil(t, validator)
assert.Equal(t, config, validator.incomingConfig)
assert.Equal(t, currentConfig, validator.currentConfig)
assert.Equal(t, currentPChannels, validator.currentPChannels)
assert.NotNil(t, validator.clusterMap)
assert.Equal(t, 0, len(validator.clusterMap)) // clusterMap is built during validation
})
}
func TestReplicateConfigValidator_Validate(t *testing.T) {
t.Run("success - valid configuration without current config", func(t *testing.T) {
config := createValidValidatorConfig()
currentPChannels := []string{"channel-1", "channel-2"}
validator := NewReplicateConfigValidator(config, nil, "cluster-1", currentPChannels)
err := validator.Validate()
assert.NoError(t, err)
})
t.Run("success - valid configuration with current config", func(t *testing.T) {
config := createValidValidatorConfig()
currentConfig := createValidValidatorConfig()
currentPChannels := []string{"channel-1", "channel-2"}
validator := NewReplicateConfigValidator(config, currentConfig, "cluster-1", currentPChannels)
err := validator.Validate()
assert.NoError(t, err)
})
t.Run("error - nil incoming config", func(t *testing.T) {
validator := NewReplicateConfigValidator(nil, nil, "cluster-1", []string{})
err := validator.Validate()
assert.Error(t, err)
assert.Contains(t, err.Error(), "config cannot be nil")
})
t.Run("error - empty clusters", func(t *testing.T) {
config := &commonpb.ReplicateConfiguration{
Clusters: []*commonpb.MilvusCluster{},
CrossClusterTopology: []*commonpb.CrossClusterTopology{},
}
validator := NewReplicateConfigValidator(config, nil, "cluster-1", []string{})
err := validator.Validate()
assert.Error(t, err)
assert.Contains(t, err.Error(), "clusters list cannot be empty")
})
}
func TestReplicateConfigValidator_validateClusterBasic(t *testing.T) {
t.Run("success - valid clusters", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
{
ClusterId: "cluster-2",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19531",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.NoError(t, err)
assert.Len(t, validator.clusterMap, 2)
assert.NotNil(t, validator.clusterMap["cluster-1"])
assert.NotNil(t, validator.clusterMap["cluster-2"])
})
t.Run("error - nil cluster", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
nil,
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "cluster at index 0 is nil")
})
t.Run("error - empty cluster ID", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "has empty clusterID")
})
t.Run("error - cluster ID with whitespace", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster 1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "containing whitespace characters")
})
t.Run("error - nil connection param", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: nil,
Pchannels: []string{"channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "has nil connection_param")
})
t.Run("error - empty URI", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "",
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "has empty URI")
})
t.Run("error - invalid URI format", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "invalid-uri-format",
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "has invalid URI format")
})
t.Run("error - empty pchannels", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "has empty pchannels")
})
t.Run("error - empty pchannel", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"", "channel-2"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "has empty pchannel at index 0")
})
t.Run("error - duplicate pchannel within cluster", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "has duplicate pchannel")
})
t.Run("error - inconsistent pchannel count", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
{
ClusterId: "cluster-2",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19531",
Token: "test-token",
},
Pchannels: []string{"channel-1"}, // Only 1 channel instead of 2
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "has 1 pchannels, but expected 2")
})
t.Run("error - duplicate cluster ID", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
{
ClusterId: "cluster-1", // Duplicate cluster ID
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19531",
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "duplicate clusterID found")
})
t.Run("error - duplicate URI across clusters", func(t *testing.T) {
clusters := []*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
{
ClusterId: "cluster-2",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530", // Same URI as cluster-1
Token: "test-token",
},
Pchannels: []string{"channel-1"},
},
}
validator := &ReplicateConfigValidator{
clusterMap: make(map[string]*commonpb.MilvusCluster),
}
err := validator.validateClusterBasic(clusters)
assert.Error(t, err)
assert.Contains(t, err.Error(), "duplicate URI found")
})
}
func TestReplicateConfigValidator_validateRelevance(t *testing.T) {
t.Run("success - current cluster included and pchannels match", func(t *testing.T) {
validator := &ReplicateConfigValidator{
currentClusterID: "cluster-1",
currentPChannels: []string{"channel-1", "channel-2"},
clusterMap: map[string]*commonpb.MilvusCluster{
"cluster-1": {
ClusterId: "cluster-1",
Pchannels: []string{"channel-1", "channel-2"},
},
},
}
err := validator.validateRelevance()
assert.NoError(t, err)
})
t.Run("error - current cluster not included", func(t *testing.T) {
validator := &ReplicateConfigValidator{
currentClusterID: "cluster-1",
currentPChannels: []string{"channel-1"},
clusterMap: map[string]*commonpb.MilvusCluster{
"cluster-2": {
ClusterId: "cluster-2",
Pchannels: []string{"channel-1"},
},
},
}
err := validator.validateRelevance()
assert.Error(t, err)
assert.Contains(t, err.Error(), "must be included in the clusters list")
})
t.Run("error - pchannels don't match", func(t *testing.T) {
validator := &ReplicateConfigValidator{
currentClusterID: "cluster-1",
currentPChannels: []string{"channel-1", "channel-2"},
clusterMap: map[string]*commonpb.MilvusCluster{
"cluster-1": {
ClusterId: "cluster-1",
Pchannels: []string{"channel-1", "channel-3"}, // Different channels
},
},
}
err := validator.validateRelevance()
assert.Error(t, err)
assert.Contains(t, err.Error(), "current pchannels do not match")
})
}
func TestReplicateConfigValidator_validateTopologyEdgeUniqueness(t *testing.T) {
validator := &ReplicateConfigValidator{
clusterMap: map[string]*commonpb.MilvusCluster{
"cluster-1": {ClusterId: "cluster-1"},
"cluster-2": {ClusterId: "cluster-2"},
"cluster-3": {ClusterId: "cluster-3"},
},
}
t.Run("success - unique edges", func(t *testing.T) {
topologies := []*commonpb.CrossClusterTopology{
{
SourceClusterId: "cluster-1",
TargetClusterId: "cluster-2",
},
{
SourceClusterId: "cluster-1",
TargetClusterId: "cluster-3",
},
}
err := validator.validateTopologyEdgeUniqueness(topologies)
assert.NoError(t, err)
})
t.Run("success - empty topologies", func(t *testing.T) {
topologies := []*commonpb.CrossClusterTopology{}
err := validator.validateTopologyEdgeUniqueness(topologies)
assert.NoError(t, err)
})
t.Run("error - nil topology", func(t *testing.T) {
topologies := []*commonpb.CrossClusterTopology{
nil,
{
SourceClusterId: "cluster-1",
TargetClusterId: "cluster-2",
},
}
err := validator.validateTopologyEdgeUniqueness(topologies)
assert.Error(t, err)
assert.Contains(t, err.Error(), "topology at index 0 is nil")
})
t.Run("error - source cluster not exists", func(t *testing.T) {
topologies := []*commonpb.CrossClusterTopology{
{
SourceClusterId: "non-existent-cluster",
TargetClusterId: "cluster-2",
},
}
err := validator.validateTopologyEdgeUniqueness(topologies)
assert.Error(t, err)
assert.Contains(t, err.Error(), "references non-existent source cluster")
})
t.Run("error - target cluster not exists", func(t *testing.T) {
topologies := []*commonpb.CrossClusterTopology{
{
SourceClusterId: "cluster-1",
TargetClusterId: "non-existent-cluster",
},
}
err := validator.validateTopologyEdgeUniqueness(topologies)
assert.Error(t, err)
assert.Contains(t, err.Error(), "references non-existent target cluster")
})
t.Run("error - duplicate edge", func(t *testing.T) {
topologies := []*commonpb.CrossClusterTopology{
{
SourceClusterId: "cluster-1",
TargetClusterId: "cluster-2",
},
{
SourceClusterId: "cluster-1",
TargetClusterId: "cluster-2", // Duplicate edge
},
}
err := validator.validateTopologyEdgeUniqueness(topologies)
assert.Error(t, err)
assert.Contains(t, err.Error(), "duplicate topology relationship found")
})
}
func TestReplicateConfigValidator_validateTopologyTypeConstraint(t *testing.T) {
t.Run("success - valid star topology", func(t *testing.T) {
validator := &ReplicateConfigValidator{
clusterMap: map[string]*commonpb.MilvusCluster{
"center-cluster": {ClusterId: "center-cluster"},
"leaf-cluster-1": {ClusterId: "leaf-cluster-1"},
"leaf-cluster-2": {ClusterId: "leaf-cluster-2"},
},
}
topologies := []*commonpb.CrossClusterTopology{
{
SourceClusterId: "center-cluster",
TargetClusterId: "leaf-cluster-1",
},
{
SourceClusterId: "center-cluster",
TargetClusterId: "leaf-cluster-2",
},
}
err := validator.validateTopologyTypeConstraint(topologies)
assert.NoError(t, err)
})
t.Run("success - empty topologies", func(t *testing.T) {
validator := &ReplicateConfigValidator{
clusterMap: map[string]*commonpb.MilvusCluster{
"cluster-1": {ClusterId: "cluster-1"},
},
}
topologies := []*commonpb.CrossClusterTopology{}
err := validator.validateTopologyTypeConstraint(topologies)
assert.NoError(t, err)
})
t.Run("error - no center node", func(t *testing.T) {
validator := &ReplicateConfigValidator{
clusterMap: map[string]*commonpb.MilvusCluster{
"cluster-1": {ClusterId: "cluster-1"},
"cluster-2": {ClusterId: "cluster-2"},
},
}
topologies := []*commonpb.CrossClusterTopology{
{
SourceClusterId: "cluster-1",
TargetClusterId: "cluster-2",
},
{
SourceClusterId: "cluster-2",
TargetClusterId: "cluster-1",
},
}
err := validator.validateTopologyTypeConstraint(topologies)
assert.Error(t, err)
assert.Contains(t, err.Error(), "no center node found")
})
t.Run("error - leaf node with wrong degrees", func(t *testing.T) {
validator := &ReplicateConfigValidator{
clusterMap: map[string]*commonpb.MilvusCluster{
"center-cluster": {ClusterId: "center-cluster"},
"leaf-cluster-1": {ClusterId: "leaf-cluster-1"},
"leaf-cluster-2": {ClusterId: "leaf-cluster-2"},
},
}
topologies := []*commonpb.CrossClusterTopology{
{
SourceClusterId: "center-cluster",
TargetClusterId: "leaf-cluster-1",
},
{
SourceClusterId: "center-cluster",
TargetClusterId: "leaf-cluster-2",
},
{
SourceClusterId: "leaf-cluster-1",
TargetClusterId: "leaf-cluster-2",
},
}
err := validator.validateTopologyTypeConstraint(topologies)
assert.Error(t, err)
assert.Contains(t, err.Error(), "does not follow star topology pattern")
})
}
func TestEqualIgnoreOrder(t *testing.T) {
t.Run("success - same slices in different order", func(t *testing.T) {
a := []string{"a", "b", "c"}
b := []string{"c", "a", "b"}
result := equalIgnoreOrder(a, b)
assert.True(t, result)
})
t.Run("success - same slices in same order", func(t *testing.T) {
a := []string{"a", "b", "c"}
b := []string{"a", "b", "c"}
result := equalIgnoreOrder(a, b)
assert.True(t, result)
})
t.Run("success - empty slices", func(t *testing.T) {
a := []string{}
b := []string{}
result := equalIgnoreOrder(a, b)
assert.True(t, result)
})
t.Run("failure - different lengths", func(t *testing.T) {
a := []string{"a", "b"}
b := []string{"a", "b", "c"}
result := equalIgnoreOrder(a, b)
assert.False(t, result)
})
t.Run("failure - different content", func(t *testing.T) {
a := []string{"a", "b", "c"}
b := []string{"a", "b", "d"}
result := equalIgnoreOrder(a, b)
assert.False(t, result)
})
t.Run("failure - different counts of same elements", func(t *testing.T) {
a := []string{"a", "a", "b"}
b := []string{"a", "b", "b"}
result := equalIgnoreOrder(a, b)
assert.False(t, result)
})
}
func TestReplicateConfigValidator_validateConfigComparison(t *testing.T) {
// Helper function to create a config with specific clusters
createConfigWithClusters := func(clusters []*commonpb.MilvusCluster) *commonpb.ReplicateConfiguration {
return &commonpb.ReplicateConfiguration{
Clusters: clusters,
CrossClusterTopology: []*commonpb.CrossClusterTopology{},
}
}
t.Run("success - no current config", func(t *testing.T) {
config := createValidValidatorConfig()
currentPChannels := []string{"channel-1", "channel-2"}
validator := NewReplicateConfigValidator(config, nil, "cluster-1", currentPChannels)
err := validator.Validate()
assert.NoError(t, err)
})
t.Run("success - new cluster added", func(t *testing.T) {
currentConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
incomingConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
{
ClusterId: "cluster-2", // New cluster
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19531",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
validator := NewReplicateConfigValidator(incomingConfig, currentConfig, "cluster-1", []string{"channel-1", "channel-2"})
err := validator.Validate()
assert.NoError(t, err)
})
t.Run("error - ConnectionParam changed", func(t *testing.T) {
currentConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "old-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
incomingConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "new-token", // Token changed - should fail
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
// Test the config comparison validation directly
validator := &ReplicateConfigValidator{
incomingConfig: incomingConfig,
currentConfig: currentConfig,
}
err := validator.validateConfigComparison()
assert.Error(t, err)
assert.Contains(t, err.Error(), "connection_param.token cannot be changed")
})
t.Run("error - pchannels changed", func(t *testing.T) {
currentConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
incomingConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-3"}, // Different pchannels
},
})
// Test the config comparison validation directly
validator := &ReplicateConfigValidator{
incomingConfig: incomingConfig,
currentConfig: currentConfig,
}
err := validator.validateConfigComparison()
assert.Error(t, err)
assert.Contains(t, err.Error(), "pchannels cannot be changed")
})
t.Run("error - ConnectionParam URI changed", func(t *testing.T) {
currentConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
incomingConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19531", // URI changed - should fail
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
// Test the config comparison validation directly
validator := &ReplicateConfigValidator{
incomingConfig: incomingConfig,
currentConfig: currentConfig,
}
err := validator.validateConfigComparison()
assert.Error(t, err)
assert.Contains(t, err.Error(), "connection_param.uri cannot be changed")
})
t.Run("success - same cluster with no changes", func(t *testing.T) {
currentConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1",
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
incomingConfig := createConfigWithClusters([]*commonpb.MilvusCluster{
{
ClusterId: "cluster-1", // Same cluster ID
ConnectionParam: &commonpb.ConnectionParam{
Uri: "localhost:19530",
Token: "test-token",
},
Pchannels: []string{"channel-1", "channel-2"},
},
})
// Test the config comparison validation directly
validator := &ReplicateConfigValidator{
incomingConfig: incomingConfig,
currentConfig: currentConfig,
}
err := validator.validateConfigComparison()
assert.NoError(t, err) // This should pass since it's the same cluster
})
}