mirror of
https://gitee.com/milvus-io/milvus.git
synced 2025-12-06 17:18:35 +08:00
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>
144 lines
3.7 KiB
Go
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
|
|
}
|