package scrobbler import ( "context" "errors" "time" "github.com/navidrome/navidrome/log" "github.com/navidrome/navidrome/model" ) // Loader is a function that loads a scrobbler by name. // It returns the scrobbler and true if found, or nil and false if not available. // This allows the buffered scrobbler to always get the current plugin instance. type Loader func() (Scrobbler, bool) // newBufferedScrobbler creates a buffered scrobbler that wraps a static scrobbler instance. // Use this for builtin scrobblers that don't change. func newBufferedScrobbler(ds model.DataStore, s Scrobbler, service string) *bufferedScrobbler { return newBufferedScrobblerWithLoader(ds, service, func() (Scrobbler, bool) { return s, true }) } // newBufferedScrobblerWithLoader creates a buffered scrobbler that dynamically loads // the underlying scrobbler on each call. Use this for plugin scrobblers that may be // reloaded (e.g., after configuration changes). func newBufferedScrobblerWithLoader(ds model.DataStore, service string, loader Loader) *bufferedScrobbler { ctx, cancel := context.WithCancel(context.Background()) b := &bufferedScrobbler{ ds: ds, loader: loader, service: service, wakeSignal: make(chan struct{}, 1), ctx: ctx, cancel: cancel, } go b.run(ctx) return b } type bufferedScrobbler struct { ds model.DataStore loader Loader service string wakeSignal chan struct{} ctx context.Context cancel context.CancelFunc } func (b *bufferedScrobbler) Stop() { if b.cancel != nil { b.cancel() } } func (b *bufferedScrobbler) IsAuthorized(ctx context.Context, userId string) bool { s, ok := b.loader() if !ok { return false } return s.IsAuthorized(ctx, userId) } func (b *bufferedScrobbler) NowPlaying(ctx context.Context, userId string, track *model.MediaFile, position int) error { s, ok := b.loader() if !ok { return errors.New("scrobbler not available") } return s.NowPlaying(ctx, userId, track, position) } func (b *bufferedScrobbler) Scrobble(ctx context.Context, userId string, s Scrobble) error { err := b.ds.ScrobbleBuffer(ctx).Enqueue(b.service, userId, s.ID, s.TimeStamp) if err != nil { return err } b.sendWakeSignal() return nil } func (b *bufferedScrobbler) sendWakeSignal() { // Don't block if the previous signal was not read yet select { case b.wakeSignal <- struct{}{}: default: } } func (b *bufferedScrobbler) run(ctx context.Context) { for { if !b.processQueue(ctx) { time.AfterFunc(5*time.Second, func() { b.sendWakeSignal() }) } select { case <-b.wakeSignal: continue case <-ctx.Done(): return } } } func (b *bufferedScrobbler) processQueue(ctx context.Context) bool { buffer := b.ds.ScrobbleBuffer(ctx) userIds, err := buffer.UserIDs(b.service) if err != nil { log.Error(ctx, "Error retrieving userIds from scrobble buffer", "scrobbler", b.service, err) return false } result := true for _, userId := range userIds { if !b.processUserQueue(ctx, userId) { result = false } } return result } func (b *bufferedScrobbler) processUserQueue(ctx context.Context, userId string) bool { buffer := b.ds.ScrobbleBuffer(ctx) for { entry, err := buffer.Next(b.service, userId) if err != nil { log.Error(ctx, "Error reading from scrobble buffer", "scrobbler", b.service, err) return false } if entry == nil { return true } s, ok := b.loader() if !ok { log.Warn(ctx, "Scrobbler not available, will retry later", "scrobbler", b.service) return false } log.Debug(ctx, "Sending scrobble", "scrobbler", b.service, "track", entry.Title, "artist", entry.Artist) err = s.Scrobble(ctx, entry.UserID, Scrobble{ MediaFile: entry.MediaFile, TimeStamp: entry.PlayTime, }) if errors.Is(err, ErrRetryLater) { log.Warn(ctx, "Could not send scrobble. Will be retried", "userId", entry.UserID, "track", entry.Title, "artist", entry.Artist, "scrobbler", b.service, err) return false } if err != nil { log.Error(ctx, "Error sending scrobble to service. Discarding", "scrobbler", b.service, "userId", entry.UserID, "artist", entry.Artist, "track", entry.Title, err) } err = buffer.Dequeue(entry) if err != nil { log.Error(ctx, "Error removing entry from scrobble buffer", "userId", entry.UserID, "track", entry.Title, "artist", entry.Artist, "scrobbler", b.service, err) return false } } } var _ Scrobbler = (*bufferedScrobbler)(nil)