207 lines
7.7 KiB
Go
207 lines
7.7 KiB
Go
package artwork
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"image"
|
|
"io"
|
|
|
|
"github.com/navidrome/navidrome/conf"
|
|
"github.com/navidrome/navidrome/conf/configtest"
|
|
"github.com/navidrome/navidrome/consts"
|
|
"github.com/navidrome/navidrome/log"
|
|
"github.com/navidrome/navidrome/model"
|
|
"github.com/navidrome/navidrome/tests"
|
|
. "github.com/onsi/ginkgo/v2"
|
|
. "github.com/onsi/gomega"
|
|
)
|
|
|
|
var _ = Describe("Artwork", func() {
|
|
var aw *artwork
|
|
var ds model.DataStore
|
|
var ffmpeg *tests.MockFFmpeg
|
|
ctx := log.NewContext(context.TODO())
|
|
var alOnlyEmbed, alEmbedNotFound, alOnlyExternal, alExternalNotFound, alMultipleCovers model.Album
|
|
var mfWithEmbed, mfWithoutEmbed, mfCorruptedCover model.MediaFile
|
|
|
|
BeforeEach(func() {
|
|
DeferCleanup(configtest.SetupConfig())
|
|
conf.Server.ImageCacheSize = "0" // Disable cache
|
|
conf.Server.CoverArtPriority = "folder.*, cover.*, embedded , front.*"
|
|
|
|
ds = &tests.MockDataStore{MockedTranscoding: &tests.MockTranscodingRepo{}}
|
|
alOnlyEmbed = model.Album{ID: "222", Name: "Only embed", EmbedArtPath: "tests/fixtures/test.mp3"}
|
|
alEmbedNotFound = model.Album{ID: "333", Name: "Embed not found", EmbedArtPath: "tests/fixtures/NON_EXISTENT.mp3"}
|
|
alOnlyExternal = model.Album{ID: "444", Name: "Only external", ImageFiles: "tests/fixtures/front.png"}
|
|
alExternalNotFound = model.Album{ID: "555", Name: "External not found", ImageFiles: "tests/fixtures/NON_EXISTENT.png"}
|
|
alMultipleCovers = model.Album{ID: "666", Name: "All options", EmbedArtPath: "tests/fixtures/test.mp3",
|
|
ImageFiles: "tests/fixtures/cover.jpg" + consts.Zwsp + "tests/fixtures/front.png",
|
|
}
|
|
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"}
|
|
mfCorruptedCover = model.MediaFile{ID: "45", Path: "tests/fixtures/test.ogg", HasCoverArt: true, AlbumID: "444"}
|
|
|
|
cache := GetImageCache()
|
|
ffmpeg = tests.NewMockFFmpeg("content from ffmpeg")
|
|
aw = NewArtwork(ds, cache, ffmpeg, nil).(*artwork)
|
|
})
|
|
|
|
Describe("albumArtworkReader", func() {
|
|
Context("ID not found", func() {
|
|
It("returns ErrNotFound if album is not in the DB", func() {
|
|
_, err := newAlbumArtworkReader(ctx, aw, model.MustParseArtworkID("al-NOT-FOUND"), nil)
|
|
Expect(err).To(MatchError(model.ErrNotFound))
|
|
})
|
|
})
|
|
Context("Embed images", func() {
|
|
BeforeEach(func() {
|
|
ds.Album(ctx).(*tests.MockAlbumRepo).SetData(model.Albums{
|
|
alOnlyEmbed,
|
|
alEmbedNotFound,
|
|
})
|
|
})
|
|
It("returns embed cover", func() {
|
|
aw, err := newAlbumArtworkReader(ctx, aw, alOnlyEmbed.CoverArtID(), nil)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
_, path, err := aw.Reader(ctx)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(path).To(Equal("tests/fixtures/test.mp3"))
|
|
})
|
|
It("returns ErrUnavailable if embed path is not available", func() {
|
|
ffmpeg.Error = errors.New("not available")
|
|
aw, err := newAlbumArtworkReader(ctx, aw, alEmbedNotFound.CoverArtID(), nil)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
_, _, err = aw.Reader(ctx)
|
|
Expect(err).To(MatchError(ErrUnavailable))
|
|
})
|
|
})
|
|
Context("External images", func() {
|
|
BeforeEach(func() {
|
|
ds.Album(ctx).(*tests.MockAlbumRepo).SetData(model.Albums{
|
|
alOnlyExternal,
|
|
alExternalNotFound,
|
|
})
|
|
})
|
|
It("returns external cover", func() {
|
|
aw, err := newAlbumArtworkReader(ctx, aw, alOnlyExternal.CoverArtID(), nil)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
_, path, err := aw.Reader(ctx)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(path).To(Equal("tests/fixtures/front.png"))
|
|
})
|
|
It("returns ErrUnavailable if external file is not available", func() {
|
|
aw, err := newAlbumArtworkReader(ctx, aw, alExternalNotFound.CoverArtID(), nil)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
_, _, err = aw.Reader(ctx)
|
|
Expect(err).To(MatchError(ErrUnavailable))
|
|
})
|
|
})
|
|
Context("Multiple covers", func() {
|
|
BeforeEach(func() {
|
|
ds.Album(ctx).(*tests.MockAlbumRepo).SetData(model.Albums{
|
|
alMultipleCovers,
|
|
})
|
|
})
|
|
DescribeTable("CoverArtPriority",
|
|
func(priority string, expected string) {
|
|
conf.Server.CoverArtPriority = priority
|
|
aw, err := newAlbumArtworkReader(ctx, aw, alMultipleCovers.CoverArtID(), nil)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
_, path, err := aw.Reader(ctx)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(path).To(Equal(expected))
|
|
},
|
|
Entry(nil, " folder.* , cover.*,embedded,front.*", "tests/fixtures/cover.jpg"),
|
|
Entry(nil, "front.* , cover.*, embedded ,folder.*", "tests/fixtures/front.png"),
|
|
Entry(nil, " embedded , front.* , cover.*,folder.*", "tests/fixtures/test.mp3"),
|
|
)
|
|
})
|
|
})
|
|
Describe("mediafileArtworkReader", func() {
|
|
Context("ID not found", func() {
|
|
It("returns ErrNotFound if mediafile is not in the DB", func() {
|
|
_, err := newAlbumArtworkReader(ctx, aw, alMultipleCovers.CoverArtID(), nil)
|
|
Expect(err).To(MatchError(model.ErrNotFound))
|
|
})
|
|
})
|
|
Context("Embed images", func() {
|
|
BeforeEach(func() {
|
|
ds.Album(ctx).(*tests.MockAlbumRepo).SetData(model.Albums{
|
|
alOnlyEmbed,
|
|
alOnlyExternal,
|
|
})
|
|
ds.MediaFile(ctx).(*tests.MockMediaFileRepo).SetData(model.MediaFiles{
|
|
mfWithEmbed,
|
|
mfWithoutEmbed,
|
|
mfCorruptedCover,
|
|
})
|
|
})
|
|
It("returns embed cover", func() {
|
|
aw, err := newMediafileArtworkReader(ctx, aw, mfWithEmbed.CoverArtID())
|
|
Expect(err).ToNot(HaveOccurred())
|
|
_, path, err := aw.Reader(ctx)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(path).To(Equal("tests/fixtures/test.mp3"))
|
|
})
|
|
It("returns embed cover if successfully extracted by ffmpeg", func() {
|
|
aw, err := newMediafileArtworkReader(ctx, aw, mfCorruptedCover.CoverArtID())
|
|
Expect(err).ToNot(HaveOccurred())
|
|
r, path, err := aw.Reader(ctx)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(io.ReadAll(r)).To(Equal([]byte("content from ffmpeg")))
|
|
Expect(path).To(Equal("tests/fixtures/test.ogg"))
|
|
})
|
|
It("returns album cover if cannot read embed artwork", func() {
|
|
ffmpeg.Error = errors.New("not available")
|
|
aw, err := newMediafileArtworkReader(ctx, aw, mfCorruptedCover.CoverArtID())
|
|
Expect(err).ToNot(HaveOccurred())
|
|
_, path, err := aw.Reader(ctx)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(path).To(Equal("al-444_0"))
|
|
})
|
|
It("returns album cover if media file has no cover art", func() {
|
|
aw, err := newMediafileArtworkReader(ctx, aw, model.MustParseArtworkID("mf-"+mfWithoutEmbed.ID))
|
|
Expect(err).ToNot(HaveOccurred())
|
|
_, path, err := aw.Reader(ctx)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(path).To(Equal("al-444_0"))
|
|
})
|
|
})
|
|
})
|
|
Describe("resizedArtworkReader", func() {
|
|
BeforeEach(func() {
|
|
ds.Album(ctx).(*tests.MockAlbumRepo).SetData(model.Albums{
|
|
alMultipleCovers,
|
|
})
|
|
})
|
|
It("returns a PNG if original image is a PNG", func() {
|
|
conf.Server.CoverArtPriority = "front.png"
|
|
r, _, err := aw.Get(context.Background(), alMultipleCovers.CoverArtID(), 15)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
|
|
br, format, err := asImageReader(r)
|
|
Expect(format).To(Equal("image/png"))
|
|
Expect(err).ToNot(HaveOccurred())
|
|
|
|
img, _, err := image.Decode(br)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(img.Bounds().Size().X).To(Equal(15))
|
|
Expect(img.Bounds().Size().Y).To(Equal(15))
|
|
})
|
|
It("returns a JPEG if original image is not a PNG", func() {
|
|
conf.Server.CoverArtPriority = "cover.jpg"
|
|
r, _, err := aw.Get(context.Background(), alMultipleCovers.CoverArtID(), 200)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
|
|
br, format, err := asImageReader(r)
|
|
Expect(format).To(Equal("image/jpeg"))
|
|
Expect(err).ToNot(HaveOccurred())
|
|
|
|
img, _, err := image.Decode(br)
|
|
Expect(err).ToNot(HaveOccurred())
|
|
Expect(img.Bounds().Size().X).To(Equal(200))
|
|
Expect(img.Bounds().Size().Y).To(Equal(200))
|
|
})
|
|
})
|
|
})
|