1
0
mirror of https://github.com/MontFerret/ferret.git synced 2024-12-14 11:23:02 +02:00
ferret/pkg/drivers/common/lazy.go
Tim Voronov a5cbdb435c
Bugfix/#332 nav deadlock (#333)
* Added failing e2e test

* Fixed deadlock on navigation

* Removed filter for e2e tests

* Updated method name in LazyValue struct

* Custom atomic value

* Fixed linting issue

* Updated comments
2019-07-16 10:34:13 -04:00

118 lines
2.3 KiB
Go

package common
import (
"context"
"sync"
"github.com/MontFerret/ferret/pkg/runtime/core"
"github.com/MontFerret/ferret/pkg/runtime/values"
)
type (
// LazyValueFactory represents a value initializer
LazyValueFactory func(ctx context.Context) (core.Value, error)
// LazyValue represents a value with late initialization
LazyValue struct {
mu sync.Mutex
factory LazyValueFactory
ready bool
value core.Value
err error
}
)
func NewLazyValue(factory LazyValueFactory) *LazyValue {
lz := new(LazyValue)
lz.ready = false
lz.factory = factory
lz.value = values.None
return lz
}
// Ready indicates whether the value is ready.
// @returns (Boolean) - Boolean value indicating whether the value is ready.
func (lv *LazyValue) Ready() bool {
lv.mu.Lock()
defer lv.mu.Unlock()
return lv.ready
}
// Read returns an underlying value.
// Not thread safe. Should not mutated.
// @returns (Value) - Underlying value if successfully loaded, otherwise error
func (lv *LazyValue) Read(ctx context.Context) (core.Value, error) {
lv.mu.Lock()
defer lv.mu.Unlock()
if !lv.ready {
lv.load(ctx)
}
return lv.value, lv.err
}
// Mutate safely mutates an underlying value.
// Loads a value if it's not ready.
// Thread safe.
func (lv *LazyValue) Mutate(ctx context.Context, mutator func(v core.Value, err error)) {
lv.mu.Lock()
defer lv.mu.Unlock()
if !lv.ready {
lv.load(ctx)
}
mutator(lv.value, lv.err)
}
// MutateIfReady safely mutates an underlying value only if it's ready.
func (lv *LazyValue) MutateIfReady(mutator func(v core.Value, err error)) {
lv.mu.Lock()
defer lv.mu.Unlock()
if lv.ready {
mutator(lv.value, lv.err)
}
}
// Reload resets the storage and loads data.
func (lv *LazyValue) Reload(ctx context.Context) {
lv.mu.Lock()
defer lv.mu.Unlock()
lv.resetInternal()
lv.load(ctx)
}
// Reset resets the storage.
// Next call of Read will trigger the factory function again.
func (lv *LazyValue) Reset() {
lv.mu.Lock()
defer lv.mu.Unlock()
lv.resetInternal()
}
func (lv *LazyValue) resetInternal() {
lv.ready = false
lv.value = values.None
lv.err = nil
}
func (lv *LazyValue) load(ctx context.Context) {
val, err := lv.factory(ctx)
if err == nil {
lv.value = val
lv.err = nil
} else {
lv.value = values.None
lv.err = err
}
lv.ready = true
}