mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 20:47:35 +03:00
Wrap ttlcache in our own SimpleCache implementation
This commit is contained in:
parent
4044642abf
commit
29bc17acd7
4 changed files with 157 additions and 17 deletions
60
utils/cache/simple_cache.go
vendored
Normal file
60
utils/cache/simple_cache.go
vendored
Normal file
|
@ -0,0 +1,60 @@
|
|||
package cache
|
||||
|
||||
import (
|
||||
"time"
|
||||
|
||||
"github.com/jellydator/ttlcache/v2"
|
||||
)
|
||||
|
||||
type SimpleCache[V any] interface {
|
||||
Add(key string, value V) error
|
||||
AddWithTTL(key string, value V, ttl time.Duration) error
|
||||
Get(key string) (V, error)
|
||||
GetWithLoader(key string, loader func(key string) (V, time.Duration, error)) (V, error)
|
||||
Keys() []string
|
||||
}
|
||||
|
||||
func NewSimpleCache[V any]() SimpleCache[V] {
|
||||
c := ttlcache.NewCache()
|
||||
c.SkipTTLExtensionOnHit(true)
|
||||
return &simpleCache[V]{
|
||||
data: c,
|
||||
}
|
||||
}
|
||||
|
||||
type simpleCache[V any] struct {
|
||||
data *ttlcache.Cache
|
||||
}
|
||||
|
||||
func (c *simpleCache[V]) Add(key string, value V) error {
|
||||
return c.data.Set(key, value)
|
||||
}
|
||||
|
||||
func (c *simpleCache[V]) AddWithTTL(key string, value V, ttl time.Duration) error {
|
||||
return c.data.SetWithTTL(key, value, ttl)
|
||||
}
|
||||
|
||||
func (c *simpleCache[V]) Get(key string) (V, error) {
|
||||
v, err := c.data.Get(key)
|
||||
if err != nil {
|
||||
var zero V
|
||||
return zero, err
|
||||
}
|
||||
return v.(V), nil
|
||||
}
|
||||
|
||||
func (c *simpleCache[V]) GetWithLoader(key string, loader func(key string) (V, time.Duration, error)) (V, error) {
|
||||
v, err := c.data.GetByLoader(key, func(key string) (interface{}, time.Duration, error) {
|
||||
v, ttl, err := loader(key)
|
||||
return v, ttl, err
|
||||
})
|
||||
if err != nil {
|
||||
var zero V
|
||||
return zero, err
|
||||
}
|
||||
return v.(V), nil
|
||||
}
|
||||
|
||||
func (c *simpleCache[V]) Keys() []string {
|
||||
return c.data.GetKeys()
|
||||
}
|
85
utils/cache/simple_cache_test.go
vendored
Normal file
85
utils/cache/simple_cache_test.go
vendored
Normal file
|
@ -0,0 +1,85 @@
|
|||
package cache
|
||||
|
||||
import (
|
||||
"errors"
|
||||
"time"
|
||||
|
||||
. "github.com/onsi/ginkgo/v2"
|
||||
. "github.com/onsi/gomega"
|
||||
)
|
||||
|
||||
var _ = Describe("SimpleCache", func() {
|
||||
var (
|
||||
cache SimpleCache[string]
|
||||
)
|
||||
|
||||
BeforeEach(func() {
|
||||
cache = NewSimpleCache[string]()
|
||||
})
|
||||
|
||||
Describe("Add and Get", func() {
|
||||
It("should add and retrieve a value", func() {
|
||||
err := cache.Add("key", "value")
|
||||
Expect(err).NotTo(HaveOccurred())
|
||||
|
||||
value, err := cache.Get("key")
|
||||
Expect(err).NotTo(HaveOccurred())
|
||||
Expect(value).To(Equal("value"))
|
||||
})
|
||||
})
|
||||
|
||||
Describe("AddWithTTL and Get", func() {
|
||||
It("should add a value with TTL and retrieve it", func() {
|
||||
err := cache.AddWithTTL("key", "value", 1*time.Second)
|
||||
Expect(err).NotTo(HaveOccurred())
|
||||
|
||||
value, err := cache.Get("key")
|
||||
Expect(err).NotTo(HaveOccurred())
|
||||
Expect(value).To(Equal("value"))
|
||||
})
|
||||
|
||||
It("should not retrieve a value after its TTL has expired", func() {
|
||||
err := cache.AddWithTTL("key", "value", 10*time.Millisecond)
|
||||
Expect(err).NotTo(HaveOccurred())
|
||||
|
||||
time.Sleep(50 * time.Millisecond)
|
||||
|
||||
_, err = cache.Get("key")
|
||||
Expect(err).To(HaveOccurred())
|
||||
})
|
||||
})
|
||||
|
||||
Describe("GetWithLoader", func() {
|
||||
It("should retrieve a value using the loader function", func() {
|
||||
loader := func(key string) (string, time.Duration, error) {
|
||||
return "value", 1 * time.Second, nil
|
||||
}
|
||||
|
||||
value, err := cache.GetWithLoader("key", loader)
|
||||
Expect(err).NotTo(HaveOccurred())
|
||||
Expect(value).To(Equal("value"))
|
||||
})
|
||||
|
||||
It("should return the error returned by the loader function", func() {
|
||||
loader := func(key string) (string, time.Duration, error) {
|
||||
return "", 0, errors.New("some error")
|
||||
}
|
||||
|
||||
_, err := cache.GetWithLoader("key", loader)
|
||||
Expect(err).To(MatchError("some error"))
|
||||
})
|
||||
})
|
||||
|
||||
Describe("Keys", func() {
|
||||
It("should return all keys", func() {
|
||||
err := cache.Add("key1", "value1")
|
||||
Expect(err).NotTo(HaveOccurred())
|
||||
|
||||
err = cache.Add("key2", "value2")
|
||||
Expect(err).NotTo(HaveOccurred())
|
||||
|
||||
keys := cache.Keys()
|
||||
Expect(keys).To(ConsistOf("key1", "key2"))
|
||||
})
|
||||
})
|
||||
})
|
Loading…
Add table
Add a link
Reference in a new issue