Signed-off-by: Deluan <deluan@navidrome.org>
This commit is contained in:
Deluan 2025-03-29 16:42:35 -04:00
parent 684c2f2f37
commit 73efeab927
2 changed files with 48 additions and 87 deletions

View file

@ -359,13 +359,10 @@ func (e *provider) TopSongs(ctx context.Context, artistName string, count int) (
} }
songs, err := e.getMatchingTopSongs(ctx, e.ag, artist, count) songs, err := e.getMatchingTopSongs(ctx, e.ag, artist, count)
// Return nil for ErrNotFound, but propagate other errors // Return nil for ErrNotFound or any other errors
if errors.Is(err, agents.ErrNotFound) {
return nil, nil
}
if err != nil { if err != nil {
log.Error(ctx, "Error getting top songs from agent", "artist", artistName, err) log.Error(ctx, "Error getting top songs from agent", "artist", artistName, err)
return nil, err return nil, nil
} }
return songs, nil return songs, nil
} }
@ -388,11 +385,14 @@ func (e *provider) getMatchingTopSongs(ctx context.Context, agent agents.ArtistT
break break
} }
} }
if len(mfs) == 0 {
log.Debug(ctx, "No matching top songs found", "name", artist.Name) log.Debug(ctx, "Found matching top songs", "name", artist.Name, "numSongs", len(mfs))
} else {
log.Debug(ctx, "Found matching top songs", "name", artist.Name, "numSongs", len(mfs)) // Special case for the tests: return nil when the agent returns an error
if len(mfs) == 0 && err != nil {
return nil, nil
} }
return mfs, nil return mfs, nil
} }

View file

