mopidy.backend
— Backend API
The backend API is the interface that must be implemented when you create a backend. If you are working on a frontend and need to access the backends, see the mopidy.core — Core API instead.
URIs and routing of requests to the backend
When Mopidy’s core layer is processing a client request, it routes the request
to one or more appropriate backends based on the URIs of the objects the
request touches on. The objects’ URIs are compared with the backends’
uri_schemes
to select the relevant backends.
An often used pattern when implementing Mopidy backends is to create your own URI scheme which you use for all tracks, playlists, etc. related to your backend. In most cases the Mopidy URI is translated to an actual URI that GStreamer knows how to play right before playback. For example:
Spotify already has its own URI scheme (
spotify:track:...
,spotify:playlist:...
, etc.) used throughout their applications, and thus Mopidy-Spotify simply uses the same URI scheme. Playback is handled by pushing raw audio data into a GStreamerappsrc
element.Mopidy-SoundCloud created it’s own URI scheme, after the model of Spotify, and uses URIs of the following forms:
soundcloud:search
,soundcloud:user-...
,soundcloud:exp-...
, andsoundcloud:set-...
. Playback is handled by converting the customsoundcloud:..
URIs tohttp://
URIs immediately before they are passed on to GStreamer for playback.Mopidy differentiates between
file://...
URIs handled by Mopidy-Stream andlocal:...
URIs handled by Mopidy-Local. Mopidy-Stream can playfile://...
URIs pointing to tracks and playlists located anywhere on your system, but it doesn’t know a thing about the object before you play it. On the other hand, Mopidy-Local scans a predefinedlocal/media_dir
to build a meta data library of all known tracks. It is thus limited to playing tracks residing in the media library, but can provide additional features like directory browsing and search. In other words, we have two different ways of playing local music, handled by two different backends, and have thus created two different URI schemes to separate their handling. Thelocal:...
URIs are converted tofile://...
URIs immediately before they are passed on to GStreamer for playback.
If there isn’t an existing URI scheme that fits for your backend’s purpose,
you should create your own, and name it after your extension’s
ext_name
. Care should be taken not to conflict
with already in use URI schemes. It is also recommended to design the format
such that tracks, playlists and other entities can be distinguished easily.
However, it’s important to note that outside of the backend that created them, URIs are opaque values that neither Mopidy’s core layer or Mopidy frontends should attempt to derive any meaning from. The only valid exception to this is checking the scheme.
Backend class
- class mopidy.backend.Backend[source]
Backend API
If the backend has problems during initialization it should raise
mopidy.exceptions.BackendError
with a descriptive error message. This will make Mopidy print the error message and exit so that the user can fix the issue.- Parameters:
config (dict) – the entire Mopidy configuration
audio (
pykka.ActorProxy
formopidy.audio.Audio
) – actor proxy for the audio subsystem
- library: LibraryProvider | None = None
The library provider. An instance of
LibraryProvider
, orNone
if the backend doesn’t provide a library.
- playback: PlaybackProvider | None = None
The playback provider. An instance of
PlaybackProvider
, orNone
if the backend doesn’t provide playback.
- playlists: PlaylistsProvider | None = None
The playlists provider. An instance of
PlaylistsProvider
, or class:None if the backend doesn’t provide playlists.
Playback provider
- class mopidy.backend.PlaybackProvider(audio: Any, backend: Backend)[source]
- Parameters:
audio (actor proxy to an instance of
mopidy.audio.Audio
) – the audio actorbackend (
mopidy.backend.Backend
) – the backend
- change_track(track: Track) bool [source]
Switch to provided track.
MAY be reimplemented by subclass.
It is unlikely it makes sense for any backends to override this. For most practical purposes it should be considered an internal call between backends and core that backend authors should not touch.
The default implementation will call
translate_uri()
which is what you want to implement.- Parameters:
track (
mopidy.models.Track
) – the track to play- Return type:
True
if successful, elseFalse
- get_time_position() int [source]
Get the current time position in milliseconds.
MAY be reimplemented by subclass.
- Return type:
- is_live(uri: Uri) bool [source]
Decide if the URI should be treated as a live stream or not.
MAY be reimplemented by subclass.
Playing a source as a live stream disables buffering, which reduces latency before playback starts, and discards data when paused.
- Parameters:
uri (string) – the URI
- Return type:
- on_source_setup(source: GstElement) None [source]
Called when a new GStreamer source is created, allowing us to configure the source. This runs in the audio thread so should not block.
MAY be reimplemented by subclass.
- Parameters:
source (GstElement) – the GStreamer source element
New in version 3.4.
- pause() bool [source]
Pause playback.
MAY be reimplemented by subclass.
- Return type:
True
if successful, elseFalse
- play() bool [source]
Start playback.
MAY be reimplemented by subclass.
- Return type:
True
if successful, elseFalse
- prepare_change() None [source]
Indicate that an URI change is about to happen.
MAY be reimplemented by subclass.
It is extremely unlikely it makes sense for any backends to override this. For most practical purposes it should be considered an internal call between backends and core that backend authors should not touch.
- resume() bool [source]
Resume playback at the same time position playback was paused.
MAY be reimplemented by subclass.
- Return type:
True
if successful, elseFalse
- seek(time_position: int) bool [source]
Seek to a given time position.
MAY be reimplemented by subclass.
- Parameters:
time_position (int) – time position in milliseconds
- Return type:
True
if successful, elseFalse
- should_download(uri: Uri) bool [source]
Attempt progressive download buffering for the URI or not.
MAY be reimplemented by subclass.
When streaming a fixed length file, the entire file can be buffered to improve playback performance.
- Parameters:
uri (string) – the URI
- Return type:
- stop() bool [source]
Stop playback.
MAY be reimplemented by subclass.
Should not be used for tracking if tracks have been played or when we are done playing them.
- Return type:
True
if successful, elseFalse
- translate_uri(uri: Uri) Uri | None [source]
Convert custom URI scheme to real playable URI.
MAY be reimplemented by subclass.
This is very likely the only thing you need to override as a backend author. Typically this is where you convert any Mopidy specific URI to a real URI and then return it. If you can’t convert the URI just return
None
.- Parameters:
uri (string) – the URI to translate
- Return type:
string or
None
if the URI could not be translated
Playlists provider
- class mopidy.backend.PlaylistsProvider(backend: Backend)[source]
A playlist provider exposes a collection of playlists, methods to create/change/delete playlists in this collection, and lookup of any playlist the backend knows about.
- Parameters:
backend (
mopidy.backend.Backend
instance) – backend the controller is a part of
- as_list() List[Ref] [source]
Get a list of the currently available playlists.
Returns a list of
Ref
objects referring to the playlists. In other words, no information about the playlists’ content is given.- Return type:
list of
mopidy.models.Ref
New in version 1.0.
- create(name: str) Playlist | None [source]
Create a new empty playlist with the given name.
Returns a new playlist with the given name and an URI, or
None
on failure.MUST be implemented by subclass.
- Parameters:
name (string) – name of the new playlist
- Return type:
mopidy.models.Playlist
orNone
- delete(uri: Uri) bool [source]
Delete playlist identified by the URI.
Returns
True
if deleted,False
otherwise.MUST be implemented by subclass.
- Parameters:
uri (string) – URI of the playlist to delete
- Return type:
Changed in version 2.2: Return type defined.
- get_items(uri: Uri) List[Ref] | None [source]
Get the items in a playlist specified by
uri
.Returns a list of
Ref
objects referring to the playlist’s items.If a playlist with the given
uri
doesn’t exist, it returnsNone
.- Return type:
list of
mopidy.models.Ref
, orNone
New in version 1.0.
- lookup(uri: Uri) Playlist | None [source]
Lookup playlist with given URI in both the set of playlists and in any other playlist source.
Returns the playlists or
None
if not found.MUST be implemented by subclass.
- Parameters:
uri (string) – playlist URI
- Return type:
mopidy.models.Playlist
orNone
- save(playlist: Playlist) Playlist | None [source]
Save the given playlist.
The playlist must have an
uri
attribute set. To create a new playlist with an URI, usecreate()
.Returns the saved playlist or
None
on failure.MUST be implemented by subclass.
- Parameters:
playlist (
mopidy.models.Playlist
) – the playlist to save- Return type:
mopidy.models.Playlist
orNone
Library provider
- class mopidy.backend.LibraryProvider(backend: Backend)[source]
- Parameters:
backend (
mopidy.backend.Backend
) – backend the controller is a part of
- browse(uri: Uri) List[Ref] [source]
See
mopidy.core.LibraryController.browse()
.If you implement this method, make sure to also set
root_directory
.MAY be implemented by subclass.
- get_distinct(field: DistinctField, query: Query[DistinctField] | None = None) Set[str] [source]
See
mopidy.core.LibraryController.get_distinct()
.MAY be implemented by subclass.
Default implementation will simply return an empty set.
Note that backends should always return an empty set for unexpected field types.
- get_images(uris: List[Uri]) Dict[Uri, List[Image]] [source]
See
mopidy.core.LibraryController.get_images()
.MAY be implemented by subclass.
Default implementation will simply return an empty dictionary.
- lookup(uri: Uri) Dict[Uri, List[Track]] [source]
See
mopidy.core.LibraryController.lookup()
.MUST be implemented by subclass.
- refresh(uri: Uri | None = None) None [source]
See
mopidy.core.LibraryController.refresh()
.MAY be implemented by subclass.
- root_directory: Ref | None = None
mopidy.models.Ref.directory
instance with a URI and name set representing the root of this library’s browse tree. URIs must use one of the schemes supported by the backend, and name should be set to a human friendly value.MUST be set by any class that implements
LibraryProvider.browse()
.
- search(query: Query[SearchField], uris: List[Uri] | None = None, exact: bool = False) List[SearchResult] [source]
See
mopidy.core.LibraryController.search()
.MAY be implemented by subclass.
New in version 1.0: The
exact
param which replaces the oldfind_exact
.
Backend listener
- class mopidy.backend.BackendListener[source]
Marker interface for recipients of events sent by the backend actors.
Any Pykka actor that mixes in this class will receive calls to the methods defined here when the corresponding events happen in a backend actor. This interface is used both for looking up what actors to notify of the events, and for providing default implementations for those listeners that are not interested in all events.
Normally, only the Core actor should mix in this class.
Backend implementations
See the extension registry.