mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Evict expired items from SimpleCache
This commit is contained in:
parent
3993c4d17f
commit
3bc9e75b28
3 changed files with 72 additions and 20 deletions
|
@ -107,14 +107,7 @@ func (p *playTracker) dispatchNowPlaying(ctx context.Context, userId string, t *
|
||||||
}
|
}
|
||||||
|
|
||||||
func (p *playTracker) GetNowPlaying(_ context.Context) ([]NowPlayingInfo, error) {
|
func (p *playTracker) GetNowPlaying(_ context.Context) ([]NowPlayingInfo, error) {
|
||||||
var res []NowPlayingInfo
|
res := p.playMap.Values()
|
||||||
for _, playerId := range p.playMap.Keys() {
|
|
||||||
info, err := p.playMap.Get(playerId)
|
|
||||||
if err != nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
res = append(res, info)
|
|
||||||
}
|
|
||||||
sort.Slice(res, func(i, j int) bool {
|
sort.Slice(res, func(i, j int) bool {
|
||||||
return res[i].Start.After(res[j].Start)
|
return res[i].Start.After(res[j].Start)
|
||||||
})
|
})
|
||||||
|
|
40
utils/cache/simple_cache.go
vendored
40
utils/cache/simple_cache.go
vendored
|
@ -2,9 +2,11 @@ package cache
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
|
"sync/atomic"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/jellydator/ttlcache/v3"
|
"github.com/jellydator/ttlcache/v3"
|
||||||
|
. "github.com/navidrome/navidrome/utils/gg"
|
||||||
)
|
)
|
||||||
|
|
||||||
type SimpleCache[K comparable, V any] interface {
|
type SimpleCache[K comparable, V any] interface {
|
||||||
|
@ -13,6 +15,7 @@ type SimpleCache[K comparable, V any] interface {
|
||||||
Get(key K) (V, error)
|
Get(key K) (V, error)
|
||||||
GetWithLoader(key K, loader func(key K) (V, time.Duration, error)) (V, error)
|
GetWithLoader(key K, loader func(key K) (V, time.Duration, error)) (V, error)
|
||||||
Keys() []K
|
Keys() []K
|
||||||
|
Values() []V
|
||||||
}
|
}
|
||||||
|
|
||||||
type Options struct {
|
type Options struct {
|
||||||
|
@ -40,15 +43,20 @@ func NewSimpleCache[K comparable, V any](options ...Options) SimpleCache[K, V] {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const evictionTimeout = 1 * time.Hour
|
||||||
|
|
||||||
type simpleCache[K comparable, V any] struct {
|
type simpleCache[K comparable, V any] struct {
|
||||||
data *ttlcache.Cache[K, V]
|
data *ttlcache.Cache[K, V]
|
||||||
|
evictionDeadline atomic.Pointer[time.Time]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *simpleCache[K, V]) Add(key K, value V) error {
|
func (c *simpleCache[K, V]) Add(key K, value V) error {
|
||||||
|
c.evictExpired()
|
||||||
return c.AddWithTTL(key, value, ttlcache.DefaultTTL)
|
return c.AddWithTTL(key, value, ttlcache.DefaultTTL)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *simpleCache[K, V]) AddWithTTL(key K, value V, ttl time.Duration) error {
|
func (c *simpleCache[K, V]) AddWithTTL(key K, value V, ttl time.Duration) error {
|
||||||
|
c.evictExpired()
|
||||||
item := c.data.Set(key, value, ttl)
|
item := c.data.Set(key, value, ttl)
|
||||||
if item == nil {
|
if item == nil {
|
||||||
return errors.New("failed to add item")
|
return errors.New("failed to add item")
|
||||||
|
@ -68,6 +76,7 @@ func (c *simpleCache[K, V]) Get(key K) (V, error) {
|
||||||
func (c *simpleCache[K, V]) GetWithLoader(key K, loader func(key K) (V, time.Duration, error)) (V, error) {
|
func (c *simpleCache[K, V]) GetWithLoader(key K, loader func(key K) (V, time.Duration, error)) (V, error) {
|
||||||
loaderWrapper := ttlcache.LoaderFunc[K, V](
|
loaderWrapper := ttlcache.LoaderFunc[K, V](
|
||||||
func(t *ttlcache.Cache[K, V], key K) *ttlcache.Item[K, V] {
|
func(t *ttlcache.Cache[K, V], key K) *ttlcache.Item[K, V] {
|
||||||
|
c.evictExpired()
|
||||||
value, ttl, err := loader(key)
|
value, ttl, err := loader(key)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil
|
return nil
|
||||||
|
@ -83,6 +92,31 @@ func (c *simpleCache[K, V]) GetWithLoader(key K, loader func(key K) (V, time.Dur
|
||||||
return item.Value(), nil
|
return item.Value(), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *simpleCache[K, V]) Keys() []K {
|
func (c *simpleCache[K, V]) evictExpired() {
|
||||||
return c.data.Keys()
|
if c.evictionDeadline.Load() == nil || c.evictionDeadline.Load().Before(time.Now()) {
|
||||||
|
c.data.DeleteExpired()
|
||||||
|
c.evictionDeadline.Store(P(time.Now().Add(evictionTimeout)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *simpleCache[K, V]) Keys() []K {
|
||||||
|
var res []K
|
||||||
|
c.data.Range(func(item *ttlcache.Item[K, V]) bool {
|
||||||
|
if !item.IsExpired() {
|
||||||
|
res = append(res, item.Key())
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
return res
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *simpleCache[K, V]) Values() []V {
|
||||||
|
var res []V
|
||||||
|
c.data.Range(func(item *ttlcache.Item[K, V]) bool {
|
||||||
|
if !item.IsExpired() {
|
||||||
|
res = append(res, item.Value())
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
})
|
||||||
|
return res
|
||||||
}
|
}
|
||||||
|
|
43
utils/cache/simple_cache_test.go
vendored
43
utils/cache/simple_cache_test.go
vendored
|
@ -31,7 +31,7 @@ var _ = Describe("SimpleCache", func() {
|
||||||
|
|
||||||
Describe("AddWithTTL and Get", func() {
|
Describe("AddWithTTL and Get", func() {
|
||||||
It("should add a value with TTL and retrieve it", func() {
|
It("should add a value with TTL and retrieve it", func() {
|
||||||
err := cache.AddWithTTL("key", "value", 1*time.Second)
|
err := cache.AddWithTTL("key", "value", 1*time.Minute)
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
|
||||||
value, err := cache.Get("key")
|
value, err := cache.Get("key")
|
||||||
|
@ -53,12 +53,12 @@ var _ = Describe("SimpleCache", func() {
|
||||||
Describe("GetWithLoader", func() {
|
Describe("GetWithLoader", func() {
|
||||||
It("should retrieve a value using the loader function", func() {
|
It("should retrieve a value using the loader function", func() {
|
||||||
loader := func(key string) (string, time.Duration, error) {
|
loader := func(key string) (string, time.Duration, error) {
|
||||||
return "value", 1 * time.Second, nil
|
return fmt.Sprintf("%s=value", key), 1 * time.Minute, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
value, err := cache.GetWithLoader("key", loader)
|
value, err := cache.GetWithLoader("key", loader)
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
Expect(value).To(Equal("value"))
|
Expect(value).To(Equal("key=value"))
|
||||||
})
|
})
|
||||||
|
|
||||||
It("should return the error returned by the loader function", func() {
|
It("should return the error returned by the loader function", func() {
|
||||||
|
@ -71,8 +71,8 @@ var _ = Describe("SimpleCache", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
Describe("Keys", func() {
|
Describe("Keys and Values", func() {
|
||||||
It("should return all keys", func() {
|
It("should return all keys and all values", func() {
|
||||||
err := cache.Add("key1", "value1")
|
err := cache.Add("key1", "value1")
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
|
||||||
|
@ -81,6 +81,24 @@ var _ = Describe("SimpleCache", func() {
|
||||||
|
|
||||||
keys := cache.Keys()
|
keys := cache.Keys()
|
||||||
Expect(keys).To(ConsistOf("key1", "key2"))
|
Expect(keys).To(ConsistOf("key1", "key2"))
|
||||||
|
|
||||||
|
values := cache.Values()
|
||||||
|
Expect(values).To(ConsistOf("value1", "value2"))
|
||||||
|
})
|
||||||
|
|
||||||
|
Context("when there are expired items in the cache", func() {
|
||||||
|
It("should not return expired items", func() {
|
||||||
|
Expect(cache.Add("key0", "value0")).To(Succeed())
|
||||||
|
for i := 1; i <= 3; i++ {
|
||||||
|
err := cache.AddWithTTL(fmt.Sprintf("key%d", i), fmt.Sprintf("value%d", i), 10*time.Millisecond)
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
}
|
||||||
|
|
||||||
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
|
||||||
|
Expect(cache.Keys()).To(ConsistOf("key0"))
|
||||||
|
Expect(cache.Values()).To(ConsistOf("value0"))
|
||||||
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -92,7 +110,7 @@ var _ = Describe("SimpleCache", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
It("should not add more items than the size limit", func() {
|
It("should drop the oldest item when the size limit is reached", func() {
|
||||||
for i := 1; i <= 3; i++ {
|
for i := 1; i <= 3; i++ {
|
||||||
err := cache.Add(fmt.Sprintf("key%d", i), fmt.Sprintf("value%d", i))
|
err := cache.Add(fmt.Sprintf("key%d", i), fmt.Sprintf("value%d", i))
|
||||||
Expect(err).NotTo(HaveOccurred())
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
@ -110,12 +128,19 @@ var _ = Describe("SimpleCache", func() {
|
||||||
})
|
})
|
||||||
|
|
||||||
It("should expire items after the default TTL", func() {
|
It("should expire items after the default TTL", func() {
|
||||||
_ = cache.Add("key", "value")
|
Expect(cache.AddWithTTL("key0", "value0", 1*time.Minute)).To(Succeed())
|
||||||
|
for i := 1; i <= 3; i++ {
|
||||||
|
err := cache.Add(fmt.Sprintf("key%d", i), fmt.Sprintf("value%d", i))
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
}
|
||||||
|
|
||||||
time.Sleep(50 * time.Millisecond)
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
|
||||||
_, err := cache.Get("key")
|
for i := 1; i <= 3; i++ {
|
||||||
Expect(err).To(HaveOccurred())
|
_, err := cache.Get(fmt.Sprintf("key%d", i))
|
||||||
|
Expect(err).To(HaveOccurred())
|
||||||
|
}
|
||||||
|
Expect(cache.Get("key0")).To(Equal("value0"))
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue