package lastfm import ( "context" "errors" "fmt" "net/http" "regexp" "strconv" "strings" "github.com/andybalholm/cascadia" "github.com/navidrome/navidrome/conf" "github.com/navidrome/navidrome/consts" "github.com/navidrome/navidrome/core/agents" "github.com/navidrome/navidrome/core/scrobbler" "github.com/navidrome/navidrome/log" "github.com/navidrome/navidrome/model" "github.com/navidrome/navidrome/utils/cache" "golang.org/x/net/html" ) const ( lastFMAgentName = "lastfm" sessionKeyProperty = "LastFMSessionKey" ) var ignoredBiographies = []string{ // Unknown Artist ` head > meta[property="og:image"]`) func (l *lastfmAgent) GetArtistImages(ctx context.Context, _, name, mbid string) ([]agents.ExternalImage, error) { a, err := l.callArtistGetInfo(ctx, name, mbid) if err != nil { return nil, fmt.Errorf("get artist info: %w", err) } req, err := http.NewRequest(http.MethodGet, a.URL, nil) if err != nil { return nil, fmt.Errorf("create artist image request: %w", err) } resp, err := l.client.hc.Do(req) if err != nil { return nil, fmt.Errorf("get artist url: %w", err) } defer resp.Body.Close() node, err := html.Parse(resp.Body) if err != nil { return nil, fmt.Errorf("parse html: %w", err) } var res []agents.ExternalImage n := cascadia.Query(node, artistOpenGraphQuery) if n == nil { return res, nil } for _, attr := range n.Attr { if attr.Key == "content" { res = []agents.ExternalImage{ {URL: attr.Val}, } break } } return res, nil } func (l *lastfmAgent) callAlbumGetInfo(ctx context.Context, name, artist, mbid string) (*Album, error) { a, err := l.client.albumGetInfo(ctx, name, artist, mbid) var lfErr *lastFMError isLastFMError := errors.As(err, &lfErr) if mbid != "" && (isLastFMError && lfErr.Code == 6) { log.Debug(ctx, "LastFM/album.getInfo could not find album by mbid, trying again", "album", name, "mbid", mbid) return l.callAlbumGetInfo(ctx, name, artist, "") } if err != nil { if isLastFMError && lfErr.Code == 6 { log.Debug(ctx, "Album not found", "album", name, "mbid", mbid, err) } else { log.Error(ctx, "Error calling LastFM/album.getInfo", "album", name, "mbid", mbid, err) } return nil, err } return a, nil } func (l *lastfmAgent) callArtistGetInfo(ctx context.Context, name string, mbid string) (*Artist, error) { a, err := l.client.artistGetInfo(ctx, name, mbid) var lfErr *lastFMError isLastFMError := errors.As(err, &lfErr) if mbid != "" && ((err == nil && a.Name == "[unknown]") || (isLastFMError && lfErr.Code == 6)) { log.Debug(ctx, "LastFM/artist.getInfo could not find artist by mbid, trying again", "artist", name, "mbid", mbid) return l.callArtistGetInfo(ctx, name, "") } if err != nil { log.Error(ctx, "Error calling LastFM/artist.getInfo", "artist", name, "mbid", mbid, err) return nil, err } return a, nil } func (l *lastfmAgent) callArtistGetSimilar(ctx context.Context, name string, mbid string, limit int) ([]Artist, error) { s, err := l.client.artistGetSimilar(ctx, name, mbid, limit) var lfErr *lastFMError isLastFMError := errors.As(err, &lfErr) if mbid != "" && ((err == nil && s.Attr.Artist == "[unknown]") || (isLastFMError && lfErr.Code == 6)) { log.Debug(ctx, "LastFM/artist.getSimilar could not find artist by mbid, trying again", "artist", name, "mbid", mbid) return l.callArtistGetSimilar(ctx, name, "", limit) } if err != nil { log.Error(ctx, "Error calling LastFM/artist.getSimilar", "artist", name, "mbid", mbid, err) return nil, err } return s.Artists, nil } func (l *lastfmAgent) callArtistGetTopTracks(ctx context.Context, artistName, mbid string, count int) ([]Track, error) { t, err := l.client.artistGetTopTracks(ctx, artistName, mbid, count) var lfErr *lastFMError isLastFMError := errors.As(err, &lfErr) if mbid != "" && ((err == nil && t.Attr.Artist == "[unknown]") || (isLastFMError && lfErr.Code == 6)) { log.Debug(ctx, "LastFM/artist.getTopTracks could not find artist by mbid, trying again", "artist", artistName, "mbid", mbid) return l.callArtistGetTopTracks(ctx, artistName, "", count) } if err != nil { log.Error(ctx, "Error calling LastFM/artist.getTopTracks", "artist", artistName, "mbid", mbid, err) return nil, err } return t.Track, nil } func (l *lastfmAgent) NowPlaying(ctx context.Context, userId string, track *model.MediaFile) error { sk, err := l.sessionKeys.Get(ctx, userId) if err != nil || sk == "" { return scrobbler.ErrNotAuthorized } err = l.client.updateNowPlaying(ctx, sk, ScrobbleInfo{ artist: track.Artist, track: track.Title, album: track.Album, trackNumber: track.TrackNumber, mbid: track.MbzRecordingID, duration: int(track.Duration), albumArtist: track.AlbumArtist, }) if err != nil { log.Warn(ctx, "Last.fm client.updateNowPlaying returned error", "track", track.Title, err) return scrobbler.ErrUnrecoverable } return nil } func (l *lastfmAgent) Scrobble(ctx context.Context, userId string, s scrobbler.Scrobble) error { sk, err := l.sessionKeys.Get(ctx, userId) if err != nil || sk == "" { return scrobbler.ErrNotAuthorized } if s.Duration <= 30 { log.Debug(ctx, "Skipping Last.fm scrobble for short song", "track", s.Title, "duration", s.Duration) return nil } err = l.client.scrobble(ctx, sk, ScrobbleInfo{ artist: s.Artist, track: s.Title, album: s.Album, trackNumber: s.TrackNumber, mbid: s.MbzRecordingID, duration: int(s.Duration), albumArtist: s.AlbumArtist, timestamp: s.TimeStamp, }) if err == nil { return nil } var lfErr *lastFMError isLastFMError := errors.As(err, &lfErr) if !isLastFMError { log.Warn(ctx, "Last.fm client.scrobble returned error", "track", s.Title, err) return scrobbler.ErrRetryLater } if lfErr.Code == 11 || lfErr.Code == 16 { return scrobbler.ErrRetryLater } return scrobbler.ErrUnrecoverable } func (l *lastfmAgent) IsAuthorized(ctx context.Context, userId string) bool { sk, err := l.sessionKeys.Get(ctx, userId) return err == nil && sk != "" } func init() { conf.AddHook(func() { if conf.Server.LastFM.Enabled { if conf.Server.LastFM.ApiKey != "" && conf.Server.LastFM.Secret != "" { agents.Register(lastFMAgentName, func(ds model.DataStore) agents.Interface { return lastFMConstructor(ds) }) scrobbler.Register(lastFMAgentName, func(ds model.DataStore) scrobbler.Scrobbler { return lastFMConstructor(ds) }) } } }) }