From 61bb873e76437d99bc4d5d325dbe0bb4977b32fd Mon Sep 17 00:00:00 2001 From: wisp3rwind <17089248+wisp3rwind@users.noreply.github.com> Date: Tue, 1 Oct 2024 11:36:41 +0200 Subject: [PATCH] add initial MPRIS support using zbus - following the spec at https://specifications.freedesktop.org/mpris-spec/latest/ - some properties/commands are not fully supported, yet --- Cargo.lock | 443 +++++++++++++- Cargo.toml | 7 +- src/main.rs | 20 + src/mpris_event_handler.rs | 1192 ++++++++++++++++++++++++++++++++++++ 4 files changed, 1650 insertions(+), 12 deletions(-) create mode 100644 src/mpris_event_handler.rs diff --git a/Cargo.lock b/Cargo.lock index 0a97b0f87..1607c24de 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -135,6 +135,114 @@ version = "0.7.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50" +[[package]] +name = "async-broadcast" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20cd0e2e25ea8e5f7e9df04578dc6cf5c83577fd09b1a46aaf5c85e1c33f2a7e" +dependencies = [ + "event-listener", + "event-listener-strategy", + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "async-channel" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "89b47800b0be77592da0afd425cc03468052844aff33b84e33cc696f64e77b6a" +dependencies = [ + "concurrent-queue", + "event-listener-strategy", + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "async-io" +version = "2.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "444b0228950ee6501b3568d3c93bf1176a1fdbc3b758dcd9475046d30f4dc7e8" +dependencies = [ + "async-lock", + "cfg-if", + "concurrent-queue", + "futures-io", + "futures-lite", + "parking", + "polling", + "rustix", + "slab", + "tracing", + "windows-sys 0.59.0", +] + +[[package]] +name = "async-lock" +version = "3.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff6e472cdea888a4bd64f342f09b3f50e1886d32afe8df3d663c01140b811b18" +dependencies = [ + "event-listener", + "event-listener-strategy", + "pin-project-lite", +] + +[[package]] +name = "async-process" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "63255f1dc2381611000436537bbedfe83183faa303a5a0edaf191edef06526bb" +dependencies = [ + "async-channel", + "async-io", + "async-lock", + "async-signal", + "async-task", + "blocking", + "cfg-if", + "event-listener", + "futures-lite", + "rustix", + "tracing", +] + +[[package]] +name = "async-recursion" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b43422f69d8ff38f95f1b2bb76517c91589a924d1559a0e935d7c8ce0274c11" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.79", +] + +[[package]] +name = "async-signal" +version = "0.2.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "637e00349800c0bdf8bfc21ebbc0b6524abea702b0da4168ac00d070d0c0b9f3" +dependencies = [ + "async-io", + "async-lock", + "atomic-waker", + "cfg-if", + "futures-core", + "futures-io", + "rustix", + "signal-hook-registry", + "slab", + "windows-sys 0.59.0", +] + +[[package]] +name = "async-task" +version = "4.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b75356056920673b02621b35afd0f7dda9306d03c79a30f5c56c44cf256e3de" + [[package]] name = "async-trait" version = "0.1.83" @@ -292,6 +400,19 @@ dependencies = [ "generic-array", ] +[[package]] +name = "blocking" +version = "1.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "703f41c54fc768e63e091340b424302bb1c29ef4aa0c7f10fe849dfb114d29ea" +dependencies = [ + "async-channel", + "async-task", + "futures-io", + "futures-lite", + "piper", +] + [[package]] name = "bumpalo" version = "3.16.0" @@ -358,6 +479,12 @@ version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" +[[package]] +name = "cfg_aliases" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724" + [[package]] name = "chrono" version = "0.4.38" @@ -419,6 +546,15 @@ dependencies = [ "memchr", ] +[[package]] +name = "concurrent-queue" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ca0197aee26d1ae37445ee532fefce43251d24cc7c166799f4d46817f1d3973" +dependencies = [ + "crossbeam-utils", +] + [[package]] name = "const-oid" version = "0.9.6" @@ -494,6 +630,12 @@ dependencies = [ "libc", ] +[[package]] +name = "crossbeam-utils" +version = "0.8.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80" + [[package]] name = "crypto-common" version = "0.1.6" @@ -654,6 +796,33 @@ dependencies = [ "cfg-if", ] +[[package]] +name = "endi" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a3d8a32ae18130a3c84dd492d4215c3d913c3b07c6b63c2eb3eb7ff1101ab7bf" + +[[package]] +name = "enumflags2" +version = "0.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d232db7f5956f3f14313dc2f87985c58bd2c695ce124c8cdd984e08e15ac133d" +dependencies = [ + "enumflags2_derive", + "serde", +] + +[[package]] +name = "enumflags2_derive" +version = "0.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "de0d48a183585823424a4ce1aa132d174a6a81bd540895822eb4c8373a8e49e8" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.79", +] + [[package]] name = "env_filter" version = "0.1.2" @@ -692,6 +861,27 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "event-listener" +version = "5.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6032be9bd27023a771701cc49f9f053c751055f71efb2e0ae5c15809093675ba" +dependencies = [ + "concurrent-queue", + "parking", + "pin-project-lite", +] + +[[package]] +name = "event-listener-strategy" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0f214dc438f977e6d4e3500aaa277f5ad94ca83fbbd9b1a15713ce2344ccc5a1" +dependencies = [ + "event-listener", + "pin-project-lite", +] + [[package]] name = "fastrand" version = "2.1.1" @@ -773,6 +963,19 @@ version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" +[[package]] +name = "futures-lite" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "52527eb5074e35e9339c6b4e8d12600c7128b68fb25dcb9fa9dec18f7c25f3a5" +dependencies = [ + "fastrand", + "futures-core", + "futures-io", + "parking", + "pin-project-lite", +] + [[package]] name = "futures-macro" version = "0.3.30" @@ -1152,6 +1355,12 @@ version = "0.3.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024" +[[package]] +name = "hermit-abi" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fbf6a919d6cf397374f7dfeeea91d974c7c0a7221d0d0f4f20d859d329e53fcc" + [[package]] name = "hex" version = "0.4.3" @@ -1245,9 +1454,9 @@ dependencies = [ [[package]] name = "httparse" -version = "1.9.4" +version = "1.9.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0fcc0b4a115bf80b728eb8ea024ad5bd707b615bfed49e0665b6e0f86fd082d9" +checksum = "7d71d3574edd2771538b901e6549113b4006ece66150fb69c0fb6d9a2adae946" [[package]] name = "httpdate" @@ -1702,8 +1911,11 @@ dependencies = [ "sha1", "sysinfo", "thiserror", + "time", "tokio", "url", + "zbus", + "zvariant", ] [[package]] @@ -1930,6 +2142,15 @@ version = "2.7.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3" +[[package]] +name = "memoffset" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "488016bfae457b036d996092f6cb448677611ce4449e970ceaf42695203f218a" +dependencies = [ + "autocfg", +] + [[package]] name = "mime" version = "0.3.17" @@ -1957,7 +2178,7 @@ version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "80e04d1dcff3aae0704555fe5fee3bcfaf3d1fdf8a7e521d5b9d2b42acb52cec" dependencies = [ - "hermit-abi", + "hermit-abi 0.3.9", "libc", "wasi", "windows-sys 0.52.0", @@ -2013,6 +2234,19 @@ dependencies = [ "jni-sys", ] +[[package]] +name = "nix" +version = "0.29.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "71e2746dc3a24dd78b3cfcb7be93368c6de9963d30f43a6a73998a9cf4b17b46" +dependencies = [ + "bitflags 2.6.0", + "cfg-if", + "cfg_aliases", + "libc", + "memoffset", +] + [[package]] name = "no-std-compat" version = "0.4.1" @@ -2255,6 +2489,22 @@ dependencies = [ "paste", ] +[[package]] +name = "ordered-stream" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9aa2b01e1d916879f73a53d01d1d6cee68adbb31d6d9177a8cfce093cced1d50" +dependencies = [ + "futures-core", + "pin-project-lite", +] + +[[package]] +name = "parking" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f38d5652c16fde515bb1ecef450ab0f6a219d619a7274976324d5e377f7dceba" + [[package]] name = "parking_lot" version = "0.12.3" @@ -2334,6 +2584,17 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" +[[package]] +name = "piper" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96c8c490f422ef9a4efd2cb5b42b76c8613d7e7dfc1caf667b8a3350a5acc066" +dependencies = [ + "atomic-waker", + "fastrand", + "futures-io", +] + [[package]] name = "pkcs1" version = "0.7.5" @@ -2361,6 +2622,21 @@ version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2" +[[package]] +name = "polling" +version = "3.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc2790cd301dec6cd3b7a025e4815cf825724a51c98dccfe6a3e55f05ffb6511" +dependencies = [ + "cfg-if", + "concurrent-queue", + "hermit-abi 0.4.0", + "pin-project-lite", + "rustix", + "tracing", + "windows-sys 0.59.0", +] + [[package]] name = "portable-atomic" version = "1.9.0" @@ -2444,9 +2720,9 @@ dependencies = [ [[package]] name = "protobuf" -version = "3.5.1" +version = "3.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0bcc343da15609eaecd65f8aa76df8dc4209d325131d8219358c0aaaebab0bf6" +checksum = "3018844a02746180074f621e847703737d27d89d7f0721a7a4da317f88b16385" dependencies = [ "once_cell", "protobuf-support", @@ -2455,9 +2731,9 @@ dependencies = [ [[package]] name = "protobuf-codegen" -version = "3.5.1" +version = "3.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c4d0cde5642ea4df842b13eb9f59ea6fafa26dcb43e3e1ee49120e9757556189" +checksum = "411c15a212b4de05eb8bc989fd066a74c86bd3c04e27d6e86bd7703b806d7734" dependencies = [ "anyhow", "once_cell", @@ -2470,9 +2746,9 @@ dependencies = [ [[package]] name = "protobuf-parse" -version = "3.5.1" +version = "3.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1b0e9b447d099ae2c4993c0cbb03c7a9d6c937b17f2d56cfc0b1550e6fcfdb76" +checksum = "06f45f16b522d92336e839b5e40680095a045e36a1e7f742ba682ddc85236772" dependencies = [ "anyhow", "indexmap", @@ -2486,9 +2762,9 @@ dependencies = [ [[package]] name = "protobuf-support" -version = "3.5.1" +version = "3.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f0766e3675a627c327e4b3964582594b0e8741305d628a98a5de75a1d15f99b9" +checksum = "faf96d872914fcda2b66d66ea3fff2be7c66865d31c7bb2790cff32c0e714880" dependencies = [ "thiserror", ] @@ -2948,6 +3224,17 @@ dependencies = [ "serde", ] +[[package]] +name = "serde_repr" +version = "0.1.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c64451ba24fc7a6a2d60fc75dd9c83c90903b19028d4eff35e88fc1e86564e9" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.79", +] + [[package]] name = "serde_spanned" version = "0.6.8" @@ -3081,6 +3368,12 @@ dependencies = [ "der", ] +[[package]] +name = "static_assertions" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" + [[package]] name = "strsim" version = "0.11.1" @@ -3364,6 +3657,7 @@ dependencies = [ "signal-hook-registry", "socket2", "tokio-macros", + "tracing", "windows-sys 0.52.0", ] @@ -3497,9 +3791,21 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c3523ab5a71916ccf420eebdf5521fcef02141234bbc0b8a49f2fdc4544364ef" dependencies = [ "pin-project-lite", + "tracing-attributes", "tracing-core", ] +[[package]] +name = "tracing-attributes" +version = "0.1.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.79", +] + [[package]] name = "tracing-core" version = "0.1.32" @@ -3541,6 +3847,17 @@ version = "1.17.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" +[[package]] +name = "uds_windows" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "89daebc3e6fd160ac4aa9fc8b3bf71e1f74fbf92367ae71fb83a037e8bf164b9" +dependencies = [ + "memoffset", + "tempfile", + "winapi", +] + [[package]] name = "unicode-bidi" version = "0.3.15" @@ -4150,6 +4467,73 @@ dependencies = [ "windows-sys 0.48.0", ] +[[package]] +name = "xdg-home" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec1cdab258fb55c0da61328dc52c8764709b249011b2cad0454c72f0bf10a1f6" +dependencies = [ + "libc", + "windows-sys 0.59.0", +] + +[[package]] +name = "zbus" +version = "4.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bb97012beadd29e654708a0fdb4c84bc046f537aecfde2c3ee0a9e4b4d48c725" +dependencies = [ + "async-broadcast", + "async-process", + "async-recursion", + "async-trait", + "enumflags2", + "event-listener", + "futures-core", + "futures-sink", + "futures-util", + "hex", + "nix", + "ordered-stream", + "rand", + "serde", + "serde_repr", + "sha1", + "static_assertions", + "tokio", + "tracing", + "uds_windows", + "windows-sys 0.52.0", + "xdg-home", + "zbus_macros", + "zbus_names", + "zvariant", +] + +[[package]] +name = "zbus_macros" +version = "4.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "267db9407081e90bbfa46d841d3cbc60f59c0351838c4bc65199ecd79ab1983e" +dependencies = [ + "proc-macro-crate", + "proc-macro2", + "quote", + "syn 2.0.79", + "zvariant_utils", +] + +[[package]] +name = "zbus_names" +version = "3.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4b9b1fef7d021261cc16cba64c351d291b715febe0fa10dc3a443ac5a5022e6c" +dependencies = [ + "serde", + "static_assertions", + "zvariant", +] + [[package]] name = "zerocopy" version = "0.7.35" @@ -4176,3 +4560,40 @@ name = "zeroize" version = "1.8.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde" + +[[package]] +name = "zvariant" +version = "4.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2084290ab9a1c471c38fc524945837734fbf124487e105daec2bb57fd48c81fe" +dependencies = [ + "endi", + "enumflags2", + "serde", + "static_assertions", + "zvariant_derive", +] + +[[package]] +name = "zvariant_derive" +version = "4.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73e2ba546bda683a90652bac4a279bc146adad1386f25379cf73200d2002c449" +dependencies = [ + "proc-macro-crate", + "proc-macro2", + "quote", + "syn 2.0.79", + "zvariant_utils", +] + +[[package]] +name = "zvariant_utils" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c51bcff7cc3dbb5055396bcf774748c3dab426b4b8659046963523cee4808340" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.79", +] diff --git a/Cargo.toml b/Cargo.toml index d23be2e10..b632b0299 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -62,8 +62,11 @@ log = "0.4" sha1 = "0.10" sysinfo = { version = "0.31.3", default-features = false, features = ["system"] } thiserror = "1.0" +time = { version = "0.3", features = ["formatting"] } tokio = { version = "1", features = ["rt", "macros", "signal", "sync", "parking_lot", "process"] } url = "2.2" +zbus = { version = "4", default-features = false, features = ["tokio"], optional = true } +zvariant = { version = "4", default-features = false, optional = true } [features] alsa-backend = ["librespot-playback/alsa-backend"] @@ -79,7 +82,9 @@ with-dns-sd = ["librespot-core/with-dns-sd", "librespot-discovery/with-dns-sd"] passthrough-decoder = ["librespot-playback/passthrough-decoder"] -default = ["rodio-backend"] +with-mpris = ["dep:zbus", "dep:zvariant"] + +default = ["rodio-backend", "with-mpris"] [package.metadata.deb] maintainer = "librespot-org" diff --git a/src/main.rs b/src/main.rs index 2735ddf71..afa8f7fd9 100644 --- a/src/main.rs +++ b/src/main.rs @@ -39,6 +39,11 @@ use librespot::playback::mixer::alsamixer::AlsaMixer; mod player_event_handler; use player_event_handler::{run_program_on_sink_events, EventHandler}; +#[cfg(feature = "with-mpris")] +mod mpris_event_handler; +#[cfg(feature = "with-mpris")] +use mpris_event_handler::MprisEventHandler; + fn device_id(name: &str) -> String { HEXLOWER.encode(&Sha1::digest(name.as_bytes())) } @@ -1853,6 +1858,14 @@ async fn main() { } } + #[cfg(feature = "with-mpris")] + let mpris = MprisEventHandler::spawn(player.clone()) + .await + .unwrap_or_else(|e| { + error!("could not initialize MPRIS: {}", e); + exit(1); + }); + loop { tokio::select! { credentials = async { @@ -1906,6 +1919,10 @@ async fn main() { exit(1); } }; + + #[cfg(feature = "with-mpris")] + mpris.set_spirc(spirc_.clone()); + spirc = Some(spirc_); spirc_task = Some(Box::pin(spirc_task_)); @@ -1949,6 +1966,9 @@ async fn main() { info!("Gracefully shutting down"); + #[cfg(feature = "with-mpris")] + mpris.quit_and_join().await; + // Shutdown spirc if necessary if let Some(spirc) = spirc { if let Err(e) = spirc.shutdown() { diff --git a/src/mpris_event_handler.rs b/src/mpris_event_handler.rs new file mode 100644 index 000000000..6750ce6db --- /dev/null +++ b/src/mpris_event_handler.rs @@ -0,0 +1,1192 @@ +#![cfg(feature = "with-mpris")] + +use std::{ + collections::HashMap, + sync::Arc, +}; + +use librespot_connect::spirc::Spirc; +use log::{debug, warn}; +use thiserror::Error; +use time::format_description::well_known::Iso8601; +use tokio::sync::mpsc; +use zbus::connection; + +use librespot::{ + core::Error, + metadata::audio::UniqueFields, + playback::player::{Player, PlayerEvent}, +}; + +/// A playback state. +#[derive(Clone, Copy, Debug)] +enum PlaybackStatus { + /// A track is currently playing. + Playing, + + /// A track is currently paused. + Paused, + + /// There is no track currently playing. + Stopped +} + +impl zvariant::Type for PlaybackStatus { + fn signature() -> zvariant::Signature<'static> { + zvariant::Signature::try_from("s").unwrap() + } +} + +impl TryFrom> for PlaybackStatus { + type Error = zvariant::Error; + + fn try_from(value: zvariant::Value<'_>) -> Result { + if let zvariant::Value::Str(s) = value { + match s.as_str() { + "Playing" => Ok(Self::Playing), + "Paused" => Ok(Self::Paused), + "Stopped" => Ok(Self::Stopped), + _ => Err(zvariant::Error::Message("invalid enum value".to_owned())), + } + } else { + Err(zvariant::Error::IncorrectType) + } + } +} + +impl From for zvariant::Value<'_> { + fn from(value: PlaybackStatus) -> Self { + let s = match value { + PlaybackStatus::Playing => "Playing", + PlaybackStatus::Paused => "Paused", + PlaybackStatus::Stopped => "Stopped", + }; + + s.into() + } +} + +/// A repeat / loop status +#[derive(Clone, Copy, Debug)] +enum LoopStatus { + /// The playback will stop when there are no more tracks to play + None, + + /// The current track will start again from the begining once it has finished playing + Track, + + /// The playback loops through a list of tracks + Playlist +} + +impl zvariant::Type for LoopStatus { + fn signature() -> zvariant::Signature<'static> { + zvariant::Signature::try_from("s").unwrap() + } +} + +impl TryFrom> for LoopStatus { + type Error = zvariant::Error; + + fn try_from(value: zvariant::Value<'_>) -> Result { + if let zvariant::Value::Str(s) = value { + match s.as_str() { + "None" => Ok(Self::None), + "Track" => Ok(Self::Track), + "Playlist" => Ok(Self::Playlist), + _ => Err(zvariant::Error::Message("invalid enum value".to_owned())), + } + } else { + Err(zvariant::Error::IncorrectType) + } + } +} + +impl From for zvariant::Value<'_> { + fn from(value: LoopStatus) -> Self { + let s = match value { + LoopStatus::None => "None", + LoopStatus::Track => "Track", + LoopStatus::Playlist => "Playlist", + }; + + s.into() + } +} + + +/// Unique track identifier. +/// +/// If the media player implements the TrackList interface and allows +/// the same track to appear multiple times in the tracklist, +/// this must be unique within the scope of the tracklist. +/// +/// Note that this should be a valid D-Bus object id, although clients +/// should not assume that any object is actually exported with any +/// interfaces at that path. +/// +/// Media players may not use any paths starting with +/// `/org/mpris` unless explicitly allowed by this specification. +/// Such paths are intended to have special meaning, such as +/// `/org/mpris/MediaPlayer2/TrackList/NoTrack` +/// to indicate "no track". +/// +/// This is a D-Bus object id as that is the definitive way to have +/// unique identifiers on D-Bus. It also allows for future optional +/// expansions to the specification where tracks are exported to D-Bus +/// with an interface similar to org.gnome.UPnP.MediaItem2. +// type TrackId = ...; + +// A playback rate +// +// This is a multiplier, so a value of 0.5 indicates that playback is +// happening at half speed, while 1.5 means that 1.5 seconds of "track time" +// is consumed every second. +type PlaybackRate = f64; + +// Audio volume level +// +// - 0.0 means mute. +// - 1.0 is a sensible maximum volume level (ex: 0dB). +// +// Note that the volume may be higher than 1.0, although generally +// clients should not attempt to set it above 1.0. +type Volume = f64; + +// Time in microseconds. +type TimeInUs = i64; + +struct MprisService { } + +#[zbus::interface(name = "org.mpris.MediaPlayer2")] +impl MprisService { + // Brings the media player's user interface to the front using any appropriate mechanism + // available. + // + // The media player may be unable to control how its user interface is displayed, or it may not + // have a graphical user interface at all. In this case, the `CanRaise` property is `false` and + // this method does nothing. + async fn raise(&self) { + debug!("org.mpris.MediaPlayer2::Raise"); + } + + // Causes the media player to stop running. + // + // The media player may refuse to allow clients to shut it down. In this case, the `CanQuit` + // property is `false` and this method does nothing. + // + // Note: Media players which can be D-Bus activated, or for which there is no sensibly easy way + // to terminate a running instance (via the main interface or a notification area icon for + // example) should allow clients to use this method. Otherwise, it should not be needed. + // + // If the media player does not have a UI, this should be implemented. + async fn quit(&self) { + debug!("org.mpris.MediaPlayer2::Quit"); + } + + // If `false`, calling `Quit` will have no effect, and may raise a `NotSupported` error. If + // `true`, calling `Quit` will cause the media application to attempt to quit (although it may + // still be prevented from quitting by the user, for example). + #[zbus(property)] + async fn can_quit(&self) -> bool { + debug!("org.mpris.MediaPlayer2::CanQuit"); + false + } + + // Whether the media player is occupying the fullscreen. + // + // This is typically used for videos. A value of `true` indicates that the media player is + // taking up the full screen. + // + // Media centre software may well have this value fixed to `true` + // + // If `CanSetFullscreen` is `true`, clients may set this property to `true` to tell the media + // player to enter fullscreen mode, or to `false` to return to windowed mode. + // + // If `CanSetFullscreen` is `false`, then attempting to set this property should have no + // effect, and may raise an error. However, even if it is `true`, the media player may still + // be unable to fulfil the request, in which case attempting to set this property will have no + // effect (but should not raise an error). + // + // Rationale: + // + // This allows remote control interfaces, such as LIRC or mobile devices like + // phones, to control whether a video is shown in fullscreen. + #[zbus(property)] + async fn fullscreen(&self) -> bool { + debug!("org.mpris.MediaPlayer2::Fullscreen"); + false + } + + #[zbus(property)] + async fn set_fullscreen(&self, _value: bool) { + debug!("org.mpris.MediaPlayer2::SetFullscreen"); + } + + // If `false`, attempting to set `Fullscreen` will have no effect, and may raise an error. If + // `true`, attempting to set `Fullscreen` will not raise an error, and (if it is different from + // the current value) will cause the media player to attempt to enter or exit fullscreen mode. + // + // Note that the media player may be unable to fulfil the request. In this case, the value will + // not change. If the media player knows in advance that it will not be able to fulfil the + // request, however, this property should be `false`. + // + // Rationale: + // + // This allows clients to choose whether to display controls for entering + // or exiting fullscreen mode. + #[zbus(property)] + async fn can_set_fullscreen(&self) -> bool { + debug!("org.mpris.MediaPlayer2::CanSetFullscreen"); + false + } + + // If `false`, calling `Raise` will have no effect, and may raise a NotSupported error. If + // `true`, calling `Raise` will cause the media application to attempt to bring its user + // interface to the front, although it may be prevented from doing so (by the window manager, + // for example). + #[zbus(property)] + async fn can_raise(&self) -> bool { + debug!("org.mpris.MediaPlayer2::CanRaise"); + false + } + + // Indicates whether the `/org/mpris/MediaPlayer2` object implements the + // `org.mpris.MediaPlayer2.TrackList` interface. + #[zbus(property)] + async fn has_track_list(&self) -> bool { + debug!("org.mpris.MediaPlayer2::HasTrackList"); + // TODO: Eventually implement + false + } + + // A friendly name to identify the media player to users. This should usually match the name + // found in .desktop files (eg: "VLC media player"). + #[zbus(property)] + async fn identity(&self) -> String { + debug!("org.mpris.MediaPlayer2::Identity"); + // TOOD: use name from config + "Librespot".to_owned() + } + + // The basename of an installed .desktop file which complies with the + // [Desktop entry specification](http://standards.freedesktop.org/desktop-entry-spec/latest/) + // with the `.desktop` extension stripped. + // + // Example: The desktop entry file is "/usr/share/applications/vlc.desktop", and this property + // contains "vlc" + // + #[zbus(property)] + async fn desktop_entry(&self) -> String { + debug!("org.mpris.MediaPlayer2::DesktopEntry"); + // FIXME: The spec doesn't say anything about the case when there is no .desktop. + // Is there any convention? Any value that common clients handle in a sane way? + "".to_owned() + } + + // The URI schemes supported by the media player. + // + // This can be viewed as protocols supported by the player in almost all cases. Almost every + // media player will include support for the `"file"` scheme. Other common schemes are + // `"http"` and `"rtsp"`. + // + // Note that URI schemes should be lower-case. + // + // Rationale: + // + // This is important for clients to know when using the editing + // capabilities of the Playlist interface, for example. + #[zbus(property)] + async fn supported_uri_schemes(&self) -> Vec { + debug!("org.mpris.MediaPlayer2::SupportedUriSchemes"); + vec![] + } + + // The mime-types supported by the media player. + // + // Mime-types should be in the standard format (eg: `audio/mpeg` or `application/ogg`). + // + // Rationale: + // + // This is important for clients to know when using the editing + // capabilities of the Playlist interface, for example. + #[zbus(property)] + async fn supported_mime_types(&self) -> Vec { + debug!("org.mpris.MediaPlayer2::SupportedMimeTypes"); + vec![] + } +} + +struct MprisPlayerService { + spirc: Option, + repeat: bool, + shuffle: bool, + playback_status: PlaybackStatus, + volume: f64, + metadata: HashMap, +} + +// This interface implements the methods for querying and providing basic +// control over what is currently playing. +#[zbus::interface(name = "org.mpris.MediaPlayer2.Player")] +impl MprisPlayerService { + /// Skips to the next track in the tracklist. If there is no next track (and endless playback + /// and track repeat are both off), stop playback. + /// + /// If playback is paused or stopped, it remains that way. + /// + /// If self.can_go_next is `false`, attempting to call this method should have no effect. + async fn next(&self) { + if let Some(spirc) = &self.spirc { + let _ = spirc.next(); + } + } + + // Skips to the previous track in the tracklist. + // + // If there is no previous track (and endless playback and track repeat are both off), stop + // playback. + // + // If playback is paused or stopped, it remains that way. + // + // If `self.can_go_previous` is `false`, attempting to call this method should have no effect. + async fn previous(&self) { + if let Some(spirc) = &self.spirc { + let _ = spirc.prev(); + } + } + + // Pauses playback. + // + // If playback is already paused, this has no effect. + // + // Calling Play after this should cause playback to start again from the same position. + // + // If `self.can_pause` is `false`, attempting to call this method should have no effect. + async fn pause(&self) { + if let Some(spirc) = &self.spirc { + let _ = spirc.pause(); + } + } + + // Pauses playback. + // + // If playback is already paused, resumes playback. + // + // If playback is stopped, starts playback. + // + // If `self.can_pause` is `false`, attempting to call this method should have no effect and + // raise an error. + async fn play_pause(&self) { + // ignore for now + // TODO: implement + } + + // Stops playback. + // + // If playback is already stopped, this has no effect. + // + // Calling Play after this should cause playback to start again from the beginning of the + // track. + // + // If `CanControl` is `false`, attempting to call this method should have no effect and raise + // an error. + async fn stop(&self) { + if let Some(spirc) = &self.spirc { + let _ = spirc.pause(); + let _ = spirc.set_position_ms(0); + } + } + + // Starts or resumes playback. + // + // If already playing, this has no effect. + // + // If paused, playback resumes from the current position. + // + // If there is no track to play, this has no effect. + // + // If `self.can_play` is `false`, attempting to call this method should have no effect. + async fn play(&self) { + if let Some(spirc) = &self.spirc { + let _ = spirc.activate(); + let _ = spirc.play(); + } + } + + // Seeks forward in the current track by the specified number of microseconds. + // + // A negative value seeks back. If this would mean seeking back further than the start of the + // track, the position is set to 0. + // + // If the value passed in would mean seeking beyond the end of the track, acts like a call to + // Next. + // + // If the `self.can_seek` property is `false`, this has no effect. + // + // Arguments: + // + // * `offset`: The number of microseconds to seek forward. + async fn seek(&self, offset: TimeInUs) { + if let Some(spirc) = &self.spirc { + let _ = spirc.seek_offset((offset / 1000) as i32); + } + } + + // Sets the current track position in microseconds. + // + // If the Position argument is less than 0, do nothing. + // + // If the Position argument is greater than the track length, do nothing. + // + // If the `CanSeek` property is `false`, this has no effect. + // + // Rationale: + // + // The reason for having this method, rather than making `self.position` writable, is to + // include the `track_id` argument to avoid race conditions where a client tries to seek to + // a position when the track has already changed. + // + // Arguments: + // + // * `track_id`: The currently playing track's identifier. + // If this does not match the id of the currently-playing track, the call is + // ignored as "stale". + // `/org/mpris/MediaPlayer2/TrackList/NoTrack` is _not_ a valid value for this + // argument. + // * `position`: Track position in microseconds. This must be between 0 and `track_length`. + async fn set_position( + &self, + _track_id: zbus::zvariant::ObjectPath<'_>, + position: TimeInUs, + ) { + // FIXME: handle track_id + if position < 0 { + return; + } + if let Some(spirc) = &self.spirc { + let _ = spirc.set_position_ms((position / 1000) as u32); + } + } + + // Opens the Uri given as an argument + // + // If the playback is stopped, starts playing + // + // If the uri scheme or the mime-type of the uri to open is not supported, this method does + // nothing and may raise an error. In particular, if the list of available uri schemes is + // empty, this method may not be implemented. + // + // Clients should not assume that the Uri has been opened as soon as this method returns. They + // should wait until the mpris:trackid field in the `Metadata` property changes. + // + // If the media player implements the TrackList interface, then the opened track should be made + // part of the tracklist, the `org.mpris.MediaPlayer2.TrackList.TrackAdded` or + // `org.mpris.MediaPlayer2.TrackList.TrackListReplaced` signal should be fired, as well as the + // `org.freedesktop.DBus.Properties.PropertiesChanged` signal on the tracklist interface. + // + // Arguments: + // + // * `uri`: Uri of the track to load. Its uri scheme should be an element of the + // `org.mpris.MediaPlayer2.SupportedUriSchemes` property and the mime-type should + // match one of the elements of the `org.mpris.MediaPlayer2.SupportedMimeTypes`. + async fn open_uri(&self, _uri: &str) -> zbus::fdo::Result<()> { + Err(zbus::fdo::Error::NotSupported("OpenUri not supported".to_owned())) + } + + // The current playback status. + // + // May be "Playing", "Paused" or "Stopped". + #[zbus(property(emits_changed_signal = "true"))] + async fn playback_status(&self) -> PlaybackStatus { + self.playback_status + } + + // The current loop / repeat status + // + // May be: + // - "None" if the playback will stop when there are no more tracks to play + // - "Track" if the current track will start again from the begining once it has finished playing + // - "Playlist" if the playback loops through a list of tracks + // + // If `self.can_control` is `false`, attempting to set this property should have no effect and + // raise an error. + // + #[zbus(property(emits_changed_signal = "true"))] + async fn loop_status(&self) -> LoopStatus { + if self.repeat { + // FIXME: How does Spotify handle single track repeat? + LoopStatus::Playlist + } else { + LoopStatus::None + } + } + + #[zbus(property)] + async fn set_loop_status(&mut self, value: LoopStatus) -> zbus::fdo::Result<()> { + // TODO: implement, notify change + match value { + LoopStatus::None => { + if let Some(spirc) = &self.spirc { + let _ = spirc.repeat(false); + } + } + LoopStatus::Track => { + return Err(zbus::fdo::Error::NotSupported("Player control not implemented".to_owned())); + } + LoopStatus::Playlist => { + if let Some(spirc) = &self.spirc { + let _ = spirc.repeat(true); + } + } + } + + Ok(()) + } + + // The current playback rate. + // + // The value must fall in the range described by `MinimumRate` and `MaximumRate`, and must not + // be 0.0. If playback is paused, the `PlaybackStatus` property should be used to indicate + // this. A value of 0.0 should not be set by the client. If it is, the media player should + // act as though `Pause` was called. + // + // If the media player has no ability to play at speeds other than the normal playback rate, + // this must still be implemented, and must return 1.0. The `MinimumRate` and `MaximumRate` + // properties must also be set to 1.0. + // + // Not all values may be accepted by the media player. It is left to media player + // implementations to decide how to deal with values they cannot use; they may either ignore + // them or pick a "best fit" value. Clients are recommended to only use sensible fractions or + // multiples of 1 (eg: 0.5, 0.25, 1.5, 2.0, etc). + // + // Rationale: + // + // This allows clients to display (reasonably) accurate progress bars + // without having to regularly query the media player for the current + // position. + #[zbus(property(emits_changed_signal = "true"))] + async fn rate(&self) -> PlaybackRate { + 1.0 + } + + #[zbus(property)] + async fn set_rate(&mut self, _value: PlaybackRate) { + // ignore + } + + + // A value of `false` indicates that playback is progressing linearly through a playlist, while + // `true` means playback is progressing through a playlist in some other order. + // + // If `CanControl` is `false`, attempting to set this property should have no effect and raise + // an error. + // + #[zbus(property(emits_changed_signal = "true"))] + async fn shuffle(&self) -> bool { + self.shuffle + } + + #[zbus(property)] + async fn set_shuffle(&mut self, value: bool) { + if let Some(spirc) = &self.spirc { + let _ = spirc.shuffle(value); + } + } + + // The metadata of the current element. + // + // If there is a current track, this must have a "mpris:trackid" entry (of D-Bus type "o") at + // the very least, which contains a D-Bus path that uniquely identifies this track. + // + // See the type documentation for more details. + #[zbus(property(emits_changed_signal = "true"))] + async fn metadata( + &self, + ) -> zbus::fdo::Result> { + let meta = if self.metadata.is_empty() { + let mut meta = HashMap::new(); + meta.insert( + "mpris:trackid".to_owned(), + zvariant::Str::from(" /org/mpris/MediaPlayer2/TrackList/NoTrack").into() + ); + meta + } else { + self.metadata.iter() + .map(|(k, v)| (k.clone(), v.try_clone().unwrap())) + .collect() + }; + Ok(meta) + } + + // The volume level. + // + // When setting, if a negative value is passed, the volume should be set to 0.0. + // + // If `CanControl` is `false`, attempting to set this property should have no effect and raise + // an error. + #[zbus(property(emits_changed_signal = "true"))] + async fn volume(&self) -> Volume { + self.volume + } + + #[zbus(property)] + async fn set_volume(&mut self, _value: Volume) -> zbus::fdo::Result<()> { + // TODO: implement + Err(zbus::fdo::Error::NotSupported("Player control not implemented".to_owned())) + } + + // The current track position in microseconds, between 0 and the 'mpris:length' metadata entry + // (see Metadata). + // + // Note: If the media player allows it, the current playback position can be changed either the + // SetPosition method or the Seek method on this interface. If this is not the case, the + // `CanSeek` property is false, and setting this property has no effect and can raise an error. + // + // If the playback progresses in a way that is inconstistant with the `Rate` property, the + // `Seeked` signal is emited. + #[zbus(property(emits_changed_signal = "false"))] + async fn position(&self) -> zbus::fdo::Result { + // todo!("fetch up-to-date position from player") + Ok(0) + } + + // Note that the `Position` property is not writable intentionally, see + // the `set_position` method above. + // #[zbus(property)] + // async fn set_position(&self, _value: TimeInUs) -> zbus::fdo::Result<()> { + // // TODO: implement + // Err(zbus::fdo::Error::NotSupported("Player control not implemented".to_owned())) + // } + + // The minimum value which the `Rate` property can take. Clients should not attempt to set the + // `Rate` property below this value. + // + // Note that even if this value is 0.0 or negative, clients should not attempt to set the + // `Rate` property to 0.0. + // + // This value should always be 1.0 or less. + #[zbus(property(emits_changed_signal = "true"))] + async fn minimum_rate(&self) -> PlaybackRate { + // TODO: implement + 1.0 + } + + // The maximum value which the `Rate` property can take. Clients should not attempt to set the + // `Rate` property above this value. + // + // This value should always be 1.0 or greater. + #[zbus(property(emits_changed_signal = "true"))] + async fn maximum_rate(&self) -> PlaybackRate { + // TODO: implement + 1.0 + } + + // Whether the client can call the `Next` method on this interface and expect the current track + // to change. + // + // If it is unknown whether a call to `Next` will be successful (for example, when streaming + // tracks), this property should be set to `true`. + // + // If `CanControl` is `false`, this property should also be `false`. + // + // Rationale: + // + // Even when playback can generally be controlled, there may not + // always be a next track to move to. + #[zbus(property(emits_changed_signal = "true"))] + async fn can_go_next(&self) -> bool { + true + } + + // Whether the client can call the `Previous` method on this interface and expect the current + // track to change. + // + // If it is unknown whether a call to `Previous` will be successful (for example, when + // streaming tracks), this property should be set to `true`. + // + // If `CanControl` is `false`, this property should also be `false`. + // + // Rationale: + // + // Even when playback can generally be controlled, there may not + // always be a next previous to move to. + #[zbus(property(emits_changed_signal = "true"))] + async fn can_go_previous(&self) -> bool { + true + } + + // Whether playback can be started using `Play` or `PlayPause`. + // + // Note that this is related to whether there is a "current track": the value should not depend + // on whether the track is currently paused or playing. In fact, if a track is currently + // playing (and `CanControl` is `true`), this should be `true`. + // + // If `CanControl` is `false`, this property should also be `false`. + // + // Rationale: + // + // Even when playback can generally be controlled, it may not be + // possible to enter a "playing" state, for example if there is no + // "current track". + #[zbus(property(emits_changed_signal = "true"))] + async fn can_play(&self) -> bool { + !self.metadata.is_empty() + } + + // Whether playback can be paused using `Pause` or `PlayPause`. + // + // Note that this is an intrinsic property of the current track: its value should not depend on + // whether the track is currently paused or playing. In fact, if playback is currently paused + // (and `CanControl` is `true`), this should be `true`. + // + // + // If `CanControl` is `false`, this property should also be `false`. + // + // Rationale: + // + // Not all media is pausable: it may not be possible to pause some + // streamed media, for example. + #[zbus(property(emits_changed_signal = "true"))] + async fn can_pause(&self) -> bool { + !self.metadata.is_empty() + } + + // Whether the client can control the playback position using `Seek` and `SetPosition`. This + // may be different for different tracks. + // + // If `CanControl` is `false`, this property should also be `false`. + // + // Rationale: + // + // Not all media is seekable: it may not be possible to seek when + // playing some streamed media, for example. + #[zbus(property(emits_changed_signal = "true"))] + async fn can_seek(&self) -> bool { + true + } + + // Whether the media player may be controlled over this interface. + // + // This property is not expected to change, as it describes an intrinsic capability of the + // implementation. + // + // If this is `false`, clients should assume that all properties on this interface are + // read-only (and will raise errors if writing to them is attempted), no methods are + // implemented and all other properties starting with "can_" are also `false`. + // + // Rationale: + // + // This allows clients to determine whether to present and enable controls to the user in + // advance of attempting to call methods and write to properties. + #[zbus(property(emits_changed_signal = "const"))] + async fn can_control(&self) -> bool { + true + } + + // Indicates that the track position has changed in a way that is inconsistant with the current + // playing state. + // + // When this signal is not received, clients should assume that: + // - When playing, the position progresses according to the rate property. + // - When paused, it remains constant. + // + // This signal does not need to be emitted when playback starts or when the track changes, + // unless the track is starting at an unexpected position. An expected position would be the + // last known one when going from Paused to Playing, and 0 when going from Stopped to Playing. + // + // Arguments: + // + // * `position`: The new position, in microseconds. + #[zbus(signal)] + async fn seeked(signal_ctxt: &zbus::SignalContext<'_>, position: TimeInUs) -> zbus::Result<()>; + // FIXME: signal on appropriate player events! +} + +#[derive(Debug, Error)] +pub enum MprisError { + #[error("zbus error: {0}")] + DbusError(zbus::Error), +} + +impl From for Error { + fn from(err: MprisError) -> Self { + use MprisError::*; + match err { + DbusError(_) => Error::internal(err), + } + } +} + +impl From for MprisError { + fn from(err: zbus::Error) -> Self { + Self::DbusError(err) + } +} + +enum MprisCommand { + SetSpirc(Spirc), + Quit, +} + +pub struct MprisEventHandler { + cmd_tx: mpsc::UnboundedSender, + join_handle: tokio::task::JoinHandle<()>, +} + +impl MprisEventHandler { + pub async fn spawn( + player: Arc, + ) -> Result { + let (cmd_tx, cmd_rx) = mpsc::unbounded_channel(); + + let mpris_service = MprisService { }; + let mpris_player_service = MprisPlayerService { + spirc: None, + // FIXME: obtain current values from Player + repeat: false, + shuffle: false, + playback_status: PlaybackStatus::Stopped, + volume: 1.0, + metadata: HashMap::new(), + }; + + let connection = connection::Builder::session()? + // FIXME: retry with "org.mpris.MediaPlayer2.librespot.instance" + // on error + .name("org.mpris.MediaPlayer2.librespot")? + .serve_at("/org/mpris/MediaPlayer2", mpris_service)? + .serve_at("/org/mpris/MediaPlayer2", mpris_player_service)? + .build() + .await?; + + let mpris_task = MprisTask { player, connection, cmd_rx }; + + let join_handle = tokio::spawn(mpris_task.run()); + + Ok(MprisEventHandler { cmd_tx, join_handle }) + } + + pub fn set_spirc(&self, spirc: Spirc) { + let _ = self.cmd_tx.send(MprisCommand::SetSpirc(spirc)); + } + + pub async fn quit_and_join(self) { + let _ = self.cmd_tx.send(MprisCommand::Quit); + let _ = self.join_handle.await; + } +} + +struct MprisTask { + player: Arc, + connection: zbus::Connection, + cmd_rx: mpsc::UnboundedReceiver, +} + +impl MprisTask { + async fn run(mut self) { + let mut player_events = self.player.get_player_event_channel(); + + loop { + tokio::select! { + Some(event) = player_events.recv() => { + if let Err(e) = self.handle_event(event).await { + warn!("Error handling PlayerEvent: {}", e); + } + } + + cmd = self.cmd_rx.recv() => { + match cmd { + Some(MprisCommand::SetSpirc(spirc)) => { + // TODO: Update playback status, metadata, etc (?) + self.mpris_player_iface().await + .get_mut().await + .spirc = Some(spirc); + + } + Some(MprisCommand::Quit) => break, + + // Keep running if the cmd sender was dropped + None => (), + } + } + + // If player_events yields None, shutdown + else => break, + } + } + + debug!("Shutting down MprisTask ..."); + } + + #[allow(dead_code)] + async fn mpris_iface(&self) -> zbus::object_server::InterfaceRef { + self.connection + .object_server() + .interface::<_, MprisService>("/org/mpris/MediaPlayer2") + .await + .expect("iface missing on object server") + } + + async fn mpris_player_iface(&self) -> zbus::object_server::InterfaceRef { + self.connection + .object_server() + .interface::<_, MprisPlayerService>("/org/mpris/MediaPlayer2") + .await + .expect("iface missing on object server") + } + + async fn handle_event(&self, event: PlayerEvent) -> zbus::Result<()> { + match event { + PlayerEvent::PlayRequestIdChanged { play_request_id: _ } => { }, + PlayerEvent::TrackChanged { audio_item } => { + match audio_item.track_id.to_base62() { + Err(e) => { + warn!("PlayerEvent::TrackChanged: Invalid track id: {}", e) + } + Ok(track_id) => { + let iface_ref = self.mpris_player_iface().await; + let mut iface = iface_ref.get_mut().await; + + let meta = &mut iface.metadata; + meta.clear(); + + let mut trackid = String::new(); + trackid.push_str("/org/librespot/track/"); + trackid.push_str(&track_id); + meta.insert( + "mpris:trackid".into(), + zvariant::ObjectPath::try_from(trackid).unwrap().into() + ); + + meta.insert( + "xesam:title".into(), + zvariant::Str::from(audio_item.name).into() + ); + + if audio_item.covers.is_empty() { + meta.remove("mpris:artUrl"); + } else { + // TODO: Select image by size + let url = &audio_item .covers[0].url; + meta.insert( + "mpris.artUrl".into(), + zvariant::Str::from(url).into() + ); + } + + meta.insert( + "mpris:length".into(), + (audio_item.duration_ms as i64 * 1000).into(), + ); + + match audio_item.unique_fields { + UniqueFields::Track { + artists, + album, + album_date, + album_artists, + popularity: _, + number, + disc_number, + } => { + let artists = artists + .0 + .into_iter() + .map(|a| a.name) + .collect::>(); + meta.insert( + "xesam:artist".into(), + // try_to_owned only fails if the Value contains file + // descriptors, so the unwrap never panics here + zvariant::Value::from(artists).try_to_owned().unwrap() + ); + + + meta.insert( + "xesam:albumArtist".into(), + // try_to_owned only fails if the Value contains file + // descriptors, so the unwrap never panics here + zvariant::Value::from(&album_artists).try_to_owned().unwrap() + ); + + meta.insert( + "xesam:album".into(), + zvariant::Str::from(album).into() + ); + + meta.insert( + "xesam:trackNumber".into(), + (number as i32).into(), + ); + + meta.insert( + "xesam:discNumber".into(), + (disc_number as i32).into(), + ); + + meta.insert( + "xesam:contentCreated".into(), + zvariant::Str::from(album_date.0.format(&Iso8601::DATE).unwrap()).into() + ); + } + UniqueFields::Episode { + description, + publish_time, + show_name, + } => { + meta.insert( + "xesam:album".into(), + zvariant::Str::from(show_name).into() + ); + + meta.insert( + "xesam:comment".into(), + zvariant::Str::from(description).into() + ); + + meta.insert( + "xesam:contentCreated".into(), + zvariant::Str::from(publish_time.0.format(&Iso8601::DATE).unwrap()).into() + ); + } + } + + iface.metadata_changed(iface_ref.signal_context()).await?; + } + } + } + PlayerEvent::Stopped { track_id, .. } => match track_id.to_base62() { + Err(e) => warn!("PlayerEvent::Stopped: Invalid track id: {}", e), + Ok(track_id) => { + let iface_ref = self.mpris_player_iface().await; + let mut iface = iface_ref.get_mut().await; + let meta = &mut iface.metadata; + + // TODO: Check if metadata changed, if so clear + let mut trackid = String::new(); + trackid.push_str("/org/librespot/track/"); + trackid.push_str(&track_id); + meta.insert("mpris:trackid".into(), zvariant::ObjectPath::try_from(trackid).unwrap().into()); + iface.metadata_changed(iface_ref.signal_context()).await?; + + iface.playback_status = PlaybackStatus::Stopped; + iface.playback_status_changed(iface_ref.signal_context()).await?; + } + }, + PlayerEvent::Playing { + track_id, + // position_ms, + .. + } => match track_id.to_base62() { + Err(e) => warn!("PlayerEvent::Playing: Invalid track id: {}", e), + Ok(track_id) => { + // TODO: update position + let iface_ref = self.mpris_player_iface().await; + let mut iface = iface_ref.get_mut().await; + let meta = &mut iface.metadata; + + // TODO: Check if metadata changed, if so clear + let mut trackid = String::new(); + trackid.push_str("/org/librespot/track/"); + trackid.push_str(&track_id); + meta.insert("mpris:trackid".into(), zvariant::ObjectPath::try_from(trackid).unwrap().into()); + iface.metadata_changed(iface_ref.signal_context()).await?; + + iface.playback_status = PlaybackStatus::Playing; + iface.playback_status_changed(iface_ref.signal_context()).await?; + } + }, + PlayerEvent::Paused { + track_id, + // position_ms, + .. + } => match track_id.to_base62() { + Err(e) => warn!("PlayerEvent::Paused: Invalid track id: {}", e), + Ok(track_id) => { + // TODO: update position + let iface_ref = self.mpris_player_iface().await; + let mut iface = iface_ref.get_mut().await; + let meta = &mut iface.metadata; + + // TODO: Check if metadata changed, if so clear + let mut trackid = String::new(); + trackid.push_str("/org/librespot/track/"); + trackid.push_str(&track_id); + meta.insert("mpris:trackid".into(), zvariant::ObjectPath::try_from(trackid).unwrap().into()); + iface.metadata_changed(iface_ref.signal_context()).await?; + + iface.playback_status = PlaybackStatus::Paused; + iface.playback_status_changed(iface_ref.signal_context()).await?; + } + }, + PlayerEvent::Loading { .. } => { }, + PlayerEvent::Preloading { .. } => { }, + PlayerEvent::TimeToPreloadNextTrack { .. } => { }, + PlayerEvent::EndOfTrack { track_id, .. } => match track_id.to_base62() { + Err(e) => warn!("PlayerEvent::EndOfTrack: Invalid track id: {}", e), + Ok(_id) => { + // TODO: ? + } + }, + PlayerEvent::Unavailable { .. } => { }, + PlayerEvent::VolumeChanged { + // volume + .. + } => { + // TODO: Handle volume + }, + PlayerEvent::Seeked { + track_id, + // position_ms, + .. + } => match track_id.to_base62() { + Err(e) => warn!("PlayerEvent::Seeked: Invalid track id: {}", e), + Ok(track_id) => { + // TODO: Update position + track_id + let iface_ref = self.mpris_player_iface().await; + let mut iface = iface_ref.get_mut().await; + let meta = &mut iface.metadata; + + // TODO: Check if metadata changed, if so clear + let mut trackid = String::new(); + trackid.push_str("/org/librespot/track/"); + trackid.push_str(&track_id); + meta.insert("mpris:trackid".into(), zvariant::ObjectPath::try_from(trackid).unwrap().into()); + iface.metadata_changed(iface_ref.signal_context()).await?; + } + }, + PlayerEvent::PositionCorrection { + track_id, + // position_ms, + .. + } => match track_id.to_base62() { + Err(e) => { + warn!("PlayerEvent::PositionCorrection: Invalid track id: {}", e) + } + Ok(_id) => { + // TODO: Update position + track_id + } + }, + PlayerEvent::SessionConnected { .. } => { }, + PlayerEvent::SessionDisconnected { .. } => { }, + PlayerEvent::SessionClientChanged { .. } => { }, + PlayerEvent::ShuffleChanged { shuffle } => { + let iface_ref = self.mpris_player_iface().await; + let mut iface = iface_ref.get_mut().await; + iface.shuffle = shuffle; + iface.shuffle_changed(iface_ref.signal_context()).await?; + }, + PlayerEvent::RepeatChanged { repeat } => { + let iface_ref = self.mpris_player_iface().await; + let mut iface = iface_ref.get_mut().await; + iface.repeat = repeat; + iface.loop_status_changed(iface_ref.signal_context()).await?; + }, + PlayerEvent::AutoPlayChanged { .. } => { }, + PlayerEvent::FilterExplicitContentChanged { .. } => { }, + } + + Ok(()) + } +}