Zhen Ye 496331ffa8
enhance: support alias with WAL-based DDL framework (#44865)
issue: #43897

- Alias related DDL is implemented by WAL-based DDL framework now.
- Support following message type in wal AlterAlias, DropAlias.
- Alias DDL can be synced by new CDC now.
- Refactor some UT for Alias DDL.

Signed-off-by: chyezh <chyezh@outlook.com>
2025-10-18 15:12:01 +08:00

144 lines
3.7 KiB
Go

package broadcaster
import (
"sort"
"github.com/cockroachdb/errors"
"github.com/milvus-io/milvus/pkg/v2/proto/messagespb"
"github.com/milvus-io/milvus/pkg/v2/streaming/util/message"
"github.com/milvus-io/milvus/pkg/v2/util/lock"
"github.com/milvus-io/milvus/pkg/v2/util/typeutil"
)
// errFastLockFailed is the error for fast lock failed.
var errFastLockFailed = errors.New("fast lock failed")
// newResourceKeyLocker creates a new resource key locker.
func newResourceKeyLocker(metrics *broadcasterMetrics) *resourceKeyLocker {
return &resourceKeyLocker{
inner: lock.NewKeyLock[resourceLockKey](),
}
}
// newResourceLockKey creates a new resource lock key.
func newResourceLockKey(key message.ResourceKey) resourceLockKey {
return resourceLockKey{
Domain: key.Domain,
Key: key.Key,
}
}
// resourceLockKey is the key for the resource lock.
type resourceLockKey struct {
Domain messagespb.ResourceDomain
Key string
}
// resourceKeyLocker is the locker for the resource keys.
// It's a low performance implementation, but the broadcaster is only used at low frequency of ddl.
// So it's acceptable to use this implementation.
type resourceKeyLocker struct {
inner *lock.KeyLock[resourceLockKey]
}
// lockGuards is the guards for multiple resource keys.
type lockGuards struct {
guards []*lockGuard
}
// ResourceKeys returns the resource keys.
func (l *lockGuards) ResourceKeys() []message.ResourceKey {
keys := make([]message.ResourceKey, 0, len(l.guards))
for _, guard := range l.guards {
keys = append(keys, guard.key)
}
return keys
}
// append appends the guard to the guards.
func (l *lockGuards) append(guard *lockGuard) {
l.guards = append(l.guards, guard)
}
// Unlock unlocks the resource keys.
func (l *lockGuards) Unlock() {
// release the locks in reverse order to avoid deadlock.
for i := len(l.guards) - 1; i >= 0; i-- {
l.guards[i].Unlock()
}
l.guards = nil
}
// lockGuard is the guard for the resource key.
type lockGuard struct {
locker *resourceKeyLocker
key message.ResourceKey
}
// Unlock unlocks the resource key.
func (l *lockGuard) Unlock() {
l.locker.unlockWithKey(l.key)
}
// FastLock locks the resource keys without waiting.
// return error if the resource key is already locked.
func (r *resourceKeyLocker) FastLock(keys ...message.ResourceKey) (*lockGuards, error) {
keys = uniqueSortResourceKeys(keys)
g := &lockGuards{}
for _, key := range keys {
var locked bool
if key.Shared {
locked = r.inner.TryRLock(newResourceLockKey(key))
} else {
locked = r.inner.TryLock(newResourceLockKey(key))
}
if locked {
g.append(&lockGuard{locker: r, key: key})
continue
}
g.Unlock()
return nil, errors.Wrapf(errFastLockFailed, "fast lock failed at resource key %s", key.String())
}
return g, nil
}
// Lock locks the resource keys.
func (r *resourceKeyLocker) Lock(keys ...message.ResourceKey) *lockGuards {
// lock the keys in order to avoid deadlock.
keys = uniqueSortResourceKeys(keys)
g := &lockGuards{}
for _, key := range keys {
if key.Shared {
r.inner.RLock(newResourceLockKey(key))
} else {
r.inner.Lock(newResourceLockKey(key))
}
g.append(&lockGuard{locker: r, key: key})
}
return g
}
// unlockWithKey unlocks the resource key.
func (r *resourceKeyLocker) unlockWithKey(key message.ResourceKey) {
if key.Shared {
r.inner.RUnlock(newResourceLockKey(key))
return
}
r.inner.Unlock(newResourceLockKey(key))
}
// uniqueSortResourceKeys sorts the resource keys.
func uniqueSortResourceKeys(keys []message.ResourceKey) []message.ResourceKey {
keys = typeutil.NewSet(keys...).Collect()
sort.Slice(keys, func(i, j int) bool {
if keys[i].Domain != keys[j].Domain {
return keys[i].Domain < keys[j].Domain
}
return keys[i].Key < keys[j].Key
})
return keys
}