Signed-off-by: Deluan <deluan@navidrome.org>
This commit is contained in:
Deluan 2025-03-30 15:49:36 -04:00
parent 018b139a79
commit 61240375db

View file

@ -4,7 +4,6 @@ import (
"context"
"errors"
"github.com/Masterminds/squirrel"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/core/agents"
_ "github.com/navidrome/navidrome/core/agents/lastfm"
@ -57,7 +56,7 @@ var _ = Describe("Provider - TopSongs", func() {
It("returns top songs for a known artist", func() {
// Mock finding the artist
artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
artistRepo.On("GetAll", mock.MatchedBy(matchArtistByNameFilter)).Return(model.Artists{artist1}, nil).Once()
artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
// Mock agent response
agentSongs := []agents.Song{
@ -69,8 +68,8 @@ var _ = Describe("Provider - TopSongs", func() {
// Mock finding matching tracks
song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-song-1"}
song2 := model.MediaFile{ID: "song-2", Title: "Song Two", ArtistID: "artist-1", MbzRecordingID: "mbid-song-2"}
mediaFileRepo.On("GetAll", mock.MatchedBy(matchMediaFileByMBIDFilter("mbid-song-1"))).Return(model.MediaFiles{song1}, nil).Once()
mediaFileRepo.On("GetAll", mock.MatchedBy(matchMediaFileByMBIDFilter("mbid-song-2"))).Return(model.MediaFiles{song2}, nil).Once()
mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1}, nil).Once()
mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song2}, nil).Once()
songs, err := p.TopSongs(ctx, "Artist One", 2)
@ -85,7 +84,7 @@ var _ = Describe("Provider - TopSongs", func() {
It("returns nil for an unknown artist", func() {
// Mock artist not found
artistRepo.On("GetAll", mock.MatchedBy(matchArtistByNameFilter)).Return(model.Artists{}, nil).Once()
artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{}, nil).Once()
songs, err := p.TopSongs(ctx, "Unknown Artist", 5)
@ -98,7 +97,7 @@ var _ = Describe("Provider - TopSongs", func() {
It("returns error when the agent returns an error", func() {
// Mock finding the artist
artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
artistRepo.On("GetAll", mock.MatchedBy(matchArtistByNameFilter)).Return(model.Artists{artist1}, nil).Once()
artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
// Mock agent error
agentErr := errors.New("agent error")
@ -115,7 +114,7 @@ var _ = Describe("Provider - TopSongs", func() {
It("returns ErrNotFound when the agent returns ErrNotFound", func() {
// Mock finding the artist
artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
artistRepo.On("GetAll", mock.MatchedBy(matchArtistByNameFilter)).Return(model.Artists{artist1}, nil).Once()
artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
// Mock agent ErrNotFound
ag.On("GetArtistTopSongs", ctx, "artist-1", "Artist One", "mbid-artist-1", 5).Return(nil, agents.ErrNotFound).Once()
@ -131,7 +130,7 @@ var _ = Describe("Provider - TopSongs", func() {
It("returns fewer songs if count is less than available top songs", func() {
// Mock finding the artist
artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
artistRepo.On("GetAll", mock.MatchedBy(matchArtistByNameFilter)).Return(model.Artists{artist1}, nil).Once()
artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
// Mock agent response (only need 1 for the test)
agentSongs := []agents.Song{{Name: "Song One", MBID: "mbid-song-1"}}
@ -139,7 +138,7 @@ var _ = Describe("Provider - TopSongs", func() {
// Mock finding matching track
song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-song-1"}
mediaFileRepo.On("GetAll", mock.MatchedBy(matchMediaFileByMBIDFilter("mbid-song-1"))).Return(model.MediaFiles{song1}, nil).Once()
mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1}, nil).Once()
songs, err := p.TopSongs(ctx, "Artist One", 1)
@ -154,7 +153,7 @@ var _ = Describe("Provider - TopSongs", func() {
It("returns fewer songs if fewer matching tracks are found", func() {
// Mock finding the artist
artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
artistRepo.On("GetAll", mock.MatchedBy(matchArtistByNameFilter)).Return(model.Artists{artist1}, nil).Once()
artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
// Mock agent response
agentSongs := []agents.Song{
@ -165,10 +164,9 @@ var _ = Describe("Provider - TopSongs", func() {
// Mock finding matching tracks (only find song 1)
song1 := model.MediaFile{ID: "song-1", Title: "Song One", ArtistID: "artist-1", MbzRecordingID: "mbid-song-1"}
mediaFileRepo.On("GetAll", mock.MatchedBy(matchMediaFileByMBIDFilter("mbid-song-1"))).Return(model.MediaFiles{song1}, nil).Once()
mediaFileRepo.On("GetAll", mock.MatchedBy(matchMediaFileByMBIDFilter("mbid-song-2"))).Return(model.MediaFiles{}, nil).Once() // MBID lookup fails
// Mock fallback lookup by title (also fails)
mediaFileRepo.On("GetAll", mock.MatchedBy(matchMediaFileByTitleFilter("artist-1", "Song Two"))).Return(model.MediaFiles{}, nil).Once()
mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{song1}, nil).Once()
mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{}, nil).Once() // For mbid-song-2 (fails)
mediaFileRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.MediaFiles{}, nil).Once() // For title fallback (fails)
songs, err := p.TopSongs(ctx, "Artist One", 2)
@ -183,7 +181,7 @@ var _ = Describe("Provider - TopSongs", func() {
It("returns error when context is canceled during agent call", func() {
// Mock finding the artist
artist1 := model.Artist{ID: "artist-1", Name: "Artist One", MbzArtistID: "mbid-artist-1"}
artistRepo.On("GetAll", mock.MatchedBy(matchArtistByNameFilter)).Return(model.Artists{artist1}, nil).Once()
artistRepo.On("GetAll", mock.AnythingOfType("model.QueryOptions")).Return(model.Artists{artist1}, nil).Once()
// Setup context that will be canceled
canceledCtx, cancel := context.WithCancel(ctx)
@ -201,77 +199,3 @@ var _ = Describe("Provider - TopSongs", func() {
})
})
})
// Helper functions to match squirrel filters for testify/mock
func matchArtistByNameFilter(opt model.QueryOptions) bool {
if opt.Max != 1 || opt.Filters == nil {
return false
}
_, ok := opt.Filters.(squirrel.Like)
// Could add more specific checks on the Like clause if needed
return ok
}
func matchMediaFileByMBIDFilter(expectedMBID string) func(opt model.QueryOptions) bool {
return func(opt model.QueryOptions) bool {
if opt.Filters == nil {
return false
}
andClause, ok := opt.Filters.(squirrel.And)
if !ok || len(andClause) < 2 {
return false
}
foundMBID := false
foundMissing := false
for _, condition := range andClause {
if eqClause, ok := condition.(squirrel.Eq); ok {
if mbid, exists := eqClause["mbz_recording_id"]; exists && mbid == expectedMBID {
foundMBID = true
}
if missing, exists := eqClause["missing"]; exists && missing == false {
foundMissing = true
}
}
}
return foundMBID && foundMissing
}
}
func matchMediaFileByTitleFilter(expectedArtistID, expectedTitle string) func(opt model.QueryOptions) bool {
return func(opt model.QueryOptions) bool {
if opt.Filters == nil || opt.Max != 1 {
return false
}
andClause, ok := opt.Filters.(squirrel.And)
if !ok || len(andClause) < 3 {
return false
}
foundArtist := false
foundTitle := false
foundMissing := false
for _, condition := range andClause {
if orClause, ok := condition.(squirrel.Or); ok {
for _, orCond := range orClause {
if eq, ok := orCond.(squirrel.Eq); ok {
if id, exists := eq["artist_id"]; exists && id == expectedArtistID {
foundArtist = true
}
if id, exists := eq["album_artist_id"]; exists && id == expectedArtistID {
foundArtist = true
}
}
}
} else if likeClause, ok := condition.(squirrel.Like); ok {
if _, exists := likeClause["order_title"]; exists {
// We assume the title matches here for simplicity; could compare sanitized title if needed
foundTitle = true
}
} else if eqClause, ok := condition.(squirrel.Eq); ok {
if missing, exists := eqClause["missing"]; exists && missing == false {
foundMissing = true
}
}
}
return foundArtist && foundTitle && foundMissing
}
}