mirror of
https://gitee.com/milvus-io/milvus.git
synced 2026-01-07 19:31:51 +08:00
<!-- This is an auto-generated comment: release notes by coderabbit.ai --> - Core invariant: config refresh events must reliably propagate updated values and evict cached entries within a bounded time window; tests must observe this deterministically without relying on fixed sleeps. - Logic simplified: brittle fixed time.Sleep delays and separate error assertions were replaced by assert.Eventually polling blocks that combine value checks and cache-eviction verification, and consolidated checks to reduce redundant assertions. - Why no data loss / no behavior regression: only test synchronization and assertions were changed—production config manager code paths (value propagation, KV puts, cache eviction) are untouched; tests now wait for the same outcomes more robustly, so no mutation of runtime behavior or storage occurs. - Enhancement scope: this is a test-stability improvement (no new runtime capability); it fixes flaky unit tests (root cause: timing assumptions) by replacing fixed waits with bounded polling and by using t.Context for KV puts to align test context usage. <!-- end of auto-generated comment: release notes by coderabbit.ai --> Signed-off-by: aoiasd <zhicheng.yue@zilliz.com>
335 lines
8.7 KiB
Go
335 lines
8.7 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 config
|
|
|
|
import (
|
|
"context"
|
|
"os"
|
|
"path"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/cockroachdb/errors"
|
|
"github.com/stretchr/testify/assert"
|
|
"go.etcd.io/etcd/server/v3/embed"
|
|
"go.etcd.io/etcd/server/v3/etcdserver/api/v3client"
|
|
"golang.org/x/sync/errgroup"
|
|
)
|
|
|
|
func TestAllConfigFromManager(t *testing.T) {
|
|
mgr, _ := Init()
|
|
all := mgr.GetConfigs()
|
|
assert.Equal(t, 0, len(all))
|
|
|
|
mgr, _ = Init(WithEnvSource(formatKey))
|
|
all = mgr.GetConfigs()
|
|
assert.Less(t, 0, len(all))
|
|
}
|
|
|
|
func TestConfigChangeEvent(t *testing.T) {
|
|
dir, _ := os.MkdirTemp("", "milvus")
|
|
os.WriteFile(path.Join(dir, "milvus.yaml"), []byte("a.b: 1\nc.d: 2"), 0o600)
|
|
os.WriteFile(path.Join(dir, "user.yaml"), []byte("a.b: 3"), 0o600)
|
|
|
|
fs := NewFileSource(&FileInfo{[]string{path.Join(dir, "milvus.yaml"), path.Join(dir, "user.yaml")}, 1})
|
|
mgr, _ := Init()
|
|
err := mgr.AddSource(fs)
|
|
assert.NoError(t, err)
|
|
_, res, err := mgr.GetConfig("a.b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, res, "3")
|
|
os.WriteFile(path.Join(dir, "user.yaml"), []byte("a.b: 6"), 0o600)
|
|
time.Sleep(3 * time.Second)
|
|
_, res, err = mgr.GetConfig("a.b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, res, "6")
|
|
}
|
|
|
|
func TestAllDupliateSource(t *testing.T) {
|
|
mgr, _ := Init()
|
|
err := mgr.AddSource(NewEnvSource(formatKey))
|
|
assert.NoError(t, err)
|
|
err = mgr.AddSource(NewEnvSource(formatKey))
|
|
assert.Error(t, err)
|
|
|
|
err = mgr.AddSource(ErrSource{})
|
|
assert.Error(t, err, "error")
|
|
|
|
err = mgr.pullSourceConfigs("ErrSource")
|
|
assert.Error(t, err, "invalid source or source not added")
|
|
}
|
|
|
|
func TestBasic(t *testing.T) {
|
|
mgr, _ := Init()
|
|
|
|
// test set config
|
|
mgr.SetConfig("a.b", "aaa")
|
|
_, value, err := mgr.GetConfig("a.b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, value, "aaa")
|
|
_, _, err = mgr.GetConfig("a.a")
|
|
assert.Error(t, err)
|
|
|
|
// test delete config
|
|
mgr.SetConfig("a.b", "aaa")
|
|
mgr.DeleteConfig("a.b")
|
|
assert.Error(t, err)
|
|
|
|
// test reset config
|
|
mgr.ResetConfig("a.b")
|
|
assert.Error(t, err)
|
|
|
|
// test forbid config
|
|
envSource := NewEnvSource(formatKey)
|
|
err = mgr.AddSource(envSource)
|
|
assert.NoError(t, err)
|
|
|
|
envSource.configs.Insert("ab", "aaa")
|
|
mgr.OnEvent(&Event{
|
|
EventSource: envSource.GetSourceName(),
|
|
EventType: CreateType,
|
|
Key: "ab",
|
|
Value: "aaa",
|
|
})
|
|
_, value, err = mgr.GetConfig("a.b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, value, "aaa")
|
|
|
|
mgr.ForbidUpdate("a.b")
|
|
mgr.OnEvent(&Event{
|
|
EventSource: envSource.GetSourceName(),
|
|
EventType: UpdateType,
|
|
Key: "a.b",
|
|
Value: "bbb",
|
|
})
|
|
_, value, err = mgr.GetConfig("a.b")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, value, "aaa")
|
|
|
|
configs := mgr.FileConfigs()
|
|
assert.Len(t, configs, 0)
|
|
}
|
|
|
|
func TestOnEvent(t *testing.T) {
|
|
cfg, _ := embed.ConfigFromFile("../../configs/advanced/etcd.yaml")
|
|
cfg.Dir = "/tmp/milvus/test"
|
|
e, err := embed.StartEtcd(cfg)
|
|
assert.NoError(t, err)
|
|
defer e.Close()
|
|
defer os.RemoveAll(cfg.Dir)
|
|
|
|
client := v3client.New(e.Server)
|
|
|
|
dir, _ := os.MkdirTemp("", "milvus")
|
|
yamlFile := path.Join(dir, "milvus.yaml")
|
|
os.WriteFile(yamlFile, []byte("a.b: \"\""), 0o600)
|
|
mgr, _ := Init(WithEnvSource(formatKey),
|
|
WithFilesSource(&FileInfo{
|
|
Files: []string{yamlFile},
|
|
RefreshInterval: 10 * time.Millisecond,
|
|
}),
|
|
WithEtcdSource(&EtcdInfo{
|
|
Endpoints: []string{cfg.AdvertiseClientUrls[0].Host},
|
|
KeyPrefix: "test",
|
|
RefreshInterval: 10 * time.Millisecond,
|
|
}))
|
|
os.WriteFile(yamlFile, []byte("a.b: aaa"), 0o600)
|
|
assert.Eventually(t, func() bool {
|
|
_, value, err := mgr.GetConfig("a.b")
|
|
return err == nil && value == "aaa"
|
|
}, time.Second*5, time.Second)
|
|
|
|
ctx := context.Background()
|
|
client.KV.Put(ctx, "test/config/a/b", "bbb")
|
|
|
|
assert.Eventually(t, func() bool {
|
|
_, value, err := mgr.GetConfig("a.b")
|
|
return err == nil && value == "bbb"
|
|
}, time.Second*5, time.Second)
|
|
|
|
client.KV.Put(ctx, "test/config/a/b", "ccc")
|
|
assert.Eventually(t, func() bool {
|
|
_, value, err := mgr.GetConfig("a.b")
|
|
return err == nil && value == "ccc"
|
|
}, time.Second*5, time.Second)
|
|
|
|
os.WriteFile(yamlFile, []byte("a.b: ddd"), 0o600)
|
|
assert.Eventually(t, func() bool {
|
|
_, value, err := mgr.GetConfig("a.b")
|
|
return err == nil && value == "ccc"
|
|
}, time.Second*5, time.Second)
|
|
|
|
client.KV.Delete(ctx, "test/config/a/b")
|
|
assert.Eventually(t, func() bool {
|
|
_, value, err := mgr.GetConfig("a.b")
|
|
return err == nil && value == "ddd"
|
|
}, time.Second*5, time.Second)
|
|
}
|
|
|
|
func TestGetConfigAndSource(t *testing.T) {
|
|
mgr, _ := Init()
|
|
envSource := NewEnvSource(formatKey)
|
|
err := mgr.AddSource(envSource)
|
|
assert.NoError(t, err)
|
|
|
|
envSource.configs.Insert("ab-key", "ab-value")
|
|
mgr.OnEvent(&Event{
|
|
EventSource: envSource.GetSourceName(),
|
|
EventType: CreateType,
|
|
Key: "ab-key",
|
|
})
|
|
|
|
mgr.SetConfig("ac-key", "ac-value")
|
|
_, value, err := mgr.GetConfig("ac-key")
|
|
assert.NoError(t, err)
|
|
assert.Equal(t, value, "ac-value")
|
|
|
|
// test get all configs
|
|
configs := mgr.GetConfigsView()
|
|
v, ok := configs["ab-key"]
|
|
assert.True(t, ok)
|
|
assert.Contains(t, v, "EnvironmentSource")
|
|
|
|
v, ok = configs["ac-key"]
|
|
assert.True(t, ok)
|
|
assert.Contains(t, v, RuntimeSource)
|
|
}
|
|
|
|
func TestDeadlock(t *testing.T) {
|
|
mgr, _ := Init()
|
|
|
|
// test concurrent lock and recursive rlock
|
|
wg, _ := errgroup.WithContext(context.Background())
|
|
wg.Go(func() error {
|
|
for i := 0; i < 100; i++ {
|
|
mgr.GetBy(WithPrefix("rootcoord."))
|
|
}
|
|
return nil
|
|
})
|
|
|
|
wg.Go(func() error {
|
|
for i := 0; i < 100; i++ {
|
|
mgr.SetConfig("rootcoord.xxx", "111")
|
|
}
|
|
return nil
|
|
})
|
|
|
|
wg.Wait()
|
|
}
|
|
|
|
func TestCachedConfig(t *testing.T) {
|
|
cfg, _ := embed.ConfigFromFile("../../configs/advanced/etcd.yaml")
|
|
cfg.Dir = "/tmp/milvus/test"
|
|
e, err := embed.StartEtcd(cfg)
|
|
assert.NoError(t, err)
|
|
defer e.Close()
|
|
defer os.RemoveAll(cfg.Dir)
|
|
|
|
dir, _ := os.MkdirTemp("", "milvus")
|
|
yamlFile := path.Join(dir, "milvus.yaml")
|
|
os.WriteFile(yamlFile, []byte("a.b: aaa"), 0o600)
|
|
mgr, _ := Init(WithEnvSource(formatKey),
|
|
WithFilesSource(&FileInfo{
|
|
Files: []string{yamlFile},
|
|
RefreshInterval: 10 * time.Millisecond,
|
|
}),
|
|
WithEtcdSource(&EtcdInfo{
|
|
Endpoints: []string{cfg.AdvertiseClientUrls[0].Host},
|
|
KeyPrefix: "test",
|
|
RefreshInterval: 10 * time.Millisecond,
|
|
}))
|
|
// test get cached value from file
|
|
{
|
|
time.Sleep(time.Second)
|
|
_, exist := mgr.GetCachedValue("a.b")
|
|
assert.False(t, exist)
|
|
mgr.CASCachedValue("a.b", "aaa", "aaa")
|
|
val, exist := mgr.GetCachedValue("a.b")
|
|
assert.True(t, exist)
|
|
assert.Equal(t, "aaa", val.(string))
|
|
|
|
// after refresh, the cached value should be reset
|
|
os.WriteFile(yamlFile, []byte("a.b: xxx"), 0o600)
|
|
assert.Eventually(t, func() bool {
|
|
// make sure the config is refreshed
|
|
_, value, err := mgr.GetConfig("a.b")
|
|
if err != nil || value != "xxx" {
|
|
return false
|
|
}
|
|
|
|
// make sure the cached value is evicted
|
|
_, exist := mgr.GetCachedValue("a.b")
|
|
return !exist
|
|
}, time.Second*5, 500*time.Millisecond)
|
|
}
|
|
client := v3client.New(e.Server)
|
|
{
|
|
_, exist := mgr.GetCachedValue("c.d")
|
|
assert.False(t, exist)
|
|
mgr.CASCachedValue("cd", "", "xxx")
|
|
_, exist = mgr.GetCachedValue("cd")
|
|
assert.True(t, exist)
|
|
|
|
// after refresh, the cached value should be reset
|
|
client.KV.Put(t.Context(), "test/config/c/d", "www")
|
|
assert.Eventually(t, func() bool {
|
|
// make sure the config is refreshed
|
|
_, value, err := mgr.GetConfig("cd")
|
|
if err != nil || value != "www" {
|
|
return false
|
|
}
|
|
|
|
// make sure the cached value is evicted
|
|
_, exist := mgr.GetCachedValue("cd")
|
|
return !exist
|
|
}, time.Second*5, 500*time.Millisecond)
|
|
}
|
|
}
|
|
|
|
type ErrSource struct{}
|
|
|
|
func (e ErrSource) Close() {
|
|
}
|
|
|
|
func (e ErrSource) GetConfigurationByKey(string) (string, error) {
|
|
return "", errors.New("error")
|
|
}
|
|
|
|
// GetConfigurations implements Source
|
|
func (ErrSource) GetConfigurations() (map[string]string, error) {
|
|
return nil, errors.New("error")
|
|
}
|
|
|
|
// GetPriority implements Source
|
|
func (ErrSource) GetPriority() int {
|
|
return 2
|
|
}
|
|
|
|
func (ErrSource) SetManager(m ConfigManager) {
|
|
}
|
|
|
|
// GetSourceName implements Source
|
|
func (ErrSource) GetSourceName() string {
|
|
return "ErrSource"
|
|
}
|
|
|
|
func (e ErrSource) SetEventHandler(eh EventHandler) {
|
|
}
|
|
|
|
func (e ErrSource) UpdateOptions(opt Options) {
|
|
}
|