mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Always update artist info, even if info is fresh
This commit is contained in:
parent
b398053223
commit
4e0177ee53
2 changed files with 34 additions and 23 deletions
|
@ -30,7 +30,7 @@ const (
|
||||||
RequestThrottleBacklogLimit = 100
|
RequestThrottleBacklogLimit = 100
|
||||||
RequestThrottleBacklogTimeout = time.Minute
|
RequestThrottleBacklogTimeout = time.Minute
|
||||||
|
|
||||||
ArtistInfoTimeToLive = 1 * time.Hour
|
ArtistInfoTimeToLive = 3 * 24 * time.Hour
|
||||||
|
|
||||||
I18nFolder = "i18n"
|
I18nFolder = "i18n"
|
||||||
SkipScanFile = ".ndignore"
|
SkipScanFile = ".ndignore"
|
||||||
|
|
|
@ -16,7 +16,10 @@ import (
|
||||||
"github.com/navidrome/navidrome/model"
|
"github.com/navidrome/navidrome/model"
|
||||||
)
|
)
|
||||||
|
|
||||||
const unavailableArtistID = "-1"
|
const (
|
||||||
|
unavailableArtistID = "-1"
|
||||||
|
maxSimilarArtists = 100
|
||||||
|
)
|
||||||
|
|
||||||
type ExternalMetadata interface {
|
type ExternalMetadata interface {
|
||||||
UpdateArtistInfo(ctx context.Context, id string, count int, includeNotPresent bool) (*model.Artist, error)
|
UpdateArtistInfo(ctx context.Context, id string, count int, includeNotPresent bool) (*model.Artist, error)
|
||||||
|
@ -88,19 +91,35 @@ func clearName(name string) string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *externalMetadata) UpdateArtistInfo(ctx context.Context, id string, similarCount int, includeNotPresent bool) (*model.Artist, error) {
|
func (e *externalMetadata) UpdateArtistInfo(ctx context.Context, id string, similarCount int, includeNotPresent bool) (*model.Artist, error) {
|
||||||
allAgents := e.initAgents(ctx)
|
|
||||||
artist, err := e.getArtist(ctx, id)
|
artist, err := e.getArtist(ctx, id)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we have fresh info, just return it
|
// If we have fresh info, just return it and trigger a refresh in the background
|
||||||
if time.Since(artist.ExternalInfoUpdatedAt) < consts.ArtistInfoTimeToLive {
|
if time.Since(artist.ExternalInfoUpdatedAt) < consts.ArtistInfoTimeToLive {
|
||||||
log.Debug("Found cached ArtistInfo", "updatedAt", artist.ExternalInfoUpdatedAt, "name", artist.Name)
|
go func() {
|
||||||
err := e.loadSimilar(ctx, artist, includeNotPresent)
|
err := e.refreshArtistInfo(ctx, artist)
|
||||||
|
if err != nil {
|
||||||
|
log.Error("Error refreshing ArtistInfo", "id", id, "name", artist.Name, err)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
log.Debug("Found cached ArtistInfo, refreshing in the background", "updatedAt", artist.ExternalInfoUpdatedAt, "name", artist.Name)
|
||||||
|
err := e.loadSimilar(ctx, artist, similarCount, includeNotPresent)
|
||||||
return &artist.Artist, err
|
return &artist.Artist, err
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Debug(ctx, "ArtistInfo not cached or expired", "updatedAt", artist.ExternalInfoUpdatedAt, "id", id, "name", artist.Name)
|
log.Debug(ctx, "ArtistInfo not cached or expired", "updatedAt", artist.ExternalInfoUpdatedAt, "id", id, "name", artist.Name)
|
||||||
|
err = e.refreshArtistInfo(ctx, artist)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
err = e.loadSimilar(ctx, artist, similarCount, includeNotPresent)
|
||||||
|
return &artist.Artist, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (e *externalMetadata) refreshArtistInfo(ctx context.Context, artist *auxArtist) error {
|
||||||
|
allAgents := e.initAgents(ctx)
|
||||||
|
|
||||||
// Get MBID first, if it is not yet available
|
// Get MBID first, if it is not yet available
|
||||||
if artist.MbzArtistID == "" {
|
if artist.MbzArtistID == "" {
|
||||||
|
@ -112,33 +131,22 @@ func (e *externalMetadata) UpdateArtistInfo(ctx context.Context, id string, simi
|
||||||
e.callGetBiography(ctx, allAgents, artist, wg)
|
e.callGetBiography(ctx, allAgents, artist, wg)
|
||||||
e.callGetURL(ctx, allAgents, artist, wg)
|
e.callGetURL(ctx, allAgents, artist, wg)
|
||||||
e.callGetImage(ctx, allAgents, artist, wg)
|
e.callGetImage(ctx, allAgents, artist, wg)
|
||||||
e.callGetSimilar(ctx, allAgents, artist, similarCount, wg)
|
e.callGetSimilar(ctx, allAgents, artist, maxSimilarArtists, wg)
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
|
|
||||||
if isDone(ctx) {
|
if isDone(ctx) {
|
||||||
log.Warn(ctx, "ArtistInfo update canceled", ctx.Err())
|
log.Warn(ctx, "ArtistInfo update canceled", ctx.Err())
|
||||||
return nil, ctx.Err()
|
return ctx.Err()
|
||||||
}
|
}
|
||||||
|
|
||||||
artist.ExternalInfoUpdatedAt = time.Now()
|
artist.ExternalInfoUpdatedAt = time.Now()
|
||||||
err = e.ds.Artist(ctx).Put(&artist.Artist)
|
err := e.ds.Artist(ctx).Put(&artist.Artist)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error(ctx, "Error trying to update artist external information", "id", id, "name", artist.Name, err)
|
log.Error(ctx, "Error trying to update artist external information", "id", artist.ID, "name", artist.Name, err)
|
||||||
}
|
|
||||||
|
|
||||||
if !includeNotPresent {
|
|
||||||
similar := artist.SimilarArtists
|
|
||||||
artist.SimilarArtists = nil
|
|
||||||
for _, s := range similar {
|
|
||||||
if s.ID == unavailableArtistID {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
artist.SimilarArtists = append(artist.SimilarArtists, s)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
log.Trace(ctx, "ArtistInfo collected", "artist", artist)
|
log.Trace(ctx, "ArtistInfo collected", "artist", artist)
|
||||||
return &artist.Artist, nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *externalMetadata) SimilarSongs(ctx context.Context, id string, count int) (model.MediaFiles, error) {
|
func (e *externalMetadata) SimilarSongs(ctx context.Context, id string, count int) (model.MediaFiles, error) {
|
||||||
|
@ -425,7 +433,7 @@ func (e *externalMetadata) findArtistByName(ctx context.Context, artistName stri
|
||||||
return artist, nil
|
return artist, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (e *externalMetadata) loadSimilar(ctx context.Context, artist *auxArtist, includeNotPresent bool) error {
|
func (e *externalMetadata) loadSimilar(ctx context.Context, artist *auxArtist, count int, includeNotPresent bool) error {
|
||||||
var ids []string
|
var ids []string
|
||||||
for _, sa := range artist.SimilarArtists {
|
for _, sa := range artist.SimilarArtists {
|
||||||
if sa.ID == unavailableArtistID {
|
if sa.ID == unavailableArtistID {
|
||||||
|
@ -449,6 +457,9 @@ func (e *externalMetadata) loadSimilar(ctx context.Context, artist *auxArtist, i
|
||||||
|
|
||||||
var loaded model.Artists
|
var loaded model.Artists
|
||||||
for _, sa := range artist.SimilarArtists {
|
for _, sa := range artist.SimilarArtists {
|
||||||
|
if len(loaded) >= count {
|
||||||
|
break
|
||||||
|
}
|
||||||
la, ok := artistMap[sa.ID]
|
la, ok := artistMap[sa.ID]
|
||||||
if !ok {
|
if !ok {
|
||||||
if !includeNotPresent {
|
if !includeNotPresent {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue