mirror of
https://github.com/navidrome/navidrome.git
synced 2026-03-04 06:35:52 +00:00
* feat(plugins): add lyrics provider plugin capability Refactor the lyrics system from a static function to an interface-based service that supports WASM plugin providers. Plugins listed in the LyricsPriority config (alongside "embedded" and file extensions) are now resolved through the plugin system. Includes capability definition, Go/Rust PDK, adapter, Wire integration, and tests for plugin fallback behavior. * test(plugins): add lyrics capability integration test with test plugin * fix(plugins): default lyrics language to 'xxx' when plugin omits it Per the OpenSubsonic spec, the server must return 'und' or 'xxx' when the lyrics language is unknown. The lyrics plugin adapter was passing an empty string through when a plugin didn't provide a language value. This defaults the language to 'xxx', consistent with all other callers of model.ToLyrics() in the codebase. * refactor(plugins): rename lyrics import to improve clarity Signed-off-by: Deluan <deluan@navidrome.org> * refactor(lyrics): update TrackInfo description for clarity Signed-off-by: Deluan <deluan@navidrome.org> * fix(lyrics): enhance lyrics plugin handling and case sensitivity Signed-off-by: Deluan <deluan@navidrome.org> * fix(plugins): update payload type to string with byte format for task data Signed-off-by: Deluan <deluan@navidrome.org> --------- Signed-off-by: Deluan <deluan@navidrome.org>
107 lines
4.0 KiB
Go
107 lines
4.0 KiB
Go
package capabilities
|
|
|
|
// Scrobbler provides scrobbling functionality to external services.
|
|
// This capability allows plugins to submit listening history to services like Last.fm,
|
|
// ListenBrainz, or custom scrobbling backends.
|
|
//
|
|
// All methods are required - plugins implementing this capability must provide
|
|
// all three functions: IsAuthorized, NowPlaying, and Scrobble.
|
|
//
|
|
//nd:capability name=scrobbler required=true
|
|
type Scrobbler interface {
|
|
// IsAuthorized checks if a user is authorized to scrobble to this service.
|
|
//nd:export name=nd_scrobbler_is_authorized
|
|
IsAuthorized(IsAuthorizedRequest) (bool, error)
|
|
|
|
// NowPlaying sends a now playing notification to the scrobbling service.
|
|
//nd:export name=nd_scrobbler_now_playing
|
|
NowPlaying(NowPlayingRequest) error
|
|
|
|
// Scrobble submits a completed scrobble to the scrobbling service.
|
|
//nd:export name=nd_scrobbler_scrobble
|
|
Scrobble(ScrobbleRequest) error
|
|
}
|
|
|
|
// IsAuthorizedRequest is the request for authorization check.
|
|
type IsAuthorizedRequest struct {
|
|
// Username is the username of the user.
|
|
Username string `json:"username"`
|
|
}
|
|
|
|
// ArtistRef is a reference to an artist with name and optional MBID.
|
|
type ArtistRef struct {
|
|
// ID is the internal Navidrome artist ID (if known).
|
|
ID string `json:"id,omitempty"`
|
|
// Name is the artist name.
|
|
Name string `json:"name"`
|
|
// MBID is the MusicBrainz ID for the artist.
|
|
MBID string `json:"mbid,omitempty"`
|
|
}
|
|
|
|
// TrackInfo contains track metadata.
|
|
type TrackInfo struct {
|
|
// ID is the internal Navidrome track ID.
|
|
ID string `json:"id"`
|
|
// Title is the track title.
|
|
Title string `json:"title"`
|
|
// Album is the album name.
|
|
Album string `json:"album"`
|
|
// Artist is the formatted artist name for display (e.g., "Artist1 • Artist2").
|
|
Artist string `json:"artist"`
|
|
// AlbumArtist is the formatted album artist name for display.
|
|
AlbumArtist string `json:"albumArtist"`
|
|
// Artists is the list of track artists.
|
|
Artists []ArtistRef `json:"artists"`
|
|
// AlbumArtists is the list of album artists.
|
|
AlbumArtists []ArtistRef `json:"albumArtists"`
|
|
// Duration is the track duration in seconds.
|
|
Duration float32 `json:"duration"`
|
|
// TrackNumber is the track number on the album.
|
|
TrackNumber int32 `json:"trackNumber"`
|
|
// DiscNumber is the disc number.
|
|
DiscNumber int32 `json:"discNumber"`
|
|
// MBZRecordingID is the MusicBrainz recording ID.
|
|
MBZRecordingID string `json:"mbzRecordingId,omitempty"`
|
|
// MBZAlbumID is the MusicBrainz album/release ID.
|
|
MBZAlbumID string `json:"mbzAlbumId,omitempty"`
|
|
// MBZReleaseGroupID is the MusicBrainz release group ID.
|
|
MBZReleaseGroupID string `json:"mbzReleaseGroupId,omitempty"`
|
|
// MBZReleaseTrackID is the MusicBrainz release track ID.
|
|
MBZReleaseTrackID string `json:"mbzReleaseTrackId,omitempty"`
|
|
}
|
|
|
|
// NowPlayingRequest is the request for now playing notification.
|
|
type NowPlayingRequest struct {
|
|
// Username is the username of the user.
|
|
Username string `json:"username"`
|
|
// Track is the track currently playing.
|
|
Track TrackInfo `json:"track"`
|
|
// Position is the current playback position in seconds.
|
|
Position int32 `json:"position"`
|
|
}
|
|
|
|
// ScrobbleRequest is the request for submitting a scrobble.
|
|
type ScrobbleRequest struct {
|
|
// Username is the username of the user.
|
|
Username string `json:"username"`
|
|
// Track is the track that was played.
|
|
Track TrackInfo `json:"track"`
|
|
// Timestamp is the Unix timestamp when the track started playing.
|
|
Timestamp int64 `json:"timestamp"`
|
|
}
|
|
|
|
// ScrobblerError represents an error type for scrobbling operations.
|
|
type ScrobblerError string
|
|
|
|
const (
|
|
// ScrobblerErrorNotAuthorized indicates the user is not authorized.
|
|
ScrobblerErrorNotAuthorized ScrobblerError = "scrobbler(not_authorized)"
|
|
// ScrobblerErrorRetryLater indicates the operation should be retried later.
|
|
ScrobblerErrorRetryLater ScrobblerError = "scrobbler(retry_later)"
|
|
// ScrobblerErrorUnrecoverable indicates an unrecoverable error.
|
|
ScrobblerErrorUnrecoverable ScrobblerError = "scrobbler(unrecoverable)"
|
|
)
|
|
|
|
// Error implements the error interface for ScrobblerError.
|
|
func (e ScrobblerError) Error() string { return string(e) }
|