From 945c2abc155a8e9cd3bf73d477e4b5209b5d059a Mon Sep 17 00:00:00 2001 From: h7x4 Date: Mon, 6 Jan 2025 19:11:40 +0100 Subject: [PATCH 1/2] add websocket API --- Cargo.lock | 650 +++++++++++++++++++++++++++++----------- Cargo.toml | 4 +- src/api.rs | 2 + src/api/websocket_v1.rs | 387 ++++++++++++++++++++++++ src/main.rs | 8 +- 5 files changed, 867 insertions(+), 184 deletions(-) create mode 100644 src/api/websocket_v1.rs diff --git a/Cargo.lock b/Cargo.lock index a6ece72..4a01d38 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -43,9 +43,9 @@ dependencies = [ [[package]] name = "anstream" -version = "0.6.15" +version = "0.6.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "64e15c1ab1f89faffbf04a634d5e1962e9074f2741eef6d97f3c4e322426d526" +checksum = "8acc5369981196006228e28809f761875c0327210a891e941f4c683b3a99529b" dependencies = [ "anstyle", "anstyle-parse", @@ -58,58 +58,58 @@ dependencies = [ [[package]] name = "anstyle" -version = "1.0.8" +version = "1.0.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1bec1de6f59aedf83baf9ff929c98f2ad654b97c9510f4e70cf6f661d49fd5b1" +checksum = "55cc3b69f167a1ef2e161439aa98aed94e6028e5f9a59be9a6ffb47aef1651f9" [[package]] name = "anstyle-parse" -version = "0.2.5" +version = "0.2.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eb47de1e80c2b463c735db5b217a0ddc39d612e7ac9e2e96a5aed1f57616c1cb" +checksum = "3b2d16507662817a6a20a9ea92df6652ee4f94f914589377d69f3b21bc5798a9" dependencies = [ "utf8parse", ] [[package]] name = "anstyle-query" -version = "1.1.1" +version = "1.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6d36fc52c7f6c869915e99412912f22093507da8d9e942ceaf66fe4b7c14422a" +checksum = "79947af37f4177cfead1110013d678905c37501914fba0efea834c3fe9a8d60c" dependencies = [ - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] name = "anstyle-wincon" -version = "3.0.4" +version = "3.0.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5bf74e1b6e971609db8ca7a9ce79fd5768ab6ae46441c572e46cf596f59e57f8" +checksum = "2109dbce0e72be3ec00bed26e6a7479ca384ad226efdd66db8fa2e3a38c83125" dependencies = [ "anstyle", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] name = "anyhow" -version = "1.0.90" +version = "1.0.94" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "37bf3594c4c988a53154954629820791dde498571819ae4ca50ca811e060cc95" +checksum = "c1fd03a028ef38ba2276dce7e33fcd6369c158a1bca17946c4b1b701891c1ff7" [[package]] name = "arbitrary" -version = "1.3.2" +version = "1.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7d5a26814d8dcb93b0e5a0ff3c6d80a8843bafb21b39e8e18a6f05471870e110" +checksum = "dde20b3d026af13f561bdd0f15edf01fc734f0dafcedbaf42bba506a9517f223" dependencies = [ "derive_arbitrary", ] [[package]] name = "async-compression" -version = "0.4.17" +version = "0.4.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0cb8f1d480b0ea3783ab015936d2a55c87e219676f0c0b7dec61494043f21857" +checksum = "df895a515f70646414f4b45c0b79082783b80552b373a68283012928df56f522" dependencies = [ "brotli", "flate2", @@ -140,16 +140,17 @@ checksum = "ace50bade8e6234aa140d9a2f552bbee1db4d353f69b8217bc503490fc1a9f26" [[package]] name = "axum" -version = "0.7.7" +version = "0.7.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "504e3947307ac8326a5437504c517c4b56716c9d98fac0028c2acc7ca47d70ae" +checksum = "edca88bc138befd0323b20752846e6587272d3b03b0343c8ea28a6f819e6e71f" dependencies = [ "async-trait", "axum-core", "axum-macros", + "base64 0.22.1", "bytes", "futures-util", - "http 1.1.0", + "http 1.2.0", "http-body 1.0.1", "http-body-util", "hyper", @@ -165,9 +166,11 @@ dependencies = [ "serde_json", "serde_path_to_error", "serde_urlencoded", - "sync_wrapper 1.0.1", + "sha1", + "sync_wrapper", "tokio", - "tower 0.5.1", + "tokio-tungstenite", + "tower 0.5.2", "tower-layer", "tower-service", "tracing", @@ -182,13 +185,13 @@ dependencies = [ "async-trait", "bytes", "futures-util", - "http 1.1.0", + "http 1.2.0", "http-body 1.0.1", "http-body-util", "mime", "pin-project-lite", "rustversion", - "sync_wrapper 1.0.1", + "sync_wrapper", "tower-layer", "tower-service", "tracing", @@ -226,6 +229,12 @@ version = "0.21.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567" +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + [[package]] name = "bitflags" version = "2.6.0" @@ -276,15 +285,15 @@ checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" [[package]] name = "bytes" -version = "1.7.2" +version = "1.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "428d9aa8fbc0670b7b8d6030a7fadd0f86151cae55e4dbbece15f3780a3dfaf3" +checksum = "325918d6fe32f23b19878fe4b34794ae41fc19ddbe53b10571a4874d44ffd39b" [[package]] name = "cc" -version = "1.1.31" +version = "1.2.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c2e7962b54006dcfcc61cb72735f4d89bb97061dd6a7ed882ec6b8ee53714c6f" +checksum = "9157bbaa6b165880c27a4293a474c91cdcf265cc68cc829bf10be0964a391caf" dependencies = [ "jobserver", "libc", @@ -299,9 +308,9 @@ checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" [[package]] name = "clap" -version = "4.5.20" +version = "4.5.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b97f376d85a664d5837dbae44bf546e6477a679ff6610010f17276f686d867e8" +checksum = "3135e7ec2ef7b10c6ed8950f0f792ed96ee093fa088608f1c76e569722700c84" dependencies = [ "clap_builder", "clap_derive", @@ -309,9 +318,9 @@ dependencies = [ [[package]] name = "clap-verbosity-flag" -version = "2.2.2" +version = "2.2.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e099138e1807662ff75e2cebe4ae2287add879245574489f9b1588eb5e5564ed" +checksum = "34c77f67047557f62582784fd7482884697731b2932c7d37ced54bce2312e1e2" dependencies = [ "clap", "log", @@ -319,9 +328,9 @@ dependencies = [ [[package]] name = "clap_builder" -version = "4.5.20" +version = "4.5.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "19bc80abd44e4bed93ca373a0704ccbd1b710dc5749406201bb018272808dc54" +checksum = "30582fc632330df2bd26877bde0c1f4470d57c582bbc070376afcd04d8cb4838" dependencies = [ "anstream", "anstyle", @@ -343,21 +352,21 @@ dependencies = [ [[package]] name = "clap_lex" -version = "0.7.2" +version = "0.7.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1462739cb27611015575c0c11df5df7601141071f07518d56fcc1be504cbec97" +checksum = "f46ad14479a25103f283c0f10005961cf086d8dc42205bb44c46ac563475dca6" [[package]] name = "colorchoice" -version = "1.0.2" +version = "1.0.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d3fd119d74b830634cea2a0f58bbd0d54540518a14397557951e79340abc28c0" +checksum = "5b63caa9aa9397e2d9480a9b13673856c78d8ac123288526c37d7839f2a86990" [[package]] name = "cpufeatures" -version = "0.2.14" +version = "0.2.16" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "608697df725056feaccfa42cffdaeeec3fccc4ffc38358ecd19b243e716a78e0" +checksum = "16b80225097f2e5ae4e7179dd2266824648f3e2f49d9134d584b76389d31c4c3" dependencies = [ "libc", ] @@ -373,9 +382,9 @@ dependencies = [ [[package]] name = "crossbeam-utils" -version = "0.8.20" +version = "0.8.21" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80" +checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28" [[package]] name = "crypto-common" @@ -388,10 +397,16 @@ dependencies = [ ] [[package]] -name = "derive_arbitrary" -version = "1.3.2" +name = "data-encoding" +version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "67e77553c4162a157adbf834ebae5b415acbecbeafc7a74b0e886657506a7611" +checksum = "e8566979429cf69b49a5c740c60791108e86440e8be149bbea4fe54d2c32d6e2" + +[[package]] +name = "derive_arbitrary" +version = "1.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "30542c1ad912e0e3d22a1935c290e12e8a29d704a420177a31faad4a601a0800" dependencies = [ "proc-macro2", "quote", @@ -440,25 +455,25 @@ checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" [[package]] name = "errno" -version = "0.3.9" +version = "0.3.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "534c5cf6194dfab3db3242765c03bbe257cf92f22b38f6bc0c58d59108a820ba" +checksum = "33d852cb9b869c2a9b3df2f71a3074817f01e1844f839a144f5fcef059a4eb5d" dependencies = [ "libc", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] name = "fastrand" -version = "2.1.1" +version = "2.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e8c02a5121d4ea3eb16a80748c74f5549a5665e4c21333c6098f283870fbdea6" +checksum = "37909eebbb50d72f9059c3b6d82c0463f2ff062c9e95845c43a6c9c0355411be" [[package]] name = "flate2" -version = "1.0.34" +version = "1.0.35" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a1b589b4dc103969ad3cf85c950899926ec64300a1a46d76c03a6072957036f0" +checksum = "c936bfdafb507ebbf50b8074c54fa31c5be9a1e7e5f467dd659697041407d07c" dependencies = [ "crc32fast", "miniz_oxide", @@ -628,9 +643,9 @@ checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" [[package]] name = "hashbrown" -version = "0.15.0" +version = "0.15.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1e087f84d4f86bf4b218b927129862374b72199ae7d8657835f1e89000eea4fb" +checksum = "bf151400ff0baff5465007dd2f3e717f3fe502074ca563069ce3a6629d07b289" [[package]] name = "hdrhistogram" @@ -648,12 +663,6 @@ version = "0.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea" -[[package]] -name = "hermit-abi" -version = "0.3.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024" - [[package]] name = "hermit-abi" version = "0.4.0" @@ -673,9 +682,9 @@ dependencies = [ [[package]] name = "http" -version = "1.1.0" +version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "21b9ddb458710bc376481b842f5da65cdf31522de232c1ca8146abce2a358258" +checksum = "f16ca2af56261c99fba8bac40a10251ce8188205a4c448fbb745a2e4daa76fea" dependencies = [ "bytes", "fnv", @@ -700,7 +709,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" dependencies = [ "bytes", - "http 1.1.0", + "http 1.2.0", ] [[package]] @@ -711,7 +720,7 @@ checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" dependencies = [ "bytes", "futures-util", - "http 1.1.0", + "http 1.2.0", "http-body 1.0.1", "pin-project-lite", ] @@ -742,14 +751,14 @@ checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4" [[package]] name = "hyper" -version = "1.5.0" +version = "1.5.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bbbff0a806a4728c99295b254c8838933b5b082d75e3cb70c8dab21fdfbcfa9a" +checksum = "97818827ef4f364230e16705d4706e2897df2bb60617d6ca15d598025a3c481f" dependencies = [ "bytes", "futures-channel", "futures-util", - "http 1.1.0", + "http 1.2.0", "http-body 1.0.1", "httparse", "httpdate", @@ -761,13 +770,13 @@ dependencies = [ [[package]] name = "hyper-util" -version = "0.1.9" +version = "0.1.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "41296eb09f183ac68eec06e03cdbea2e759633d4067b2f6552fc2e009bcad08b" +checksum = "df2dcfbe0677734ab2f3ffa7fa7bfd4706bfdc1ef393f2ee30184aed67e631b4" dependencies = [ "bytes", "futures-util", - "http 1.1.0", + "http 1.2.0", "http-body 1.0.1", "hyper", "pin-project-lite", @@ -776,13 +785,142 @@ dependencies = [ ] [[package]] -name = "idna" -version = "0.5.0" +name = "icu_collections" +version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" +checksum = "db2fa452206ebee18c4b5c2274dbf1de17008e874b4dc4f0aea9d01ca79e4526" dependencies = [ - "unicode-bidi", - "unicode-normalization", + "displaydoc", + "yoke", + "zerofrom", + "zerovec", +] + +[[package]] +name = "icu_locid" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13acbb8371917fc971be86fc8057c41a64b521c184808a698c02acc242dbf637" +dependencies = [ + "displaydoc", + "litemap", + "tinystr", + "writeable", + "zerovec", +] + +[[package]] +name = "icu_locid_transform" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "01d11ac35de8e40fdeda00d9e1e9d92525f3f9d887cdd7aa81d727596788b54e" +dependencies = [ + "displaydoc", + "icu_locid", + "icu_locid_transform_data", + "icu_provider", + "tinystr", + "zerovec", +] + +[[package]] +name = "icu_locid_transform_data" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fdc8ff3388f852bede6b579ad4e978ab004f139284d7b28715f773507b946f6e" + +[[package]] +name = "icu_normalizer" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19ce3e0da2ec68599d193c93d088142efd7f9c5d6fc9b803774855747dc6a84f" +dependencies = [ + "displaydoc", + "icu_collections", + "icu_normalizer_data", + "icu_properties", + "icu_provider", + "smallvec", + "utf16_iter", + "utf8_iter", + "write16", + "zerovec", +] + +[[package]] +name = "icu_normalizer_data" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8cafbf7aa791e9b22bec55a167906f9e1215fd475cd22adfcf660e03e989516" + +[[package]] +name = "icu_properties" +version = "1.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93d6020766cfc6302c15dbbc9c8778c37e62c14427cb7f6e601d849e092aeef5" +dependencies = [ + "displaydoc", + "icu_collections", + "icu_locid_transform", + "icu_properties_data", + "icu_provider", + "tinystr", + "zerovec", +] + +[[package]] +name = "icu_properties_data" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67a8effbc3dd3e4ba1afa8ad918d5684b8868b3b26500753effea8d2eed19569" + +[[package]] +name = "icu_provider" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ed421c8a8ef78d3e2dbc98a973be2f3770cb42b606e3ab18d6237c4dfde68d9" +dependencies = [ + "displaydoc", + "icu_locid", + "icu_provider_macros", + "stable_deref_trait", + "tinystr", + "writeable", + "yoke", + "zerofrom", + "zerovec", +] + +[[package]] +name = "icu_provider_macros" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1ec89e9337638ecdc08744df490b221a7399bf8d164eb52a665454e60e075ad6" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "idna" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "686f825264d630750a544639377bae737628043f20d38bbc029e8f29ea968a7e" +dependencies = [ + "idna_adapter", + "smallvec", + "utf8_iter", +] + +[[package]] +name = "idna_adapter" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "daca1df1c957320b2cf139ac61e7bd64fed304c5040df000a745aa1de3b4ef71" +dependencies = [ + "icu_normalizer", + "icu_properties", ] [[package]] @@ -797,12 +935,12 @@ dependencies = [ [[package]] name = "indexmap" -version = "2.6.0" +version = "2.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "707907fe3c25f5424cce2cb7e1cbcafee6bdbe735ca90ef77c29e84591e5b9da" +checksum = "62f822373a4fe84d4bb149bf54e584a7f4abec90e072ed49cda0edea5b95471f" dependencies = [ "equivalent", - "hashbrown 0.15.0", + "hashbrown 0.15.2", "serde", ] @@ -822,7 +960,7 @@ version = "0.4.13" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "261f68e344040fbd0edea105bef17c66edf46f984ddb1115b775ce31be948f4b" dependencies = [ - "hermit-abi 0.4.0", + "hermit-abi", "libc", "windows-sys 0.52.0", ] @@ -835,9 +973,9 @@ checksum = "7943c866cc5cd64cbc25b2e01621d07fa8eb2a1a23160ee81ce38704e97b8ecf" [[package]] name = "itoa" -version = "1.0.11" +version = "1.0.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "49f1f14873335454500d59611f1cf4a4b0f786f9ac11f4312a78e4cf2566695b" +checksum = "d75a2a4b1b190afb6f5425f10f6a8f959d2ea0b9c2b1d79553551850539e4674" [[package]] name = "jobserver" @@ -850,9 +988,9 @@ dependencies = [ [[package]] name = "libc" -version = "0.2.161" +version = "0.2.168" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8e9489c2807c139ffd9c1794f4af0ebe86a828db53ecdc7fea2111d0fed085d1" +checksum = "5aaeb2981e0606ca11d79718f8bb01164f1d6ed75080182d3abf017e6d244b6d" [[package]] name = "linux-raw-sys" @@ -860,6 +998,12 @@ version = "0.4.14" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "78b3ae25bc7c8c38cec158d1f2757ee79e9b3740fbc7ccf0e59e4b08d793fa89" +[[package]] +name = "litemap" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ee93343901ab17bd981295f2cf0026d4ad018c7c31ba84549a4ddbb47a45104" + [[package]] name = "lock_api" version = "0.4.12" @@ -921,11 +1065,10 @@ dependencies = [ [[package]] name = "mio" -version = "1.0.2" +version = "1.0.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "80e04d1dcff3aae0704555fe5fee3bcfaf3d1fdf8a7e521d5b9d2b42acb52cec" +checksum = "2886843bf800fba2e3377cff24abf6379b4c4d5c6681eaf9ea5b0d15090450bd" dependencies = [ - "hermit-abi 0.3.9", "libc", "wasi", "windows-sys 0.52.0", @@ -934,13 +1077,13 @@ dependencies = [ [[package]] name = "mpvipc-async" version = "0.1.0" -source = "git+https://git.pvv.ntnu.no/Projects/mpvipc-async.git?rev=v0.1.0#467cac3c503887c4d6371ec5fdf1b23b3e0eb515" +source = "git+https://git.pvv.ntnu.no/Grzegorz/mpvipc-async.git?branch=main#a6c6bf4388c0e349c08b127a98ed480646618b3f" dependencies = [ "futures", "log", "serde", "serde_json", - "thiserror", + "thiserror 1.0.69", "tokio", "tokio-stream", "tokio-util", @@ -1007,18 +1150,18 @@ checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" [[package]] name = "pin-project" -version = "1.1.6" +version = "1.1.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "baf123a161dde1e524adf36f90bc5d8d3462824a9c43553ad07a8183161189ec" +checksum = "be57f64e946e500c8ee36ef6331845d40a93055567ec57e8fae13efd33759b95" dependencies = [ "pin-project-internal", ] [[package]] name = "pin-project-internal" -version = "1.1.6" +version = "1.1.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a4502d8515ca9f32f1fb543d987f63d95a14934883db45bdb48060b6b69257f8" +checksum = "3c0f5fad0874fc7abcd4d750e76917eaebbecaa2c20bde22e1dbeeba8beb758c" dependencies = [ "proc-macro2", "quote", @@ -1027,9 +1170,9 @@ dependencies = [ [[package]] name = "pin-project-lite" -version = "0.2.14" +version = "0.2.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bda66fc9667c18cb2758a2ac84d1167245054bcf85d5d1aaa6923f45801bdd02" +checksum = "915a1e146535de9163f3987b8944ed8cf49a18bb0056bcebcdcece385cece4ff" [[package]] name = "pin-utils" @@ -1054,9 +1197,9 @@ dependencies = [ [[package]] name = "proc-macro2" -version = "1.0.88" +version = "1.0.92" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7c3a7fc5db1e57d5a779a352c8cdb57b29aa4c40cc69c3a68a7fedc815fbf2f9" +checksum = "37d3544b3f2748c54e147655edb5025752e2303145b5aefb3c3ea2c78b973bb0" dependencies = [ "unicode-ident", ] @@ -1102,18 +1245,18 @@ dependencies = [ [[package]] name = "redox_syscall" -version = "0.5.7" +version = "0.5.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9b6dfecf2c74bce2466cabf93f6664d6998a69eb21e39f4207930065b27b771f" +checksum = "03a862b389f93e68874fbf580b9de08dd02facb9a788ebadaf4a3fd33cf58834" dependencies = [ "bitflags", ] [[package]] name = "regex" -version = "1.11.0" +version = "1.11.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "38200e5ee88914975b69f657f0801b6f6dccafd44fd9326302a4aaeecfacb1d8" +checksum = "b544ef1b4eac5dc2db33ea63606ae9ffcfac26c1416a2806ae0bf5f56b201191" dependencies = [ "aho-corasick", "memchr", @@ -1123,9 +1266,9 @@ dependencies = [ [[package]] name = "regex-automata" -version = "0.4.8" +version = "0.4.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "368758f23274712b504848e9d5a6f010445cc8b87a7cdb4d7cbee666c1288da3" +checksum = "809e8dc61f6de73b46c85f4c96486310fe304c434cfa43669d7b40f711150908" dependencies = [ "aho-corasick", "memchr", @@ -1180,15 +1323,15 @@ checksum = "719b953e2095829ee67db738b3bfa9fa368c94900df327b3f07fe6e794d2fe1f" [[package]] name = "rustix" -version = "0.38.37" +version = "0.38.42" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8acb788b847c24f28525660c4d7758620a7210875711f79e7f663cc152726811" +checksum = "f93dc38ecbab2eb790ff964bb77fa94faf256fd3e73285fd7ba0903b76bedb85" dependencies = [ "bitflags", "errno", "libc", "linux-raw-sys", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -1226,18 +1369,18 @@ checksum = "1be20c5f7f393ee700f8b2f28ea35812e4e212f40774b550cd2a93ea91684451" [[package]] name = "serde" -version = "1.0.210" +version = "1.0.216" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c8e3592472072e6e22e0a54d5904d9febf8508f65fb8552499a1abc7d1078c3a" +checksum = "0b9781016e935a97e8beecf0c933758c97a5520d32930e460142b4cd80c6338e" dependencies = [ "serde_derive", ] [[package]] name = "serde_derive" -version = "1.0.210" +version = "1.0.216" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "243902eda00fad750862fc144cea25caca5e20d615af0a81bee94ca738f1df1f" +checksum = "46f859dbbf73865c6627ed570e78961cd3ac92407a2d117204c49232485da55e" dependencies = [ "proc-macro2", "quote", @@ -1246,9 +1389,9 @@ dependencies = [ [[package]] name = "serde_json" -version = "1.0.132" +version = "1.0.133" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d726bfaff4b320266d395898905d0eba0345aae23b54aee3a737e260fd46db03" +checksum = "c7fceb2473b9166b2294ef05efcb65a3db80803f0b03ef86a5fc88a2b85ee377" dependencies = [ "itoa", "memchr", @@ -1278,6 +1421,17 @@ dependencies = [ "serde", ] +[[package]] +name = "sha1" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + [[package]] name = "sha2" version = "0.10.8" @@ -1327,14 +1481,20 @@ checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" [[package]] name = "socket2" -version = "0.5.7" +version = "0.5.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ce305eb0b4296696835b71df73eb912e0f1ffd2556a501fcede6e0c50349191c" +checksum = "c970269d99b64e60ec3bd6ad27270092a5394c4e309314b18ae3fe575695fbe8" dependencies = [ "libc", "windows-sys 0.52.0", ] +[[package]] +name = "stable_deref_trait" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3" + [[package]] name = "strsim" version = "0.11.1" @@ -1343,9 +1503,9 @@ checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" [[package]] name = "syn" -version = "2.0.82" +version = "2.0.90" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "83540f837a8afc019423a8edb95b52a8effe46957ee402287f4292fae35be021" +checksum = "919d3b74a5dd0ccd15aeb8f93e7006bd9e14c295087c9896a110f490752bcf31" dependencies = [ "proc-macro2", "quote", @@ -1354,15 +1514,20 @@ dependencies = [ [[package]] name = "sync_wrapper" -version = "0.1.2" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" +checksum = "0bf256ce5efdfa370213c1dabab5935a12e49f2c58d15e9eac2870d3b4f27263" [[package]] -name = "sync_wrapper" -version = "1.0.1" +name = "synstructure" +version = "0.13.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a7065abeca94b6a8a577f9bd45aa0867a2238b74e8eb67cf10d492bc39351394" +checksum = "c8af7666ab7b6390ab78131fb5b0fce11d6b7a6951602017c35fa82800708971" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] [[package]] name = "systemd-journal-logger" @@ -1376,9 +1541,9 @@ dependencies = [ [[package]] name = "tempfile" -version = "3.13.0" +version = "3.14.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f0f2c9fc62d0beef6951ccffd757e241266a2c833136efbe35af6cd2567dca5b" +checksum = "28cce251fcbc87fac86a866eeb0d6c2d536fc16d06f184bb61aeae11aa4cee0c" dependencies = [ "cfg-if", "fastrand", @@ -1398,18 +1563,27 @@ dependencies = [ [[package]] name = "thiserror" -version = "1.0.64" +version = "1.0.69" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d50af8abc119fb8bb6dbabcfa89656f46f84aa0ac7688088608076ad2b459a84" +checksum = "b6aaf5339b578ea85b50e080feb250a3e8ae8cfcdff9a461c9ec2904bc923f52" dependencies = [ - "thiserror-impl", + "thiserror-impl 1.0.69", +] + +[[package]] +name = "thiserror" +version = "2.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93605438cbd668185516ab499d589afb7ee1859ea3d5fc8f6b0755e1c7443767" +dependencies = [ + "thiserror-impl 2.0.7", ] [[package]] name = "thiserror-impl" -version = "1.0.64" +version = "1.0.69" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "08904e7672f5eb876eaaf87e0ce17857500934f4981c4a0ab2b4aa98baac7fc3" +checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1" dependencies = [ "proc-macro2", "quote", @@ -1417,25 +1591,31 @@ dependencies = [ ] [[package]] -name = "tinyvec" -version = "1.8.0" +name = "thiserror-impl" +version = "2.0.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "445e881f4f6d382d5f27c034e25eb92edd7c784ceab92a0937db7f2e9471b938" +checksum = "e1d8749b4531af2117677a5fcd12b1348a3fe2b81e36e61ffeac5c4aa3273e36" dependencies = [ - "tinyvec_macros", + "proc-macro2", + "quote", + "syn", ] [[package]] -name = "tinyvec_macros" -version = "0.1.1" +name = "tinystr" +version = "0.7.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" +checksum = "9117f5d4db391c1cf6927e7bea3db74b9a1c1add8f7eda9ffd5364f40f57b82f" +dependencies = [ + "displaydoc", + "zerovec", +] [[package]] name = "tokio" -version = "1.40.0" +version = "1.42.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e2b070231665d27ad9ec9b8df639893f46727666c6767db40317fbe920a5d998" +checksum = "5cec9b21b0450273377fc97bd4c33a8acffc8c996c987a7c5b319a0083707551" dependencies = [ "backtrace", "bytes", @@ -1462,9 +1642,9 @@ dependencies = [ [[package]] name = "tokio-stream" -version = "0.1.16" +version = "0.1.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4f4e6ce100d0eb49a2734f8c0812bcd324cf357d21810932c5df6b96ef2b86f1" +checksum = "eca58d7bba4a75707817a2c44174253f9236b2d5fbd055602e9d5c07c139a047" dependencies = [ "futures-core", "pin-project-lite", @@ -1473,10 +1653,22 @@ dependencies = [ ] [[package]] -name = "tokio-util" -version = "0.7.12" +name = "tokio-tungstenite" +version = "0.24.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "61e7c3654c13bcd040d4a03abee2c75b1d14a37b423cf5a813ceae1cc903ec6a" +checksum = "edc5f74e248dc973e0dbb7b74c7e0d6fcc301c694ff50049504004ef4d0cdcd9" +dependencies = [ + "futures-util", + "log", + "tokio", + "tungstenite", +] + +[[package]] +name = "tokio-util" +version = "0.7.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7fcaa8d55a2bdd6b83ace262b016eca0d79ee02818c5c1bcdf0305114081078" dependencies = [ "bytes", "futures-core", @@ -1508,14 +1700,14 @@ dependencies = [ [[package]] name = "tower" -version = "0.5.1" +version = "0.5.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2873938d487c3cfb9aed7546dc9f2711d867c9f90c46b889989a2cb84eba6b4f" +checksum = "d039ad9159c98b70ecfd540b2573b97f7f52c3e8d9f8ad57a24b916a536975f9" dependencies = [ "futures-core", "futures-util", "pin-project-lite", - "sync_wrapper 0.1.2", + "sync_wrapper", "tokio", "tower-layer", "tower-service", @@ -1529,7 +1721,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "61c5bb1d698276a2443e5ecfabc1008bf15a36c12e6a7176e7bf089ea9131140" dependencies = [ "async-compression", - "base64", + "base64 0.21.7", "bitflags", "bytes", "futures-core", @@ -1566,9 +1758,9 @@ checksum = "8df9b6e13f2d32c91b9bd719c00d1958837bc7dec474d94952798cc8e69eeec3" [[package]] name = "tracing" -version = "0.1.40" +version = "0.1.41" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c3523ab5a71916ccf420eebdf5521fcef02141234bbc0b8a49f2fdc4544364ef" +checksum = "784e0ac535deb450455cbfa28a6f0df145ea1bb7ae51b821cf5e7927fdcfbdd0" dependencies = [ "log", "pin-project-lite", @@ -1577,13 +1769,31 @@ dependencies = [ [[package]] name = "tracing-core" -version = "0.1.32" +version = "0.1.33" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c06d3da6113f116aaee68e4d601191614c9053067f9ab7f6edbcb161237daa54" +checksum = "e672c95779cf947c5311f83787af4fa8fffd12fb27e4993211a84bdfd9610f9c" dependencies = [ "once_cell", ] +[[package]] +name = "tungstenite" +version = "0.24.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "18e5b8366ee7a95b16d32197d0b2604b43a0be89dc5fac9f8e96ccafbaedda8a" +dependencies = [ + "byteorder", + "bytes", + "data-encoding", + "http 1.2.0", + "httparse", + "log", + "rand", + "sha1", + "thiserror 1.0.69", + "utf-8", +] + [[package]] name = "typenum" version = "1.17.0" @@ -1596,38 +1806,41 @@ version = "2.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7e51b68083f157f853b6379db119d1c1be0e6e4dec98101079dec41f6f5cf6df" -[[package]] -name = "unicode-bidi" -version = "0.3.17" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5ab17db44d7388991a428b2ee655ce0c212e862eff1768a455c58f9aad6e7893" - [[package]] name = "unicode-ident" -version = "1.0.13" +version = "1.0.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e91b56cd4cadaeb79bbf1a5645f6b4f8dc5bde8834ad5894a8db35fda9efa1fe" - -[[package]] -name = "unicode-normalization" -version = "0.1.24" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5033c97c4262335cded6d6fc3e5c18ab755e1a3dc96376350f3d8e9f009ad956" -dependencies = [ - "tinyvec", -] +checksum = "adb9e6ca4f869e1180728b7950e35922a7fc6397f7b641499e8f3ef06e50dc83" [[package]] name = "url" -version = "2.5.2" +version = "2.5.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "22784dbdf76fdde8af1aeda5622b546b422b6fc585325248a2bf9f5e41e94d6c" +checksum = "32f8b686cadd1473f4bd0117a5d28d36b1ade384ea9b5069a1c40aefed7fda60" dependencies = [ "form_urlencoded", "idna", "percent-encoding", ] +[[package]] +name = "utf-8" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" + +[[package]] +name = "utf16_iter" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8232dd3cdaed5356e0f716d285e4b40b932ac434100fe9b7e0e8e935b9e6246" + +[[package]] +name = "utf8_iter" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6c140620e7ffbb22c2dee59cafe6084a59b5ffc27a8859a5f0d494b5d52b6be" + [[package]] name = "utf8parse" version = "0.2.2" @@ -1636,11 +1849,11 @@ checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821" [[package]] name = "utoipa" -version = "5.1.3" +version = "5.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9d9ba0ade4e2f024cd1842dfbaf9dbc540639fc082299acf7649d71bd14eaca3" +checksum = "514a48569e4e21c86d0b84b5612b5e73c0b2cf09db63260134ba426d4e8ea714" dependencies = [ - "indexmap 2.6.0", + "indexmap 2.7.0", "serde", "serde_json", "utoipa-gen", @@ -1661,9 +1874,9 @@ dependencies = [ [[package]] name = "utoipa-gen" -version = "5.1.3" +version = "5.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4cf390d6503c9c9eac988447c38ba934a707b0b768b14511a493b4fc0e8ecb00" +checksum = "5629efe65599d0ccd5d493688cbf6e03aa7c1da07fe59ff97cf5977ed0637f66" dependencies = [ "proc-macro2", "quote", @@ -1817,6 +2030,42 @@ version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" +[[package]] +name = "write16" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d1890f4022759daae28ed4fe62859b1236caebfc61ede2f63ed4e695f3f6d936" + +[[package]] +name = "writeable" +version = "0.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e9df38ee2d2c3c5948ea468a8406ff0db0b29ae1ffde1bcf20ef305bcc95c51" + +[[package]] +name = "yoke" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "120e6aef9aa629e3d4f52dc8cc43a015c7724194c97dfaf45180d2daf2b77f40" +dependencies = [ + "serde", + "stable_deref_trait", + "yoke-derive", + "zerofrom", +] + +[[package]] +name = "yoke-derive" +version = "0.7.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2380878cad4ac9aac1e2435f3eb4020e8374b5f13c296cb75b4620ff8e229154" +dependencies = [ + "proc-macro2", + "quote", + "syn", + "synstructure", +] + [[package]] name = "zerocopy" version = "0.7.35" @@ -1839,19 +2088,62 @@ dependencies = [ ] [[package]] -name = "zip" -version = "2.2.0" +name = "zerofrom" +version = "0.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dc5e4288ea4057ae23afc69a4472434a87a2495cafce6632fd1c4ec9f5cf3494" +checksum = "cff3ee08c995dee1859d998dea82f7374f2826091dd9cd47def953cae446cd2e" +dependencies = [ + "zerofrom-derive", +] + +[[package]] +name = "zerofrom-derive" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "595eed982f7d355beb85837f651fa22e90b3c044842dc7f2c2842c086f295808" +dependencies = [ + "proc-macro2", + "quote", + "syn", + "synstructure", +] + +[[package]] +name = "zerovec" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aa2b893d79df23bfb12d5461018d408ea19dfafe76c2c7ef6d4eba614f8ff079" +dependencies = [ + "yoke", + "zerofrom", + "zerovec-derive", +] + +[[package]] +name = "zerovec-derive" +version = "0.10.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6eafa6dfb17584ea3e2bd6e76e0cc15ad7af12b09abdd1ca55961bed9b1063c6" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] + +[[package]] +name = "zip" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "99d52293fc86ea7cf13971b3bb81eb21683636e7ae24c729cdaf1b7c4157a352" dependencies = [ "arbitrary", "crc32fast", "crossbeam-utils", "displaydoc", "flate2", - "indexmap 2.6.0", + "indexmap 2.7.0", "memchr", - "thiserror", + "thiserror 2.0.7", "zopfli", ] diff --git a/Cargo.toml b/Cargo.toml index 2679008..5760878 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -10,13 +10,13 @@ readme = "README.md" [dependencies] anyhow = "1.0.82" -axum = { version = "0.7.7", features = ["macros"] } +axum = { version = "0.7.7", features = ["macros", "ws"] } clap = { version = "4.4.1", features = ["derive"] } clap-verbosity-flag = "2.2.2" env_logger = "0.10.0" futures = "0.3.31" log = "0.4.20" -mpvipc-async = { git = "https://git.pvv.ntnu.no/Grzegorz/mpvipc-async.git", rev = "v0.1.0" } +mpvipc-async = { git = "https://git.pvv.ntnu.no/Grzegorz/mpvipc-async.git", branch = "main" } sd-notify = "0.4.3" serde = { version = "1.0.188", features = ["derive"] } serde_json = "1.0.105" diff --git a/src/api.rs b/src/api.rs index b62472e..604e99e 100644 --- a/src/api.rs +++ b/src/api.rs @@ -1,4 +1,6 @@ mod base; mod rest_wrapper_v1; +mod websocket_v1; pub use rest_wrapper_v1::{rest_api_docs, rest_api_routes}; +pub use websocket_v1::websocket_api; diff --git a/src/api/websocket_v1.rs b/src/api/websocket_v1.rs new file mode 100644 index 0000000..3ef1f32 --- /dev/null +++ b/src/api/websocket_v1.rs @@ -0,0 +1,387 @@ +use std::net::SocketAddr; + +use anyhow::Context; +use futures::{stream::FuturesUnordered, StreamExt}; +use serde::{Deserialize, Serialize}; + +use axum::{ + extract::{ + ws::{Message, WebSocket}, + ConnectInfo, State, WebSocketUpgrade, + }, + response::IntoResponse, + routing::any, + Router, +}; +use mpvipc_async::{ + LoopProperty, Mpv, MpvExt, NumberChangeOptions, Playlist, PlaylistAddTypeOptions, SeekOptions, + Switch, +}; +use serde_json::{json, Value}; +use tokio::select; + +pub fn websocket_api(mpv: Mpv) -> Router { + Router::new() + .route("/", any(websocket_handler)) + .with_state(mpv) +} + +async fn websocket_handler( + ws: WebSocketUpgrade, + ConnectInfo(addr): ConnectInfo, + State(mpv): State, +) -> impl IntoResponse { + let mpv = mpv.clone(); + + // TODO: get an id provisioned by the id pool + ws.on_upgrade(move |socket| handle_connection(socket, addr, mpv, 1)) +} + +#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] +pub struct InitialState { + pub cached_timestamp: Option, + pub chapters: Vec, + pub current_percent_pos: Option, + pub current_track: String, + pub duration: f64, + pub is_looping: bool, + pub is_muted: bool, + pub is_playing: bool, + pub is_paused_for_cache: bool, + pub playlist: Playlist, + pub tracks: Vec, + pub volume: f64, +} + +async fn get_initial_state(mpv: &Mpv) -> InitialState { + let cached_timestamp = mpv + .get_property_value("demuxer-cache-state") + .await + .unwrap_or(None) + .and_then(|v| { + v.as_object() + .and_then(|o| o.get("data")) + .and_then(|v| v.as_object()) + .and_then(|o| o.get("cache-end")) + .and_then(|v| v.as_f64()) + }); + let chapters = match mpv.get_property_value("chapter-list").await { + Ok(Some(Value::Array(chapters))) => chapters, + _ => vec![], + }; + let current_percent_pos = mpv.get_property("percent-pos").await.unwrap_or(None); + let current_track = mpv.get_file_path().await.unwrap_or("".to_string()); + let duration = mpv.get_duration().await.unwrap_or(0.0); + let is_looping = + mpv.playlist_is_looping().await.unwrap_or(LoopProperty::No) != LoopProperty::No; + let is_muted = mpv + .get_property("mute") + .await + .unwrap_or(Some(false)) + .unwrap_or(false); + let is_playing = mpv.is_playing().await.unwrap_or(false); + let is_paused_for_cache = mpv + .get_property("paused-for-cache") + .await + .unwrap_or(Some(false)) + .unwrap_or(false); + let playlist = mpv.get_playlist().await.unwrap_or(Playlist(vec![])); + let tracks = match mpv.get_property_value("track-list").await { + Ok(Some(Value::Array(tracks))) => tracks + .into_iter() + .filter(|t| { + t.as_object() + .and_then(|o| o.get("type")) + .and_then(|t| t.as_str()) + .unwrap_or("") + == "sub" + }) + .collect(), + _ => vec![], + }; + let volume = mpv.get_volume().await.unwrap_or(0.0); + // TODO: use default when new version is released + InitialState { + cached_timestamp, + chapters, + current_percent_pos, + current_track, + duration, + is_looping, + is_muted, + is_playing, + is_paused_for_cache, + playlist, + tracks, + volume, + } +} + +const DEFAULT_PROPERTY_SUBSCRIPTIONS: [&str; 11] = [ + "chapter-list", + "demuxer-cache-state", + "duration", + "loop-playlist", + "mute", + "pause", + "paused-for-cache", + "percent-pos", + "playlist", + "track-list", + "volume", +]; + +async fn setup_default_subscribes(mpv: &Mpv) -> anyhow::Result<()> { + let mut futures = FuturesUnordered::new(); + + futures.extend( + DEFAULT_PROPERTY_SUBSCRIPTIONS + .iter() + .map(|property| mpv.observe_property(0, property)), + ); + + while let Some(result) = futures.next().await { + result?; + } + + Ok(()) +} + +async fn handle_connection(mut socket: WebSocket, addr: SocketAddr, mpv: Mpv, channel_id: u64) { + // TODO: There is an asynchronous gap between gathering the initial state and subscribing to the properties + // This could lead to missing events if they happen in that gap. Send initial state, but also ensure + // that there is an additional "initial state" sent upon subscription to all properties to ensure that + // the state is correct. + let initial_state = get_initial_state(&mpv).await; + + let message = Message::Text( + json!({ + "type": "initial_state", + "value": initial_state, + }) + .to_string(), + ); + + socket.send(message).await.unwrap(); + + setup_default_subscribes(&mpv).await.unwrap(); + + let connection_loop_mpv = mpv.clone(); + let connection_loop = tokio::spawn(async move { + let mut event_stream = connection_loop_mpv.get_event_stream().await; + loop { + select! { + message = socket.recv() => { + log::trace!("Received command from {:?}: {:?}", addr, message); + + let ws_message_content = message + .ok_or(anyhow::anyhow!("Event stream ended for {:?}", addr)) + .and_then(|message| { + match message { + Ok(message) => Ok(message), + err => Err(anyhow::anyhow!("Error reading message for {:?}: {:?}", addr, err)), + } + })?; + + if let Message::Close(_) = ws_message_content { + log::trace!("Closing connection for {:?}", addr); + return Ok(()); + } + + if let Message::Ping(xs) = ws_message_content { + log::trace!("Ponging {:?} with {:?}", addr, xs); + socket.send(Message::Pong(xs)).await?; + continue; + } + + let message_content = match ws_message_content { + Message::Text(text) => text, + m => anyhow::bail!("Unexpected message type: {:?}", m), + }; + + let message_json = match serde_json::from_str::(&message_content) { + Ok(json) => json, + Err(e) => anyhow::bail!("Error parsing message from {:?}: {:?}", addr, e), + }; + + log::trace!("Handling command from {:?}: {:?}", addr, message_json); + + // TODO: handle errors + match handle_message(message_json, connection_loop_mpv.clone(), channel_id).await { + Ok(Some(response)) => { + log::trace!("Handled command from {:?} successfully, sending response", addr); + let message = Message::Text(json!({ + "type": "response", + "value": response, + }).to_string()); + socket.send(message).await?; + } + Ok(None) => { + log::trace!("Handled command from {:?} successfully", addr); + } + Err(e) => { + log::error!("Error handling message from {:?}: {:?}", addr, e); + } + } + } + event = event_stream.next() => { + match event { + Some(Ok(event)) => { + log::trace!("Sending event to {:?}: {:?}", addr, event); + let message = Message::Text(json!({ + "type": "event", + "value": event, + }).to_string()); + socket.send(message).await?; + } + Some(Err(e)) => { + log::error!("Error reading event stream for {:?}: {:?}", addr, e); + anyhow::bail!("Error reading event stream for {:?}: {:?}", addr, e); + } + None => { + log::trace!("Event stream ended for {:?}", addr); + return Ok(()); + } + } + } + } + } + }); + + match connection_loop.await { + Ok(Ok(())) => { + log::trace!("Connection loop ended for {:?}", addr); + } + Ok(Err(e)) => { + log::error!("Error in connection loop for {:?}: {:?}", addr, e); + } + Err(e) => { + log::error!("Error in connection loop for {:?}: {:?}", addr, e); + } + } + + match mpv.unobserve_property(channel_id).await { + Ok(()) => { + log::trace!("Unsubscribed from properties for {:?}", addr); + } + Err(e) => { + log::error!( + "Error unsubscribing from properties for {:?}: {:?}", + addr, + e + ); + } + } +} + +#[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] +#[serde(tag = "type", rename_all = "snake_case")] +pub enum WSCommand { + // Subscribe { property: String }, + // UnsubscribeAll, + + Load { urls: Vec }, + TogglePlayback, + Volume { volume: f64 }, + Time { time: f64 }, + PlaylistNext, + PlaylistPrevious, + PlaylistGoto { position: usize }, + PlaylistClear, + PlaylistRemove { positions: Vec }, + PlaylistMove { from: usize, to: usize }, + Shuffle, + SetSubtitleTrack { track: Option }, + SetLooping { value: bool }, +} + +async fn handle_message( + message: Value, + mpv: Mpv, + _channel_id: u64, +) -> anyhow::Result> { + let command = + serde_json::from_value::(message).context("Failed to parse message")?; + + log::trace!("Successfully parsed message: {:?}", command); + + match command { + // WSCommand::Subscribe { property } => { + // mpv.observe_property(channel_id, &property).await?; + // Ok(None) + // } + // WSCommand::UnsubscribeAll => { + // mpv.unobserve_property(channel_id).await?; + // Ok(None) + // } + WSCommand::Load { urls } => { + for url in urls { + mpv.playlist_add( + &url, + PlaylistAddTypeOptions::File, + mpvipc_async::PlaylistAddOptions::Append, + ) + .await?; + } + Ok(None) + } + WSCommand::TogglePlayback => { + mpv.set_playback(mpvipc_async::Switch::Toggle).await?; + Ok(None) + } + WSCommand::Volume { volume } => { + mpv.set_volume(volume, NumberChangeOptions::Absolute) + .await?; + Ok(None) + } + WSCommand::Time { time } => { + mpv.seek(time, SeekOptions::AbsolutePercent).await?; + Ok(None) + } + WSCommand::PlaylistNext => { + mpv.next().await?; + Ok(None) + } + WSCommand::PlaylistPrevious => { + mpv.prev().await?; + Ok(None) + } + WSCommand::PlaylistGoto { position } => { + mpv.playlist_play_id(position).await?; + Ok(None) + } + WSCommand::PlaylistClear => { + mpv.playlist_clear().await?; + Ok(None) + } + + // FIXME: this could lead to a race condition between `playlist_remove_id` commands + WSCommand::PlaylistRemove { mut positions } => { + positions.sort(); + + for position in positions.iter().rev() { + mpv.playlist_remove_id(*position).await?; + } + + Ok(None) + } + + WSCommand::PlaylistMove { from, to } => { + mpv.playlist_move_id(from, to).await?; + Ok(None) + } + WSCommand::Shuffle => { + mpv.playlist_shuffle().await?; + Ok(None) + } + WSCommand::SetSubtitleTrack { track } => { + mpv.set_property("sid", track).await?; + Ok(None) + } + WSCommand::SetLooping { value } => { + mpv.set_loop_playlist(if value { Switch::On } else { Switch::Off }) + .await?; + Ok(None) + } + } +} diff --git a/src/main.rs b/src/main.rs index aad9afa..0028575 100644 --- a/src/main.rs +++ b/src/main.rs @@ -228,7 +228,9 @@ async fn main() -> anyhow::Result<()> { let app = Router::new() .nest("/api", api::rest_api_routes(mpv.clone())) - .merge(api::rest_api_docs(mpv.clone())); + .nest("/ws", api::websocket_api(mpv.clone())) + .merge(api::rest_api_docs(mpv.clone())) + .into_make_service_with_connect_info::(); let listener = match tokio::net::TcpListener::bind(&socket_addr) .await @@ -265,7 +267,7 @@ async fn main() -> anyhow::Result<()> { log::info!("Received Ctrl-C, exiting"); shutdown(mpv, Some(proc)).await; } - result = axum::serve(listener, app.into_make_service()) => { + result = axum::serve(listener, app) => { log::info!("API server exited"); shutdown(mpv, Some(proc)).await; result?; @@ -277,7 +279,7 @@ async fn main() -> anyhow::Result<()> { log::info!("Received Ctrl-C, exiting"); shutdown(mpv.clone(), None).await; } - result = axum::serve(listener, app.into_make_service()) => { + result = axum::serve(listener, app) => { log::info!("API server exited"); shutdown(mpv.clone(), None).await; result?; -- 2.47.1 From c4e2ade27db6433e691bccb76927c714f07f4a02 Mon Sep 17 00:00:00 2001 From: h7x4 Date: Mon, 6 Jan 2025 19:11:40 +0100 Subject: [PATCH 2/2] websocket-api: use `IdPool` to manage connection ids --- src/api/websocket_v1.rs | 246 +++++++++++++++++++++++++--------------- src/main.rs | 50 ++++---- src/util.rs | 3 + src/util/id_pool.rs | 145 +++++++++++++++++++++++ 4 files changed, 329 insertions(+), 115 deletions(-) create mode 100644 src/util.rs create mode 100644 src/util/id_pool.rs diff --git a/src/api/websocket_v1.rs b/src/api/websocket_v1.rs index 3ef1f32..8436e4f 100644 --- a/src/api/websocket_v1.rs +++ b/src/api/websocket_v1.rs @@ -1,4 +1,7 @@ -use std::net::SocketAddr; +use std::{ + net::SocketAddr, + sync::{Arc, Mutex}, +}; use anyhow::Context; use futures::{stream::FuturesUnordered, StreamExt}; @@ -18,29 +21,45 @@ use mpvipc_async::{ Switch, }; use serde_json::{json, Value}; -use tokio::select; +use tokio::{select, sync::watch}; -pub fn websocket_api(mpv: Mpv) -> Router { +use crate::util::IdPool; + +#[derive(Debug, Clone)] +struct WebsocketState { + mpv: Mpv, + id_pool: Arc>, +} + +pub fn websocket_api(mpv: Mpv, id_pool: Arc>) -> Router { + let state = WebsocketState { mpv, id_pool }; Router::new() .route("/", any(websocket_handler)) - .with_state(mpv) + .with_state(state) } async fn websocket_handler( ws: WebSocketUpgrade, ConnectInfo(addr): ConnectInfo, - State(mpv): State, + State(WebsocketState { mpv, id_pool }): State, ) -> impl IntoResponse { let mpv = mpv.clone(); + let id = match id_pool.lock().unwrap().request_id() { + Ok(id) => id, + Err(e) => { + log::error!("Failed to get id from id pool: {:?}", e); + return axum::http::StatusCode::INTERNAL_SERVER_ERROR.into_response(); + } + }; - // TODO: get an id provisioned by the id pool - ws.on_upgrade(move |socket| handle_connection(socket, addr, mpv, 1)) + ws.on_upgrade(move |socket| handle_connection(socket, addr, mpv, id, id_pool)) } #[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] pub struct InitialState { pub cached_timestamp: Option, pub chapters: Vec, + pub connections: u64, pub current_percent_pos: Option, pub current_track: String, pub duration: f64, @@ -53,7 +72,7 @@ pub struct InitialState { pub volume: f64, } -async fn get_initial_state(mpv: &Mpv) -> InitialState { +async fn get_initial_state(mpv: &Mpv, id_pool: Arc>) -> InitialState { let cached_timestamp = mpv .get_property_value("demuxer-cache-state") .await @@ -69,6 +88,7 @@ async fn get_initial_state(mpv: &Mpv) -> InitialState { Ok(Some(Value::Array(chapters))) => chapters, _ => vec![], }; + let connections = id_pool.lock().unwrap().id_count(); let current_percent_pos = mpv.get_property("percent-pos").await.unwrap_or(None); let current_track = mpv.get_file_path().await.unwrap_or("".to_string()); let duration = mpv.get_duration().await.unwrap_or(0.0); @@ -104,6 +124,7 @@ async fn get_initial_state(mpv: &Mpv) -> InitialState { InitialState { cached_timestamp, chapters, + connections, current_percent_pos, current_track, duration, @@ -147,12 +168,18 @@ async fn setup_default_subscribes(mpv: &Mpv) -> anyhow::Result<()> { Ok(()) } -async fn handle_connection(mut socket: WebSocket, addr: SocketAddr, mpv: Mpv, channel_id: u64) { +async fn handle_connection( + mut socket: WebSocket, + addr: SocketAddr, + mpv: Mpv, + channel_id: u64, + id_pool: Arc>, +) { // TODO: There is an asynchronous gap between gathering the initial state and subscribing to the properties // This could lead to missing events if they happen in that gap. Send initial state, but also ensure // that there is an additional "initial state" sent upon subscription to all properties to ensure that // the state is correct. - let initial_state = get_initial_state(&mpv).await; + let initial_state = get_initial_state(&mpv, id_pool.clone()).await; let message = Message::Text( json!({ @@ -166,89 +193,17 @@ async fn handle_connection(mut socket: WebSocket, addr: SocketAddr, mpv: Mpv, ch setup_default_subscribes(&mpv).await.unwrap(); - let connection_loop_mpv = mpv.clone(); - let connection_loop = tokio::spawn(async move { - let mut event_stream = connection_loop_mpv.get_event_stream().await; - loop { - select! { - message = socket.recv() => { - log::trace!("Received command from {:?}: {:?}", addr, message); + let id_count_watch_receiver = id_pool.lock().unwrap().get_id_count_watch_receiver(); - let ws_message_content = message - .ok_or(anyhow::anyhow!("Event stream ended for {:?}", addr)) - .and_then(|message| { - match message { - Ok(message) => Ok(message), - err => Err(anyhow::anyhow!("Error reading message for {:?}: {:?}", addr, err)), - } - })?; + let connection_loop_result = tokio::spawn(connection_loop( + socket, + addr, + mpv.clone(), + channel_id, + id_count_watch_receiver, + )); - if let Message::Close(_) = ws_message_content { - log::trace!("Closing connection for {:?}", addr); - return Ok(()); - } - - if let Message::Ping(xs) = ws_message_content { - log::trace!("Ponging {:?} with {:?}", addr, xs); - socket.send(Message::Pong(xs)).await?; - continue; - } - - let message_content = match ws_message_content { - Message::Text(text) => text, - m => anyhow::bail!("Unexpected message type: {:?}", m), - }; - - let message_json = match serde_json::from_str::(&message_content) { - Ok(json) => json, - Err(e) => anyhow::bail!("Error parsing message from {:?}: {:?}", addr, e), - }; - - log::trace!("Handling command from {:?}: {:?}", addr, message_json); - - // TODO: handle errors - match handle_message(message_json, connection_loop_mpv.clone(), channel_id).await { - Ok(Some(response)) => { - log::trace!("Handled command from {:?} successfully, sending response", addr); - let message = Message::Text(json!({ - "type": "response", - "value": response, - }).to_string()); - socket.send(message).await?; - } - Ok(None) => { - log::trace!("Handled command from {:?} successfully", addr); - } - Err(e) => { - log::error!("Error handling message from {:?}: {:?}", addr, e); - } - } - } - event = event_stream.next() => { - match event { - Some(Ok(event)) => { - log::trace!("Sending event to {:?}: {:?}", addr, event); - let message = Message::Text(json!({ - "type": "event", - "value": event, - }).to_string()); - socket.send(message).await?; - } - Some(Err(e)) => { - log::error!("Error reading event stream for {:?}: {:?}", addr, e); - anyhow::bail!("Error reading event stream for {:?}: {:?}", addr, e); - } - None => { - log::trace!("Event stream ended for {:?}", addr); - return Ok(()); - } - } - } - } - } - }); - - match connection_loop.await { + match connection_loop_result.await { Ok(Ok(())) => { log::trace!("Connection loop ended for {:?}", addr); } @@ -272,6 +227,114 @@ async fn handle_connection(mut socket: WebSocket, addr: SocketAddr, mpv: Mpv, ch ); } } + + match id_pool.lock().unwrap().release_id(channel_id) { + Ok(()) => { + log::trace!("Released id {} for {:?}", channel_id, addr); + } + Err(e) => { + log::error!("Error releasing id {} for {:?}: {:?}", channel_id, addr, e); + } + } +} + +async fn connection_loop( + mut socket: WebSocket, + addr: SocketAddr, + mpv: Mpv, + channel_id: u64, + mut id_count_watch_receiver: watch::Receiver, +) -> Result<(), anyhow::Error> { + let mut event_stream = mpv.get_event_stream().await; + loop { + select! { + id_count = id_count_watch_receiver.changed() => { + if let Err(e) = id_count { + anyhow::bail!("Error reading id count watch receiver for {:?}: {:?}", addr, e); + } + + let message = Message::Text(json!({ + "type": "connection_count", + "value": id_count_watch_receiver.borrow().clone(), + }).to_string()); + + socket.send(message).await?; + } + message = socket.recv() => { + log::trace!("Received command from {:?}: {:?}", addr, message); + + let ws_message_content = message + .ok_or(anyhow::anyhow!("Event stream ended for {:?}", addr)) + .and_then(|message| { + match message { + Ok(message) => Ok(message), + err => Err(anyhow::anyhow!("Error reading message for {:?}: {:?}", addr, err)), + } + })?; + + if let Message::Close(_) = ws_message_content { + log::trace!("Closing connection for {:?}", addr); + return Ok(()); + } + + if let Message::Ping(xs) = ws_message_content { + log::trace!("Ponging {:?} with {:?}", addr, xs); + socket.send(Message::Pong(xs)).await?; + continue; + } + + let message_content = match ws_message_content { + Message::Text(text) => text, + m => anyhow::bail!("Unexpected message type: {:?}", m), + }; + + let message_json = match serde_json::from_str::(&message_content) { + Ok(json) => json, + Err(e) => anyhow::bail!("Error parsing message from {:?}: {:?}", addr, e), + }; + + log::trace!("Handling command from {:?}: {:?}", addr, message_json); + + // TODO: handle errors + match handle_message(message_json, mpv.clone(), channel_id).await { + Ok(Some(response)) => { + log::trace!("Handled command from {:?} successfully, sending response", addr); + let message = Message::Text(json!({ + "type": "response", + "value": response, + }).to_string()); + socket.send(message).await?; + } + Ok(None) => { + log::trace!("Handled command from {:?} successfully", addr); + } + Err(e) => { + log::error!("Error handling message from {:?}: {:?}", addr, e); + } + } + } + event = event_stream.next() => { + match event { + Some(Ok(event)) => { + log::trace!("Sending event to {:?}: {:?}", addr, event); + let message = Message::Text(json!({ + "type": "event", + "value": event, + }).to_string()); + socket.send(message).await?; + } + Some(Err(e)) => { + log::error!("Error reading event stream for {:?}: {:?}", addr, e); + anyhow::bail!("Error reading event stream for {:?}: {:?}", addr, e); + } + None => { + log::trace!("Event stream ended for {:?}", addr); + return Ok(()); + } + } + } + } + } } #[derive(Debug, Clone, PartialEq, Serialize, Deserialize)] @@ -279,7 +342,6 @@ async fn handle_connection(mut socket: WebSocket, addr: SocketAddr, mpv: Mpv, ch pub enum WSCommand { // Subscribe { property: String }, // UnsubscribeAll, - Load { urls: Vec }, TogglePlayback, Volume { volume: f64 }, diff --git a/src/main.rs b/src/main.rs index 0028575..bd9d29a 100644 --- a/src/main.rs +++ b/src/main.rs @@ -5,13 +5,18 @@ use clap_verbosity_flag::Verbosity; use futures::StreamExt; use mpv_setup::{connect_to_mpv, create_mpv_config_file, show_grzegorz_image}; use mpvipc_async::{Event, Mpv, MpvDataType, MpvExt}; -use std::net::{IpAddr, SocketAddr}; +use std::{ + net::{IpAddr, SocketAddr}, + sync::{Arc, Mutex}, +}; use systemd_journal_logger::JournalLog; use tempfile::NamedTempFile; use tokio::task::JoinHandle; +use util::IdPool; mod api; mod mpv_setup; +mod util; #[derive(Parser)] struct Args { @@ -119,29 +124,26 @@ async fn setup_systemd_notifier(mpv: Mpv) -> anyhow::Result> { systemd_update_play_status(playing, ¤t_song); loop { - match event_stream.next().await { - Some(Ok(Event::PropertyChange { name, data, .. })) => { - match (name.as_str(), data) { - ("media-title", Some(MpvDataType::String(s))) => { - current_song = Some(s); - } - ("media-title", None) => { - current_song = None; - } - ("pause", Some(MpvDataType::Bool(b))) => { - playing = !b; - } - (event_name, _) => { - log::trace!( - "Received unexpected property change on systemd notifier thread: {}", - event_name - ); - } + if let Some(Ok(Event::PropertyChange { name, data, .. })) = event_stream.next().await { + match (name.as_str(), data) { + ("media-title", Some(MpvDataType::String(s))) => { + current_song = Some(s); + } + ("media-title", None) => { + current_song = None; + } + ("pause", Some(MpvDataType::Bool(b))) => { + playing = !b; + } + (event_name, _) => { + log::trace!( + "Received unexpected property change on systemd notifier thread: {}", + event_name + ); } - - systemd_update_play_status(playing, ¤t_song) } - _ => {} + + systemd_update_play_status(playing, ¤t_song) } } }); @@ -226,9 +228,11 @@ async fn main() -> anyhow::Result<()> { let socket_addr = SocketAddr::new(addr, args.port); log::info!("Starting API on {}", socket_addr); + let id_pool = Arc::new(Mutex::new(IdPool::new_with_max_limit(1024))); + let app = Router::new() .nest("/api", api::rest_api_routes(mpv.clone())) - .nest("/ws", api::websocket_api(mpv.clone())) + .nest("/ws", api::websocket_api(mpv.clone(), id_pool.clone())) .merge(api::rest_api_docs(mpv.clone())) .into_make_service_with_connect_info::(); diff --git a/src/util.rs b/src/util.rs new file mode 100644 index 0000000..451206e --- /dev/null +++ b/src/util.rs @@ -0,0 +1,3 @@ +mod id_pool; + +pub use id_pool::IdPool; diff --git a/src/util/id_pool.rs b/src/util/id_pool.rs new file mode 100644 index 0000000..747ae07 --- /dev/null +++ b/src/util/id_pool.rs @@ -0,0 +1,145 @@ +use std::{collections::BTreeSet, fmt::Debug}; + +use tokio::sync::watch; + +/// A relatively naive ID pool implementation. +pub struct IdPool { + max_id: u64, + free_ids: BTreeSet, + id_count: u64, + id_count_watch_sender: watch::Sender, + id_count_watch_receiver: watch::Receiver, +} + +impl Debug for IdPool { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + f.debug_struct("IdPool") + .field("max_id", &self.max_id) + .field("free_ids", &self.free_ids) + .field("id_count", &self.id_count) + .finish() + } +} + +impl Default for IdPool { + fn default() -> Self { + let (id_count_watch_sender, id_count_watch_receiver) = watch::channel(0); + Self { + max_id: u64::MAX, + free_ids: BTreeSet::new(), + id_count: 0, + id_count_watch_sender, + id_count_watch_receiver, + } + } +} + +//TODO: thiserror + +#[derive(Debug, Clone, PartialEq, Eq)] +pub enum IdPoolError { + NoFreeIds, + IdNotInUse(u64), + IdOutOfBound(u64), +} + +impl IdPool { + pub fn new_with_max_limit(max_id: u64) -> Self { + let (id_count_watch_sender, id_count_watch_receiver) = watch::channel(0); + Self { + max_id, + free_ids: BTreeSet::new(), + id_count: 0, + id_count_watch_sender, + id_count_watch_receiver, + } + } + + pub fn id_count(&self) -> u64 { + self.id_count - self.free_ids.len() as u64 + } + + pub fn id_is_used(&self, id: u64) -> Result { + if id > self.max_id { + Err(IdPoolError::IdOutOfBound(id)) + } else if self.free_ids.contains(&id) { + return Ok(false); + } else { + return Ok(id <= self.id_count); + } + } + + pub fn request_id(&mut self) -> Result { + if !self.free_ids.is_empty() { + let id = self.free_ids.pop_first().unwrap(); + self.update_watch(); + Ok(id) + } else if self.id_count < self.max_id { + self.id_count += 1; + self.update_watch(); + Ok(self.id_count) + } else { + Err(IdPoolError::NoFreeIds) + } + } + + pub fn release_id(&mut self, id: u64) -> Result<(), IdPoolError> { + if !self.id_is_used(id)? { + Err(IdPoolError::IdNotInUse(id)) + } else { + self.free_ids.insert(id); + self.update_watch(); + Ok(()) + } + } + + fn update_watch(&self) { + self.id_count_watch_sender.send(self.id_count()).unwrap(); + } + + pub fn get_id_count_watch_receiver(&self) -> watch::Receiver { + self.id_count_watch_receiver.clone() + } +} + +#[cfg(test)] +mod tests { + use super::*; + + #[test] + fn test_id_pool() { + let mut pool = IdPool::new_with_max_limit(10); + assert_eq!(pool.request_id(), Ok(1)); + assert_eq!(pool.request_id(), Ok(2)); + assert_eq!(pool.request_id(), Ok(3)); + assert_eq!(pool.request_id(), Ok(4)); + assert_eq!(pool.id_count(), 4); + assert_eq!(pool.request_id(), Ok(5)); + assert_eq!(pool.request_id(), Ok(6)); + assert_eq!(pool.request_id(), Ok(7)); + assert_eq!(pool.request_id(), Ok(8)); + assert_eq!(pool.request_id(), Ok(9)); + assert_eq!(pool.request_id(), Ok(10)); + assert_eq!(pool.id_count(), 10); + assert_eq!(pool.request_id(), Err(IdPoolError::NoFreeIds)); + assert_eq!(pool.release_id(5), Ok(())); + assert_eq!(pool.release_id(5), Err(IdPoolError::IdNotInUse(5))); + assert_eq!(pool.id_count(), 9); + assert_eq!(pool.request_id(), Ok(5)); + assert_eq!(pool.release_id(11), Err(IdPoolError::IdOutOfBound(11))); + } + + #[test] + fn test_id_pool_watch() { + let mut pool = IdPool::new_with_max_limit(10); + let receiver = pool.get_id_count_watch_receiver(); + + assert_eq!(receiver.borrow().clone(), 0); + pool.request_id().unwrap(); + assert_eq!(receiver.borrow().clone(), 1); + pool.request_id().unwrap(); + assert_eq!(receiver.borrow().clone(), 2); + pool.release_id(1).unwrap(); + assert_eq!(receiver.borrow().clone(), 1); + } +} -- 2.47.1