mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 13:07:36 +03:00
Also use SimpleCache in cache.HTTPClient
This commit is contained in:
parent
29bc17acd7
commit
29b7b740ce
3 changed files with 67 additions and 25 deletions
43
utils/cache/cached_http_client.go
vendored
43
utils/cache/cached_http_client.go
vendored
|
@ -9,16 +9,14 @@ import (
|
||||||
"net/http"
|
"net/http"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/jellydator/ttlcache/v2"
|
|
||||||
"github.com/navidrome/navidrome/log"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
const cacheSizeLimit = 100
|
const cacheSizeLimit = 100
|
||||||
|
|
||||||
type HTTPClient struct {
|
type HTTPClient struct {
|
||||||
cache *ttlcache.Cache
|
cache SimpleCache[string]
|
||||||
hc httpDoer
|
hc httpDoer
|
||||||
|
ttl time.Duration
|
||||||
}
|
}
|
||||||
|
|
||||||
type httpDoer interface {
|
type httpDoer interface {
|
||||||
|
@ -33,35 +31,32 @@ type requestData struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewHTTPClient(wrapped httpDoer, ttl time.Duration) *HTTPClient {
|
func NewHTTPClient(wrapped httpDoer, ttl time.Duration) *HTTPClient {
|
||||||
c := &HTTPClient{hc: wrapped}
|
c := &HTTPClient{hc: wrapped, ttl: ttl}
|
||||||
c.cache = ttlcache.NewCache()
|
c.cache = NewSimpleCache[string](Options{
|
||||||
c.cache.SetCacheSizeLimit(cacheSizeLimit)
|
SizeLimit: cacheSizeLimit,
|
||||||
c.cache.SkipTTLExtensionOnHit(true)
|
DefaultTTL: ttl,
|
||||||
c.cache.SetLoaderFunction(func(key string) (interface{}, time.Duration, error) {
|
|
||||||
req, err := c.deserializeReq(key)
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, err
|
|
||||||
}
|
|
||||||
resp, err := c.hc.Do(req)
|
|
||||||
if err != nil {
|
|
||||||
return nil, 0, err
|
|
||||||
}
|
|
||||||
defer resp.Body.Close()
|
|
||||||
return c.serializeResponse(resp), ttl, nil
|
|
||||||
})
|
|
||||||
c.cache.SetNewItemCallback(func(key string, value interface{}) {
|
|
||||||
log.Trace("New request cached", "req", key, "resp", value)
|
|
||||||
})
|
})
|
||||||
return c
|
return c
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *HTTPClient) Do(req *http.Request) (*http.Response, error) {
|
func (c *HTTPClient) Do(req *http.Request) (*http.Response, error) {
|
||||||
key := c.serializeReq(req)
|
key := c.serializeReq(req)
|
||||||
respStr, err := c.cache.Get(key)
|
respStr, err := c.cache.GetWithLoader(key, func(key string) (string, time.Duration, error) {
|
||||||
|
req, err := c.deserializeReq(key)
|
||||||
|
if err != nil {
|
||||||
|
return "", 0, err
|
||||||
|
}
|
||||||
|
resp, err := c.hc.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return "", 0, err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
return c.serializeResponse(resp), c.ttl, nil
|
||||||
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
return c.deserializeResponse(req, respStr.(string))
|
return c.deserializeResponse(req, respStr)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *HTTPClient) serializeReq(req *http.Request) string {
|
func (c *HTTPClient) serializeReq(req *http.Request) string {
|
||||||
|
|
12
utils/cache/simple_cache.go
vendored
12
utils/cache/simple_cache.go
vendored
|
@ -14,9 +14,19 @@ type SimpleCache[V any] interface {
|
||||||
Keys() []string
|
Keys() []string
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewSimpleCache[V any]() SimpleCache[V] {
|
type Options struct {
|
||||||
|
SizeLimit int
|
||||||
|
DefaultTTL time.Duration
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewSimpleCache[V any](options ...Options) SimpleCache[V] {
|
||||||
c := ttlcache.NewCache()
|
c := ttlcache.NewCache()
|
||||||
c.SkipTTLExtensionOnHit(true)
|
c.SkipTTLExtensionOnHit(true)
|
||||||
|
if len(options) > 0 {
|
||||||
|
c.SetCacheSizeLimit(options[0].SizeLimit)
|
||||||
|
_ = c.SetTTL(options[0].DefaultTTL)
|
||||||
|
}
|
||||||
|
|
||||||
return &simpleCache[V]{
|
return &simpleCache[V]{
|
||||||
data: c,
|
data: c,
|
||||||
}
|
}
|
||||||
|
|
37
utils/cache/simple_cache_test.go
vendored
37
utils/cache/simple_cache_test.go
vendored
|
@ -2,6 +2,7 @@ package cache
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
|
"fmt"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
. "github.com/onsi/ginkgo/v2"
|
. "github.com/onsi/ginkgo/v2"
|
||||||
|
@ -82,4 +83,40 @@ var _ = Describe("SimpleCache", func() {
|
||||||
Expect(keys).To(ConsistOf("key1", "key2"))
|
Expect(keys).To(ConsistOf("key1", "key2"))
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
Describe("Options", func() {
|
||||||
|
Context("when size limit is set", func() {
|
||||||
|
BeforeEach(func() {
|
||||||
|
cache = NewSimpleCache[string](Options{
|
||||||
|
SizeLimit: 2,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
It("should not add more items than the size limit", func() {
|
||||||
|
for i := 1; i <= 3; i++ {
|
||||||
|
err := cache.Add(fmt.Sprintf("key%d", i), fmt.Sprintf("value%d", i))
|
||||||
|
Expect(err).NotTo(HaveOccurred())
|
||||||
|
}
|
||||||
|
|
||||||
|
Expect(cache.Keys()).To(ConsistOf("key2", "key3"))
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
Context("when default TTL is set", func() {
|
||||||
|
BeforeEach(func() {
|
||||||
|
cache = NewSimpleCache[string](Options{
|
||||||
|
DefaultTTL: 10 * time.Millisecond,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
It("should expire items after the default TTL", func() {
|
||||||
|
_ = cache.Add("key", "value")
|
||||||
|
|
||||||
|
time.Sleep(50 * time.Millisecond)
|
||||||
|
|
||||||
|
_, err := cache.Get("key")
|
||||||
|
Expect(err).To(HaveOccurred())
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
})
|
})
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue