mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
When trying to PreCache, wait for ImageCache to be available
This commit is contained in:
parent
0c7c6ba020
commit
cc14485194
3 changed files with 30 additions and 15 deletions
|
@ -84,7 +84,7 @@ func createScanner() scanner.Scanner {
|
||||||
fileCache := artwork.GetImageCache()
|
fileCache := artwork.GetImageCache()
|
||||||
fFmpeg := ffmpeg.New()
|
fFmpeg := ffmpeg.New()
|
||||||
artworkArtwork := artwork.NewArtwork(dataStore, fileCache, fFmpeg)
|
artworkArtwork := artwork.NewArtwork(dataStore, fileCache, fFmpeg)
|
||||||
cacheWarmer := artwork.NewCacheWarmer(artworkArtwork)
|
cacheWarmer := artwork.NewCacheWarmer(artworkArtwork, fileCache)
|
||||||
broker := events.GetBroker()
|
broker := events.GetBroker()
|
||||||
scannerScanner := scanner.New(dataStore, playlists, cacheWarmer, broker)
|
scannerScanner := scanner.New(dataStore, playlists, cacheWarmer, broker)
|
||||||
return scannerScanner
|
return scannerScanner
|
||||||
|
|
|
@ -11,14 +11,16 @@ import (
|
||||||
"github.com/navidrome/navidrome/log"
|
"github.com/navidrome/navidrome/log"
|
||||||
"github.com/navidrome/navidrome/model"
|
"github.com/navidrome/navidrome/model"
|
||||||
"github.com/navidrome/navidrome/model/request"
|
"github.com/navidrome/navidrome/model/request"
|
||||||
|
"github.com/navidrome/navidrome/utils/cache"
|
||||||
"github.com/navidrome/navidrome/utils/pl"
|
"github.com/navidrome/navidrome/utils/pl"
|
||||||
|
"golang.org/x/exp/maps"
|
||||||
)
|
)
|
||||||
|
|
||||||
type CacheWarmer interface {
|
type CacheWarmer interface {
|
||||||
PreCache(artID model.ArtworkID)
|
PreCache(artID model.ArtworkID)
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewCacheWarmer(artwork Artwork) CacheWarmer {
|
func NewCacheWarmer(artwork Artwork, cache cache.FileCache) CacheWarmer {
|
||||||
// If image cache is disabled, return a NOOP implementation
|
// If image cache is disabled, return a NOOP implementation
|
||||||
if conf.Server.ImageCacheSize == "0" {
|
if conf.Server.ImageCacheSize == "0" {
|
||||||
return &noopCacheWarmer{}
|
return &noopCacheWarmer{}
|
||||||
|
@ -26,6 +28,8 @@ func NewCacheWarmer(artwork Artwork) CacheWarmer {
|
||||||
|
|
||||||
a := &cacheWarmer{
|
a := &cacheWarmer{
|
||||||
artwork: artwork,
|
artwork: artwork,
|
||||||
|
cache: cache,
|
||||||
|
buffer: make(map[string]struct{}),
|
||||||
wakeSignal: make(chan struct{}, 1),
|
wakeSignal: make(chan struct{}, 1),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -37,15 +41,16 @@ func NewCacheWarmer(artwork Artwork) CacheWarmer {
|
||||||
|
|
||||||
type cacheWarmer struct {
|
type cacheWarmer struct {
|
||||||
artwork Artwork
|
artwork Artwork
|
||||||
buffer []string
|
buffer map[string]struct{}
|
||||||
mutex sync.Mutex
|
mutex sync.Mutex
|
||||||
|
cache cache.FileCache
|
||||||
wakeSignal chan struct{}
|
wakeSignal chan struct{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *cacheWarmer) PreCache(artID model.ArtworkID) {
|
func (a *cacheWarmer) PreCache(artID model.ArtworkID) {
|
||||||
a.mutex.Lock()
|
a.mutex.Lock()
|
||||||
defer a.mutex.Unlock()
|
defer a.mutex.Unlock()
|
||||||
a.buffer = append(a.buffer, artID.String())
|
a.buffer[artID.String()] = struct{}{}
|
||||||
a.sendWakeSignal()
|
a.sendWakeSignal()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -59,22 +64,32 @@ func (a *cacheWarmer) sendWakeSignal() {
|
||||||
|
|
||||||
func (a *cacheWarmer) run(ctx context.Context) {
|
func (a *cacheWarmer) run(ctx context.Context) {
|
||||||
for {
|
for {
|
||||||
time.AfterFunc(5*time.Second, func() {
|
time.AfterFunc(10*time.Second, func() {
|
||||||
a.sendWakeSignal()
|
a.sendWakeSignal()
|
||||||
})
|
})
|
||||||
<-a.wakeSignal
|
<-a.wakeSignal
|
||||||
|
|
||||||
a.mutex.Lock()
|
// If cache not available, keep waiting
|
||||||
var batch []string
|
if !a.cache.Available(ctx) {
|
||||||
if len(a.buffer) > 0 {
|
if len(a.buffer) > 0 {
|
||||||
batch = a.buffer
|
log.Trace(ctx, "Cache not available, buffering precache request", "bufferLen", len(a.buffer))
|
||||||
a.buffer = nil
|
}
|
||||||
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
|
a.mutex.Lock()
|
||||||
|
|
||||||
|
// If there's nothing to send, keep waiting
|
||||||
|
if len(a.buffer) == 0 {
|
||||||
|
a.mutex.Unlock()
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
batch := maps.Keys(a.buffer)
|
||||||
|
a.buffer = make(map[string]struct{})
|
||||||
a.mutex.Unlock()
|
a.mutex.Unlock()
|
||||||
|
|
||||||
if len(batch) > 0 {
|
a.processBatch(ctx, batch)
|
||||||
a.processBatch(ctx, batch)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -102,4 +117,4 @@ func (a *cacheWarmer) doCacheImage(ctx context.Context, id string) error {
|
||||||
|
|
||||||
type noopCacheWarmer struct{}
|
type noopCacheWarmer struct{}
|
||||||
|
|
||||||
func (a *noopCacheWarmer) PreCache(id model.ArtworkID) {}
|
func (a *noopCacheWarmer) PreCache(model.ArtworkID) {}
|
||||||
|
|
2
utils/cache/file_caches.go
vendored
2
utils/cache/file_caches.go
vendored
|
@ -70,7 +70,7 @@ type fileCache struct {
|
||||||
mutex *sync.RWMutex
|
mutex *sync.RWMutex
|
||||||
}
|
}
|
||||||
|
|
||||||
func (fc *fileCache) Ready(ctx context.Context) bool {
|
func (fc *fileCache) Ready(_ context.Context) bool {
|
||||||
fc.mutex.RLock()
|
fc.mutex.RLock()
|
||||||
defer fc.mutex.RUnlock()
|
defer fc.mutex.RUnlock()
|
||||||
return fc.ready
|
return fc.ready
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue