mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Add image cache back
This commit is contained in:
parent
40bb211b39
commit
0da27e8a3f
5 changed files with 96 additions and 41 deletions
|
@ -45,7 +45,8 @@ func CreateNativeAPIRouter() *nativeapi.Router {
|
||||||
func CreateSubsonicAPIRouter() *subsonic.Router {
|
func CreateSubsonicAPIRouter() *subsonic.Router {
|
||||||
sqlDB := db.Db()
|
sqlDB := db.Db()
|
||||||
dataStore := persistence.New(sqlDB)
|
dataStore := persistence.New(sqlDB)
|
||||||
artwork := core.NewArtwork(dataStore)
|
fileCache := core.GetImageCache()
|
||||||
|
artwork := core.NewArtwork(dataStore, fileCache)
|
||||||
transcoderTranscoder := transcoder.New()
|
transcoderTranscoder := transcoder.New()
|
||||||
transcodingCache := core.GetTranscodingCache()
|
transcodingCache := core.GetTranscodingCache()
|
||||||
mediaStreamer := core.NewMediaStreamer(dataStore, transcoderTranscoder, transcodingCache)
|
mediaStreamer := core.NewMediaStreamer(dataStore, transcoderTranscoder, transcodingCache)
|
||||||
|
|
|
@ -13,6 +13,7 @@ import (
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/dhowden/tag"
|
"github.com/dhowden/tag"
|
||||||
"github.com/disintegration/imaging"
|
"github.com/disintegration/imaging"
|
||||||
|
@ -21,6 +22,8 @@ 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/resources"
|
"github.com/navidrome/navidrome/resources"
|
||||||
|
"github.com/navidrome/navidrome/utils/cache"
|
||||||
|
"github.com/navidrome/navidrome/utils/singleton"
|
||||||
_ "golang.org/x/image/webp"
|
_ "golang.org/x/image/webp"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -28,53 +31,60 @@ type Artwork interface {
|
||||||
Get(ctx context.Context, id string, size int) (io.ReadCloser, error)
|
Get(ctx context.Context, id string, size int) (io.ReadCloser, error)
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewArtwork(ds model.DataStore) Artwork {
|
func NewArtwork(ds model.DataStore, cache cache.FileCache) Artwork {
|
||||||
return &artwork{ds: ds}
|
return &artwork{ds: ds, cache: cache}
|
||||||
}
|
}
|
||||||
|
|
||||||
type artwork struct {
|
type artwork struct {
|
||||||
ds model.DataStore
|
ds model.DataStore
|
||||||
|
cache cache.FileCache
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *artwork) Get(ctx context.Context, id string, size int) (io.ReadCloser, error) {
|
func (a *artwork) Get(ctx context.Context, id string, size int) (io.ReadCloser, error) {
|
||||||
r, _, err := a.get(ctx, id, size)
|
artID, err := model.ParseArtworkID(id)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.New("invalid ID")
|
||||||
|
}
|
||||||
|
|
||||||
|
key := &artworkKey{a: a, artID: artID, size: size}
|
||||||
|
|
||||||
|
r, err := a.cache.Get(ctx, key)
|
||||||
|
if err != nil {
|
||||||
|
log.Error(ctx, "Error accessing image cache", "id", id, "size", size, err)
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
return r, err
|
return r, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *artwork) get(ctx context.Context, id string, size int) (reader io.ReadCloser, path string, err error) {
|
func (a *artwork) get(ctx context.Context, artID model.ArtworkID, size int) (reader io.ReadCloser, path string, err error) {
|
||||||
artId, err := model.ParseArtworkID(id)
|
|
||||||
if err != nil {
|
|
||||||
return nil, "", errors.New("invalid ID")
|
|
||||||
}
|
|
||||||
|
|
||||||
// If requested a resized image
|
// If requested a resized image
|
||||||
if size > 0 {
|
if size > 0 {
|
||||||
return a.resizedFromOriginal(ctx, id, size)
|
return a.resizedFromOriginal(ctx, artID, size)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch artId.Kind {
|
switch artID.Kind {
|
||||||
case model.KindAlbumArtwork:
|
case model.KindAlbumArtwork:
|
||||||
reader, path = a.extractAlbumImage(ctx, artId)
|
reader, path = a.extractAlbumImage(ctx, artID)
|
||||||
case model.KindMediaFileArtwork:
|
case model.KindMediaFileArtwork:
|
||||||
reader, path = a.extractMediaFileImage(ctx, artId)
|
reader, path = a.extractMediaFileImage(ctx, artID)
|
||||||
default:
|
default:
|
||||||
reader, path = fromPlaceholder()()
|
reader, path = fromPlaceholder()()
|
||||||
}
|
}
|
||||||
return reader, path, nil
|
return reader, path, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *artwork) extractAlbumImage(ctx context.Context, artId model.ArtworkID) (io.ReadCloser, string) {
|
func (a *artwork) extractAlbumImage(ctx context.Context, artID model.ArtworkID) (io.ReadCloser, string) {
|
||||||
al, err := a.ds.Album(ctx).Get(artId.ID)
|
al, err := a.ds.Album(ctx).Get(artID.ID)
|
||||||
if errors.Is(err, model.ErrNotFound) {
|
if errors.Is(err, model.ErrNotFound) {
|
||||||
r, path := fromPlaceholder()()
|
r, path := fromPlaceholder()()
|
||||||
return r, path
|
return r, path
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error(ctx, "Could not retrieve album", "id", artId.ID, err)
|
log.Error(ctx, "Could not retrieve album", "id", artID.ID, err)
|
||||||
return nil, ""
|
return nil, ""
|
||||||
}
|
}
|
||||||
|
|
||||||
return extractImage(ctx, artId,
|
return extractImage(ctx, artID,
|
||||||
fromExternalFile(al.ImageFiles, "cover.png", "cover.jpg", "cover.jpeg", "cover.webp"),
|
fromExternalFile(al.ImageFiles, "cover.png", "cover.jpg", "cover.jpeg", "cover.webp"),
|
||||||
fromExternalFile(al.ImageFiles, "folder.png", "folder.jpg", "folder.jpeg", "folder.webp"),
|
fromExternalFile(al.ImageFiles, "folder.png", "folder.jpg", "folder.jpeg", "folder.webp"),
|
||||||
fromExternalFile(al.ImageFiles, "album.png", "album.jpg", "album.jpeg", "album.webp"),
|
fromExternalFile(al.ImageFiles, "album.png", "album.jpg", "album.jpeg", "album.webp"),
|
||||||
|
@ -85,18 +95,18 @@ func (a *artwork) extractAlbumImage(ctx context.Context, artId model.ArtworkID)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *artwork) extractMediaFileImage(ctx context.Context, artId model.ArtworkID) (reader io.ReadCloser, path string) {
|
func (a *artwork) extractMediaFileImage(ctx context.Context, artID model.ArtworkID) (reader io.ReadCloser, path string) {
|
||||||
mf, err := a.ds.MediaFile(ctx).Get(artId.ID)
|
mf, err := a.ds.MediaFile(ctx).Get(artID.ID)
|
||||||
if errors.Is(err, model.ErrNotFound) {
|
if errors.Is(err, model.ErrNotFound) {
|
||||||
r, path := fromPlaceholder()()
|
r, path := fromPlaceholder()()
|
||||||
return r, path
|
return r, path
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error(ctx, "Could not retrieve mediafile", "id", artId.ID, err)
|
log.Error(ctx, "Could not retrieve mediafile", "id", artID.ID, err)
|
||||||
return nil, ""
|
return nil, ""
|
||||||
}
|
}
|
||||||
|
|
||||||
return extractImage(ctx, artId,
|
return extractImage(ctx, artID,
|
||||||
fromTag(mf.Path),
|
fromTag(mf.Path),
|
||||||
a.fromAlbum(ctx, mf.AlbumCoverArtID()),
|
a.fromAlbum(ctx, mf.AlbumCoverArtID()),
|
||||||
)
|
)
|
||||||
|
@ -104,7 +114,7 @@ func (a *artwork) extractMediaFileImage(ctx context.Context, artId model.Artwork
|
||||||
|
|
||||||
func (a *artwork) fromAlbum(ctx context.Context, id model.ArtworkID) func() (io.ReadCloser, string) {
|
func (a *artwork) fromAlbum(ctx context.Context, id model.ArtworkID) func() (io.ReadCloser, string) {
|
||||||
return func() (io.ReadCloser, string) {
|
return func() (io.ReadCloser, string) {
|
||||||
r, path, err := a.get(ctx, id.String(), 0)
|
r, path, err := a.get(ctx, id, 0)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, ""
|
return nil, ""
|
||||||
}
|
}
|
||||||
|
@ -112,8 +122,8 @@ func (a *artwork) fromAlbum(ctx context.Context, id model.ArtworkID) func() (io.
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *artwork) resizedFromOriginal(ctx context.Context, id string, size int) (io.ReadCloser, string, error) {
|
func (a *artwork) resizedFromOriginal(ctx context.Context, artID model.ArtworkID, size int) (io.ReadCloser, string, error) {
|
||||||
r, path, err := a.get(ctx, id, 0)
|
r, path, err := a.get(ctx, artID, 0)
|
||||||
if err != nil || r == nil {
|
if err != nil || r == nil {
|
||||||
return nil, "", err
|
return nil, "", err
|
||||||
}
|
}
|
||||||
|
@ -127,15 +137,15 @@ func (a *artwork) resizedFromOriginal(ctx context.Context, id string, size int)
|
||||||
return r, fmt.Sprintf("%s@%d", path, size), nil
|
return r, fmt.Sprintf("%s@%d", path, size), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func extractImage(ctx context.Context, artId model.ArtworkID, extractFuncs ...func() (io.ReadCloser, string)) (io.ReadCloser, string) {
|
func extractImage(ctx context.Context, artID model.ArtworkID, extractFuncs ...func() (io.ReadCloser, string)) (io.ReadCloser, string) {
|
||||||
for _, f := range extractFuncs {
|
for _, f := range extractFuncs {
|
||||||
r, path := f()
|
r, path := f()
|
||||||
if r != nil {
|
if r != nil {
|
||||||
log.Trace(ctx, "Found artwork", "artId", artId, "path", path)
|
log.Trace(ctx, "Found artwork", "artID", artID, "path", path)
|
||||||
return r, path
|
return r, path
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
log.Error(ctx, "extractImage should never reach this point!", "artId", artId, "path")
|
log.Error(ctx, "extractImage should never reach this point!", "artID", artID, "path")
|
||||||
return nil, ""
|
return nil, ""
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -215,3 +225,31 @@ func resizeImage(reader io.Reader, size int, usePng bool) (io.ReadCloser, error)
|
||||||
}
|
}
|
||||||
return io.NopCloser(buf), err
|
return io.NopCloser(buf), err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type ArtworkCache struct {
|
||||||
|
cache.FileCache
|
||||||
|
}
|
||||||
|
|
||||||
|
type artworkKey struct {
|
||||||
|
a *artwork
|
||||||
|
artID model.ArtworkID
|
||||||
|
size int
|
||||||
|
lastUpdate time.Time
|
||||||
|
}
|
||||||
|
|
||||||
|
func (k *artworkKey) Key() string {
|
||||||
|
return fmt.Sprintf("%s.%d.%d.%d", k.artID.ID, k.size, k.artID.LastUpdate.UnixNano(), conf.Server.CoverJpegQuality)
|
||||||
|
}
|
||||||
|
|
||||||
|
func GetImageCache() cache.FileCache {
|
||||||
|
return singleton.GetInstance(func() *ArtworkCache {
|
||||||
|
return &ArtworkCache{
|
||||||
|
FileCache: cache.NewFileCache("Image", conf.Server.ImageCacheSize, consts.ImageCacheDir, consts.DefaultImageCacheMaxItems,
|
||||||
|
func(ctx context.Context, arg cache.Item) (io.Reader, error) {
|
||||||
|
info := arg.(*artworkKey)
|
||||||
|
r, _, err := info.a.get(ctx, info.artID, info.size)
|
||||||
|
return r, err
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
|
@ -4,6 +4,8 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"image"
|
"image"
|
||||||
|
|
||||||
|
"github.com/navidrome/navidrome/conf"
|
||||||
|
"github.com/navidrome/navidrome/conf/configtest"
|
||||||
"github.com/navidrome/navidrome/consts"
|
"github.com/navidrome/navidrome/consts"
|
||||||
"github.com/navidrome/navidrome/log"
|
"github.com/navidrome/navidrome/log"
|
||||||
"github.com/navidrome/navidrome/model"
|
"github.com/navidrome/navidrome/model"
|
||||||
|
@ -31,13 +33,18 @@ var _ = Describe("Artwork", func() {
|
||||||
mfWithEmbed = model.MediaFile{ID: "22", Path: "tests/fixtures/test.mp3", HasCoverArt: true, AlbumID: "222"}
|
mfWithEmbed = model.MediaFile{ID: "22", Path: "tests/fixtures/test.mp3", HasCoverArt: true, AlbumID: "222"}
|
||||||
mfWithoutEmbed = model.MediaFile{ID: "44", Path: "tests/fixtures/test.ogg", AlbumID: "444"}
|
mfWithoutEmbed = model.MediaFile{ID: "44", Path: "tests/fixtures/test.ogg", AlbumID: "444"}
|
||||||
mfCorruptedCover = model.MediaFile{ID: "45", Path: "tests/fixtures/test.ogg", HasCoverArt: true, AlbumID: "444"}
|
mfCorruptedCover = model.MediaFile{ID: "45", Path: "tests/fixtures/test.ogg", HasCoverArt: true, AlbumID: "444"}
|
||||||
aw = NewArtwork(ds).(*artwork)
|
|
||||||
|
DeferCleanup(configtest.SetupConfig())
|
||||||
|
conf.Server.ImageCacheSize = "0" // Disable cache
|
||||||
|
|
||||||
|
cache := GetImageCache()
|
||||||
|
aw = NewArtwork(ds, cache).(*artwork)
|
||||||
})
|
})
|
||||||
|
|
||||||
Context("Albums", func() {
|
Context("Albums", func() {
|
||||||
Context("ID not found", func() {
|
Context("ID not found", func() {
|
||||||
It("returns placeholder if album is not in the DB", func() {
|
It("returns placeholder if album is not in the DB", func() {
|
||||||
_, path, err := aw.get(context.Background(), "al-NOT_FOUND-0", 0)
|
_, path, err := aw.get(context.Background(), model.MustParseArtworkID("al-NOT_FOUND-0"), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal(consts.PlaceholderAlbumArt))
|
Expect(path).To(Equal(consts.PlaceholderAlbumArt))
|
||||||
})
|
})
|
||||||
|
@ -50,12 +57,12 @@ var _ = Describe("Artwork", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
It("returns embed cover", func() {
|
It("returns embed cover", func() {
|
||||||
_, path, err := aw.get(context.Background(), alOnlyEmbed.CoverArtID().String(), 0)
|
_, path, err := aw.get(context.Background(), alOnlyEmbed.CoverArtID(), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/test.mp3"))
|
Expect(path).To(Equal("tests/fixtures/test.mp3"))
|
||||||
})
|
})
|
||||||
It("returns placeholder if embed path is not available", func() {
|
It("returns placeholder if embed path is not available", func() {
|
||||||
_, path, err := aw.get(context.Background(), alEmbedNotFound.CoverArtID().String(), 0)
|
_, path, err := aw.get(context.Background(), alEmbedNotFound.CoverArtID(), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal(consts.PlaceholderAlbumArt))
|
Expect(path).To(Equal(consts.PlaceholderAlbumArt))
|
||||||
})
|
})
|
||||||
|
@ -68,17 +75,17 @@ var _ = Describe("Artwork", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
It("returns external cover", func() {
|
It("returns external cover", func() {
|
||||||
_, path, err := aw.get(context.Background(), alOnlyExternal.CoverArtID().String(), 0)
|
_, path, err := aw.get(context.Background(), alOnlyExternal.CoverArtID(), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/front.png"))
|
Expect(path).To(Equal("tests/fixtures/front.png"))
|
||||||
})
|
})
|
||||||
It("returns the first image if more than one is available", func() {
|
It("returns the first image if more than one is available", func() {
|
||||||
_, path, err := aw.get(context.Background(), alAllOptions.CoverArtID().String(), 0)
|
_, path, err := aw.get(context.Background(), alAllOptions.CoverArtID(), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/cover.jpg"))
|
Expect(path).To(Equal("tests/fixtures/cover.jpg"))
|
||||||
})
|
})
|
||||||
It("returns placeholder if external file is not available", func() {
|
It("returns placeholder if external file is not available", func() {
|
||||||
_, path, err := aw.get(context.Background(), alExternalNotFound.CoverArtID().String(), 0)
|
_, path, err := aw.get(context.Background(), alExternalNotFound.CoverArtID(), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal(consts.PlaceholderAlbumArt))
|
Expect(path).To(Equal(consts.PlaceholderAlbumArt))
|
||||||
})
|
})
|
||||||
|
@ -87,7 +94,7 @@ var _ = Describe("Artwork", func() {
|
||||||
Context("MediaFiles", func() {
|
Context("MediaFiles", func() {
|
||||||
Context("ID not found", func() {
|
Context("ID not found", func() {
|
||||||
It("returns placeholder if album is not in the DB", func() {
|
It("returns placeholder if album is not in the DB", func() {
|
||||||
_, path, err := aw.get(context.Background(), "mf-NOT_FOUND-0", 0)
|
_, path, err := aw.get(context.Background(), model.MustParseArtworkID("mf-NOT_FOUND-0"), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal(consts.PlaceholderAlbumArt))
|
Expect(path).To(Equal(consts.PlaceholderAlbumArt))
|
||||||
})
|
})
|
||||||
|
@ -105,17 +112,17 @@ var _ = Describe("Artwork", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
It("returns embed cover", func() {
|
It("returns embed cover", func() {
|
||||||
_, path, err := aw.get(context.Background(), mfWithEmbed.CoverArtID().String(), 0)
|
_, path, err := aw.get(context.Background(), mfWithEmbed.CoverArtID(), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/test.mp3"))
|
Expect(path).To(Equal("tests/fixtures/test.mp3"))
|
||||||
})
|
})
|
||||||
It("returns album cover if media file has no cover art", func() {
|
It("returns album cover if media file has no cover art", func() {
|
||||||
_, path, err := aw.get(context.Background(), mfWithoutEmbed.CoverArtID().String(), 0)
|
_, path, err := aw.get(context.Background(), mfWithoutEmbed.CoverArtID(), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/front.png"))
|
Expect(path).To(Equal("tests/fixtures/front.png"))
|
||||||
})
|
})
|
||||||
It("returns album cover if cannot read embed artwork", func() {
|
It("returns album cover if cannot read embed artwork", func() {
|
||||||
_, path, err := aw.get(context.Background(), mfCorruptedCover.CoverArtID().String(), 0)
|
_, path, err := aw.get(context.Background(), mfCorruptedCover.CoverArtID(), 0)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/front.png"))
|
Expect(path).To(Equal("tests/fixtures/front.png"))
|
||||||
})
|
})
|
||||||
|
@ -128,7 +135,7 @@ var _ = Describe("Artwork", func() {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
It("returns external cover resized", func() {
|
It("returns external cover resized", func() {
|
||||||
r, path, err := aw.get(context.Background(), alOnlyExternal.CoverArtID().String(), 300)
|
r, path, err := aw.get(context.Background(), alOnlyExternal.CoverArtID(), 300)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/front.png@300"))
|
Expect(path).To(Equal("tests/fixtures/front.png@300"))
|
||||||
img, _, err := image.Decode(r)
|
img, _, err := image.Decode(r)
|
||||||
|
|
|
@ -11,6 +11,7 @@ var Set = wire.NewSet(
|
||||||
NewArtwork,
|
NewArtwork,
|
||||||
NewMediaStreamer,
|
NewMediaStreamer,
|
||||||
GetTranscodingCache,
|
GetTranscodingCache,
|
||||||
|
GetImageCache,
|
||||||
NewArchiver,
|
NewArchiver,
|
||||||
NewExternalMetadata,
|
NewExternalMetadata,
|
||||||
NewPlayers,
|
NewPlayers,
|
||||||
|
|
|
@ -48,6 +48,14 @@ func ParseArtworkID(id string) (ArtworkID, error) {
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func MustParseArtworkID(id string) ArtworkID {
|
||||||
|
artID, err := ParseArtworkID(id)
|
||||||
|
if err != nil {
|
||||||
|
panic(artID)
|
||||||
|
}
|
||||||
|
return artID
|
||||||
|
}
|
||||||
|
|
||||||
func artworkIDFromAlbum(al Album) ArtworkID {
|
func artworkIDFromAlbum(al Album) ArtworkID {
|
||||||
return ArtworkID{
|
return ArtworkID{
|
||||||
Kind: KindAlbumArtwork,
|
Kind: KindAlbumArtwork,
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue