mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Implement artist art priority (#2266)
* implement artist art priority * add tests
This commit is contained in:
parent
406554f1c4
commit
2ccc5bc941
9 changed files with 73 additions and 16 deletions
|
@ -56,6 +56,7 @@ type configOptions struct {
|
||||||
FFmpegPath string
|
FFmpegPath string
|
||||||
CoverArtPriority string
|
CoverArtPriority string
|
||||||
CoverJpegQuality int
|
CoverJpegQuality int
|
||||||
|
ArtistArtPriority string
|
||||||
EnableGravatar bool
|
EnableGravatar bool
|
||||||
EnableFavourites bool
|
EnableFavourites bool
|
||||||
EnableStarRating bool
|
EnableStarRating bool
|
||||||
|
@ -272,6 +273,7 @@ func init() {
|
||||||
viper.SetDefault("ffmpegpath", "")
|
viper.SetDefault("ffmpegpath", "")
|
||||||
viper.SetDefault("coverartpriority", "cover.*, folder.*, front.*, embedded, external")
|
viper.SetDefault("coverartpriority", "cover.*, folder.*, front.*, embedded, external")
|
||||||
viper.SetDefault("coverjpegquality", 75)
|
viper.SetDefault("coverjpegquality", 75)
|
||||||
|
viper.SetDefault("artistartpriority", "artist.*, album/artist.*, external")
|
||||||
viper.SetDefault("enablegravatar", false)
|
viper.SetDefault("enablegravatar", false)
|
||||||
viper.SetDefault("enablefavourites", true)
|
viper.SetDefault("enablefavourites", true)
|
||||||
viper.SetDefault("enablestarrating", true)
|
viper.SetDefault("enablestarrating", true)
|
||||||
|
|
|
@ -22,7 +22,8 @@ var _ = Describe("Artwork", func() {
|
||||||
var ffmpeg *tests.MockFFmpeg
|
var ffmpeg *tests.MockFFmpeg
|
||||||
ctx := log.NewContext(context.TODO())
|
ctx := log.NewContext(context.TODO())
|
||||||
var alOnlyEmbed, alEmbedNotFound, alOnlyExternal, alExternalNotFound, alMultipleCovers model.Album
|
var alOnlyEmbed, alEmbedNotFound, alOnlyExternal, alExternalNotFound, alMultipleCovers model.Album
|
||||||
var mfWithEmbed, mfWithoutEmbed, mfCorruptedCover model.MediaFile
|
var arMultipleCovers model.Artist
|
||||||
|
var mfWithEmbed, mfAnotherWithEmbed, mfWithoutEmbed, mfCorruptedCover model.MediaFile
|
||||||
|
|
||||||
BeforeEach(func() {
|
BeforeEach(func() {
|
||||||
DeferCleanup(configtest.SetupConfig())
|
DeferCleanup(configtest.SetupConfig())
|
||||||
|
@ -30,14 +31,23 @@ var _ = Describe("Artwork", func() {
|
||||||
conf.Server.CoverArtPriority = "folder.*, cover.*, embedded , front.*"
|
conf.Server.CoverArtPriority = "folder.*, cover.*, embedded , front.*"
|
||||||
|
|
||||||
ds = &tests.MockDataStore{MockedTranscoding: &tests.MockTranscodingRepo{}}
|
ds = &tests.MockDataStore{MockedTranscoding: &tests.MockTranscodingRepo{}}
|
||||||
alOnlyEmbed = model.Album{ID: "222", Name: "Only embed", EmbedArtPath: "tests/fixtures/test.mp3"}
|
alOnlyEmbed = model.Album{ID: "222", Name: "Only embed", EmbedArtPath: "tests/fixtures/artist/an-album/test.mp3"}
|
||||||
alEmbedNotFound = model.Album{ID: "333", Name: "Embed not found", EmbedArtPath: "tests/fixtures/NON_EXISTENT.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"}
|
alOnlyExternal = model.Album{ID: "444", Name: "Only external", ImageFiles: "tests/fixtures/artist/an-album/front.png"}
|
||||||
alExternalNotFound = model.Album{ID: "555", Name: "External not found", ImageFiles: "tests/fixtures/NON_EXISTENT.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",
|
arMultipleCovers = model.Artist{ID: "777", Name: "All options"}
|
||||||
ImageFiles: "tests/fixtures/cover.jpg" + consts.Zwsp + "tests/fixtures/front.png",
|
alMultipleCovers = model.Album{
|
||||||
|
ID: "666",
|
||||||
|
Name: "All options",
|
||||||
|
EmbedArtPath: "tests/fixtures/artist/an-album/test.mp3",
|
||||||
|
Paths: "tests/fixtures/artist/an-album",
|
||||||
|
ImageFiles: "tests/fixtures/artist/an-album/cover.jpg" + consts.Zwsp +
|
||||||
|
"tests/fixtures/artist/an-album/front.png" + consts.Zwsp +
|
||||||
|
"tests/fixtures/artist/an-album/artist.png",
|
||||||
|
AlbumArtistID: "777",
|
||||||
}
|
}
|
||||||
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"}
|
||||||
|
mfAnotherWithEmbed = model.MediaFile{ID: "23", Path: "tests/fixtures/artist/an-album/test.mp3", HasCoverArt: true, AlbumID: "666"}
|
||||||
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"}
|
||||||
|
|
||||||
|
@ -65,7 +75,7 @@ var _ = Describe("Artwork", func() {
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
_, path, err := aw.Reader(ctx)
|
_, path, err := aw.Reader(ctx)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/test.mp3"))
|
Expect(path).To(Equal("tests/fixtures/artist/an-album/test.mp3"))
|
||||||
})
|
})
|
||||||
It("returns ErrUnavailable if embed path is not available", func() {
|
It("returns ErrUnavailable if embed path is not available", func() {
|
||||||
ffmpeg.Error = errors.New("not available")
|
ffmpeg.Error = errors.New("not available")
|
||||||
|
@ -87,7 +97,7 @@ var _ = Describe("Artwork", func() {
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
_, path, err := aw.Reader(ctx)
|
_, path, err := aw.Reader(ctx)
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal("tests/fixtures/front.png"))
|
Expect(path).To(Equal("tests/fixtures/artist/an-album/front.png"))
|
||||||
})
|
})
|
||||||
It("returns ErrUnavailable if external file is not available", func() {
|
It("returns ErrUnavailable if external file is not available", func() {
|
||||||
aw, err := newAlbumArtworkReader(ctx, aw, alExternalNotFound.CoverArtID(), nil)
|
aw, err := newAlbumArtworkReader(ctx, aw, alExternalNotFound.CoverArtID(), nil)
|
||||||
|
@ -111,9 +121,36 @@ var _ = Describe("Artwork", func() {
|
||||||
Expect(err).ToNot(HaveOccurred())
|
Expect(err).ToNot(HaveOccurred())
|
||||||
Expect(path).To(Equal(expected))
|
Expect(path).To(Equal(expected))
|
||||||
},
|
},
|
||||||
Entry(nil, " folder.* , cover.*,embedded,front.*", "tests/fixtures/cover.jpg"),
|
Entry(nil, " folder.* , cover.*,embedded,front.*", "tests/fixtures/artist/an-album/cover.jpg"),
|
||||||
Entry(nil, "front.* , cover.*, embedded ,folder.*", "tests/fixtures/front.png"),
|
Entry(nil, "front.* , cover.*, embedded ,folder.*", "tests/fixtures/artist/an-album/front.png"),
|
||||||
Entry(nil, " embedded , front.* , cover.*,folder.*", "tests/fixtures/test.mp3"),
|
Entry(nil, " embedded , front.* , cover.*,folder.*", "tests/fixtures/artist/an-album/test.mp3"),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
Describe("artistArtworkReader", func() {
|
||||||
|
Context("Multiple covers", func() {
|
||||||
|
BeforeEach(func() {
|
||||||
|
ds.Artist(ctx).(*tests.MockArtistRepo).SetData(model.Artists{
|
||||||
|
arMultipleCovers,
|
||||||
|
})
|
||||||
|
ds.Album(ctx).(*tests.MockAlbumRepo).SetData(model.Albums{
|
||||||
|
alMultipleCovers,
|
||||||
|
})
|
||||||
|
ds.MediaFile(ctx).(*tests.MockMediaFileRepo).SetData(model.MediaFiles{
|
||||||
|
mfAnotherWithEmbed,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
DescribeTable("ArtistArtPriority",
|
||||||
|
func(priority string, expected string) {
|
||||||
|
conf.Server.ArtistArtPriority = priority
|
||||||
|
aw, err := newArtistReader(ctx, aw, arMultipleCovers.CoverArtID(), nil)
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
_, path, err := aw.Reader(ctx)
|
||||||
|
Expect(err).ToNot(HaveOccurred())
|
||||||
|
Expect(path).To(Equal(expected))
|
||||||
|
},
|
||||||
|
Entry(nil, " folder.* , artist.*,album/artist.*", "tests/fixtures/artist/artist.jpg"),
|
||||||
|
Entry(nil, "album/artist.*, folder.*,artist.*", "tests/fixtures/artist/an-album/artist.png"),
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
|
@ -79,11 +79,24 @@ func (a *artistReader) LastUpdated() time.Time {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *artistReader) Reader(ctx context.Context) (io.ReadCloser, string, error) {
|
func (a *artistReader) Reader(ctx context.Context) (io.ReadCloser, string, error) {
|
||||||
return selectImageReader(ctx, a.artID,
|
var ff = a.fromArtistArtPriority(ctx, conf.Server.ArtistArtPriority)
|
||||||
fromArtistFolder(ctx, a.artistFolder, "artist.*"),
|
return selectImageReader(ctx, a.artID, ff...)
|
||||||
fromExternalFile(ctx, a.files, "artist.*"),
|
}
|
||||||
fromArtistExternalSource(ctx, a.artist, a.em),
|
|
||||||
)
|
func (a *artistReader) fromArtistArtPriority(ctx context.Context, priority string) []sourceFunc {
|
||||||
|
var ff []sourceFunc
|
||||||
|
for _, pattern := range strings.Split(strings.ToLower(priority), ",") {
|
||||||
|
pattern = strings.TrimSpace(pattern)
|
||||||
|
switch {
|
||||||
|
case pattern == "external":
|
||||||
|
ff = append(ff, fromArtistExternalSource(ctx, a.artist, a.em))
|
||||||
|
case strings.HasPrefix(pattern, "album/"):
|
||||||
|
ff = append(ff, fromExternalFile(ctx, a.files, strings.TrimPrefix(pattern, "album/")))
|
||||||
|
default:
|
||||||
|
ff = append(ff, fromArtistFolder(ctx, a.artistFolder, pattern))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ff
|
||||||
}
|
}
|
||||||
|
|
||||||
func fromArtistFolder(ctx context.Context, artistFolder string, pattern string) sourceFunc {
|
func fromArtistFolder(ctx context.Context, artistFolder string, pattern string) sourceFunc {
|
||||||
|
|
|
@ -34,10 +34,15 @@ var _ = Describe("walk_dir_tree", func() {
|
||||||
|
|
||||||
Eventually(errC).Should(Receive(nil))
|
Eventually(errC).Should(Receive(nil))
|
||||||
Expect(collected[baseDir]).To(MatchFields(IgnoreExtras, Fields{
|
Expect(collected[baseDir]).To(MatchFields(IgnoreExtras, Fields{
|
||||||
"Images": ConsistOf("cover.jpg", "front.png"),
|
"Images": BeEmpty(),
|
||||||
"HasPlaylist": BeFalse(),
|
"HasPlaylist": BeFalse(),
|
||||||
"AudioFilesCount": BeNumerically("==", 5),
|
"AudioFilesCount": BeNumerically("==", 5),
|
||||||
}))
|
}))
|
||||||
|
Expect(collected[filepath.Join(baseDir, "artist", "an-album")]).To(MatchFields(IgnoreExtras, Fields{
|
||||||
|
"Images": ConsistOf("cover.jpg", "front.png", "artist.png"),
|
||||||
|
"HasPlaylist": BeFalse(),
|
||||||
|
"AudioFilesCount": BeNumerically("==", 1),
|
||||||
|
}))
|
||||||
Expect(collected[filepath.Join(baseDir, "playlists")].HasPlaylist).To(BeTrue())
|
Expect(collected[filepath.Join(baseDir, "playlists")].HasPlaylist).To(BeTrue())
|
||||||
Expect(collected).To(HaveKey(filepath.Join(baseDir, "symlink2dir")))
|
Expect(collected).To(HaveKey(filepath.Join(baseDir, "symlink2dir")))
|
||||||
Expect(collected).To(HaveKey(filepath.Join(baseDir, "empty_folder")))
|
Expect(collected).To(HaveKey(filepath.Join(baseDir, "empty_folder")))
|
||||||
|
|
Before Width: | Height: | Size: 3.9 KiB After Width: | Height: | Size: 3.9 KiB |
Before Width: | Height: | Size: 26 KiB After Width: | Height: | Size: 26 KiB |
BIN
tests/fixtures/artist/an-album/front.png
vendored
Normal file
BIN
tests/fixtures/artist/an-album/front.png
vendored
Normal file
Binary file not shown.
After Width: | Height: | Size: 3.9 KiB |
BIN
tests/fixtures/artist/an-album/test.mp3
vendored
Normal file
BIN
tests/fixtures/artist/an-album/test.mp3
vendored
Normal file
Binary file not shown.
BIN
tests/fixtures/artist/artist.jpg
vendored
Normal file
BIN
tests/fixtures/artist/artist.jpg
vendored
Normal file
Binary file not shown.
After Width: | Height: | Size: 26 KiB |
Loading…
Add table
Add a link
Reference in a new issue