navidrome/core/extdata/external_metadata_test.go
Deluan eccf34c6f2 rename external metadata -wip
Signed-off-by: Deluan <deluan@navidrome.org>
2025-03-30 13:43:55 -04:00

538 lines
16 KiB
Go

package extdata
import (
"context"
"errors"
"fmt"
"reflect"
"strings"
"unsafe"
"github.com/Masterminds/squirrel"
"github.com/navidrome/navidrome/conf"
"github.com/navidrome/navidrome/core/agents"
_ "github.com/navidrome/navidrome/core/agents/lastfm"
_ "github.com/navidrome/navidrome/core/agents/listenbrainz"
_ "github.com/navidrome/navidrome/core/agents/spotify"
"github.com/navidrome/navidrome/log"
"github.com/navidrome/navidrome/model"
"github.com/navidrome/navidrome/tests"
"github.com/navidrome/navidrome/utils/str"
. "github.com/onsi/ginkgo/v2"
. "github.com/onsi/gomega"
)
// Mock agent implementation for testing
type mockArtistTopSongsAgent struct {
agents.Interface
err error
topSongs []agents.Song
lastArtistID string
lastArtistName string
lastMBID string
lastCount int
getArtistTopSongsFn func(ctx context.Context, id, artistName, mbid string, count int) ([]agents.Song, error)
}
func (m *mockArtistTopSongsAgent) AgentName() string {
return "mock"
}
func (m *mockArtistTopSongsAgent) GetArtistTopSongs(ctx context.Context, id, artistName, mbid string, count int) ([]agents.Song, error) {
m.lastCount = count
m.lastArtistID = id
m.lastArtistName = artistName
m.lastMBID = mbid
log.Debug(ctx, "MockAgent.GetArtistTopSongs called", "id", id, "name", artistName, "mbid", mbid, "count", count)
// Use the custom function if available
if m.getArtistTopSongsFn != nil {
return m.getArtistTopSongsFn(ctx, id, artistName, mbid, count)
}
if m.err != nil {
log.Debug(ctx, "MockAgent.GetArtistTopSongs returning error", "err", m.err)
return nil, m.err
}
log.Debug(ctx, "MockAgent.GetArtistTopSongs returning songs", "count", len(m.topSongs))
return m.topSongs, nil
}
// Make sure the mock agent implements the necessary interface
var _ agents.ArtistTopSongsRetriever = (*mockArtistTopSongsAgent)(nil)
// Sets unexported fields in a struct using reflection and unsafe package
func setAgentField(obj interface{}, fieldName string, value interface{}) {
v := reflect.ValueOf(obj).Elem()
f := v.FieldByName(fieldName)
rf := reflect.NewAt(f.Type(), unsafe.Pointer(f.UnsafeAddr())).Elem()
rf.Set(reflect.ValueOf(value))
}
// Custom ArtistRepo that implements GetAll for our tests
type testArtistRepo struct {
*tests.MockArtistRepo
artists model.Artists
errFlag bool
err error
}
func newTestArtistRepo() *testArtistRepo {
return &testArtistRepo{
MockArtistRepo: tests.CreateMockArtistRepo(),
artists: model.Artists{},
}
}
func (m *testArtistRepo) SetError(err bool) {
m.errFlag = err
m.MockArtistRepo.SetError(err)
}
func (m *testArtistRepo) SetData(artists model.Artists) {
m.artists = artists
m.MockArtistRepo.SetData(artists)
}
func (m *testArtistRepo) GetAll(options ...model.QueryOptions) (model.Artists, error) {
if m.errFlag {
return nil, errors.New("error")
}
// No filters means return all
if len(options) == 0 || options[0].Filters == nil {
return m.artists, nil
}
// Basic implementation that returns artists matching name filter
if len(options) > 0 && options[0].Filters != nil {
switch f := options[0].Filters.(type) {
case squirrel.Like:
if nameFilter, ok := f["artist.name"]; ok {
// Convert to string and remove any SQL wildcard characters for simple comparison
name := strings.ReplaceAll(nameFilter.(string), "%", "")
log.Debug("ArtistRepo.GetAll: Looking for artist by name", "name", name)
for _, a := range m.artists {
if a.Name == name {
log.Debug("ArtistRepo.GetAll: Found artist", "id", a.ID, "name", a.Name)
return model.Artists{a}, nil
}
}
}
case squirrel.Eq:
if ids, ok := f["artist.id"]; ok {
var result model.Artists
for _, a := range m.artists {
for _, id := range ids.([]string) {
if a.ID == id {
result = append(result, a)
}
}
}
return result, nil
}
}
}
log.Debug("ArtistRepo.GetAll: No matching artist found")
// If no filter matches or no options, return empty
return model.Artists{}, nil
}
// Custom MediaFileRepo that implements GetAll for our tests
type testMediaFileRepo struct {
*tests.MockMediaFileRepo
mediaFiles model.MediaFiles
errFlag bool
}
func newTestMediaFileRepo() *testMediaFileRepo {
return &testMediaFileRepo{
MockMediaFileRepo: tests.CreateMockMediaFileRepo(),
mediaFiles: model.MediaFiles{},
}
}
func (m *testMediaFileRepo) SetError(err bool) {
m.errFlag = err
m.MockMediaFileRepo.SetError(err)
}
func (m *testMediaFileRepo) SetData(mfs model.MediaFiles) {
m.mediaFiles = mfs
m.MockMediaFileRepo.SetData(mfs)
}
func (m *testMediaFileRepo) GetAll(options ...model.QueryOptions) (model.MediaFiles, error) {
if m.errFlag {
return nil, errors.New("error")
}
if len(options) == 0 {
return m.mediaFiles, nil
}
// Process filters
if options[0].Filters != nil {
switch filter := options[0].Filters.(type) {
case squirrel.And:
// This handles the case where we search by artist ID and title
log.Debug("MediaFileRepo.GetAll: Processing AND filter")
return m.handleAndFilter(filter, options[0])
case squirrel.Eq:
// This handles the case where we search by mbz_recording_id
log.Debug("MediaFileRepo.GetAll: Processing EQ filter", "filter", fmt.Sprintf("%+v", filter))
if mbid, ok := filter["mbz_recording_id"]; ok {
log.Debug("MediaFileRepo.GetAll: Looking for MBID", "mbid", mbid)
for _, mf := range m.mediaFiles {
log.Debug("MediaFileRepo.GetAll: Comparing MBID", "file_mbid", mf.MbzReleaseTrackID, "search_mbid", mbid, "missing", mf.Missing)
if mf.MbzReleaseTrackID == mbid.(string) && !mf.Missing {
log.Debug("MediaFileRepo.GetAll: Found matching file by MBID", "id", mf.ID, "title", mf.Title)
return model.MediaFiles{mf}, nil
}
}
}
}
}
log.Debug("MediaFileRepo.GetAll: No matches found")
return model.MediaFiles{}, nil
}
func (m *testMediaFileRepo) handleAndFilter(andFilter squirrel.And, option model.QueryOptions) (model.MediaFiles, error) {
// Get matches for each condition
var artistMatches []model.MediaFile
var titleMatches []model.MediaFile
var notMissingMatches []model.MediaFile
// First identify non-missing files
for _, mf := range m.mediaFiles {
if !mf.Missing {
notMissingMatches = append(notMissingMatches, mf)
}
}
log.Debug("MediaFileRepo.handleAndFilter: Processing filters", "filterCount", len(andFilter))
// Now look for matches on other criteria
for _, sqlizer := range andFilter {
switch filter := sqlizer.(type) {
case squirrel.Or:
// Handle artist ID matching
log.Debug("MediaFileRepo.handleAndFilter: Processing OR filter")
for _, orCond := range filter {
if eqCond, ok := orCond.(squirrel.Eq); ok {
if artistID, ok := eqCond["artist_id"]; ok {
log.Debug("MediaFileRepo.handleAndFilter: Looking for artist_id", "artistID", artistID)
for _, mf := range notMissingMatches {
if mf.ArtistID == artistID.(string) {
log.Debug("MediaFileRepo.handleAndFilter: Found match by artist_id", "id", mf.ID, "title", mf.Title)
artistMatches = append(artistMatches, mf)
}
}
}
if albumArtistID, ok := eqCond["album_artist_id"]; ok {
log.Debug("MediaFileRepo.handleAndFilter: Looking for album_artist_id", "albumArtistID", albumArtistID)
for _, mf := range notMissingMatches {
if mf.AlbumArtistID == albumArtistID.(string) {
log.Debug("MediaFileRepo.handleAndFilter: Found match by album_artist_id", "id", mf.ID, "title", mf.Title)
artistMatches = append(artistMatches, mf)
}
}
}
}
}
case squirrel.Like:
// Handle title matching
log.Debug("MediaFileRepo.handleAndFilter: Processing LIKE filter", "filter", fmt.Sprintf("%+v", filter))
if orderTitle, ok := filter["order_title"]; ok {
normalizedTitle := str.SanitizeFieldForSorting(orderTitle.(string))
log.Debug("MediaFileRepo.handleAndFilter: Looking for title match", "normalizedTitle", normalizedTitle)
for _, mf := range notMissingMatches {
normalizedMfTitle := str.SanitizeFieldForSorting(mf.Title)
log.Debug("MediaFileRepo.handleAndFilter: Comparing titles", "fileTitle", mf.Title, "normalizedFileTitle", normalizedMfTitle)
if normalizedTitle == normalizedMfTitle {
log.Debug("MediaFileRepo.handleAndFilter: Found title match", "id", mf.ID, "title", mf.Title)
titleMatches = append(titleMatches, mf)
}
}
}
case squirrel.Eq:
// Handle missing check
if missingFlag, ok := filter["missing"]; ok && !missingFlag.(bool) {
// This is already handled above when we build notMissingMatches
continue
}
}
}
log.Debug("MediaFileRepo.handleAndFilter: Matching stats", "artistMatches", len(artistMatches), "titleMatches", len(titleMatches))
// Find matches that satisfy all conditions
var result model.MediaFiles
for _, am := range artistMatches {
for _, tm := range titleMatches {
if am.ID == tm.ID {
log.Debug("MediaFileRepo.handleAndFilter: Found complete match", "id", am.ID, "title", am.Title)
result = append(result, am)
}
}
}
// Apply any sort and limit from the options
if option.Max > 0 && len(result) > option.Max {
result = result[:option.Max]
}
log.Debug("MediaFileRepo.handleAndFilter: Returning results", "count", len(result))
return result, nil
}
var _ = Describe("ExternalMetadata", func() {
var ds model.DataStore
var em ExternalMetadata
var mockAgent *mockArtistTopSongsAgent
var mockArtistRepo *testArtistRepo
var mockMediaFileRepo *testMediaFileRepo
var ctx context.Context
var originalAgentsConfig string
BeforeEach(func() {
ctx = context.Background()
// Store the original agents config to restore it later
originalAgentsConfig = conf.Server.Agents
// Setup mocks
mockArtistRepo = newTestArtistRepo()
mockMediaFileRepo = newTestMediaFileRepo()
ds = &tests.MockDataStore{
MockedArtist: mockArtistRepo,
MockedMediaFile: mockMediaFileRepo,
}
// Clear the agents map to prevent interference from previous tests
agents.Map = nil
// Create a mock agent
mockAgent = &mockArtistTopSongsAgent{}
log.Debug(ctx, "Creating mock agent", "agent", mockAgent)
})
AfterEach(func() {
// Restore original config
conf.Server.Agents = originalAgentsConfig
})
Describe("TopSongs with direct agent injection", func() {
BeforeEach(func() {
// Set up artists data
mockArtistRepo.SetData(model.Artists{
{ID: "artist-1", Name: "Artist One"},
{ID: "artist-2", Name: "Artist Two"},
})
// Set up mediafiles data with all necessary fields for matching
mockMediaFileRepo.SetData(model.MediaFiles{
{
ID: "song-1",
Title: "Song One",
Artist: "Artist One",
ArtistID: "artist-1",
AlbumArtistID: "artist-1",
MbzReleaseTrackID: "mbid-1",
Missing: false,
},
{
ID: "song-2",
Title: "Song Two",
Artist: "Artist One",
ArtistID: "artist-1",
AlbumArtistID: "artist-1",
MbzReleaseTrackID: "mbid-2",
Missing: false,
},
{
ID: "song-3",
Title: "Song Three",
Artist: "Artist Two",
ArtistID: "artist-2",
AlbumArtistID: "artist-2",
MbzReleaseTrackID: "mbid-3",
Missing: false,
},
})
// Configure the mockAgent to return some top songs
mockAgent.topSongs = []agents.Song{
{Name: "Song One", MBID: "mbid-1"},
{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 externalMetadata 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)
Expect(err).ToNot(HaveOccurred())
Expect(songs).To(HaveLen(2))
Expect(songs[0].ID).To(Equal("song-1"))
Expect(songs[1].ID).To(Equal("song-2"))
// Verify the agent was called with the right parameters
Expect(mockAgent.lastArtistID).To(Equal("artist-1"))
Expect(mockAgent.lastArtistName).To(Equal("Artist One"))
Expect(mockAgent.lastCount).To(Equal(5))
})
It("returns nil when artist is not found", func() {
// Set an error for mockArtistRepo to simulate artist not found
mockArtistRepo.SetError(true)
songs, err := em.TopSongs(ctx, "Unknown Artist", 5)
Expect(err).To(BeNil())
Expect(songs).To(BeNil())
})
It("returns empty list when no matching songs are found", func() {
// Configure the agent to return songs that don't match our repo
mockAgent.topSongs = []agents.Song{
{Name: "Nonexistent Song", MBID: "unknown-mbid"},
}
songs, err := em.TopSongs(ctx, "Artist One", 5)
Expect(err).ToNot(HaveOccurred())
Expect(songs).To(HaveLen(0))
})
It("returns nil when agent returns errors", func() {
// Reset the agent error state
mockArtistRepo.SetError(false)
// Set the error
testError := errors.New("some agent error")
mockAgent.err = testError
songs, err := em.TopSongs(ctx, "Artist One", 5)
// Current behavior returns nil for both error and songs
Expect(err).To(BeNil())
Expect(songs).To(BeNil())
})
It("respects count parameter", func() {
mockAgent.topSongs = []agents.Song{
{Name: "Song One", MBID: "mbid-1"},
{Name: "Song Two", MBID: "mbid-2"},
{Name: "Song Three", MBID: "mbid-3"},
}
songs, err := em.TopSongs(ctx, "Artist One", 1)
Expect(err).ToNot(HaveOccurred())
Expect(songs).To(HaveLen(1))
Expect(songs[0].ID).To(Equal("song-1"))
})
})
Describe("TopSongs with agent registration", func() {
BeforeEach(func() {
// Set our mock agent as the only agent
conf.Server.Agents = "mock"
// Set up artists data
mockArtistRepo.SetData(model.Artists{
{ID: "artist-1", Name: "Artist One"},
})
// Set up mediafiles data
mockMediaFileRepo.SetData(model.MediaFiles{
{
ID: "song-1",
Title: "Song One",
Artist: "Artist One",
ArtistID: "artist-1",
MbzReleaseTrackID: "mbid-1",
Missing: false,
},
{
ID: "song-2",
Title: "Song Two",
Artist: "Artist One",
ArtistID: "artist-1",
MbzReleaseTrackID: "mbid-2",
Missing: false,
},
})
// Configure and register the agent
mockAgent.topSongs = []agents.Song{
{Name: "Song One", MBID: "mbid-1"},
{Name: "Song Two", MBID: "mbid-2"},
}
// Register our mock agent
agents.Register("mock", func(model.DataStore) agents.Interface { return mockAgent })
// Create the externalMetadata instance with registered agents
em = NewExternalMetadata(ds, agents.GetAgents(ds))
})
It("returns matching songs from the registered agent", func() {
songs, err := em.TopSongs(ctx, "Artist One", 5)
Expect(err).ToNot(HaveOccurred())
Expect(songs).To(HaveLen(2))
Expect(songs[0].ID).To(Equal("song-1"))
Expect(songs[1].ID).To(Equal("song-2"))
})
})
Describe("Error propagation from agents", func() {
BeforeEach(func() {
// Set up artists data
mockArtistRepo.SetData(model.Artists{
{ID: "artist-1", Name: "Artist One"},
})
// Create a direct agent that returns an error
testError := errors.New("direct agent error")
directAgent := &mockArtistTopSongsAgent{
getArtistTopSongsFn: func(ctx context.Context, id, artistName, mbid string, count int) ([]agents.Song, error) {
return nil, testError
},
}
// Create a custom implementation of agents.Agents that will return our error
directAgentsImpl := &agents.Agents{}
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() {
songs, err := em.TopSongs(ctx, "Artist One", 5)
// Current behavior returns nil for both error and songs
Expect(err).To(BeNil())
Expect(songs).To(BeNil())
})
})
})