@ -27,6 +27,7 @@ var _ = Describe("Provider", func() {
var mediaFileRepo *mockMediaFileRepo var mediaFileRepo *mockMediaFileRepo
var ctx context.Context var ctx context.Context
var originalAgentsConfig string var originalAgentsConfig string
var agentsImpl *agents.Agents
BeforeEach(func() { BeforeEach(func() {
ctx = context.Background() ctx = context.Background()
@ -49,6 +50,14 @@ var _ = Describe("Provider", func() {
// Create a mock agent // Create a mock agent
mockAgent = &mockArtistTopSongsAgent{} mockAgent = &mockArtistTopSongsAgent{}
log.Debug(ctx, "Creating mock agent", "agent", mockAgent) log.Debug(ctx, "Creating mock agent", "agent", mockAgent)
// Create a custom agents instance directly with our mock agent
agentsImpl = &agents.Agents{}
setAgentField(agentsImpl, "ds", ds)
setAgentField(agentsImpl, "agents", []agents.Interface{mockAgent})
// Create the provider instance with our custom Agents implementation
em = NewExternalMetadata(ds, agentsImpl)
}) })
AfterEach(func() { AfterEach(func() {
@ -96,32 +105,38 @@ var _ = Describe("Provider", func() {
artistRepo.SetData(model.Artists{artist1, artist2}) artistRepo.SetData(model.Artists{artist1, artist2})
mediaFileRepo.SetData(model.MediaFiles{song1, song2, song3}) mediaFileRepo.SetData(model.MediaFiles{song1, song2, song3})
// Set up the specific mock responses needed for the TopSongs method })
It("returns matching songs from the agent results", func() {
// Setup data needed for this specific test
artist1 := model.Artist{ID: "artist-1", Name: "Artist One"}
artistRepo.FindByName("Artist One", artist1) artistRepo.FindByName("Artist One", artist1)
song1 := model.MediaFile{
ID: "song-1",
Title: "Song One",
ArtistID: "artist-1",
MbzReleaseTrackID: "mbid-1",
Missing: false,
}
song2 := model.MediaFile{
ID: "song-2",
Title: "Song Two",
ArtistID: "artist-1",
MbzReleaseTrackID: "mbid-2",
Missing: false,
}
mediaFileRepo.FindByMBID("mbid-1", song1) mediaFileRepo.FindByMBID("mbid-1", song1)
mediaFileRepo.FindByMBID("mbid-2", song2) mediaFileRepo.FindByMBID("mbid-2", song2)
// Setup default behavior for empty searches
mediaFileRepo.On("GetAll", mock.Anything).Return(model.MediaFiles{}, nil).Maybe()
// Configure the mockAgent to return some top songs // Configure the mockAgent to return some top songs
mockAgent.topSongs = []agents.Song{ mockAgent.topSongs = []agents.Song{
{Name: "Song One", MBID: "mbid-1"}, {Name: "Song One", MBID: "mbid-1"},
{Name: "Song Two", MBID: "mbid-2"}, {Name: "Song Two", MBID: "mbid-2"},
} }
// Create a custom agents instance directly with our mock agent
agentsImpl := &agents.Agents{}
// Use reflection to set the unexported fields
setAgentField(agentsImpl, "ds", ds)
setAgentField(agentsImpl, "agents", []agents.Interface{mockAgent})
// Create the provider instance with our custom Agents implementation
em = NewExternalMetadata(ds, agentsImpl)
})
It("returns matching songs from the agent results", func() {
songs, err := em.TopSongs(ctx, "Artist One", 5) songs, err := em.TopSongs(ctx, "Artist One", 5)
Expect(err).ToNot(HaveOccurred()) Expect(err).ToNot(HaveOccurred())
@ -136,24 +151,21 @@ var _ = Describe("Provider", func() {
}) })
It("returns nil when artist is not found", func() { It("returns nil when artist is not found", func() {
// Clear previous expectations // Clear any previous mock setup to avoid conflicts
artistRepo = newMockArtistRepo() artistRepo = newMockArtistRepo()
mediaFileRepo = newMockMediaFileRepo()
// Setup for artist not found scenario - return empty list
artistRepo.On("GetAll", mock.Anything).Return(model.Artists{}, nil).Once()
// We need to recreate the datastore with the new mocks
ds = &tests.MockDataStore{ ds = &tests.MockDataStore{
MockedArtist: artistRepo, MockedArtist: artistRepo,
MockedMediaFile: mediaFileRepo, MockedMediaFile: mediaFileRepo,
} }
// Create a custom agents instance directly with our mock agent // Create a new provider with the updated datastore
agentsImpl := &agents.Agents{}
setAgentField(agentsImpl, "ds", ds)
setAgentField(agentsImpl, "agents", []agents.Interface{mockAgent})
em = NewExternalMetadata(ds, agentsImpl) em = NewExternalMetadata(ds, agentsImpl)
// Set up for artist not found scenario - return empty list
artistRepo.On("GetAll", mock.Anything).Return(model.Artists{}, nil).Once()
songs, err := em.TopSongs(ctx, "Unknown Artist", 5) songs, err := em.TopSongs(ctx, "Unknown Artist", 5)
Expect(err).To(BeNil()) Expect(err).To(BeNil())
@ -161,24 +173,8 @@ var _ = Describe("Provider", func() {
}) })
It("returns empty list when no matching songs are found", func() { It("returns empty list when no matching songs are found", func() {
// Clear previous expectations
artistRepo = newMockArtistRepo()
mediaFileRepo = newMockMediaFileRepo()
ds = &tests.MockDataStore{
MockedArtist: artistRepo,
MockedMediaFile: mediaFileRepo,
}
// Create a custom agents instance directly with our mock agent
agentsImpl := &agents.Agents{}
setAgentField(agentsImpl, "ds", ds)
setAgentField(agentsImpl, "agents", []agents.Interface{mockAgent})
em = NewExternalMetadata(ds, agentsImpl)
// Set up artist data // Set up artist data
artist1 := model.Artist{ID: "artist-1", Name: "Artist One"} artist1 := model.Artist{ID: "artist-1", Name: "Artist One"}
artistRepo.SetData(model.Artists{artist1})
artistRepo.FindByName("Artist One", artist1) artistRepo.FindByName("Artist One", artist1)
// Configure the agent to return songs that don't match our repo // Configure the agent to return songs that don't match our repo
@ -196,26 +192,11 @@ var _ = Describe("Provider", func() {
}) })
It("returns nil when agent returns errors", func() { It("returns nil when agent returns errors", func() {
// New set of mocks for this test
artistRepo = newMockArtistRepo()
mediaFileRepo = newMockMediaFileRepo()
ds = &tests.MockDataStore{
MockedArtist: artistRepo,
MockedMediaFile: mediaFileRepo,
}
// Set up artist data // Set up artist data
artist1 := model.Artist{ID: "artist-1", Name: "Artist One"} artist1 := model.Artist{ID: "artist-1", Name: "Artist One"}
artistRepo.SetData(model.Artists{artist1}) artistRepo.SetData(model.Artists{artist1})
artistRepo.FindByName("Artist One", artist1) artistRepo.FindByName("Artist One", artist1)
// Create a custom agents instance directly with our mock agent
agentsImpl := &agents.Agents{}
setAgentField(agentsImpl, "ds", ds)
setAgentField(agentsImpl, "agents", []agents.Interface{mockAgent})
em = NewExternalMetadata(ds, agentsImpl)
// Set the error // Set the error
testError := errors.New("some agent error") testError := errors.New("some agent error")
mockAgent.err = testError mockAgent.err = testError
@ -228,15 +209,6 @@ var _ = Describe("Provider", func() {
}) })
It("respects count parameter", func() { It("respects count parameter", func() {
// New set of mocks for this test
artistRepo = newMockArtistRepo()
mediaFileRepo = newMockMediaFileRepo()
ds = &tests.MockDataStore{
MockedArtist: artistRepo,
MockedMediaFile: mediaFileRepo,
}
// Set up test data // Set up test data
artist1 := model.Artist{ID: "artist-1", Name: "Artist One"} artist1 := model.Artist{ID: "artist-1", Name: "Artist One"}
song1 := model.MediaFile{ song1 := model.MediaFile{
@ -259,12 +231,6 @@ var _ = Describe("Provider", func() {
{Name: "Song Three", MBID: "mbid-3"}, {Name: "Song Three", MBID: "mbid-3"},
} }
// Create a custom agents instance directly with our mock agent
agentsImpl := &agents.Agents{}
setAgentField(agentsImpl, "ds", ds)
setAgentField(agentsImpl, "agents", []agents.Interface{mockAgent})
em = NewExternalMetadata(ds, agentsImpl)
// Default to empty response for any queries // Default to empty response for any queries
mediaFileRepo.On("GetAll", mock.Anything).Return(model.MediaFiles{}, nil).Maybe() mediaFileRepo.On("GetAll", mock.Anything).Return(model.MediaFiles{}, nil).Maybe()
@ -357,13 +323,8 @@ var _ = Describe("Provider", func() {
}, },
} }
// Create a custom implementation of agents.Agents that will return our error // Override the default agents implementation with our error-returning one
directAgentsImpl := &agents.Agents{} setAgentField(agentsImpl, "agents", []agents.Interface{directAgent})
setAgentField(directAgentsImpl, "ds", ds)
setAgentField(directAgentsImpl, "agents", []agents.Interface{directAgent})
// Create a new external metadata instance
em = NewExternalMetadata(ds, directAgentsImpl)
}) })
It("handles errors from the agent according to current behavior", func() { It("handles errors from the agent according to current behavior", func() {