mirror of
https://gitee.com/milvus-io/milvus.git
synced 2024-12-01 11:29:48 +08:00
465fd474de
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>
81 lines
1.9 KiB
Go
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
|
|
}
|