From dffeef56d9bd45e2c6c9bc23add9ef18fe41cd4b Mon Sep 17 00:00:00 2001 From: Nick Johnson Date: Tue, 1 Oct 2024 13:53:42 -0700 Subject: [PATCH] Add high level async protocol interface --- Cargo.lock | 503 ++++++++++++++++++++++++++--------------- protocol/Cargo.toml | 2 + protocol/README.md | 5 +- protocol/src/lib.rs | 334 ++++++++++++++++++++++----- proxy/Cargo.toml | 4 +- proxy/src/bin/proxy.rs | 118 +++------- proxy/src/lib.rs | 66 +----- 7 files changed, 651 insertions(+), 381 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index ad61fc7..a44e52b 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -4,44 +4,44 @@ version = 3 [[package]] name = "addr2line" -version = "0.14.1" +version = "0.24.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a55f82cfe485775d02112886f4169bde0c5894d75e79ead7eafe7e40a25e45f7" +checksum = "f5fb1d8e4442bd405fdfd1dacb42792696b0cf9cb15882e5d097b742a676d375" dependencies = [ "gimli", ] [[package]] -name = "adler" -version = "1.0.2" +name = "adler2" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" +checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627" [[package]] name = "arrayvec" -version = "0.7.4" +version = "0.7.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "96d30a06541fbafbc7f82ed10c06164cfbd2c401138f6addd8404629c4b16711" +checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50" [[package]] name = "autocfg" -version = "1.2.0" +version = "1.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f1fdabc7756949593fe60f30ec81974b613357de856987752631dea1e3394c80" +checksum = "ace50bade8e6234aa140d9a2f552bbee1db4d353f69b8217bc503490fc1a9f26" [[package]] name = "backtrace" -version = "0.3.58" +version = "0.3.74" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "88fb5a785d6b44fd9d6700935608639af1b8356de1e55d5f7c2740f4faa15d82" +checksum = "8d82cb332cdfaed17ae235a638438ac4d4839913cc2af585c3c6746e8f8bee1a" dependencies = [ "addr2line", - "cc", "cfg-if", "libc", "miniz_oxide", "object", "rustc-demangle", + "windows-targets", ] [[package]] @@ -65,6 +65,7 @@ name = "bip324" version = "0.3.1" dependencies = [ "bitcoin", + "futures", "hex-conservative", "rand", ] @@ -75,18 +76,18 @@ version = "0.2.0" dependencies = [ "bip324", "bitcoin", - "bytes", "configure_me", "configure_me_codegen", "hex-conservative", "tokio", + "tokio-util", ] [[package]] name = "bitcoin" -version = "0.32.0" +version = "0.32.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7170e7750a20974246f17ece04311b4205a6155f1db564c5b224af817663c3ea" +checksum = "0032b0e8ead7074cda7fc4f034409607e3f03a6f71d66ade8a307f79b4d99e73" dependencies = [ "base58ck", "bech32", @@ -113,9 +114,9 @@ checksum = "340e09e8399c7bd8912f495af6aa58bea0c9214773417ffaa8f6460f93aaee56" [[package]] name = "bitcoin-units" -version = "0.1.1" +version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cb54da0b28892f3c52203a7191534033e051b6f4b52bc15480681b57b7e036f5" +checksum = "5285c8bcaa25876d07f37e3d30c303f2609179716e11d688f51e8f1fe70063e2" dependencies = [ "bitcoin-internals", ] @@ -132,31 +133,40 @@ dependencies = [ [[package]] name = "bitflags" -version = "1.3.2" +version = "2.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" +checksum = "b048fb63fd8b5923fc5aa7b340d8e156aec7ec02f0c78fa8a6ddc2613f6f71de" + +[[package]] +name = "byteorder" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" [[package]] name = "bytes" -version = "1.6.0" +version = "1.7.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "514de17de45fdb8dc022b1a7975556c53c86f9f0aa5f534b98977b171857c2c9" +checksum = "428d9aa8fbc0670b7b8d6030a7fadd0f86151cae55e4dbbece15f3780a3dfaf3" [[package]] name = "cargo_toml" -version = "0.12.4" +version = "0.20.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6a621d5d6d6c8d086dbaf1fe659981da41a1b63c6bdbba30b4dbb592c6d3bd49" +checksum = "88da5a13c620b4ca0078845707ea9c3faf11edbc3ffd8497d11d686211cd1ac0" dependencies = [ "serde", - "toml", + "toml 0.8.19", ] [[package]] name = "cc" -version = "1.0.90" +version = "1.1.24" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8cd6604a82acf3039f1144f54b8eb34e91ffba622051189e71b781822d5ee1f5" +checksum = "812acba72f0a070b003d3697490d2b55b837230ae7c6c6497f05cc2ddbb8d938" +dependencies = [ + "shlex", +] [[package]] name = "cfg-if" @@ -173,7 +183,7 @@ dependencies = [ "parse_arg", "serde", "serde_derive", - "toml", + "toml 0.5.11", ] [[package]] @@ -186,22 +196,93 @@ dependencies = [ "fmt2io", "serde", "serde_derive", - "toml", + "toml 0.5.11", "unicode-segmentation", "void", ] +[[package]] +name = "equivalent" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" + [[package]] name = "fmt2io" version = "1.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6b6129284da9f7e5296cc22183a63f24300e945e297705dcc0672f7df01d62c8" +[[package]] +name = "futures" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "645c6916888f6cb6350d2550b80fb63e734897a8498abe35cfb732b6487804b0" +dependencies = [ + "futures-channel", + "futures-core", + "futures-io", + "futures-sink", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-channel" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eac8f7d7865dcb88bd4373ab671c8cf4508703796caa2b1985a9ca867b3fcb78" +dependencies = [ + "futures-core", + "futures-sink", +] + +[[package]] +name = "futures-core" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dfc6580bb841c5a68e9ef15c77ccc837b40a7504914d52e47b8b0e9bbda25a1d" + +[[package]] +name = "futures-io" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" + +[[package]] +name = "futures-sink" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9fb8e00e87438d937621c1c6269e53f536c14d3fbd6a042bb24879e57d474fb5" + +[[package]] +name = "futures-task" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38d84fa142264698cdce1a9f9172cf383a0c82de1bddcf3092901442c4097004" + +[[package]] +name = "futures-util" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d6401deb83407ab3da39eba7e33987a73c3df0c82b4bb5813ee871c19c41d48" +dependencies = [ + "futures-channel", + "futures-core", + "futures-io", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "pin-utils", + "slab", +] + [[package]] name = "getrandom" -version = "0.2.12" +version = "0.2.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "190092ea657667030ac6a35e305e62fc4dd69fd98ac98631e5d3a2b1575a12b5" +checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7" dependencies = [ "cfg-if", "libc", @@ -210,9 +291,15 @@ dependencies = [ [[package]] name = "gimli" -version = "0.23.0" +version = "0.31.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32085ea23f3234fc7846555e85283ba4de91e21016dc0455a16286d87a292d64" + +[[package]] +name = "hashbrown" +version = "0.14.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f6503fe142514ca4799d4c26297c4248239fe8838d827db6bd6065c6ed29a6ce" +checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" [[package]] name = "hermit-abi" @@ -222,9 +309,9 @@ checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024" [[package]] name = "hex-conservative" -version = "0.2.0" +version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e1aa273bf451e37ed35ced41c71a5e2a4e29064afb104158f2514bcd71c2c986" +checksum = "5313b072ce3c597065a808dbf612c4c8e8590bdbf8b579508bf7a762c5eae6cd" dependencies = [ "arrayvec", ] @@ -235,64 +322,73 @@ version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3011d1213f159867b13cfd6ac92d2cd5f1345762c63be3554e84092d85a50bbd" +[[package]] +name = "indexmap" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68b900aa2f7301e21c36462b170ee99994de34dff39a4a6a528e80e7376d07e5" +dependencies = [ + "equivalent", + "hashbrown", +] + [[package]] name = "libc" -version = "0.2.153" +version = "0.2.159" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9c198f91728a82281a64e1f4f9eeb25d82cb32a5de251c6bd1b5154d63a8e7bd" +checksum = "561d97a539a36e26a9a5fad1ea11a3039a67714694aaa379433e580854bc3dc5" [[package]] name = "lock_api" -version = "0.4.11" +version = "0.4.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3c168f8615b12bc01f9c17e2eb0cc07dcae1940121185446edc3744920e8ef45" +checksum = "07af8b9cdd281b7915f413fa73f29ebd5d55d0d3f0155584dade1ff18cea1b17" dependencies = [ "autocfg", "scopeguard", ] [[package]] -name = "miniz_oxide" -version = "0.4.4" +name = "memchr" +version = "2.7.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a92518e98c078586bc6c934028adcca4c92a53d6a958196de835170a01d84e4b" -dependencies = [ - "adler", - "autocfg", -] +checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3" [[package]] -name = "mio" -version = "0.8.11" +name = "miniz_oxide" +version = "0.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a4a650543ca06a924e8b371db273b2756685faae30f8487da1b56505a8f78b0c" +checksum = "e2d80299ef12ff69b16a84bb182e3b9df68b5a91574d3d4fa6e41b65deec4df1" dependencies = [ - "libc", - "wasi", - "windows-sys 0.48.0", + "adler2", ] [[package]] -name = "num_cpus" -version = "1.16.0" +name = "mio" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" +checksum = "80e04d1dcff3aae0704555fe5fee3bcfaf3d1fdf8a7e521d5b9d2b42acb52cec" dependencies = [ "hermit-abi", "libc", + "wasi", + "windows-sys", ] [[package]] name = "object" -version = "0.23.0" +version = "0.36.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a9a7ab5d64814df0fe4a4b5ead45ed6c5f181ee3ff04ba344313a6c80446c5d4" +checksum = "084f1a5821ac4c651660a94a7153d27ac9d8a53736203f58b31945ded098070a" +dependencies = [ + "memchr", +] [[package]] name = "parking_lot" -version = "0.12.1" +version = "0.12.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f" +checksum = "f1bf18183cf54e8d6059647fc3063646a1801cf30896933ec2311622cc4b9a27" dependencies = [ "lock_api", "parking_lot_core", @@ -300,15 +396,15 @@ dependencies = [ [[package]] name = "parking_lot_core" -version = "0.9.9" +version = "0.9.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4c42a9226546d68acdd9c0a280d17ce19bfe27a46bf68784e4066115788d008e" +checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" dependencies = [ "cfg-if", "libc", "redox_syscall", "smallvec", - "windows-targets 0.48.5", + "windows-targets", ] [[package]] @@ -319,30 +415,39 @@ checksum = "14248cc8eced350e20122a291613de29e4fa129ba2731818c4cdbb44fccd3e55" [[package]] name = "pin-project-lite" -version = "0.2.13" +version = "0.2.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bda66fc9667c18cb2758a2ac84d1167245054bcf85d5d1aaa6923f45801bdd02" + +[[package]] +name = "pin-utils" +version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8afb450f006bf6385ca15ef45d71d2288452bc3683ce2e2cacc0d18e4be60b58" +checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" [[package]] name = "ppv-lite86" -version = "0.2.17" +version = "0.2.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5b40af805b3121feab8a3c29f04d8ad262fa8e0561883e7653e024ae4479e6de" +checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04" +dependencies = [ + "zerocopy", +] [[package]] name = "proc-macro2" -version = "1.0.79" +version = "1.0.86" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e835ff2298f5721608eb1a980ecaee1aef2c132bf95ecc026a11b7bf3c01c02e" +checksum = "5e719e8df665df0d1c8fbfd238015744736151d4445ec0836b8e628aae103b77" dependencies = [ "unicode-ident", ] [[package]] name = "quote" -version = "1.0.35" +version = "1.0.37" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "291ec9ab5efd934aaf503a6466c5d5251535d108ee747472c3977cc5acc868ef" +checksum = "b5b9d34b8991d19d98081b46eacdd8eb58c6f2b201139f7c5f643cc155a633af" dependencies = [ "proc-macro2", ] @@ -379,18 +484,18 @@ dependencies = [ [[package]] name = "redox_syscall" -version = "0.4.1" +version = "0.5.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4722d768eff46b75989dd134e5c353f0d6296e5aaa3132e776cbdb56be7731aa" +checksum = "9b6dfecf2c74bce2466cabf93f6664d6998a69eb21e39f4207930065b27b771f" dependencies = [ "bitflags", ] [[package]] name = "rustc-demangle" -version = "0.1.23" +version = "0.1.24" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d626bb9dae77e28219937af045c257c28bfd3f69333c512553507f5f9798cb76" +checksum = "719b953e2095829ee67db738b3bfa9fa368c94900df327b3f07fe6e794d2fe1f" [[package]] name = "scopeguard" @@ -400,9 +505,9 @@ checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" [[package]] name = "secp256k1" -version = "0.29.0" +version = "0.29.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0e0cc0f1cf93f4969faf3ea1c7d8a9faed25918d96affa959720823dfe86d4f3" +checksum = "9465315bc9d4566e1724f0fffcbcc446268cb522e60f9a27bcded6b19c108113" dependencies = [ "bitcoin_hashes", "secp256k1-sys", @@ -410,42 +515,66 @@ dependencies = [ [[package]] name = "secp256k1-sys" -version = "0.10.0" +version = "0.10.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1433bd67156263443f14d603720b082dd3121779323fce20cba2aa07b874bc1b" +checksum = "d4387882333d3aa8cb20530a17c69a3752e97837832f34f6dccc760e715001d9" dependencies = [ "cc", ] [[package]] name = "serde" -version = "1.0.205" +version = "1.0.210" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e33aedb1a7135da52b7c21791455563facbbcc43d0f0f66165b42c21b3dfb150" +checksum = "c8e3592472072e6e22e0a54d5904d9febf8508f65fb8552499a1abc7d1078c3a" dependencies = [ "serde_derive", ] [[package]] name = "serde_derive" -version = "1.0.205" +version = "1.0.210" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "692d6f5ac90220161d6774db30c662202721e64aed9058d2c394f451261420c1" +checksum = "243902eda00fad750862fc144cea25caca5e20d615af0a81bee94ca738f1df1f" dependencies = [ "proc-macro2", "quote", "syn", ] +[[package]] +name = "serde_spanned" +version = "0.6.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87607cb1398ed59d48732e575a4c28a7a8ebf2454b964fe3f224f2afc07909e1" +dependencies = [ + "serde", +] + +[[package]] +name = "shlex" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64" + [[package]] name = "signal-hook-registry" -version = "1.4.1" +version = "1.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d8229b473baa5980ac72ef434c4415e70c4b5e71b423043adb4ba059f89c99a1" +checksum = "a9e9e0b4211b72e7b8b6e85c807d36c212bdb33ea8587f7569562a84df5465b1" dependencies = [ "libc", ] +[[package]] +name = "slab" +version = "0.4.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f92a496fb766b417c996b9c5e57daf2f7ad3b0bebe1ccfca4856390e3d3bb67" +dependencies = [ + "autocfg", +] + [[package]] name = "smallvec" version = "1.13.2" @@ -454,19 +583,19 @@ checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" [[package]] name = "socket2" -version = "0.5.6" +version = "0.5.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "05ffd9c0a93b7543e062e759284fcf5f5e3b098501104bfbdde4d404db792871" +checksum = "ce305eb0b4296696835b71df73eb912e0f1ffd2556a501fcede6e0c50349191c" dependencies = [ "libc", - "windows-sys 0.52.0", + "windows-sys", ] [[package]] name = "syn" -version = "2.0.55" +version = "2.0.79" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "002a1b3dbf967edfafc32655d0f377ab0bb7b994aa1d32c8cc7e9b8bf3ebb8f0" +checksum = "89132cd0bf050864e1d38dc3bbc07a0eb8e7530af26344d3d2bbbef83499f590" dependencies = [ "proc-macro2", "quote", @@ -475,34 +604,47 @@ dependencies = [ [[package]] name = "tokio" -version = "1.37.0" +version = "1.40.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1adbebffeca75fcfd058afa480fb6c0b81e165a0323f9c9d39c9697e37c46787" +checksum = "e2b070231665d27ad9ec9b8df639893f46727666c6767db40317fbe920a5d998" dependencies = [ "backtrace", "bytes", "libc", "mio", - "num_cpus", "parking_lot", "pin-project-lite", "signal-hook-registry", "socket2", "tokio-macros", - "windows-sys 0.48.0", + "windows-sys", ] [[package]] name = "tokio-macros" -version = "2.2.0" +version = "2.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5b8a1e28f2deaa14e508979454cb3a223b10b938b45af148bc0986de36f1923b" +checksum = "693d596312e88961bc67d7f1f97af8a70227d9f90c31bba5806eec004978d752" dependencies = [ "proc-macro2", "quote", "syn", ] +[[package]] +name = "tokio-util" +version = "0.7.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61e7c3654c13bcd040d4a03abee2c75b1d14a37b423cf5a813ceae1cc903ec6a" +dependencies = [ + "bytes", + "futures-core", + "futures-io", + "futures-sink", + "pin-project-lite", + "tokio", +] + [[package]] name = "toml" version = "0.5.11" @@ -512,17 +654,51 @@ dependencies = [ "serde", ] +[[package]] +name = "toml" +version = "0.8.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1ed1f98e3fdc28d6d910e6737ae6ab1a93bf1985935a1193e68f93eeb68d24e" +dependencies = [ + "serde", + "serde_spanned", + "toml_datetime", + "toml_edit", +] + +[[package]] +name = "toml_datetime" +version = "0.6.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0dd7358ecb8fc2f8d014bf86f6f638ce72ba252a2c3a2572f2a795f1d23efb41" +dependencies = [ + "serde", +] + +[[package]] +name = "toml_edit" +version = "0.22.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ae48d6208a266e853d946088ed816055e556cc6028c5e8e2b84d9fa5dd7c7f5" +dependencies = [ + "indexmap", + "serde", + "serde_spanned", + "toml_datetime", + "winnow", +] + [[package]] name = "unicode-ident" -version = "1.0.12" +version = "1.0.13" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b" +checksum = "e91b56cd4cadaeb79bbf1a5645f6b4f8dc5bde8834ad5894a8db35fda9efa1fe" [[package]] name = "unicode-segmentation" -version = "1.11.0" +version = "1.12.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d4c87d22b6e3f4a18d4d40ef354e97c90fcb14dd91d7dc0aa9d8a1172ebf7202" +checksum = "f6ccf251212114b54433ec949fd6a7841275f9ada20dddd2f29e9ceea4501493" [[package]] name = "void" @@ -536,134 +712,105 @@ version = "0.11.0+wasi-snapshot-preview1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" -[[package]] -name = "windows-sys" -version = "0.48.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9" -dependencies = [ - "windows-targets 0.48.5", -] - [[package]] name = "windows-sys" version = "0.52.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "282be5f36a8ce781fad8c8ae18fa3f9beff57ec1b52cb3de0789201425d9a33d" dependencies = [ - "windows-targets 0.52.4", -] - -[[package]] -name = "windows-targets" -version = "0.48.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c" -dependencies = [ - "windows_aarch64_gnullvm 0.48.5", - "windows_aarch64_msvc 0.48.5", - "windows_i686_gnu 0.48.5", - "windows_i686_msvc 0.48.5", - "windows_x86_64_gnu 0.48.5", - "windows_x86_64_gnullvm 0.48.5", - "windows_x86_64_msvc 0.48.5", + "windows-targets", ] [[package]] name = "windows-targets" -version = "0.52.4" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7dd37b7e5ab9018759f893a1952c9420d060016fc19a472b4bb20d1bdd694d1b" +checksum = "9b724f72796e036ab90c1021d4780d4d3d648aca59e491e6b98e725b84e99973" dependencies = [ - "windows_aarch64_gnullvm 0.52.4", - "windows_aarch64_msvc 0.52.4", - "windows_i686_gnu 0.52.4", - "windows_i686_msvc 0.52.4", - "windows_x86_64_gnu 0.52.4", - "windows_x86_64_gnullvm 0.52.4", - "windows_x86_64_msvc 0.52.4", + "windows_aarch64_gnullvm", + "windows_aarch64_msvc", + "windows_i686_gnu", + "windows_i686_gnullvm", + "windows_i686_msvc", + "windows_x86_64_gnu", + "windows_x86_64_gnullvm", + "windows_x86_64_msvc", ] [[package]] name = "windows_aarch64_gnullvm" -version = "0.48.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" - -[[package]] -name = "windows_aarch64_gnullvm" -version = "0.52.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bcf46cf4c365c6f2d1cc93ce535f2c8b244591df96ceee75d8e83deb70a9cac9" +checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3" [[package]] name = "windows_aarch64_msvc" -version = "0.48.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc" - -[[package]] -name = "windows_aarch64_msvc" -version = "0.52.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "da9f259dd3bcf6990b55bffd094c4f7235817ba4ceebde8e6d11cd0c5633b675" - -[[package]] -name = "windows_i686_gnu" -version = "0.48.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" +checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469" [[package]] name = "windows_i686_gnu" -version = "0.52.4" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b474d8268f99e0995f25b9f095bc7434632601028cf86590aea5c8a5cb7801d3" +checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b" [[package]] -name = "windows_i686_msvc" -version = "0.48.5" +name = "windows_i686_gnullvm" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" +checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66" [[package]] name = "windows_i686_msvc" -version = "0.52.4" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1515e9a29e5bed743cb4415a9ecf5dfca648ce85ee42e15873c3cd8610ff8e02" +checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66" [[package]] name = "windows_x86_64_gnu" -version = "0.48.5" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" +checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78" [[package]] -name = "windows_x86_64_gnu" -version = "0.52.4" +name = "windows_x86_64_gnullvm" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5eee091590e89cc02ad514ffe3ead9eb6b660aedca2183455434b93546371a03" +checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d" [[package]] -name = "windows_x86_64_gnullvm" -version = "0.48.5" +name = "windows_x86_64_msvc" +version = "0.52.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" +checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec" [[package]] -name = "windows_x86_64_gnullvm" -version = "0.52.4" +name = "winnow" +version = "0.6.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "77ca79f2451b49fa9e2af39f0747fe999fcda4f5e241b2898624dca97a1f2177" +checksum = "36c1fec1a2bb5866f07c25f68c26e565c4c200aebb96d7e55710c19d3e8ac49b" +dependencies = [ + "memchr", +] [[package]] -name = "windows_x86_64_msvc" -version = "0.48.5" +name = "zerocopy" +version = "0.7.35" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538" +checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0" +dependencies = [ + "byteorder", + "zerocopy-derive", +] [[package]] -name = "windows_x86_64_msvc" -version = "0.52.4" +name = "zerocopy-derive" +version = "0.7.35" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "32b752e52a2da0ddfbdbcc6fceadfeede4c939ed16d13e648833a61dfb611ed8" +checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] diff --git a/protocol/Cargo.toml b/protocol/Cargo.toml index 1e78732..525a15f 100644 --- a/protocol/Cargo.toml +++ b/protocol/Cargo.toml @@ -10,10 +10,12 @@ rust-version = "1.63.0" [features] default = ["std"] +async = ["std", "futures/std"] std = ["alloc", "bitcoin/std", "rand/std", "rand/std_rng"] alloc = [] [dependencies] +futures = { version = "0.3.30", default-features = false } rand = { version = "0.8.0", default-features = false } bitcoin = { version = "0.32.0", default-features = false } diff --git a/protocol/README.md b/protocol/README.md index 249a999..55df62c 100644 --- a/protocol/README.md +++ b/protocol/README.md @@ -10,8 +10,9 @@ Both structures are designed with a bare `no_std` and "Sans I/O" interface to ke ## Feature Flags -- `alloc` -- Expose memory allocation dependent features. -- `std` -- Includes the `alloc` memory allocation feature as well as extra standard library dependencies for I/O and random number generators. +* `alloc` -- Expose memory allocation dependent features. +* `std` -- Includes the `alloc` memory allocation feature as well as extra standard library dependencies for I/O and random number generators. +* `async` -- High level wrappers for asynchronous read and write runtimes. ## ChaCha20Poly1305 diff --git a/protocol/src/lib.rs b/protocol/src/lib.rs index 8ab2b86..be695db 100644 --- a/protocol/src/lib.rs +++ b/protocol/src/lib.rs @@ -47,6 +47,8 @@ use bitcoin::{ }, }; use fschacha20poly1305::{FSChaCha20, FSChaCha20Poly1305}; +#[cfg(feature = "async")] +use futures::{AsyncRead, AsyncReadExt, AsyncWrite, AsyncWriteExt}; use hkdf::Hkdf; use rand::Rng; @@ -111,25 +113,14 @@ impl fmt::Display for Error { write!(f, "More than 4095 bytes of garbage in the handshake.") } Error::HandshakeOutOfOrder => write!(f, "Handshake flow out of sequence."), - Error::SecretGeneration(e) => write!(f, "Cannot generate secrets: {}.", e), - Error::Decryption(e) => write!(f, "Decrytion error: {}.", e), + Error::SecretGeneration(e) => write!(f, "Cannot generate secrets: {:?}.", e), + Error::Decryption(e) => write!(f, "Decrytion error: {:?}.", e), } } } #[cfg(feature = "std")] -impl std::error::Error for Error { - fn source(&self) -> Option<&(dyn std::error::Error + 'static)> { - match self { - Error::CiphertextTooSmall => None, - Error::MaxGarbageLength => None, - Error::HandshakeOutOfOrder => None, - Error::SecretGeneration(e) => Some(e), - Error::Decryption(e) => Some(e), - Error::BufferTooSmall { required_bytes: _ } => None, - } - } -} +impl std::error::Error for Error {} impl From for Error { fn from(e: fschacha20poly1305::Error) -> Self { @@ -826,11 +817,7 @@ impl<'a> Handshake<'a> { packet_buffer: &mut [u8], ) -> Result<(), Error> { // Find the end of the garbage. - let (garbage, ciphertext) = split_garbage( - buffer, - self.remote_garbage_terminator - .ok_or(Error::HandshakeOutOfOrder)?, - )?; + let (garbage, ciphertext) = self.split_garbage(buffer)?; // Flag to track if the version packet has been received to signal the end of the handshake. let mut found_version_packet = false; @@ -925,40 +912,286 @@ impl<'a> Handshake<'a> { let packet_handler = self.packet_handler.ok_or(Error::HandshakeOutOfOrder)?; Ok(packet_handler) } + + /// Split off garbage in the given buffer on the remote garbage terminator. + /// + /// # Returns + /// + /// A `Result` containing the garbage and the remaining ciphertext not including the terminator. + /// + /// # Error + /// + /// * `CiphertextTooSmall` - Buffer did not contain a garbage terminator. + /// * `MaxGarbageLength` - Buffer did not contain the garbage terminator and contains too much garbage, should not be retried. + fn split_garbage<'b>(&self, buffer: &'b [u8]) -> Result<(&'b [u8], &'b [u8]), Error> { + let garbage_term = self + .remote_garbage_terminator + .ok_or(Error::HandshakeOutOfOrder)?; + if let Some(index) = buffer + .windows(garbage_term.len()) + .position(|window| window == garbage_term) + { + Ok((&buffer[..index], &buffer[(index + garbage_term.len())..])) + } else if buffer.len() >= (MAX_NUM_GARBAGE_BYTES + NUM_GARBAGE_TERMINTOR_BYTES) { + Err(Error::MaxGarbageLength) + } else { + // Terminator not found, the buffer needs more information. + Err(Error::CiphertextTooSmall) + } + } } -/// Split off garbage in the buffer on the garbage terminator. -/// -/// # Returns -/// -/// A `Result` containing the garbage and the remaining ciphertext not including the terminator. -/// -/// # Error -/// -/// * `CiphertextTooSmall` - Buffer did not contain a garbage terminator. -/// * `MaxGarbageLength` - Buffer did not contain the garbage terminator and contains too much garbage, should not be retried. -fn split_garbage( - buffer: &[u8], - garbage_term: [u8; NUM_GARBAGE_TERMINTOR_BYTES], -) -> Result<(&[u8], &[u8]), Error> { - if let Some(index) = buffer - .windows(garbage_term.len()) - .position(|window| window == garbage_term) - { - Ok((&buffer[..index], &buffer[(index + garbage_term.len())..])) - } else if buffer.len() >= (MAX_NUM_GARBAGE_BYTES + NUM_GARBAGE_TERMINTOR_BYTES) { - Err(Error::MaxGarbageLength) - } else { - // Terminator not found, the buffer needs more information. - Err(Error::CiphertextTooSmall) +/// High level error type for the protocol interface. +#[cfg(feature = "std")] +#[derive(Debug)] +pub enum ProtocolError { + Io(std::io::Error), + Internal(Error), +} + +#[cfg(feature = "std")] +impl From for ProtocolError { + fn from(error: std::io::Error) -> Self { + ProtocolError::Io(error) + } +} + +#[cfg(feature = "std")] +impl From for ProtocolError { + fn from(error: Error) -> Self { + ProtocolError::Internal(error) + } +} + +#[cfg(feature = "std")] +impl std::error::Error for ProtocolError {} + +#[cfg(feature = "std")] +impl fmt::Display for ProtocolError { + fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { + match self { + ProtocolError::Io(e) => write!(f, "IO error: {:?}", e), + ProtocolError::Internal(e) => write!(f, "Internal error: {:?}", e), + } + } +} + +/// A protocol session with handshake and send/receive packet management. +#[cfg(feature = "async")] +pub struct AsyncProtocol +where + R: AsyncRead + Unpin + Send, + W: AsyncWrite + Unpin + Send, +{ + reader: AsyncProtocolReader, + writer: AsyncProtocolWriter, +} + +#[cfg(feature = "async")] +impl AsyncProtocol +where + R: AsyncRead + Unpin + Send, + W: AsyncWrite + Unpin + Send, +{ + /// New protocol session which completes the initial handshake and returns a handler. + /// + /// # Returns + /// + /// A `Result` containing: + /// * `Ok(AsyncProtocol)`: An initialized protocol handler. + /// * `Err(ProtocolError)`: An error that occurred during the handshake. + pub async fn new( + network: Network, + role: Role, + garbage: Option<&[u8]>, + mut reader: R, + mut writer: W, + ) -> Result { + // Initialize buffer. + let garbage_len = match garbage { + Some(slice) => slice.len(), + None => 0, + }; + let mut ellswift_buffer = vec![0u8; NUM_ELLIGATOR_SWIFT_BYTES + garbage_len]; + let mut handshake = Handshake::new(network, role, garbage, &mut ellswift_buffer)?; + + // Send initial key to remote. + writer.write_all(&ellswift_buffer).await?; + writer.flush().await?; + + // Read remote's initial key. + let mut remote_ellswift_buffer = [0u8; 64]; + reader.read_exact(&mut remote_ellswift_buffer).await?; + + // Complete materials and send terminator to remote. + // Not exposing decoy packets yet. + let mut terminator_and_version_buffer = + vec![0u8; NUM_GARBAGE_TERMINTOR_BYTES + NUM_PACKET_OVERHEAD_BYTES]; + handshake.complete_materials( + remote_ellswift_buffer, + &mut terminator_and_version_buffer, + None, + )?; + writer.write_all(&terminator_and_version_buffer).await?; + writer.flush().await?; + + // Receive and authenticate remote garbage and version. + // Keep pulling bytes from the buffer until the garbage is flushed. + let mut remote_garbage_and_version_buffer = + Vec::with_capacity(NUM_INITIAL_HANDSHAKE_BUFFER_BYTES); + loop { + let mut temp_buffer = [0u8; NUM_INITIAL_HANDSHAKE_BUFFER_BYTES]; + match reader.read(&mut temp_buffer).await { + // No data available right now, continue. + Ok(0) => { + continue; + } + Ok(bytes_read) => { + remote_garbage_and_version_buffer.extend_from_slice(&temp_buffer[..bytes_read]); + + match handshake.authenticate_garbage_and_version_with_alloc( + &remote_garbage_and_version_buffer, + ) { + Ok(()) => break, + // Not enough data, continue reading. + Err(Error::CiphertextTooSmall) => continue, + Err(e) => return Err(ProtocolError::Internal(e)), + } + } + Err(e) => match e.kind() { + // No data available or interrupted, retry. + std::io::ErrorKind::WouldBlock | std::io::ErrorKind::Interrupted => { + continue; + } + _ => return Err(ProtocolError::Io(e)), + }, + } + } + + let packet_handler = handshake.finalize()?; + let (packet_reader, packet_writer) = packet_handler.into_split(); + + Ok(Self { + reader: AsyncProtocolReader { + buffer: reader, + packet_reader, + state: DecryptState::default(), + }, + writer: AsyncProtocolWriter { + buffer: writer, + packet_writer, + }, + }) + } + + /// Read reference for packet reading operations. + pub fn reader(&mut self) -> &mut AsyncProtocolReader { + &mut self.reader + } + + /// Write reference for packet writing operations. + pub fn writer(&mut self) -> &mut AsyncProtocolWriter { + &mut self.writer + } + + /// Split the protocol into a separate reader and writer. + pub fn into_split(self) -> (AsyncProtocolReader, AsyncProtocolWriter) { + (self.reader, self.writer) + } +} + +/// State machine of an asynchronous packet read. +#[cfg(feature = "async")] +#[derive(Default, Debug)] +enum DecryptState { + #[default] + ReadingLength, + ReadingPayload { + packet_bytes: Vec, + }, +} + +/// Manages an async buffer to automatically decrypt contents of received packets. +#[cfg(feature = "async")] +pub struct AsyncProtocolReader +where + R: AsyncRead + Unpin + Send, +{ + buffer: R, + packet_reader: PacketReader, + state: DecryptState, +} + +#[cfg(feature = "async")] +impl AsyncProtocolReader +where + R: AsyncRead + Unpin + Send, +{ + /// Decrypt contents of received packet from buffer. + /// + /// # Returns + /// + /// A `Result` containing: + /// * `Ok(Payload)`: A decrypted payload. + /// * `Err(ProtocolError)`: An error that occurred during the read or decryption. + pub async fn decrypt(&mut self) -> Result { + // Storing state between async read_exacts to make function more cancellation safe. + loop { + match &mut self.state { + DecryptState::ReadingLength => { + let mut length_bytes = [0u8; 3]; + self.buffer.read_exact(&mut length_bytes).await?; + let packet_bytes_len = self.packet_reader.decypt_len(length_bytes); + let packet_bytes = vec![0u8; packet_bytes_len]; + self.state = DecryptState::ReadingPayload { packet_bytes }; + } + DecryptState::ReadingPayload { packet_bytes } => { + self.buffer.read_exact(packet_bytes).await?; + let payload = self + .packet_reader + .decrypt_payload_with_alloc(packet_bytes, None)?; + self.state = DecryptState::ReadingLength; + return Ok(payload); + } + } + } + } +} + +/// Manages an async buffer to automatically encrypt and send contents in packets. +#[cfg(feature = "async")] +pub struct AsyncProtocolWriter +where + W: AsyncWrite + Unpin + Send, +{ + buffer: W, + packet_writer: PacketWriter, +} + +#[cfg(feature = "async")] +impl AsyncProtocolWriter +where + W: AsyncWrite + Unpin + Send, +{ + /// Encrypt contents and write packet buffer. + /// + /// # Returns + /// + /// A `Result` containing: + /// * `Ok()`: On successful contents encryption and packet send. + /// * `Err(ProtocolError)`: An error that occurred during the encryption or write. + pub async fn encrypt(&mut self, plaintext: &[u8]) -> Result<(), ProtocolError> { + let write_bytes = + self.packet_writer + .encrypt_packet_with_alloc(plaintext, None, PacketType::Genuine)?; + self.buffer.write_all(&write_bytes[..]).await?; + self.buffer.flush().await?; + Ok(()) } } #[cfg(test)] mod tests { - //! Any tests requiring a random number generator are currently - //! gated with the std feature flag. - use super::*; use core::str::FromStr; use hex::prelude::*; @@ -1066,15 +1299,6 @@ mod tests { ); } - #[test] - #[cfg(feature = "std")] - fn test_max_garbage() { - let too_much_garbage = vec![0; 4111]; - let garbage_terminator = [1; 16]; - let result = split_garbage(&too_much_garbage, garbage_terminator); - assert!(matches!(result, Err(Error::MaxGarbageLength))); - } - #[test] #[cfg(feature = "std")] fn test_packet_handler() { diff --git a/proxy/Cargo.toml b/proxy/Cargo.toml index 613e38d..1aaf20a 100644 --- a/proxy/Cargo.toml +++ b/proxy/Cargo.toml @@ -14,9 +14,9 @@ spec = "config_spec.toml" [dependencies] bitcoin = { version = "0.32.0" } tokio = { version = "1.37.0", features = ["full"] } -bytes = "1.6.0" +tokio-util = { version = "0.7.12", features = ["compat"] } hex = { package = "hex-conservative", version = "0.2.0" } -bip324 = { path = "../protocol" } +bip324 = { path = "../protocol", features = ["async"] } configure_me = "0.4.0" [build-dependencies] diff --git a/proxy/src/bin/proxy.rs b/proxy/src/bin/proxy.rs index 8ceef2d..9890180 100644 --- a/proxy/src/bin/proxy.rs +++ b/proxy/src/bin/proxy.rs @@ -1,18 +1,16 @@ // SPDX-License-Identifier: MIT OR Apache-2.0 -use core::panic; - -use bip324::{Handshake, Role}; -use bip324_proxy::{read_v1, read_v2, write_v1, write_v2}; +use bip324::{ + serde::{deserialize, serialize}, + AsyncProtocol, PacketType, Role, +}; +use bip324_proxy::{read_v1, write_v1}; use bitcoin::Network; -use bytes::BytesMut; -use tokio::io::{AsyncReadExt, AsyncWriteExt}; use tokio::net::{TcpListener, TcpStream}; +use tokio_util::compat::{TokioAsyncReadCompatExt, TokioAsyncWriteCompatExt}; configure_me::include_config!(); -const HANDSHAKE_BUFFER_BYTES: usize = 4096; - /// Validate and bootstrap proxy connection. async fn proxy_conn(client: TcpStream) -> Result<(), bip324_proxy::Error> { let remote_ip = bip324_proxy::peek_addr(&client) @@ -20,90 +18,30 @@ async fn proxy_conn(client: TcpStream) -> Result<(), bip324_proxy::Error> { .expect("peek address"); println!("Reaching out to {}.", remote_ip); - let mut remote = TcpStream::connect(remote_ip) + let remote = TcpStream::connect(remote_ip) .await .expect("connect to remote"); println!("Initiating handshake."); - let mut local_material_message = vec![0u8; 64]; - let mut handshake = Handshake::new( + let (remote_reader, remote_writer) = remote.into_split(); + // Convert to futures-compatible types. + let remote_reader = remote_reader.compat(); + let remote_writer = remote_writer.compat_write(); + + let protocol = AsyncProtocol::new( Network::Bitcoin, Role::Initiator, None, - &mut local_material_message, + remote_reader, + remote_writer, ) - .expect("generate handshake"); - - remote - .write_all(&local_material_message) - .await - .expect("send local materials"); - - println!("Sent handshake to remote."); + .await + .expect("protocol establishment"); - // 64 bytes ES. - let mut remote_material_message = [0u8; 64]; - println!("Reading handshake response from remote."); - remote - .read_exact(&mut remote_material_message) - .await - .expect("read remote materials"); - - println!("Completing materials."); - let mut local_garbage_terminator_message = [0u8; 36]; - handshake - .complete_materials( - remote_material_message, - &mut local_garbage_terminator_message, - None, - ) - .expect("complete materials"); - - println!("Sending garbage terminator and version packet."); - remote - .write_all(&local_garbage_terminator_message) - .await - .expect("send garbage and version"); - - // Keep pulling bytes from the buffer until the garbage is flushed. - let mut remote_garbage_and_version_buffer = BytesMut::with_capacity(HANDSHAKE_BUFFER_BYTES); - loop { - println!("Authenticating garbage and version packet..."); - - // Read from the remote, hopefully contains all garbage, decoy packets, and version packet. - // BytesMut is keeping track of its internal posistion, so this read should only ever - // extend the buffer on retries. Not overwrite it. The buffer will grow if required. - if let Err(e) = remote - .read_buf(&mut remote_garbage_and_version_buffer) - .await - { - panic!("unable to read garbage {}", e) - } - - // Attempt to authenticate the channel. - match handshake - .authenticate_garbage_and_version_with_alloc(&remote_garbage_and_version_buffer) - { - Ok(()) => { - println!("Channel authenticated."); - break; - } - Err(bip324::Error::CiphertextTooSmall) => { - // Attempt to pull more from the buffer and retry. - continue; - } - Err(e) => panic!("unable to authenticate garbage and version {}", e), - } - } - - let packet_handler = handshake.finalize().expect("finished handshake"); - - println!("Splitting channels."); let (mut client_reader, mut client_writer) = client.into_split(); - let (mut remote_reader, mut remote_writer) = remote.into_split(); - let (mut decrypter, mut encrypter) = packet_handler.into_split(); + let (mut remote_reader, mut remote_writer) = protocol.into_split(); - println!("Setting up proxy loops."); + println!("Setting up proxy."); // Spawning two threads instead of selecting on one due // to the IO calls not being cancellation safe. A select @@ -117,7 +55,9 @@ async fn proxy_conn(client: TcpStream) -> Result<(), bip324_proxy::Error> { "Read {} message from client, writing to remote.", msg.command() ); - write_v2(&mut remote_writer, &mut encrypter, msg) + let contents = serialize(msg).expect("serialize-able contents into network message"); + remote_writer + .encrypt(&contents) .await .expect("write to remote"); } @@ -125,9 +65,17 @@ async fn proxy_conn(client: TcpStream) -> Result<(), bip324_proxy::Error> { tokio::spawn(async move { loop { - let msg = read_v2(&mut remote_reader, &mut decrypter) - .await - .expect("read from remote"); + // Ignore any decoy packets. + let payload = loop { + let payload = remote_reader.decrypt().await.expect("read packet"); + + if payload.packet_type() == PacketType::Genuine { + break payload; + } + }; + + let msg = deserialize(payload.contents()) + .expect("deserializable contents into network message"); println!( "Read {} message from remote, writing to client.", msg.command() diff --git a/proxy/src/lib.rs b/proxy/src/lib.rs index 1f6bc03..20da666 100644 --- a/proxy/src/lib.rs +++ b/proxy/src/lib.rs @@ -11,8 +11,6 @@ use std::fmt; use std::net::SocketAddr; -use bip324::serde::{deserialize, serialize}; -use bip324::{PacketReader, PacketType, PacketWriter}; use bitcoin::consensus::{Decodable, Encodable}; use bitcoin::p2p::message::{NetworkMessage, RawNetworkMessage}; use bitcoin::p2p::{Address, Magic}; @@ -34,44 +32,34 @@ pub enum Error { WrongNetwork, WrongCommand, Serde, - Network(std::io::Error), - Cipher(bip324::Error), + Io(std::io::Error), + Protocol(bip324::Error), } impl fmt::Display for Error { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { match self { Error::WrongNetwork => write!(f, "recieved message on wrong network"), - Error::Network(e) => write!(f, "network {}", e), + Error::Io(e) => write!(f, "network {:?}", e), Error::WrongCommand => write!(f, "recieved message with wrong command"), - Error::Cipher(e) => write!(f, "cipher encryption/decrytion error {}", e), + Error::Protocol(e) => write!(f, "protocol error {:?}", e), Error::Serde => write!(f, "unable to serialize command"), } } } -impl std::error::Error for Error { - fn source(&self) -> Option<&(dyn std::error::Error + 'static)> { - match self { - Error::Network(e) => Some(e), - Error::WrongNetwork => None, - Error::WrongCommand => None, - Error::Cipher(e) => Some(e), - Error::Serde => None, - } - } -} +impl std::error::Error for Error {} impl From for Error { fn from(e: bip324::Error) -> Self { - Error::Cipher(e) + Error::Protocol(e) } } // Convert IO errors. impl From for Error { fn from(e: std::io::Error) -> Self { - Error::Network(e) + Error::Io(e) } } @@ -125,31 +113,6 @@ pub async fn read_v1(input: &mut T) -> Result( - input: &mut T, - decrypter: &mut PacketReader, -) -> Result { - // Ignore any decoy packets. - let payload = loop { - let mut length_bytes = [0u8; 3]; - input.read_exact(&mut length_bytes).await?; - let packet_bytes_len = decrypter.decypt_len(length_bytes); - let mut packet_bytes = vec![0u8; packet_bytes_len]; - input.read_exact(&mut packet_bytes).await?; - let payload = decrypter.decrypt_payload_with_alloc(&packet_bytes, None)?; - - if payload.packet_type() == PacketType::Genuine { - break payload; - } - }; - - let message = deserialize(payload.contents()).map_err(|_| Error::Serde)?; - Ok(message) -} - /// Write message to the output stream using v1. pub async fn write_v1( output: &mut T, @@ -163,18 +126,3 @@ pub async fn write_v1( output.flush().await?; Ok(()) } - -/// Write the network message to the output stream using v2. -pub async fn write_v2( - output: &mut T, - encrypter: &mut PacketWriter, - msg: NetworkMessage, -) -> Result<(), Error> { - let payload = serialize(msg).map_err(|_| Error::Serde)?; - let write_bytes = encrypter - .encrypt_packet_with_alloc(&payload, None, PacketType::Genuine) - .expect("encryption"); - output.write_all(&write_bytes[..]).await?; - output.flush().await?; - Ok(()) -}