milvus/pkg/util/syncutil/versioned_notifier.go
chyezh 465fd474de
enhance: add syncutil type ContextCond and VersionedNotifier (#30648)
issue: #30647

- ContextCond is a broadcast-only condition variable which can be
canceled by context.

- VersionedNotifier is a version-based notifier-listener implementation,
which promise no change can be ignored.

Signed-off-by: chyezh <chyezh@outlook.com>
2024-03-15 15:41:04 +08:00

81 lines
1.9 KiB
Go

package syncutil
import (
"context"
"sync"
)
const (
VersionedListenAtEarliest versionedListenAt = -1
VersionedListenAtLatest versionedListenAt = -2
)
// versionedListenerAt is the position where the listener starts to listen.
type versionedListenAt int
// NewVersionedNotifier creates a new VersionedNotifier.
func NewVersionedNotifier() *VersionedNotifier {
return &VersionedNotifier{
inner: &versionedSignal{
version: 0,
cond: NewContextCond(&sync.Mutex{}),
},
}
}
// versionedSignal is a signal with version.
type versionedSignal struct {
version int
cond *ContextCond
}
// VersionedNotifier is a notifier with version.
// A version-based notifier, any change of version could be seen by all listeners without lost.
type VersionedNotifier struct {
inner *versionedSignal
}
// NotifyAll notifies all listeners.
func (vn *VersionedNotifier) NotifyAll() {
vn.inner.cond.LockAndBroadcast()
vn.inner.version++
vn.inner.cond.L.Unlock()
}
// Listen creates a listener at given position.
func (vn *VersionedNotifier) Listen(at versionedListenAt) *VersionedListener {
var last int
if at == VersionedListenAtEarliest {
last = -1
} else if at == VersionedListenAtLatest {
vn.inner.cond.L.Lock()
last = vn.inner.version
vn.inner.cond.L.Unlock()
}
return &VersionedListener{
lastNotifiedVersion: last,
inner: vn.inner,
}
}
// VersionedListener is a listener with version.
type VersionedListener struct {
lastNotifiedVersion int
inner *versionedSignal
}
// Wait waits for the next notification.
// If the context is canceled, it returns the error.
// Otherwise it will block until the next notification.
func (vl *VersionedListener) Wait(ctx context.Context) error {
vl.inner.cond.L.Lock()
for vl.lastNotifiedVersion >= vl.inner.version {
if err := vl.inner.cond.Wait(ctx); err != nil {
return err
}
}
vl.lastNotifiedVersion = vl.inner.version
vl.inner.cond.L.Unlock()
return nil
}