mpd2mpris provide MPRIS 2 support to mpd (Music Player Daemon).
Formerly named mpDris2. Renamed to mpd2mpris (in agreement with the original maintainer) to avoid confusion with the original mpDris2, which this project no longer shares code with after the asyncio rewrite.
mpd2mpris runs in the user session and monitors a local or distant mpd server.
- Features
- Install
- Configuration
- Architecture
- Development
- Build a .deb
- Cover art
- Used in
- Contributing
- Credits
- License
- 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-mpd2anddbus-fastat runtime.
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 mpd2mprisThe shipped systemd user unit is Type=dbus and a matching D-Bus
service file is installed, so mpd2mpris 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 mpd2mpris.servicepipx install mpd2mpris # or: pip install --user mpd2mprisThis installs the mpd2mpris 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/mpd2mpris.service (copy data/user/mpd2mpris.service)
and systemctl --user enable --now mpd2mpris.service. The optional cover
deps are pipx install 'mpd2mpris[cover]'.
git clone https://github.com/b0bbywan/mpd2mpris.git
cd mpd2mpris
pipx install . # or: pip install --user .This installs the mpd2mpris console script into your $PATH. Start it
via a systemctl --user unit.
Tagged releases on GitHub also publish an sdist tarball
(mpd2mpris-X.Y.Z.tar.gz) next to the .deb, installable with
pipx install ./mpd2mpris-X.Y.Z.tar.gz.
Runtime dependencies: python-mpd2 and dbus-fast. Python 3.11+.
By default mpd2mpris 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/mpd2mpris/mpd2mpris.conf to
~/.config/mpd2mpris/mpd2mpris.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 mpd2mpris (pkill -HUP mpd2mpris, or just restart your session) to
pick up config changes.
Note: the
[Bling] mmkeysoption from the historical mpDris2 is no longer supported. Modern desktops (GNOME, KDE, sway withmpris-ctrl, …) read MPRIS directly for multimedia-key handling, so mpd2mpris doesn't need to grab the keys itself anymore.
[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 = Falsempd2mpris does not raise its own desktop notifications: modern desktops
(GNOME, KDE, sway with playerctld, …) surface track changes straight
from the MPRIS metadata mpd2mpris exports.
mpd2mpris 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.
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__.pympd2mpris/__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-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.
mpd2mpris 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.
- 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 mpd2mpris to expose its MPD over MPRIS2, including the cover pipeline that resolves artwork for CD and web-radio playback.
Issues and pull requests are welcome on GitHub.
git clone https://github.com/b0bbywan/mpd2mpris.git
cd mpd2mpris
pip install -e '.[dev,cover]' # dev tooling + optional cover deps
make lint # ruff + mypy
make test # pytestRun 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 mpd2mpris -v and paste the verbose log
into the issue.
mpd2mpris (formerly mpDris2) 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.
mpd2mpris is licensed under the GNU General Public License v3.0 or later (GPL-3.0-or-later). See COPYING for the full text.