Skip to main content

MPRIS2 D-Bus bridge for MPD

Project description

mpDris2

Build Release License: GPL-3.0-or-later Python MPD APT

mpDris2 provide MPRIS 2 support to mpd (Music Player Daemon).

mpDris2 runs in the user session and monitors a local or distant mpd server.

Contents

Features

  • Full MPRIS 2 interface (playback control, metadata, seek, volume) for any MPRIS client: playerctl, media keys, desktop applets.
  • Pure asyncio + dbus-fast: single event loop, no threads, no GLib.
  • Local or remote MPD, with automatic reconnect and capability probing.
  • A 7-step cover-art pipeline that resolves artwork for tagged files, CD (CUE/cdda) tracks and web-radio streams (see Cover art).
  • Optional remote cover sources: MusicBrainz/CAA, iTunes, Deezer, Radio Browser, myMPD WebradioDB.
  • systemd user unit with D-Bus activation: starts on the first MPRIS call.
  • Light footprint: only python-mpd2 and dbus-fast at runtime.

Install

From the Odio APT repository:

curl -fsSL https://apt.odio.love/key.gpg | sudo gpg --dearmor -o /usr/share/keyrings/odio.gpg
echo "deb [signed-by=/usr/share/keyrings/odio.gpg] https://apt.odio.love stable main" \
    | sudo tee /etc/apt/sources.list.d/odio.list
sudo apt update
sudo apt install mpdris2

The shipped systemd user unit is Type=dbus and a matching D-Bus service file is installed, so mpDris2 auto-starts on the first MPRIS call (playerctl, a media key, a desktop applet, …). Enable it explicitly only if you want it running before any client asks:

systemctl --user enable --now mpDris2.service

From PyPI

pipx install mpdris2     # or: pip install --user mpdris2

This installs the mpDris2 console script only — no systemd or D-Bus service files, unlike the .deb. To auto-start it, drop a user unit at ~/.config/systemd/user/mpDris2.service (copy data/user/mpDris2.service) and systemctl --user enable --now mpDris2.service. The optional cover deps are pipx install 'mpdris2[cover]'.

From git

git clone https://github.com/b0bbywan/mpDris2.git
cd mpDris2
pipx install .          # or: pip install --user .

This installs the mpDris2 console script into your $PATH. Start it via a systemctl --user unit.

Tagged releases on GitHub also publish an sdist tarball (mpdris2-X.Y.Z.tar.gz) next to the .deb, installable with pipx install ./mpdris2-X.Y.Z.tar.gz.

Runtime dependencies: python-mpd2 and dbus-fast. Python 3.11+.

Configuration

By default mpDris2 connects to localhost:6600. Environment variables $MPD_HOST and $MPD_PORT are honoured. To change anything else, copy the example file shipped at /usr/share/doc/mpdris2/mpDris2.conf to ~/.config/mpDris2/mpDris2.conf and edit.

Cover-art resolution needs music_dir to be set (or auto-detected over a local Unix socket connection to MPD). See Cover art below for the full pipeline.

Restart mpDris2 (pkill -HUP mpDris2, or just restart your session) to pick up config changes.

Note: the [Bling] mmkeys option from the historical mpDris2 is no longer supported. Modern desktops (GNOME, KDE, sway with mpris-ctrl, …) read MPRIS directly for multimedia-key handling, so mpDris2 doesn't need to grab the keys itself anymore.

Sample configuration

[Connection]
# Override host / port (or set $MPD_HOST / $MPD_PORT in the environment).
host = 192.168.1.5
port = 6600
password =

[Library]
# Required for cover-art resolution when MPD is remote (auto-detected
# over a local Unix socket connection).
music_dir = /media/music/
# Override the default cover-file regex; useful for non-standard names.
#cover_regex = ^(album|cover|\.?folder|front).*\.(gif|jpe?g|png|webp|bmp)$

[Cover]
# Remote cover-art sources (pipeline step 5), as an ordered, comma-separated
# list — the order is the lookup priority, and a source not listed is off.
# Valid: musicbrainz, itunes, deezer. Unset = none. MusicBrainz/CAA needs the
# [cover] extra (see below); iTunes/Deezer don't.
#sources = musicbrainz, itunes, deezer
# Web-radio stream cover sources (steps 6-7), same ordered-list rule. Valid:
# radiobrowser (station favicon), mympd (myMPD WebradioDB). Unset = none.
#stream_sources = mympd, radiobrowser
# Base URL of a myMPD instance — the data the 'mympd' stream source needs.
# Listing 'mympd' above without this is a no-op.
#mympd_uri = http://localhost:8080

[Bling]
# CD-like Previous: if elapsed >= 3 s, restart the current track instead
# of jumping to the previous one.
cdprev = False

mpDris2 does not raise its own desktop notifications: modern desktops (GNOME, KDE, sway with playerctld, …) surface track changes straight from the MPRIS metadata mpDris2 exports.

Architecture

mpDris2 is an asyncio + dbus-fast rewrite of the original PyGObject / dbus-python implementation: a single asyncio event loop replaces the GLib MainLoop + thread pool, dbus-fast replaces dbus-python, and mpd.asyncio from python-mpd2 replaces the blocking client.

Development

A top-level Makefile wraps the day-to-day commands so local dev and CI stay in sync (the GitHub workflow calls the same targets):

make lint           # ruff + mypy
make test           # pytest
make build          # python -m build (sdist + wheel)
make deb            # dpkg-buildpackage -b -us -uc (Debian toolchain)
make clean          # drop build/, dist/, *.egg-info
make version        # print the Python version (from __init__.py)
make sync-deb       # bump debian/changelog to match __init__.py

mpdris2/__init__.py is the single source of truth for the version; make sync-deb and make check-tag TAG=… keep debian/changelog and the git tag aligned with it.

