(MPRIS: Media Player Remote Interfacing Specification)
This project is a wrapper around the Snapclient binary and the Snapserver RPC API. It exposes the current playing state on the desktop bus (D-BUS), and allows control through the DBUS play/pause/stop signals.
When the main script (snapcastmpris.py
) is executed, it will hook into the communication
busses and set up and endless loop. An instance of SnapcastWrapper
, which lives
in a separate thread, is created.
SnapcastWrapper
has methods to start/pause/play snapcast audio, and keeps track of the snapclient
process. It gets help from SnapcastRpcWrapper
, which communicates with snapserver, the snapcast server.
This communication with the Snapserver RPC API
is used to control the snapcast audio level, mute status, client name. The SnapcastRpcWebsocketWrapper
is used to receive
events from the Snapserver RPC API. It receives
information about the stream that is currently playing, such as the playing state and the name, and passes this on to
the SnapcastWrapper
. Volume level changes, muting of a client, client connects, disconnects, ... is handled by this
websocket wrapper as well. SnapcastMPRISInterface
is used to communicate through the DBUS, to receive play/pause/stop signals from the OS and to relay information about the current state back to the OS.
SnapcastWrapper runs in a separate thread from the main script.
SnapcastWrapper implements the SnapcastRpcListener class and methods, which are called by SnapcastRpcWebsocketWrapper.
When ALSA <=> Snapclient volume synchronisation is enabled, SnapcastWrapper will monitor the ALSA volume level and send
any changes to Snapserver through SnapcastRPCWrapper. Volume synchronisation is disabled by default and enabled through the --sync_alsa_volume
or -s
flag or via the configuration with sync-alsa-volume = 1
. It requires the alsaaudio
library to be installed. Also the ALS mixer can be provided through the -m
or --mixer
flag or via the configuration with alsa-mixer = Softvol
.
When playing audio
- A Snapclient process is started
- Snapclient is unmuted through an RPC call
- Other audio players are stopped (through the HifiBerryOS pause-all script)
- DBUS status is updated
When pausing audio
- Snapclient is muted through an RPC call
- depending on the "pause" source, a flag in order to prevent snapclient to switch back to playing on the next stream update. It will only switch back to playing automatically when the snapcast source stream has been paused.
- Snapclient keeps running
- DBUS status is updated
When stopping audio
- The snapclient process is stopped
- DBUS information is updated
- SnapcastWrapper keeps running in order to act should a play signal come from DBUS or snapserver.
SnapcastRpcWrapper is a helper class to SnapcastWrapper, and provides access to the Snapserver RPC API. It can mute and unmute the client, set the client volume, change the client latency and change the client name. Through SnapcastRpcWrapper, the client information and server information can be obtained.
The advantage of muting is that snapserver can be configured not to send data to muted clients. This means that a muted client will reduce network traffic, compared to a running process with ignored audio.
SnapcastRpcWebsocketWrapper runs a websocket in a separate thread, and calls callback methods in SnapcastWrapper (a SnapcastRpcListener implementation) to act on stream and client status changes.
- When muted, the status is set to paused
- When unmuted, nothing is done. If a stream is already playing to the active device, and the player is in PAUSED mode, the player should switch to playing. This has not been implemented yet.
- When a stream switches from idle to playing, and the previous pause event was not caused by a DBUS event, SnapcastWrapper switches to the playing state.
- When a stream switches from playing to idle, the SnapcastWrapper pause logic is triggered to mute the client and switch to the PAUSED state.
- When the snapclient volume level is changed, and ALSA <=> Snapclient volume synchronisation is enabled, the ALSA volume is adjusted.