mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-15 11:40:36 +03:00
142 lines
4.0 KiB
Go
142 lines
4.0 KiB
Go
package artwork
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
_ "image/gif"
|
|
"io"
|
|
"time"
|
|
|
|
"github.com/lestrrat-go/jwx/v2/jwt"
|
|
"github.com/navidrome/navidrome/core"
|
|
"github.com/navidrome/navidrome/core/auth"
|
|
"github.com/navidrome/navidrome/core/ffmpeg"
|
|
"github.com/navidrome/navidrome/log"
|
|
"github.com/navidrome/navidrome/model"
|
|
"github.com/navidrome/navidrome/utils/cache"
|
|
_ "golang.org/x/image/webp"
|
|
)
|
|
|
|
type Artwork interface {
|
|
Get(ctx context.Context, id string, size int) (io.ReadCloser, time.Time, error)
|
|
}
|
|
|
|
func NewArtwork(ds model.DataStore, cache cache.FileCache, ffmpeg ffmpeg.FFmpeg, em core.ExternalMetadata) Artwork {
|
|
return &artwork{ds: ds, cache: cache, ffmpeg: ffmpeg, em: em}
|
|
}
|
|
|
|
type artwork struct {
|
|
ds model.DataStore
|
|
cache cache.FileCache
|
|
ffmpeg ffmpeg.FFmpeg
|
|
em core.ExternalMetadata
|
|
}
|
|
|
|
type artworkReader interface {
|
|
cache.Item
|
|
LastUpdated() time.Time
|
|
Reader(ctx context.Context) (io.ReadCloser, string, error)
|
|
}
|
|
|
|
func (a *artwork) Get(ctx context.Context, id string, size int) (reader io.ReadCloser, lastUpdate time.Time, err error) {
|
|
artID, err := a.getArtworkId(ctx, id)
|
|
if err != nil {
|
|
return nil, time.Time{}, err
|
|
}
|
|
|
|
artReader, err := a.getArtworkReader(ctx, artID, size)
|
|
if err != nil {
|
|
return nil, time.Time{}, err
|
|
}
|
|
|
|
r, err := a.cache.Get(ctx, artReader)
|
|
if err != nil {
|
|
if !errors.Is(err, context.Canceled) {
|
|
log.Error(ctx, "Error accessing image cache", "id", id, "size", size, err)
|
|
}
|
|
return nil, time.Time{}, err
|
|
}
|
|
return r, artReader.LastUpdated(), nil
|
|
}
|
|
|
|
func (a *artwork) getArtworkId(ctx context.Context, id string) (model.ArtworkID, error) {
|
|
if id == "" {
|
|
return model.ArtworkID{}, nil
|
|
}
|
|
artID, err := model.ParseArtworkID(id)
|
|
if err == nil {
|
|
return artID, nil
|
|
}
|
|
|
|
log.Trace(ctx, "ArtworkID invalid. Trying to figure out kind based on the ID", "id", id)
|
|
entity, err := model.GetEntityByID(ctx, a.ds, id)
|
|
if err != nil {
|
|
return model.ArtworkID{}, err
|
|
}
|
|
switch e := entity.(type) {
|
|
case *model.Artist:
|
|
artID = model.NewArtworkID(model.KindArtistArtwork, e.ID)
|
|
log.Trace(ctx, "ID is for an Artist", "id", id, "name", e.Name, "artist", e.Name)
|
|
case *model.Album:
|
|
artID = model.NewArtworkID(model.KindAlbumArtwork, e.ID)
|
|
log.Trace(ctx, "ID is for an Album", "id", id, "name", e.Name, "artist", e.AlbumArtist)
|
|
case *model.MediaFile:
|
|
artID = model.NewArtworkID(model.KindMediaFileArtwork, e.ID)
|
|
log.Trace(ctx, "ID is for a MediaFile", "id", id, "title", e.Title, "album", e.Album)
|
|
case *model.Playlist:
|
|
artID = model.NewArtworkID(model.KindPlaylistArtwork, e.ID)
|
|
log.Trace(ctx, "ID is for a Playlist", "id", id, "name", e.Name)
|
|
}
|
|
return artID, nil
|
|
}
|
|
|
|
func (a *artwork) getArtworkReader(ctx context.Context, artID model.ArtworkID, size int) (artworkReader, error) {
|
|
var artReader artworkReader
|
|
var err error
|
|
if size > 0 {
|
|
artReader, err = resizedFromOriginal(ctx, a, artID, size)
|
|
} else {
|
|
switch artID.Kind {
|
|
case model.KindArtistArtwork:
|
|
artReader, err = newArtistReader(ctx, a, artID, a.em)
|
|
case model.KindAlbumArtwork:
|
|
artReader, err = newAlbumArtworkReader(ctx, a, artID)
|
|
case model.KindMediaFileArtwork:
|
|
artReader, err = newMediafileArtworkReader(ctx, a, artID)
|
|
case model.KindPlaylistArtwork:
|
|
artReader, err = newPlaylistArtworkReader(ctx, a, artID)
|
|
default:
|
|
artReader, err = newEmptyIDReader(ctx, artID)
|
|
}
|
|
}
|
|
return artReader, err
|
|
}
|
|
|
|
func EncodeArtworkID(artID model.ArtworkID) string {
|
|
token, _ := auth.CreatePublicToken(map[string]any{"id": artID.String()})
|
|
return token
|
|
}
|
|
|
|
func DecodeArtworkID(tokenString string) (model.ArtworkID, error) {
|
|
token, err := auth.TokenAuth.Decode(tokenString)
|
|
if err != nil {
|
|
return model.ArtworkID{}, err
|
|
}
|
|
if token == nil {
|
|
return model.ArtworkID{}, errors.New("unauthorized")
|
|
}
|
|
err = jwt.Validate(token, jwt.WithRequiredClaim("id"))
|
|
if err != nil {
|
|
return model.ArtworkID{}, err
|
|
}
|
|
claims, err := token.AsMap(context.Background())
|
|
if err != nil {
|
|
return model.ArtworkID{}, err
|
|
}
|
|
id, ok := claims["id"].(string)
|
|
if !ok {
|
|
return model.ArtworkID{}, errors.New("invalid id type")
|
|
}
|
|
return model.ParseArtworkID(id)
|
|
}
|