Build a .deb

Build-deps (per debian/control): debhelper-compat (= 13), dh-python, python3, python3-setuptools. Then make deb on Debian trixie or a derivative produces the .deb. The runtime deps (python3-mpd, python3-dbus-fast) are resolved by APT at install time, not at build time.

Cover art

mpDris2 resolves mpris:artUrl through a fixed pipeline. The first step that yields a usable image wins; later steps are skipped.

# Step Source Exposed mpris:artUrl Wins when…
1 MPD readpicture Embedded picture in the audio file (FLAC PICTURE, ID3 APIC, …) file:///tmp/cover-*.{jpg,png,…} The track carries embedded art
2 FS regex scan cover_regex match in the song's directory (default matches cover.*, folder.*, album.*, front.*) file:// URI of the matched file (RFC-3986 percent-encoded) A non-standardly-named cover sits next to the audio file (local FS only)
3 MPD albumart cover.{png,jpg,jxl,webp} in the song's directory (resolved server-side by MPD) file:///tmp/cover-*.{jpg,png,…} Remote MPD, or step 2 missed (standard name only)
4 CUE/cdda fallback cover_regex match next to the loaded .cue playlist (FS scan), falling back to MPD albumart (which server-side resolves cover.{png,jpg,jxl,webp}) when music_dir isn't locally accessible file:// URI of the matched file (local FS) or file:///tmp/cover-* (remote MPD) The song is a CUE virtual track (cdda://, http://, …) and the CUE's own directory holds a cover
5 Remote cover URL MusicBrainz/CAA, iTunes, Deezer — whichever are listed in [Cover] sources, in that priority order. For web radio (title only) each source resolves the album within its own catalogue Remote image URL (image not downloaded) Earlier steps failed, a source is enabled and has cover art for the (artist, album)
6 Station favicon Community Radio Browser lookup of the stream URL (radiobrowser in [Cover] stream_sources) Station favicon URL (image not downloaded) An http(s):// web-radio stream whose station has a favicon
7 myMPD WebradioDB MYMPD_API_WEBRADIODB_RADIO_GET_BY_URI against the myMPD at [Cover] mympd_uri (mympd in [Cover] stream_sources) WebradioDB cover URL (image not downloaded) A web-radio stream that the configured myMPD's WebradioDB knows

Step 5's MusicBrainz/CAA lookup needs the optional [cover] extra (pip install '.[cover]', or the python3-musicbrainzngs package); artist/title validation is stdlib (difflib). The iTunes, Deezer and myMPD fallbacks are stdlib-only. Steps 5–7 return a remote URL used as mpris:artUrl — the MPRIS client fetches it, nothing is downloaded or cached to disk.

Used in

  • odio (odios): an open-source, self-hosted multi-protocol audio streamer that turns a Raspberry Pi or Debian box into a hi-fi network receiver. It bundles Bluetooth A2DP, AirPlay, Snapcast multi-room, UPnP/DLNA, Spotify Connect, automatic CD playback and web radio on top of MPD. Odio runs mpDris2 to expose its MPD over MPRIS2, including the cover pipeline that resolves artwork for CD and web-radio playback.

Contributing

Issues and pull requests are welcome on GitHub.

git clone https://github.com/b0bbywan/mpDris2.git
cd mpDris2
pip install -e '.[dev,cover]'   # dev tooling + optional cover deps
make lint                       # ruff + mypy
make test                       # pytest

Run make lint and make test before opening a PR, and keep commits focused (one logical change each). Module layout and responsibilities are documented in CLAUDE.md.

Filing a bug? Run the daemon with mpDris2 -v and paste the verbose log into the issue.

Credits

This is an asyncio + dbus-fast rewrite of the original mpDris2, whose authors are Erik Karlsson, Jean-Philippe Braun, Christoph Reiter and Mantas Mikulėnas. The Debian packaging traces back to Simon McVittie.

License

mpDris2 is licensed under the GNU General Public License v3.0 or later (GPL-3.0-or-later). See COPYING for the full text.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

mpdris2-0.11.1.tar.gz (77.1 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

mpdris2-0.11.1-py3-none-any.whl (54.4 kB view details)

Uploaded Python 3

File details

Details for the file mpdris2-0.11.1.tar.gz.

File metadata

  • Download URL: mpdris2-0.11.1.tar.gz
  • Upload date:
  • Size: 77.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for mpdris2-0.11.1.tar.gz
Algorithm Hash digest
SHA256 de9cb0412ff2eb98c8b5fedd6e35e9fef9aec0acd240c9e2bae399aa64839a17
MD5 febb38836a04b2561ba071052b193362
BLAKE2b-256 f22d35d0a0f50496a0b138159343eb6693e112a3c21ca5cac07905997a7867ae

See more details on using hashes here.

Provenance

The following attestation bundles were made for mpdris2-0.11.1.tar.gz:

Publisher: build.yml on b0bbywan/mpDris2

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file mpdris2-0.11.1-py3-none-any.whl.

File metadata

  • Download URL: mpdris2-0.11.1-py3-none-any.whl
  • Upload date:
  • Size: 54.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for mpdris2-0.11.1-py3-none-any.whl
Algorithm Hash digest
SHA256 42714097c1361cf8fd62664f74db977dfed61c753ddada47eed15793b6ac6e4f
MD5 4afb1ab93b5b6bececb2758cb89b4148
BLAKE2b-256 1ffb859ade251d5c85c0d79b311063aaaa6488f23340b5d57f420a43658422db

See more details on using hashes here.

Provenance

The following attestation bundles were made for mpdris2-0.11.1-py3-none-any.whl:

Publisher: build.yml on b0bbywan/mpDris2

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page