mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-03 20:47:35 +03:00
* Upgrade to Golang 1.23rc1 * Fix imports * Go 1.23 final version * Fix lint compatibility with ci-goreleaser
147 lines
3.3 KiB
Go
147 lines
3.3 KiB
Go
package artwork
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"maps"
|
|
"slices"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/navidrome/navidrome/conf"
|
|
"github.com/navidrome/navidrome/consts"
|
|
"github.com/navidrome/navidrome/log"
|
|
"github.com/navidrome/navidrome/model"
|
|
"github.com/navidrome/navidrome/model/request"
|
|
"github.com/navidrome/navidrome/utils/cache"
|
|
"github.com/navidrome/navidrome/utils/pl"
|
|
)
|
|
|
|
type CacheWarmer interface {
|
|
PreCache(artID model.ArtworkID)
|
|
}
|
|
|
|
func NewCacheWarmer(artwork Artwork, cache cache.FileCache) CacheWarmer {
|
|
// If image cache is disabled, return a NOOP implementation
|
|
if conf.Server.ImageCacheSize == "0" || !conf.Server.EnableArtworkPrecache {
|
|
return &noopCacheWarmer{}
|
|
}
|
|
|
|
a := &cacheWarmer{
|
|
artwork: artwork,
|
|
cache: cache,
|
|
buffer: make(map[model.ArtworkID]struct{}),
|
|
wakeSignal: make(chan struct{}, 1),
|
|
}
|
|
|
|
// Create a context with a fake admin user, to be able to pre-cache Playlist CoverArts
|
|
ctx := request.WithUser(context.TODO(), model.User{IsAdmin: true})
|
|
go a.run(ctx)
|
|
return a
|
|
}
|
|
|
|
type cacheWarmer struct {
|
|
artwork Artwork
|
|
buffer map[model.ArtworkID]struct{}
|
|
mutex sync.Mutex
|
|
cache cache.FileCache
|
|
wakeSignal chan struct{}
|
|
}
|
|
|
|
var ignoredIds = map[string]struct{}{
|
|
consts.VariousArtistsID: {},
|
|
consts.UnknownArtistID: {},
|
|
}
|
|
|
|
func (a *cacheWarmer) PreCache(artID model.ArtworkID) {
|
|
if _, shouldIgnore := ignoredIds[artID.ID]; shouldIgnore {
|
|
return
|
|
}
|
|
a.mutex.Lock()
|
|
defer a.mutex.Unlock()
|
|
a.buffer[artID] = struct{}{}
|
|
a.sendWakeSignal()
|
|
}
|
|
|
|
func (a *cacheWarmer) sendWakeSignal() {
|
|
// Don't block if the previous signal was not read yet
|
|
select {
|
|
case a.wakeSignal <- struct{}{}:
|
|
default:
|
|
}
|
|
}
|
|
|
|
func (a *cacheWarmer) run(ctx context.Context) {
|
|
for {
|
|
a.waitSignal(ctx, 10*time.Second)
|
|
if ctx.Err() != nil {
|
|
break
|
|
}
|
|
|
|
// If cache not available, keep waiting
|
|
if !a.cache.Available(ctx) {
|
|
if len(a.buffer) > 0 {
|
|
log.Trace(ctx, "Cache not available, buffering precache request", "bufferLen", len(a.buffer))
|
|
}
|
|
continue
|
|
}
|
|
|
|
a.mutex.Lock()
|
|
|
|
// If there's nothing to send, keep waiting
|
|
if len(a.buffer) == 0 {
|
|
a.mutex.Unlock()
|
|
continue
|
|
}
|
|
|
|
batch := slices.Collect(maps.Keys(a.buffer))
|
|
a.buffer = make(map[model.ArtworkID]struct{})
|
|
a.mutex.Unlock()
|
|
|
|
a.processBatch(ctx, batch)
|
|
}
|
|
}
|
|
|
|
func (a *cacheWarmer) waitSignal(ctx context.Context, timeout time.Duration) {
|
|
var to <-chan time.Time
|
|
if !a.cache.Available(ctx) {
|
|
tmr := time.NewTimer(timeout)
|
|
defer tmr.Stop()
|
|
to = tmr.C
|
|
}
|
|
select {
|
|
case <-to:
|
|
case <-a.wakeSignal:
|
|
case <-ctx.Done():
|
|
}
|
|
}
|
|
|
|
func (a *cacheWarmer) processBatch(ctx context.Context, batch []model.ArtworkID) {
|
|
log.Trace(ctx, "PreCaching a new batch of artwork", "batchSize", len(batch))
|
|
input := pl.FromSlice(ctx, batch)
|
|
errs := pl.Sink(ctx, 2, input, a.doCacheImage)
|
|
for err := range errs {
|
|
log.Debug(ctx, "Error warming cache", err)
|
|
}
|
|
}
|
|
|
|
func (a *cacheWarmer) doCacheImage(ctx context.Context, id model.ArtworkID) error {
|
|
ctx, cancel := context.WithTimeout(ctx, 10*time.Second)
|
|
defer cancel()
|
|
|
|
r, _, err := a.artwork.Get(ctx, id, consts.UICoverArtSize, false)
|
|
if err != nil {
|
|
return fmt.Errorf("caching id='%s': %w", id, err)
|
|
}
|
|
defer r.Close()
|
|
_, err = io.Copy(io.Discard, r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type noopCacheWarmer struct{}
|
|
|
|
func (a *noopCacheWarmer) PreCache(model.ArtworkID) {}
|