diff --git a/Cargo.lock b/Cargo.lock index 7a29258c..a82ab160 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -131,6 +131,12 @@ dependencies = [ "syn 2.0.72", ] +[[package]] +name = "atomic-waker" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0" + [[package]] name = "atty" version = "0.2.14" @@ -401,13 +407,12 @@ dependencies = [ [[package]] name = "cc" -version = "1.0.105" +version = "1.1.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5208975e568d83b6b05cc0a063c8e7e9acc2b43bee6da15616a5b73e109d7437" +checksum = "504bdec147f2cc13c8b57ed9401fd8a147cc66b67ad5cb241394244f2c947549" dependencies = [ "jobserver", "libc", - "once_cell", ] [[package]] @@ -515,9 +520,9 @@ dependencies = [ [[package]] name = "clap_lex" -version = "0.3.0" +version = "0.3.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0d4198f73e42b4936b35b5bb248d81d2b595ecb170da0bac7655c54eedfa8da8" +checksum = "033f6b7a4acb1f358c742aaca805c939ee73b4c6209ae4318ec7aca81c42e646" dependencies = [ "os_str_bytes", ] @@ -724,6 +729,15 @@ version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e8566979429cf69b49a5c740c60791108e86440e8be149bbea4fe54d2c32d6e2" +[[package]] +name = "deranged" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b42b6fa04a440b495c8b04d0e71b707c585f83cb9cb28cf8cd0d976c315e31b4" +dependencies = [ + "powerfmt", +] + [[package]] name = "digest" version = "0.9.0" @@ -799,21 +813,21 @@ checksum = "28dea519a9695b9977216879a3ebfddf92f1c08c05d984f8996aecd6ecdc811d" [[package]] name = "filetime" -version = "0.2.23" +version = "0.2.24" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1ee447700ac8aa0b2f2bd7bc4462ad686ba06baa6727ac149a2d6277f0d240fd" +checksum = "bf401df4a4e3872c4fe8151134cf483738e74b67fc934d6532c882b3d24a4550" dependencies = [ "cfg-if", "libc", - "redox_syscall 0.4.1", - "windows-sys 0.52.0", + "libredox", + "windows-sys 0.59.0", ] [[package]] name = "flate2" -version = "1.0.30" +version = "1.0.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f54427cfd1c7829e2a139fcefea601bf088ebca651d2bf53ebc600eac295dae" +checksum = "7f211bbe8e69bbd0cfdea405084f128ae8b4aaa6b0b522fc8f2b009084797920" dependencies = [ "crc32fast", "miniz_oxide", @@ -991,16 +1005,16 @@ checksum = "40ecd4077b5ae9fd2e9e169b102c6c330d0605168eb0e8bf79952b256dbefffd" [[package]] name = "h2" -version = "0.3.26" +version = "0.4.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "81fe527a889e1532da5c525686d96d4c2e74cdd345badf8dfef9f6b39dd5f5e8" +checksum = "fa82e28a107a8cc405f0839610bdc9b15f1e25ec7d696aa5cf173edbcb1486ab" dependencies = [ + "atomic-waker", "bytes", "fnv", "futures-core", "futures-sink", - "futures-util", - "http 0.2.12", + "http", "indexmap", "slab", "tokio", @@ -1103,6 +1117,15 @@ dependencies = [ "digest 0.10.7", ] +[[package]] +name = "home" +version = "0.5.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3d1354bf6b7235cb4a0576c2619fd4ed18183f689b12b006a0ee7329eeff9a5" +dependencies = [ + "windows-sys 0.52.0", +] + [[package]] name = "hpke" version = "0.11.0" @@ -1124,17 +1147,6 @@ dependencies = [ "zeroize", ] -[[package]] -name = "http" -version = "0.2.12" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "601cbb57e577e2f5ef5be8e7b83f0f63994f25aa94d673e54a92d5c516d101f1" -dependencies = [ - "bytes", - "fnv", - "itoa", -] - [[package]] name = "http" version = "1.1.0" @@ -1146,17 +1158,6 @@ dependencies = [ "itoa", ] -[[package]] -name = "http-body" -version = "0.4.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7ceab25649e9960c0311ea418d17bee82c0dcec1bd053b5f9a66e265a693bed2" -dependencies = [ - "bytes", - "http 0.2.12", - "pin-project-lite", -] - [[package]] name = "http-body" version = "1.0.1" @@ -1164,7 +1165,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" dependencies = [ "bytes", - "http 1.1.0", + "http", ] [[package]] @@ -1175,8 +1176,8 @@ checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" dependencies = [ "bytes", "futures-util", - "http 1.1.0", - "http-body 1.0.1", + "http", + "http-body", "pin-project-lite", ] @@ -1198,30 +1199,6 @@ version = "2.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4" -[[package]] -name = "hyper" -version = "0.14.30" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a152ddd61dfaec7273fe8419ab357f33aee0d914c5f4efbf0d96fa749eea5ec9" -dependencies = [ - "bytes", - "futures-channel", - "futures-core", - "futures-util", - "h2", - "http 0.2.12", - "http-body 0.4.6", - "httparse", - "httpdate", - "itoa", - "pin-project-lite", - "socket2 0.4.10", - "tokio", - "tower-service", - "tracing", - "want", -] - [[package]] name = "hyper" version = "1.4.1" @@ -1231,8 +1208,9 @@ dependencies = [ "bytes", "futures-channel", "futures-util", - "http 1.1.0", - "http-body 1.0.1", + "h2", + "http", + "http-body", "httparse", "httpdate", "itoa", @@ -1244,53 +1222,39 @@ dependencies = [ [[package]] name = "hyper-rustls" -version = "0.24.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ec3efd23720e2049821a693cbc7e65ea87c72f1c58ff2f9522ff332b1491e590" -dependencies = [ - "futures-util", - "http 0.2.12", - "hyper 0.14.30", - "log", - "rustls 0.21.12", - "rustls-native-certs 0.6.3", - "tokio", - "tokio-rustls 0.24.1", -] - -[[package]] -name = "hyper-rustls" -version = "0.25.0" +version = "0.26.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "399c78f9338483cb7e630c8474b07268983c6bd5acee012e4211f9f7bb21b070" +checksum = "a0bea761b46ae2b24eb4aef630d8d1c398157b6fc29e6350ecf090a0b70c952c" dependencies = [ "futures-util", - "http 0.2.12", - "hyper 0.14.30", + "http", + "hyper", + "hyper-util", "log", "rustls 0.22.4", - "rustls-native-certs 0.7.1", + "rustls-native-certs", "rustls-pki-types", "tokio", "tokio-rustls 0.25.0", + "tower-service", + "webpki-roots 0.26.3", ] [[package]] name = "hyper-rustls" -version = "0.26.0" +version = "0.27.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a0bea761b46ae2b24eb4aef630d8d1c398157b6fc29e6350ecf090a0b70c952c" +checksum = "5ee4be2c948921a1a5320b629c4193916ed787a7f7f293fd3f7f5a6c9de74155" dependencies = [ "futures-util", - "http 1.1.0", - "hyper 1.4.1", + "http", + "hyper", "hyper-util", - "log", - "rustls 0.22.4", - "rustls-native-certs 0.7.1", + "rustls 0.23.12", + "rustls-native-certs", "rustls-pki-types", "tokio", - "tokio-rustls 0.25.0", + "tokio-rustls 0.26.0", "tower-service", "webpki-roots 0.26.3", ] @@ -1302,7 +1266,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7a343d17fe7885302ed7252767dc7bb83609a874b6ff581142241ec4b73957ad" dependencies = [ "http-body-util", - "hyper 1.4.1", + "hyper", "hyper-util", "pin-project-lite", "tokio", @@ -1312,16 +1276,16 @@ dependencies = [ [[package]] name = "hyper-util" -version = "0.1.6" +version = "0.1.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3ab92f4f49ee4fb4f997c784b7a2e0fa70050211e0b6a287f898c3c9785ca956" +checksum = "cde7055719c54e36e95e8719f95883f22072a48ede39db7fc17a4e1d5281e9b9" dependencies = [ "bytes", "futures-channel", "futures-util", - "http 1.1.0", - "http-body 1.0.1", - "hyper 1.4.1", + "http", + "http-body", + "hyper", "pin-project-lite", "socket2 0.5.7", "tokio", @@ -1450,6 +1414,17 @@ version = "0.2.155" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "97b3888a4aecf77e811145cadf6eef5901f4782c53886191b2f693f24761847c" +[[package]] +name = "libredox" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c0ff37bd590ca25063e35af745c343cb7a0271906fb7b37e4813e8f79f00268d" +dependencies = [ + "bitflags 2.6.0", + "libc", + "redox_syscall 0.5.3", +] + [[package]] name = "linked-hash-map" version = "0.5.6" @@ -1531,13 +1506,14 @@ dependencies = [ [[package]] name = "mio" -version = "0.8.11" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a4a650543ca06a924e8b371db273b2756685faae30f8487da1b56505a8f78b0c" +checksum = "4569e456d394deccd22ce1c1913e6ea0e54519f577285001215d33557431afe4" dependencies = [ + "hermit-abi 0.3.9", "libc", "wasi", - "windows-sys 0.48.0", + "windows-sys 0.52.0", ] [[package]] @@ -1561,20 +1537,16 @@ dependencies = [ ] [[package]] -name = "num_cpus" -version = "1.16.0" +name = "num-conv" +version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" -dependencies = [ - "hermit-abi 0.3.9", - "libc", -] +checksum = "51d515d32fb182ee37cda2ccdcb92950d6a3c2893aa280e540671c2cd0f3b1d9" [[package]] name = "object" -version = "0.36.2" +version = "0.36.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3f203fa8daa7bb185f760ae12bd8e097f63d17041dcdcaf675ac54cdf863170e" +checksum = "27b64972346851a39438c60b341ebc01bba47464ae329e55cf343eb93964efd9" dependencies = [ "memchr", ] @@ -1609,9 +1581,9 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7850c40a0aebcba289d3252c0a45f93cba6ad4b0c46b88a5fc51dba6ddce8632" dependencies = [ "futures", - "http 1.1.0", + "http", "http-body-util", - "hyper 1.4.1", + "hyper", "hyper-rustls 0.26.0", "hyper-tungstenite", "hyper-util", @@ -1738,7 +1710,7 @@ dependencies = [ "bitcoin", "bitcoind", "chacha20poly1305 0.10.1", - "http 1.1.0", + "http", "log", "ohttp", "ohttp-relay", @@ -1769,9 +1741,11 @@ dependencies = [ "clap", "config", "env_logger", - "http 1.1.0", - "hyper 0.14.30", - "hyper-rustls 0.25.0", + "http", + "http-body-util", + "hyper", + "hyper-rustls 0.26.0", + "hyper-util", "log", "ohttp-relay", "once_cell", @@ -1785,6 +1759,7 @@ dependencies = [ "testcontainers", "testcontainers-modules", "tokio", + "tokio-rustls 0.25.0", "tracing", "tracing-subscriber", "url", @@ -1798,12 +1773,15 @@ dependencies = [ "bhttp", "bitcoin", "futures", - "hyper 0.14.30", - "hyper-rustls 0.24.2", + "http-body-util", + "hyper", + "hyper-rustls 0.26.0", + "hyper-util", "ohttp", "redis", - "rustls 0.21.12", + "rustls 0.22.4", "tokio", + "tokio-rustls 0.25.0", "tracing", "tracing-subscriber", ] @@ -1971,11 +1949,17 @@ dependencies = [ "universal-hash 0.5.1", ] +[[package]] +name = "powerfmt" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391" + [[package]] name = "ppv-lite86" -version = "0.2.18" +version = "0.2.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dee4364d9f3b902ef14fab8a1ddffb783a1cb6b4bba3bfc1fa3922732c7de97f" +checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04" dependencies = [ "zerocopy", ] @@ -2013,6 +1997,54 @@ dependencies = [ "unicode-ident", ] +[[package]] +name = "quinn" +version = "0.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b22d8e7369034b9a7132bc2008cac12f2013c8132b45e0554e6e20e2617f2156" +dependencies = [ + "bytes", + "pin-project-lite", + "quinn-proto", + "quinn-udp", + "rustc-hash", + "rustls 0.23.12", + "socket2 0.5.7", + "thiserror", + "tokio", + "tracing", +] + +[[package]] +name = "quinn-proto" +version = "0.11.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba92fb39ec7ad06ca2582c0ca834dfeadcaf06ddfc8e635c80aa7e1c05315fdd" +dependencies = [ + "bytes", + "rand", + "ring 0.17.8", + "rustc-hash", + "rustls 0.23.12", + "slab", + "thiserror", + "tinyvec", + "tracing", +] + +[[package]] +name = "quinn-udp" +version = "0.5.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8bffec3605b73c6f1754535084a85229fa8a30f86014e6c81aeec4abb68b0285" +dependencies = [ + "libc", + "once_cell", + "socket2 0.5.7", + "tracing", + "windows-sys 0.52.0", +] + [[package]] name = "quote" version = "1.0.36" @@ -2094,15 +2126,6 @@ dependencies = [ "bitflags 1.3.2", ] -[[package]] -name = "redox_syscall" -version = "0.4.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4722d768eff46b75989dd134e5c353f0d6296e5aaa3132e776cbdb56be7731aa" -dependencies = [ - "bitflags 1.3.2", -] - [[package]] name = "redox_syscall" version = "0.5.3" @@ -2114,14 +2137,14 @@ dependencies = [ [[package]] name = "regex" -version = "1.9.6" +version = "1.10.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ebee201405406dbf528b8b672104ae6d6d63e6d118cb10e4d51abbc7b58044ff" +checksum = "4219d74c6b67a3654a9fbebc4b419e22126d13d2f3c4a07ee0cb61ff79a79619" dependencies = [ "aho-corasick", "memchr", - "regex-automata 0.3.9", - "regex-syntax 0.7.5", + "regex-automata 0.4.7", + "regex-syntax 0.8.4", ] [[package]] @@ -2135,13 +2158,13 @@ dependencies = [ [[package]] name = "regex-automata" -version = "0.3.9" +version = "0.4.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "59b23e92ee4318893fa3fe3e6fb365258efbfe6ac6ab30f090cdcbb7aa37efa9" +checksum = "38caf58cc5ef2fed281f89292ef23f6365465ed9a41b7a7754eb4e26496c92df" dependencies = [ "aho-corasick", "memchr", - "regex-syntax 0.7.5", + "regex-syntax 0.8.4", ] [[package]] @@ -2152,25 +2175,25 @@ checksum = "f162c6dd7b008981e4d40210aca20b4bd0f9b60ca9271061b07f78537722f2e1" [[package]] name = "regex-syntax" -version = "0.7.5" +version = "0.8.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dbb5fb1acd8a1a18b3dd5be62d25485eb770e05afb408a9627d14d451bae12da" +checksum = "7a66a03ae7c801facd77a29370b4faec201768915ac14a721ba36f20bc9c209b" [[package]] name = "reqwest" -version = "0.12.4" +version = "0.12.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "566cafdd92868e0939d3fb961bd0dc25fcfaaed179291093b3d43e6b3150ea10" +checksum = "c7d6d2a27d57148378eb5e111173f4276ad26340ecc5c49a4a2152167a2d6a37" dependencies = [ "base64 0.22.1", "bytes", "futures-core", "futures-util", - "http 1.1.0", - "http-body 1.0.1", + "http", + "http-body", "http-body-util", - "hyper 1.4.1", - "hyper-rustls 0.26.0", + "hyper", + "hyper-rustls 0.27.2", "hyper-util", "ipnet", "js-sys", @@ -2179,16 +2202,17 @@ dependencies = [ "once_cell", "percent-encoding", "pin-project-lite", - "rustls 0.22.4", - "rustls-native-certs 0.7.1", - "rustls-pemfile 2.1.2", + "quinn", + "rustls 0.23.12", + "rustls-native-certs", + "rustls-pemfile", "rustls-pki-types", "serde", "serde_json", "serde_urlencoded", "sync_wrapper", "tokio", - "tokio-rustls 0.25.0", + "tokio-rustls 0.26.0", "tower-service", "url", "wasm-bindgen", @@ -2255,6 +2279,12 @@ version = "0.1.24" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "719b953e2095829ee67db738b3bfa9fa368c94900df327b3f07fe6e794d2fe1f" +[[package]] +name = "rustc-hash" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "583034fd73374156e66797ed8e5b0d5690409c9226b22d87cb7f19821c05d152" + [[package]] name = "rustc_version" version = "0.4.0" @@ -2304,15 +2334,17 @@ dependencies = [ ] [[package]] -name = "rustls-native-certs" -version = "0.6.3" +name = "rustls" +version = "0.23.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a9aace74cb666635c918e9c12bc0d348266037aa8eb599b5cba565709a8dff00" +checksum = "c58f8c84392efc0a126acce10fa59ff7b3d2ac06ab451a33f2741989b806b044" dependencies = [ - "openssl-probe", - "rustls-pemfile 1.0.4", - "schannel", - "security-framework", + "once_cell", + "ring 0.17.8", + "rustls-pki-types", + "rustls-webpki 0.102.6", + "subtle", + "zeroize", ] [[package]] @@ -2322,7 +2354,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a88d6d420651b496bdd98684116959239430022a115c1240e6c3993be0b15fba" dependencies = [ "openssl-probe", - "rustls-pemfile 2.1.2", + "rustls-pemfile", "rustls-pki-types", "schannel", "security-framework", @@ -2330,18 +2362,9 @@ dependencies = [ [[package]] name = "rustls-pemfile" -version = "1.0.4" +version = "2.1.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1c74cae0a4cf6ccbbf5f359f08efdf8ee7e1dc532573bf0db71968cb56b1448c" -dependencies = [ - "base64 0.21.7", -] - -[[package]] -name = "rustls-pemfile" -version = "2.1.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "29993a25686778eb88d4189742cd713c9bce943bc54251a33509dc63cbacf73d" +checksum = "196fe16b00e106300d3e45ecfcb764fa292a535d7326a29a5875c579c7417425" dependencies = [ "base64 0.22.1", "rustls-pki-types", @@ -2349,9 +2372,9 @@ dependencies = [ [[package]] name = "rustls-pki-types" -version = "1.7.0" +version = "1.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "976295e77ce332211c0d24d92c0e83e50f5c5f046d11082cea19f3df13a3562d" +checksum = "fc0a2ce646f8655401bb81e7927b812614bd5d91dbc968696be50603510fcaf0" [[package]] name = "rustls-webpki" @@ -2457,18 +2480,18 @@ checksum = "61697e0a1c7e512e84a621326239844a24d8207b4669b41bc18b32ea5cbf988b" [[package]] name = "serde" -version = "1.0.204" +version = "1.0.205" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bc76f558e0cbb2a839d37354c575f1dc3fdc6546b5be373ba43d95f231bf7c12" +checksum = "e33aedb1a7135da52b7c21791455563facbbcc43d0f0f66165b42c21b3dfb150" dependencies = [ "serde_derive", ] [[package]] name = "serde_derive" -version = "1.0.204" +version = "1.0.205" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e0cd7e117be63d3c3678776753929474f3b04a43a080c744d6b0ae2a8c28e222" +checksum = "692d6f5ac90220161d6774db30c662202721e64aed9058d2c394f451261420c1" dependencies = [ "proc-macro2", "quote", @@ -2477,9 +2500,9 @@ dependencies = [ [[package]] name = "serde_json" -version = "1.0.121" +version = "1.0.122" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4ab380d7d9f22ef3f21ad3e6c1ebe8e4fc7a2000ccba2e4d71fc96f15b2cb609" +checksum = "784b6203951c57ff748476b126ccb5e8e2959a5c19e5c617ab1956be3dbc68da" dependencies = [ "itoa", "memchr", @@ -2679,9 +2702,9 @@ dependencies = [ [[package]] name = "sync_wrapper" -version = "0.1.2" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" +checksum = "a7065abeca94b6a8a577f9bd45aa0867a2238b74e8eb67cf10d492bc39351394" [[package]] name = "tar" @@ -2696,14 +2719,15 @@ dependencies = [ [[package]] name = "tempfile" -version = "3.10.1" +version = "3.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "85b77fafb263dd9d05cbeac119526425676db3784113aa9295c88498cbf8bff1" +checksum = "04cbcdd0c794ebb0d4cf35e88edd2f7d2c4c3e9a5a6dab322839b321c6a87a64" dependencies = [ "cfg-if", "fastrand", + "once_cell", "rustix", - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -2773,19 +2797,22 @@ dependencies = [ [[package]] name = "time" -version = "0.3.20" +version = "0.3.36" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cd0cbfecb4d19b5ea75bb31ad904eb5b9fa13f21079c3b92017ebdf4999a5890" +checksum = "5dfd88e563464686c916c7e46e623e520ddc6d79fa6641390f2e3fa86e83e885" dependencies = [ + "deranged", + "num-conv", + "powerfmt", "serde", "time-core", ] [[package]] name = "time-core" -version = "0.1.0" +version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2e153e1f1acaef8acc537e68b44906d2db6436e2b35ac2c6b42640fff91f00fd" +checksum = "ef927ca75afb808a4d64dd374f00a2adf8d0fcff8e7b184af886c3c87ec4a3f3" [[package]] name = "tinyvec" @@ -2804,28 +2831,27 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" [[package]] name = "tokio" -version = "1.38.1" +version = "1.39.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eb2caba9f80616f438e09748d5acda951967e1ea58508ef53d9c6402485a46df" +checksum = "daa4fb1bc778bd6f04cbfc4bb2d06a7396a8f299dc33ea1900cedaa316f467b1" dependencies = [ "backtrace", "bytes", "libc", "mio", - "num_cpus", "parking_lot 0.12.3", "pin-project-lite", "signal-hook-registry", "socket2 0.5.7", "tokio-macros", - "windows-sys 0.48.0", + "windows-sys 0.52.0", ] [[package]] name = "tokio-macros" -version = "2.3.0" +version = "2.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f5ae998a069d4b5aba8ee9dad856af7d520c3699e6159b185c2acd48155d39a" +checksum = "693d596312e88961bc67d7f1f97af8a70227d9f90c31bba5806eec004978d752" dependencies = [ "proc-macro2", "quote", @@ -2834,21 +2860,22 @@ dependencies = [ [[package]] name = "tokio-rustls" -version = "0.24.1" +version = "0.25.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c28327cf380ac148141087fbfb9de9d7bd4e84ab5d2c28fbc911d753de8a7081" +checksum = "775e0c0f0adb3a2f22a00c4745d728b479985fc15ee7ca6a2608388c5569860f" dependencies = [ - "rustls 0.21.12", + "rustls 0.22.4", + "rustls-pki-types", "tokio", ] [[package]] name = "tokio-rustls" -version = "0.25.0" +version = "0.26.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "775e0c0f0adb3a2f22a00c4745d728b479985fc15ee7ca6a2608388c5569860f" +checksum = "0c7bc40d0e5a97695bb96e27995cd3a08538541b0a846f65bba7a359f36700d4" dependencies = [ - "rustls 0.22.4", + "rustls 0.23.12", "rustls-pki-types", "tokio", ] @@ -2990,7 +3017,7 @@ dependencies = [ "byteorder", "bytes", "data-encoding", - "http 1.1.0", + "http", "httparse", "log", "rand", @@ -3067,9 +3094,9 @@ checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" [[package]] name = "url" -version = "2.5.0" +version = "2.5.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "31e6302e3bb753d46e83516cae55ae196fc0c309407cf11ab35cc51a4c2a4633" +checksum = "22784dbdf76fdde8af1aeda5622b546b422b6fc585325248a2bf9f5e41e94d6c" dependencies = [ "form_urlencoded", "idna", @@ -3203,13 +3230,14 @@ dependencies = [ [[package]] name = "which" -version = "4.4.0" +version = "4.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2441c784c52b289a054b7201fc93253e288f094e2f4be9058343127c4226a269" +checksum = "87ba24419a2078cd2b0f2ede2691b6c66d8e47836da3b6db8265ebad47afbfc7" dependencies = [ "either", - "libc", + "home", "once_cell", + "rustix", ] [[package]] @@ -3230,11 +3258,11 @@ checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" [[package]] name = "winapi-util" -version = "0.1.8" +version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4d4cc384e1e73b93bafa6fb4f1df8c41695c8a91cf9c4c64358067d15a7b6c6b" +checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" dependencies = [ - "windows-sys 0.52.0", + "windows-sys 0.59.0", ] [[package]] @@ -3261,6 +3289,15 @@ dependencies = [ "windows-targets 0.52.6", ] +[[package]] +name = "windows-sys" +version = "0.59.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b" +dependencies = [ + "windows-targets 0.52.6", +] + [[package]] name = "windows-targets" version = "0.48.5" @@ -3433,9 +3470,9 @@ dependencies = [ [[package]] name = "zerocopy" -version = "0.6.6" +version = "0.7.35" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "854e949ac82d619ee9a14c66a1b674ac730422372ccb759ce0c39cabcf2bf8e6" +checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0" dependencies = [ "byteorder", "zerocopy-derive", @@ -3443,9 +3480,9 @@ dependencies = [ [[package]] name = "zerocopy-derive" -version = "0.6.6" +version = "0.7.35" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "125139de3f6b9d625c39e2efdd73d41bdac468ccd556556440e322be0e1bbd91" +checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e" dependencies = [ "proc-macro2", "quote", @@ -3513,11 +3550,10 @@ dependencies = [ [[package]] name = "zstd-sys" -version = "2.0.8+zstd.1.5.5" +version = "2.0.13+zstd.1.5.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5556e6ee25d32df2586c098bbfa278803692a20d0ab9565e049480d52707ec8c" +checksum = "38ff0f21cfee8f97d94cef41359e0c89aa6113028ab0291aa8ca0038995a95aa" dependencies = [ "cc", - "libc", "pkg-config", ] diff --git a/payjoin-cli/Cargo.toml b/payjoin-cli/Cargo.toml index 3f615204..b998daa5 100644 --- a/payjoin-cli/Cargo.toml +++ b/payjoin-cli/Cargo.toml @@ -20,7 +20,7 @@ path = "src/main.rs" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [features] native-certs = ["reqwest/rustls-tls-native-roots"] -danger-local-https = ["rcgen", "reqwest/rustls-tls", "rustls", "hyper-rustls", "payjoin/danger-local-https"] +danger-local-https = ["rcgen", "reqwest/rustls-tls", "rustls", "hyper-rustls", "payjoin/danger-local-https", "tokio-rustls"] v2 = ["payjoin/v2", "payjoin/io"] [dependencies] @@ -31,16 +31,19 @@ bitcoincore-rpc = "0.19.0" clap = { version = "~4.0.32", features = ["derive"] } config = "0.13.3" env_logger = "0.9.0" -hyper = { version = "0.14", features = ["full"] } -hyper-rustls = { version = "0.25", optional = true } +http-body-util = "0.1" +hyper = { version = "1", features = ["full"] } +hyper-rustls = { version = "0.26", optional = true } +hyper-util = "0.1" log = "0.4.7" payjoin = { version = "0.19.0", features = ["send", "receive", "base64"] } rcgen = { version = "0.11.1", optional = true } reqwest = { version = "0.12", default-features = false } -rustls = { version = "0.22.2", optional = true } +rustls = { version = "0.22.4", optional = true } serde = { version = "1.0.160", features = ["derive"] } sled = "0.34" tokio = { version = "1.12.0", features = ["full"] } +tokio-rustls = { version = "0.25", features = ["ring"], default-features = false, optional = true } url = { version = "2.3.1", features = ["serde"] } [dev-dependencies] diff --git a/payjoin-cli/src/app/v1.rs b/payjoin-cli/src/app/v1.rs index da73cadb..b2b9bab5 100644 --- a/payjoin-cli/src/app/v1.rs +++ b/payjoin-cli/src/app/v1.rs @@ -6,12 +6,18 @@ use std::sync::Arc; use anyhow::{anyhow, Context, Result}; use bitcoincore_rpc::bitcoin::Amount; use bitcoincore_rpc::RpcApi; -use hyper::service::{make_service_fn, service_fn}; -use hyper::{Body, Method, Request, Response, Server, StatusCode}; +use http_body_util::combinators::BoxBody; +use http_body_util::{BodyExt, Full}; +use hyper::body::{Buf, Bytes, Incoming}; +use hyper::server::conn::http1; +use hyper::service::service_fn; +use hyper::{Method, Request, Response, StatusCode}; +use hyper_util::rt::TokioIo; use payjoin::bitcoin::psbt::Psbt; use payjoin::bitcoin::{self}; use payjoin::receive::{PayjoinProposal, UncheckedProposal}; use payjoin::{Error, PjUriBuilder, Uri, UriExt}; +use tokio::net::TcpListener; use super::config::AppConfig; use super::App as AppTrait; @@ -124,46 +130,66 @@ impl App { async fn start_http_server(self) -> Result<()> { let addr = SocketAddr::from(([0, 0, 0, 0], self.config.port)); - #[cfg(feature = "danger-local-https")] - let server = { - use std::io::Write; - - use hyper::server::conn::AddrIncoming; - use rustls::pki_types::{CertificateDer, PrivateKeyDer, PrivatePkcs8KeyDer}; - - let cert = rcgen::generate_simple_self_signed(vec!["localhost".to_string()])?; - let cert_der = cert.serialize_der()?; - let mut local_cert_path = std::env::temp_dir(); - local_cert_path.push(LOCAL_CERT_FILE); - let mut file = std::fs::File::create(local_cert_path)?; - file.write_all(&cert_der)?; - let key = - PrivateKeyDer::from(PrivatePkcs8KeyDer::from(cert.serialize_private_key_der())); - let certs = vec![CertificateDer::from(cert.serialize_der()?)]; - let incoming = AddrIncoming::bind(&addr)?; - let acceptor = hyper_rustls::TlsAcceptor::builder() - .with_single_cert(certs, key) - .map_err(|e| anyhow::anyhow!("TLS error: {}", e))? - .with_all_versions_alpn() - .with_incoming(incoming); - Server::builder(acceptor) - }; - - #[cfg(not(feature = "danger-local-https"))] - let server = Server::bind(&addr); + let listener = TcpListener::bind(addr).await?; let app = self.clone(); - let make_svc = make_service_fn(|_| { + + #[cfg(feature = "danger-local-https")] + let tls_acceptor = Self::init_tls_acceptor()?; + while let Ok((stream, _)) = listener.accept().await { let app = app.clone(); - async move { - let handler = move |req| app.clone().handle_web_request(req); - Ok::<_, hyper::Error>(service_fn(handler)) - } - }); - server.serve(make_svc).await?; + #[cfg(feature = "danger-local-https")] + let tls_acceptor = tls_acceptor.clone(); + tokio::spawn(async move { + #[cfg(feature = "danger-local-https")] + let stream = match tls_acceptor.accept(stream).await { + Ok(tls_stream) => tls_stream, + Err(e) => { + log::error!("TLS accept error: {}", e); + return; + } + }; + + let _ = http1::Builder::new() + .serve_connection( + TokioIo::new(stream), + service_fn(move |req| app.clone().handle_web_request(req)), + ) + .await; + }); + } Ok(()) } - async fn handle_web_request(self, req: Request) -> Result> { + #[cfg(feature = "danger-local-https")] + fn init_tls_acceptor() -> Result { + use std::io::Write; + + use rustls::pki_types::{CertificateDer, PrivateKeyDer}; + use rustls::ServerConfig; + use tokio_rustls::TlsAcceptor; + + let cert = rcgen::generate_simple_self_signed(vec!["localhost".to_string()])?; + let cert_der = cert.serialize_der()?; + let mut local_cert_path = std::env::temp_dir(); + local_cert_path.push(LOCAL_CERT_FILE); + let mut file = std::fs::File::create(local_cert_path)?; + file.write_all(&cert_der)?; + let key = PrivateKeyDer::try_from(cert.serialize_private_key_der()) + .map_err(|e| anyhow::anyhow!("Could not parse key: {}", e))?; + let certs = vec![CertificateDer::from(cert_der)]; + let mut server_config = ServerConfig::builder() + .with_no_client_auth() + .with_single_cert(certs, key) + .map_err(|e| anyhow::anyhow!("TLS error: {}", e))?; + server_config.alpn_protocols = + vec![b"h2".to_vec(), b"http/1.1".to_vec(), b"http/1.0".to_vec()]; + Ok(TlsAcceptor::from(Arc::new(server_config))) + } + + async fn handle_web_request( + self, + req: Request, + ) -> Result>> { log::debug!("Received request: {:?}", req); let mut response = match (req.method(), req.uri().path()) { (&Method::GET, "/bip21") => { @@ -177,7 +203,7 @@ impl App { self.handle_get_bip21(amount) .map_err(|e| { log::error!("Error handling request: {}", e); - Response::builder().status(500).body(Body::from(e.to_string())).unwrap() + Response::builder().status(500).body(full(e.to_string())).unwrap() }) .unwrap_or_else(|err_resp| err_resp) } @@ -187,18 +213,15 @@ impl App { .map_err(|e| match e { Error::BadRequest(e) => { log::error!("Error handling request: {}", e); - Response::builder().status(400).body(Body::from(e.to_string())).unwrap() + Response::builder().status(400).body(full(e.to_string())).unwrap() } e => { log::error!("Error handling request: {}", e); - Response::builder().status(500).body(Body::from(e.to_string())).unwrap() + Response::builder().status(500).body(full(e.to_string())).unwrap() } }) .unwrap_or_else(|err_resp| err_resp), - _ => Response::builder() - .status(StatusCode::NOT_FOUND) - .body(Body::from("Not found")) - .unwrap(), + _ => Response::builder().status(StatusCode::NOT_FOUND).body(full("Not found")).unwrap(), }; response .headers_mut() @@ -206,7 +229,10 @@ impl App { Ok(response) } - fn handle_get_bip21(&self, amount: Option) -> Result, Error> { + fn handle_get_bip21( + &self, + amount: Option, + ) -> Result>, Error> { let address = self .bitcoind() .map_err(|e| Error::Server(e.into()))? @@ -227,16 +253,17 @@ impl App { .map_err(|_| Error::Server(anyhow!("Could not parse payjoin URI string.").into()))?; let _ = uri.assume_checked(); // we just got it from bitcoind above - Ok(Response::new(Body::from(uri_string))) + Ok(Response::new(full(uri_string))) } - async fn handle_payjoin_post(&self, req: Request) -> Result, Error> { + async fn handle_payjoin_post( + &self, + req: Request, + ) -> Result>, Error> { let (parts, body) = req.into_parts(); let headers = Headers(&parts.headers); let query_string = parts.uri.query().unwrap_or(""); - let body = std::io::Cursor::new( - hyper::body::to_bytes(body).await.map_err(|e| Error::Server(e.into()))?.to_vec(), - ); + let body = body.collect().await.map_err(|e| Error::Server(e.into()))?.aggregate().reader(); let proposal = payjoin::receive::UncheckedProposal::from_request(body, query_string, headers)?; @@ -247,7 +274,7 @@ impl App { "Responded with Payjoin proposal {}", psbt.clone().extract_tx_unchecked_fee_rate().compute_txid() ); - Ok(Response::new(Body::from(body))) + Ok(Response::new(full(body))) } fn process_v1_proposal(&self, proposal: UncheckedProposal) -> Result { @@ -337,3 +364,7 @@ impl App { Ok(payjoin_proposal) } } + +fn full>(chunk: T) -> BoxBody { + Full::new(chunk.into()).map_err(|never| match never {}).boxed() +} diff --git a/payjoin-directory/Cargo.toml b/payjoin-directory/Cargo.toml index 0b31b0e4..833d7cc1 100644 --- a/payjoin-directory/Cargo.toml +++ b/payjoin-directory/Cargo.toml @@ -14,18 +14,21 @@ resolver = "2" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [features] -danger-local-https = ["hyper-rustls", "rustls"] +danger-local-https = ["hyper-rustls", "rustls", "tokio-rustls"] [dependencies] anyhow = "1.0.71" bitcoin = { version = "0.32.2", features = ["base64"] } bhttp = { version = "=0.5.1", features = ["http"] } futures = "0.3.17" -hyper = { version = "0.14", features = ["full"] } -hyper-rustls = { version = "0.24", optional = true } +http-body-util = "0.1.2" +hyper = { version = "1", features = ["full"] } +hyper-rustls = { version = "0.26", optional = true } +hyper-util = "0.1" ohttp = "0.5.1" redis = { version = "0.23.3", features = ["aio", "tokio-comp"] } -rustls = { version = "0.21", optional = true } +rustls = { version = "0.22.4", optional = true } tokio = { version = "1.12.0", features = ["full"] } +tokio-rustls = { version = "0.25", features = ["ring"], default-features = false, optional = true } tracing = "0.1.37" tracing-subscriber = { version = "0.3.17", features = ["env-filter"] } diff --git a/payjoin-directory/src/lib.rs b/payjoin-directory/src/lib.rs index 024bdc51..685f6bee 100644 --- a/payjoin-directory/src/lib.rs +++ b/payjoin-directory/src/lib.rs @@ -5,9 +5,15 @@ use std::time::Duration; use anyhow::Result; use bitcoin::base64::prelude::BASE64_URL_SAFE_NO_PAD; use bitcoin::base64::Engine; +use http_body_util::combinators::BoxBody; +use http_body_util::{BodyExt, Empty, Full}; +use hyper::body::{Body, Bytes, Incoming}; use hyper::header::{HeaderValue, ACCESS_CONTROL_ALLOW_ORIGIN, CONTENT_TYPE}; -use hyper::service::{make_service_fn, service_fn}; -use hyper::{Body, Method, Request, Response, Server, StatusCode, Uri}; +use hyper::server::conn::http1; +use hyper::service::service_fn; +use hyper::{Method, Request, Response, StatusCode, Uri}; +use hyper_util::rt::TokioIo; +use tokio::net::TcpListener; use tokio::sync::Mutex; use tracing::{debug, error, info, trace}; @@ -31,20 +37,29 @@ pub async fn listen_tcp( ) -> Result<(), Box> { let pool = DbPool::new(timeout, db_host).await?; let ohttp = Arc::new(Mutex::new(init_ohttp()?)); - let make_svc = make_service_fn(|_| { + let bind_addr = SocketAddr::new(IpAddr::V6(Ipv6Addr::UNSPECIFIED), port); + let listener = TcpListener::bind(bind_addr).await?; + while let Ok((stream, _)) = listener.accept().await { let pool = pool.clone(); let ohttp = ohttp.clone(); - async move { - let handler = move |req| handle_ohttp_gateway(req, pool.clone(), ohttp.clone()); - Ok::<_, hyper::Error>(service_fn(handler)) - } - }); + let io = TokioIo::new(stream); + tokio::spawn(async move { + if let Err(err) = http1::Builder::new() + .serve_connection( + io, + service_fn(move |req| { + serve_payjoin_directory(req, pool.clone(), ohttp.clone()) + }), + ) + .with_upgrades() + .await + { + error!("Error serving connection: {:?}", err); + } + }); + } - // Parse the bind address using the provided port - let bind_addr = SocketAddr::new(IpAddr::V6(Ipv6Addr::UNSPECIFIED), port); - let server = Server::bind(&bind_addr).serve(make_svc); - info!("Payjoin Directory awaiting HTTP connection at {}", bind_addr); - Ok(server.await?) + Ok(()) } #[cfg(feature = "danger-local-https")] @@ -56,39 +71,56 @@ pub async fn listen_tcp_with_tls( ) -> Result<(), Box> { let pool = DbPool::new(timeout, db_host).await?; let ohttp = Arc::new(Mutex::new(init_ohttp()?)); - let make_svc = make_service_fn(|_| { + let bind_addr = SocketAddr::new(IpAddr::V6(Ipv6Addr::UNSPECIFIED), port); + let tls_acceptor = init_tls_acceptor(tls_config)?; + let listener = TcpListener::bind(bind_addr).await?; + while let Ok((stream, _)) = listener.accept().await { let pool = pool.clone(); let ohttp = ohttp.clone(); - async move { - let handler = move |req| handle_ohttp_gateway(req, pool.clone(), ohttp.clone()); - Ok::<_, hyper::Error>(service_fn(handler)) - } - }); + let tls_acceptor = tls_acceptor.clone(); + // TODO pass tls_stream as a closure? + tokio::spawn(async move { + let tls_stream = match tls_acceptor.accept(stream).await { + Ok(tls_stream) => tls_stream, + Err(e) => { + error!("TLS accept error: {}", e); + return; + } + }; + if let Err(err) = http1::Builder::new() + .serve_connection( + TokioIo::new(tls_stream), + service_fn(move |req| { + serve_payjoin_directory(req, pool.clone(), ohttp.clone()) + }), + ) + .with_upgrades() + .await + { + error!("Error serving connection: {:?}", err); + } + }); + } - // Parse the bind address using the provided port - let bind_addr = SocketAddr::new(IpAddr::V6(Ipv6Addr::UNSPECIFIED), port); - let server = init_tls_server(&bind_addr, tls_config)?.serve(make_svc); - info!("Payjoin Directory awaiting HTTP connection at {}", bind_addr); - Ok(server.await?) + Ok(()) } #[cfg(feature = "danger-local-https")] -fn init_tls_server( - bind_addr: &SocketAddr, - cert_key: (Vec, Vec), -) -> Result> { - use hyper::server::conn::AddrIncoming; - +fn init_tls_acceptor(cert_key: (Vec, Vec)) -> Result { + use rustls::pki_types::{CertificateDer, PrivateKeyDer}; + use rustls::ServerConfig; + use tokio_rustls::TlsAcceptor; let (cert, key) = cert_key; - let cert = rustls::Certificate(cert); - let key = rustls::PrivateKey(key); - let incoming = AddrIncoming::bind(bind_addr)?; - let acceptor = hyper_rustls::TlsAcceptor::builder() + let cert = CertificateDer::from(cert); + let key = + PrivateKeyDer::try_from(key).map_err(|e| anyhow::anyhow!("Could not parse key: {}", e))?; + + let mut server_config = ServerConfig::builder() + .with_no_client_auth() .with_single_cert(vec![cert], key) - .map_err(|e| anyhow::anyhow!("TLS error: {}", e))? - .with_all_versions_alpn() - .with_incoming(incoming); - Ok(Server::builder(acceptor)) + .map_err(|e| anyhow::anyhow!("TLS error: {}", e))?; + server_config.alpn_protocols = vec![b"h2".to_vec(), b"http/1.1".to_vec(), b"http/1.0".to_vec()]; + Ok(TlsAcceptor::from(Arc::new(server_config))) } fn init_ohttp() -> Result { @@ -108,19 +140,19 @@ fn init_ohttp() -> Result { Ok(ohttp::Server::new(server_config)?) } -async fn handle_ohttp_gateway( - req: Request, +async fn serve_payjoin_directory( + req: Request, pool: DbPool, ohttp: Arc>, -) -> Result> { +) -> Result>> { let path = req.uri().path().to_string(); let query = req.uri().query().unwrap_or_default().to_string(); let (parts, body) = req.into_parts(); let path_segments: Vec<&str> = path.split('/').collect(); - debug!("handle_ohttp_gateway: {:?}", &path_segments); + debug!("serve_payjoin_directory: {:?}", &path_segments); let mut response = match (parts.method, path_segments.as_slice()) { - (Method::POST, ["", ""]) => handle_ohttp(body, pool, ohttp).await, + (Method::POST, ["", ""]) => handle_ohttp_gateway(body, pool, ohttp).await, (Method::GET, ["", "ohttp-keys"]) => get_ohttp_keys(&ohttp).await, (Method::POST, ["", id]) => post_fallback_v1(id, query, body, pool).await, (Method::GET, ["", "health"]) => health_check().await, @@ -134,14 +166,14 @@ async fn handle_ohttp_gateway( Ok(response) } -async fn handle_ohttp( - body: Body, +async fn handle_ohttp_gateway( + body: Incoming, pool: DbPool, ohttp: Arc>, -) -> Result, HandlerError> { +) -> Result>, HandlerError> { // decapsulate let ohttp_body = - hyper::body::to_bytes(body).await.map_err(|e| HandlerError::BadRequest(e.into()))?; + body.collect().await.map_err(|e| HandlerError::BadRequest(e.into()))?.to_bytes(); let ohttp_locked = ohttp.lock().await; let (bhttp_req, res_ctx) = ohttp_locked .decapsulate(&ohttp_body) @@ -161,15 +193,14 @@ async fn handle_ohttp( for header in req.header().fields() { http_req = http_req.header(header.name(), header.value()) } - let request = http_req.body(Body::from(body))?; + let request = http_req.body(full(body))?; let response = handle_v2(pool, request).await?; let (parts, body) = response.into_parts(); let mut bhttp_res = bhttp::Message::response(parts.status.as_u16()); - let full_body = hyper::body::to_bytes(body) - .await - .map_err(|e| HandlerError::InternalServerError(e.into()))?; + let full_body = + body.collect().await.map_err(|e| HandlerError::InternalServerError(e.into()))?.to_bytes(); bhttp_res.write_content(&full_body); let mut bhttp_bytes = Vec::new(); bhttp_res @@ -178,10 +209,13 @@ async fn handle_ohttp( let ohttp_res = res_ctx .encapsulate(&bhttp_bytes) .map_err(|e| HandlerError::InternalServerError(e.into()))?; - Ok(Response::new(Body::from(ohttp_res))) + Ok(Response::new(full(ohttp_res))) } -async fn handle_v2(pool: DbPool, req: Request) -> Result, HandlerError> { +async fn handle_v2( + pool: DbPool, + req: Request>, +) -> Result>, HandlerError> { let path = req.uri().path().to_string(); let (parts, body) = req.into_parts(); @@ -196,8 +230,8 @@ async fn handle_v2(pool: DbPool, req: Request) -> Result, H } } -async fn health_check() -> Result, HandlerError> { - Ok(Response::builder().status(StatusCode::OK).body(Body::empty())?) +async fn health_check() -> Result>, HandlerError> { + Ok(Response::new(empty())) } enum HandlerError { @@ -208,8 +242,8 @@ enum HandlerError { } impl HandlerError { - fn to_response(&self) -> Response { - let mut res = Response::default(); + fn to_response(&self) -> Response> { + let mut res = Response::new(empty()); match self { HandlerError::PayloadTooLarge => *res.status_mut() = StatusCode::PAYLOAD_TOO_LARGE, HandlerError::InternalServerError(e) => { @@ -223,7 +257,7 @@ impl HandlerError { *res.status_mut() = StatusCode::BAD_REQUEST; res.headers_mut() .insert(CONTENT_TYPE, HeaderValue::from_static("application/problem+json")); - *res.body_mut() = Body::from(OHTTP_KEY_REJECTION_RES_JSON); + *res.body_mut() = full(OHTTP_KEY_REJECTION_RES_JSON); } HandlerError::BadRequest(e) => { error!("Bad request: {}", e); @@ -239,9 +273,10 @@ impl From for HandlerError { fn from(e: hyper::http::Error) -> Self { HandlerError::InternalServerError(e.into()) } } -async fn post_session(body: Body) -> Result, HandlerError> { - let bytes = - hyper::body::to_bytes(body).await.map_err(|e| HandlerError::BadRequest(e.into()))?; +async fn post_session( + body: BoxBody, +) -> Result>, HandlerError> { + let bytes = body.collect().await.map_err(|e| HandlerError::BadRequest(e.into()))?.to_bytes(); let base64_id = String::from_utf8(bytes.to_vec()).map_err(|e| HandlerError::BadRequest(e.into()))?; let pubkey_bytes: Vec = @@ -249,57 +284,58 @@ async fn post_session(body: Body) -> Result, HandlerError> { let pubkey = bitcoin::secp256k1::PublicKey::from_slice(&pubkey_bytes) .map_err(|e| HandlerError::BadRequest(e.into()))?; tracing::info!("Initialized session with pubkey: {:?}", pubkey); - Ok(Response::builder().status(StatusCode::NO_CONTENT).body(Body::empty())?) + let mut res = Response::new(empty()); + *res.status_mut() = StatusCode::NO_CONTENT; + Ok(res) } async fn post_fallback_v1( id: &str, query: String, - body: Body, + body: impl Body, pool: DbPool, -) -> Result, HandlerError> { +) -> Result>, HandlerError> { trace!("Post fallback v1"); let none_response = Response::builder() .status(StatusCode::SERVICE_UNAVAILABLE) - .body(Body::from(V1_UNAVAILABLE_RES_JSON))?; + .body(full(V1_UNAVAILABLE_RES_JSON))?; let bad_request_body_res = - Response::builder().status(StatusCode::BAD_REQUEST).body(Body::from(V1_REJECT_RES_JSON))?; + Response::builder().status(StatusCode::BAD_REQUEST).body(full(V1_REJECT_RES_JSON))?; - let body_bytes = match hyper::body::to_bytes(body).await { - Ok(bytes) => bytes.to_vec(), + let body_bytes = match body.collect().await { + Ok(bytes) => bytes.to_bytes(), Err(_) => return Ok(bad_request_body_res), }; - let body_str = match String::from_utf8(body_bytes) { + let body_str = match String::from_utf8(body_bytes.to_vec()) { Ok(body_str) => body_str, Err(_) => return Ok(bad_request_body_res), }; - let v2_compat_body = Body::from(format!("{}\n{}", body_str, query)); + let v2_compat_body = full(format!("{}\n{}", body_str, query)); post_fallback(id, v2_compat_body, pool, none_response).await } async fn post_fallback_v2( id: &str, - body: Body, + body: BoxBody, pool: DbPool, -) -> Result, HandlerError> { +) -> Result>, HandlerError> { trace!("Post fallback v2"); - let none_response = Response::builder().status(StatusCode::ACCEPTED).body(Body::empty())?; + let none_response = Response::builder().status(StatusCode::ACCEPTED).body(empty())?; post_fallback(id, body, pool, none_response).await } async fn post_fallback( id: &str, - body: Body, + body: BoxBody, pool: DbPool, - none_response: Response, -) -> Result, HandlerError> { + none_response: Response>, +) -> Result>, HandlerError> { tracing::trace!("Post fallback"); let id = shorten_string(id); - let req = hyper::body::to_bytes(body) - .await - .map_err(|e| HandlerError::InternalServerError(e.into()))?; + let req = + body.collect().await.map_err(|e| HandlerError::InternalServerError(e.into()))?.to_bytes(); if req.len() > MAX_BUFFER_SIZE { return Err(HandlerError::PayloadTooLarge); } @@ -311,55 +347,70 @@ async fn post_fallback( match pool.peek_res(&id).await { Some(result) => match result { - Ok(buffered_res) => Ok(Response::new(Body::from(buffered_res))), + Ok(buffered_res) => Ok(Response::new(full(buffered_res))), Err(e) => Err(HandlerError::BadRequest(e.into())), }, None => Ok(none_response), } } -async fn get_fallback(id: &str, pool: DbPool) -> Result, HandlerError> { +async fn get_fallback( + id: &str, + pool: DbPool, +) -> Result>, HandlerError> { trace!("GET fallback"); let id = shorten_string(id); match pool.peek_req(&id).await { Some(result) => match result { - Ok(buffered_req) => Ok(Response::new(Body::from(buffered_req))), + Ok(buffered_req) => Ok(Response::new(full(buffered_req))), Err(e) => Err(HandlerError::BadRequest(e.into())), }, - None => Ok(Response::builder().status(StatusCode::ACCEPTED).body(Body::empty())?), + None => Ok(Response::builder().status(StatusCode::ACCEPTED).body(empty())?), } } -async fn post_payjoin(id: &str, body: Body, pool: DbPool) -> Result, HandlerError> { +async fn post_payjoin( + id: &str, + body: BoxBody, + pool: DbPool, +) -> Result>, HandlerError> { trace!("POST payjoin"); let id = shorten_string(id); - let res = hyper::body::to_bytes(body) - .await - .map_err(|e| HandlerError::InternalServerError(e.into()))?; + let res = + body.collect().await.map_err(|e| HandlerError::InternalServerError(e.into()))?.to_bytes(); match pool.push_res(&id, res.into()).await { - Ok(_) => Ok(Response::builder().status(StatusCode::NO_CONTENT).body(Body::empty())?), + Ok(_) => Ok(Response::builder().status(StatusCode::NO_CONTENT).body(empty())?), Err(e) => Err(HandlerError::BadRequest(e.into())), } } -fn not_found() -> Response { +fn not_found() -> Response> { let mut res = Response::default(); *res.status_mut() = StatusCode::NOT_FOUND; res } -async fn get_ohttp_keys(ohttp: &Arc>) -> Result, HandlerError> { - let mut res = Response::default(); - res.headers_mut().insert(CONTENT_TYPE, HeaderValue::from_static("application/ohttp-keys")); +async fn get_ohttp_keys( + ohttp: &Arc>, +) -> Result>, HandlerError> { let ohttp_keys = ohttp .lock() .await .config() .encode() .map_err(|e| HandlerError::InternalServerError(e.into()))?; - *res.body_mut() = Body::from(ohttp_keys); + let mut res = Response::new(full(ohttp_keys)); + res.headers_mut().insert(CONTENT_TYPE, HeaderValue::from_static("application/ohttp-keys")); Ok(res) } fn shorten_string(input: &str) -> String { input.chars().take(8).collect() } + +fn empty() -> BoxBody { + Empty::::new().map_err(|never| match never {}).boxed() +} + +fn full>(chunk: T) -> BoxBody { + Full::new(chunk.into()).map_err(|never| match never {}).boxed() +} diff --git a/payjoin/Cargo.toml b/payjoin/Cargo.toml index ca3ce295..fcd821fb 100644 --- a/payjoin/Cargo.toml +++ b/payjoin/Cargo.toml @@ -33,7 +33,7 @@ bhttp = { version = "=0.5.1", optional = true } ohttp = { version = "0.5.1", optional = true } serde = { version = "1.0.186", default-features = false, optional = true } reqwest = { version = "0.12", default-features = false, optional = true } -rustls = { version = "0.22.2", optional = true } +rustls = { version = "0.22.4", optional = true } url = "2.2.2" serde_json = "1.0.108"