diff --git a/packages/sdk/lib/Cargo.lock b/packages/sdk/lib/Cargo.lock new file mode 100644 index 0000000000..e19d9caffe --- /dev/null +++ b/packages/sdk/lib/Cargo.lock @@ -0,0 +1,6416 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 3 + +[[package]] +name = "Inflector" +version = "0.11.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fe438c63458706e03479442743baae6c88256498e6431708f6dfc520a26515d3" +dependencies = [ + "lazy_static", + "regex", +] + +[[package]] +name = "addr2line" +version = "0.21.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a30b2e23b9e17a9f90641c7ab1549cd9b44f296d3ccbf309d2863cfe398a0cb" +dependencies = [ + "gimli", +] + +[[package]] +name = "adler" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" + +[[package]] +name = "aead" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d122413f284cf2d62fb1b7db97e02edb8cda96d769b16e443a4f6195e35662b0" +dependencies = [ + "crypto-common", + "generic-array", +] + +[[package]] +name = "aes" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0" +dependencies = [ + "cfg-if", + "cipher", + "cpufeatures", +] + +[[package]] +name = "aes-gcm" +version = "0.10.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "831010a0f742e1209b3bcea8fab6a8e149051ba6099432c8cb2cc117dec3ead1" +dependencies = [ + "aead", + "aes", + "cipher", + "ctr", + "ghash", + "subtle", +] + +[[package]] +name = "aho-corasick" +version = "1.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916" +dependencies = [ + "memchr", +] + +[[package]] +name = "android-tzdata" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e999941b234f3131b00bc13c22d06e8c5ff726d1b6318ac7eb276997bbb4fef0" + +[[package]] +name = "android_system_properties" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311" +dependencies = [ + "libc", +] + +[[package]] +name = "anstyle" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "038dfcf04a5feb68e9c60b21c9625a54c2c0616e79b72b0fd87075a056ae1d1b" + +[[package]] +name = "anyhow" +version = "1.0.91" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c042108f3ed77fd83760a5fd79b53be043192bb3b9dba91d8c574c0ada7850c8" + +[[package]] +name = "argon2" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db4ce4441f99dbd377ca8a8f57b698c44d0d6e712d8329b5040da5a64aa1ce73" +dependencies = [ + "base64ct", + "blake2", + "password-hash 0.4.2", +] + +[[package]] +name = "ark-bls12-381" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "65be532f9dd1e98ad0150b037276cde464c6f371059e6dd02c0222395761f6aa" +dependencies = [ + "ark-ec", + "ark-ff", + "ark-std", +] + +[[package]] +name = "ark-ec" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dea978406c4b1ca13c2db2373b05cc55429c3575b8b21f1b9ee859aa5b03dd42" +dependencies = [ + "ark-ff", + "ark-serialize", + "ark-std", + "derivative", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "zeroize", +] + +[[package]] +name = "ark-ff" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6b3235cc41ee7a12aaaf2c575a2ad7b46713a8a50bda2fc3b003a04845c05dd6" +dependencies = [ + "ark-ff-asm", + "ark-ff-macros", + "ark-serialize", + "ark-std", + "derivative", + "num-bigint", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "paste", + "rustc_version 0.3.3", + "zeroize", +] + +[[package]] +name = "ark-ff-asm" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db02d390bf6643fb404d3d22d31aee1c4bc4459600aef9113833d17e786c6e44" +dependencies = [ + "quote", + "syn 1.0.109", +] + +[[package]] +name = "ark-ff-macros" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db2fd794a08ccb318058009eefdf15bcaaaaf6f8161eb3345f907222bac38b20" +dependencies = [ + "num-bigint", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "ark-serialize" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d6c2b318ee6e10f8c2853e73a83adc0ccb88995aa978d8a3408d492ab2ee671" +dependencies = [ + "ark-std", + "digest 0.9.0", +] + +[[package]] +name = "ark-std" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1df2c09229cbc5a028b1d70e00fdb2acee28b1055dfb5ca73eea49c5a25c4e7c" +dependencies = [ + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.8.5", +] + +[[package]] +name = "arrayref" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6b4930d2cb77ce62f89ee5d5289b4ac049559b1c45539271f5ed4fdc7db34545" + +[[package]] +name = "arrayvec" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96d30a06541fbafbc7f82ed10c06164cfbd2c401138f6addd8404629c4b16711" + +[[package]] +name = "async-trait" +version = "0.1.80" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6fa2087f2753a7da8cc1c0dbfcf89579dd57458e36769de5ac750b4671737ca" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "async_io_stream" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6d7b9decdf35d8908a7e3ef02f64c5e9b1695e230154c0e8de3969142d9b94c" +dependencies = [ + "futures", + "pharos", + "rustc_version 0.4.0", +] + +[[package]] +name = "auto_impl" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c87f3f15e7794432337fc718554eaa4dc8f04c9677a950ffe366f20a162ae42" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "autocfg" +version = "1.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c4b4d0bd25bd0b74681c0ad21497610ce1b7c91b1022cd21c80c6fbdd9476b0" + +[[package]] +name = "backtrace" +version = "0.3.71" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26b05800d2e817c8b3b4b54abd461726265fa9789ae34330622f2db9ee696f9d" +dependencies = [ + "addr2line", + "cc", + "cfg-if", + "libc", + "miniz_oxide", + "object", + "rustc-demangle", +] + +[[package]] +name = "base16ct" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4c7f02d4ea65f2c1853089ffd8d2787bdbc63de2f0d29dedbcf8ccdfa0ccd4cf" + +[[package]] +name = "base58" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6107fe1be6682a68940da878d9e9f5e90ca5745b3dec9fd1bb393c8777d4f581" + +[[package]] +name = "base64" +version = "0.13.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8" + +[[package]] +name = "base64" +version = "0.21.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567" + +[[package]] +name = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + +[[package]] +name = "base64ct" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a32fd6af2b5827bce66c29053ba0e7c42b9dcab01835835058558c10851a46b" + +[[package]] +name = "bech32" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf9ff0bbfd639f15c74af777d81383cf53efb7c93613f6cab67c6c11e05bbf8b" + +[[package]] +name = "bech32" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d86b93f97252c47b41663388e6d155714a9d0c398b99f1005cbc5f978b29f445" + +[[package]] +name = "bellman" +version = "0.14.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9afceed28bac7f9f5a508bca8aeeff51cdfa4770c0b967ac55c621e2ddfd6171" +dependencies = [ + "bitvec", + "blake2s_simd", + "byteorder", + "crossbeam-channel", + "ff", + "group", + "lazy_static", + "log", + "num_cpus", + "pairing", + "rand_core 0.6.4", + "rayon", + "subtle", +] + +[[package]] +name = "bimap" +version = "0.6.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "230c5f1ca6a325a32553f8640d31ac9b49f2411e901e427570154868b46da4f7" +dependencies = [ + "serde", +] + +[[package]] +name = "bip0039" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef0f0152ec5cf17f49a5866afaa3439816207fd4f0a224c0211ffaf5e278426" +dependencies = [ + "hmac 0.12.1", + "pbkdf2 0.10.1", + "rand 0.8.5", + "sha2 0.10.8", + "unicode-normalization", + "zeroize", +] + +[[package]] +name = "bip0039" +version = "0.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "568b6890865156d9043af490d4c4081c385dd68ea10acd6ca15733d511e6b51c" +dependencies = [ + "hmac 0.12.1", + "pbkdf2 0.12.2", + "rand 0.8.5", + "sha2 0.10.8", + "unicode-normalization", + "zeroize", +] + +[[package]] +name = "bitflags" +version = "1.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" + +[[package]] +name = "bitflags" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf4b9d6a944f767f8e5e0db018570623c85f3d925ac718db4e06d0187adb21c1" +dependencies = [ + "serde", +] + +[[package]] +name = "bitvec" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bc2832c24239b0141d5674bb9174f9d68a8b5b3f2753311927c172ca46f7e9c" +dependencies = [ + "funty", + "radium", + "tap", + "wyz", +] + +[[package]] +name = "blake2" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "46502ad458c9a52b69d4d4d32775c788b7a1b85e8bc9d482d92250fc0e3f8efe" +dependencies = [ + "digest 0.10.7", +] + +[[package]] +name = "blake2b_simd" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "23285ad32269793932e830392f2fe2f83e26488fd3ec778883a93c8323735780" +dependencies = [ + "arrayref", + "arrayvec", + "constant_time_eq", +] + +[[package]] +name = "blake2s_simd" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94230421e395b9920d23df13ea5d77a20e1725331f90fbbf6df6040b33f756ae" +dependencies = [ + "arrayref", + "arrayvec", + "constant_time_eq", +] + +[[package]] +name = "blake3" +version = "1.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "30cca6d3674597c30ddf2c587bf8d9d65c9a84d2326d941cc79c9842dfe0ef52" +dependencies = [ + "arrayref", + "arrayvec", + "cc", + "cfg-if", + "constant_time_eq", +] + +[[package]] +name = "block-buffer" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4" +dependencies = [ + "generic-array", +] + +[[package]] +name = "block-buffer" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71" +dependencies = [ + "generic-array", +] + +[[package]] +name = "bls12_381" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7bc6d6292be3a19e6379786dac800f551e5865a5bb51ebbe3064ab80433f403" +dependencies = [ + "ff", + "group", + "pairing", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "bls12_381" +version = "0.8.0" +source = "git+https://github.com/heliaxdev/bls12_381.git?rev=d3ebe9dd6488fac1923db120a7498079e55dd838#d3ebe9dd6488fac1923db120a7498079e55dd838" +dependencies = [ + "ff", + "group", + "pairing", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "borsh" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dbe5b10e214954177fb1dc9fbd20a1a2608fe99e6c832033bdc7cea287a20d77" +dependencies = [ + "borsh-derive", + "cfg_aliases", +] + +[[package]] +name = "borsh-derive" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7a8646f94ab393e43e8b35a2558b1624bed28b97ee09c5d15456e3c9463f46d" +dependencies = [ + "once_cell", + "proc-macro-crate", + "proc-macro2", + "quote", + "syn 2.0.65", + "syn_derive", +] + +[[package]] +name = "borsh-ext" +version = "1.2.0" +source = "git+https://github.com/heliaxdev/borsh-ext?tag=v1.2.0#a62fee3e847e512cad9ac0f1fd5a900e5db9ba37" +dependencies = [ + "borsh", +] + +[[package]] +name = "bs58" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf88ba1141d185c399bee5288d850d63b8369520c1eafc32a0430b5b6c287bf4" +dependencies = [ + "sha2 0.10.8", + "tinyvec", +] + +[[package]] +name = "bumpalo" +version = "3.16.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c" + +[[package]] +name = "byte-slice-cast" +version = "1.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3ac9f8b63eca6fd385229b3675f6cc0dc5c8a5c8a54a59d4f52ffd670d87b0c" + +[[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" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "514de17de45fdb8dc022b1a7975556c53c86f9f0aa5f534b98977b171857c2c9" +dependencies = [ + "serde", +] + +[[package]] +name = "camino" +version = "1.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e0ec6b951b160caa93cc0c7b209e5a3bff7aae9062213451ac99493cd844c239" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo-platform" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "24b1f0365a6c6bb4020cd05806fd0d33c44d38046b8bd7f0e40814b9763cabfc" +dependencies = [ + "serde", +] + +[[package]] +name = "cargo_metadata" +version = "0.18.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2d886547e41f740c616ae73108f6eb70afe6d940c7bc697cb30f13daec073037" +dependencies = [ + "camino", + "cargo-platform", + "semver 1.0.23", + "serde", + "serde_json", + "thiserror", +] + +[[package]] +name = "cbc" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26b52a9543ae338f279b96b0b9fed9c8093744685043739079ce85cd58f289a6" +dependencies = [ + "cipher", +] + +[[package]] +name = "cc" +version = "1.0.98" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "41c270e7540d725e65ac7f1b212ac8ce349719624d7bcff99f8e2e488e8cf03f" + +[[package]] +name = "cfg-if" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" + +[[package]] +name = "cfg_aliases" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fd16c4719339c4530435d38e511904438d07cce7950afa3718a84ac36c10e89e" + +[[package]] +name = "chacha20" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3613f74bd2eac03dad61bd53dbe620703d4371614fe0bc3b9f04dd36fe4e818" +dependencies = [ + "cfg-if", + "cipher", + "cpufeatures", +] + +[[package]] +name = "chacha20poly1305" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "10cd79432192d1c0f4e1a0fef9527696cc039165d729fb41b3f4f4f354c2dc35" +dependencies = [ + "aead", + "chacha20", + "cipher", + "poly1305", + "zeroize", +] + +[[package]] +name = "chrono" +version = "0.4.38" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a21f936df1771bf62b77f047b726c4625ff2e8aa607c01ec06e5a05bd8463401" +dependencies = [ + "android-tzdata", + "iana-time-zone", + "js-sys", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "wasm-bindgen", + "windows-targets 0.52.5", +] + +[[package]] +name = "cipher" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad" +dependencies = [ + "crypto-common", + "inout", + "zeroize", +] + +[[package]] +name = "circular-queue" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d34327ead1c743a10db339de35fb58957564b99d248a67985c55638b22c59b5" +dependencies = [ + "version_check", +] + +[[package]] +name = "clap" +version = "4.5.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "64acc1846d54c1fe936a78dc189c34e28d3f5afc348403f28ecf53660b9b8462" +dependencies = [ + "clap_builder", +] + +[[package]] +name = "clap_builder" +version = "4.5.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6fb8393d67ba2e7bfaf28a23458e4e2b543cc73a99595511eb207fdb8aede942" +dependencies = [ + "anstyle", + "clap_lex", +] + +[[package]] +name = "clap_lex" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4b82cf0babdbd58558212896d1a4272303a57bdb245c2bf1147185fb45640e70" + +[[package]] +name = "clru" +version = "0.5.0" +source = "git+https://github.com/marmeladema/clru-rs.git?rev=71ca566#71ca566915f21f3c308091ca7756a91b0f8b5afc" + +[[package]] +name = "coins-bip32" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b6be4a5df2098cd811f3194f64ddb96c267606bffd9689ac7b0160097b01ad3" +dependencies = [ + "bs58", + "coins-core", + "digest 0.10.7", + "hmac 0.12.1", + "k256", + "serde", + "sha2 0.10.8", + "thiserror", +] + +[[package]] +name = "coins-bip39" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3db8fba409ce3dc04f7d804074039eb68b960b0829161f8e06c95fea3f122528" +dependencies = [ + "bitvec", + "coins-bip32", + "hmac 0.12.1", + "once_cell", + "pbkdf2 0.12.2", + "rand 0.8.5", + "sha2 0.10.8", + "thiserror", +] + +[[package]] +name = "coins-core" +version = "0.8.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5286a0843c21f8367f7be734f89df9b822e0321d8bcce8d6e735aadff7d74979" +dependencies = [ + "base64 0.21.7", + "bech32 0.9.1", + "bs58", + "digest 0.10.7", + "generic-array", + "hex", + "ripemd", + "serde", + "serde_derive", + "sha2 0.10.8", + "sha3", + "thiserror", +] + +[[package]] +name = "console_error_panic_hook" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a06aeb73f470f66dcdbf7223caeebb85984942f22f1adb2a088cf9668146bbbc" +dependencies = [ + "cfg-if", + "wasm-bindgen", +] + +[[package]] +name = "const-hex" +version = "1.11.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70ff96486ccc291d36a958107caf2c0af8c78c0af7d31ae2f35ce055130de1a6" +dependencies = [ + "cfg-if", + "cpufeatures", + "hex", + "proptest", + "serde", +] + +[[package]] +name = "const-oid" +version = "0.9.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c2459377285ad874054d797f3ccebf984978aa39129f6eafde5cdc8315b612f8" + +[[package]] +name = "const_panic" +version = "0.2.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6051f239ecec86fde3410901ab7860d458d160371533842974fc61f96d15879b" + +[[package]] +name = "constant_time_eq" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f7144d30dcf0fafbce74250a3963025d8d52177934239851c917d29f1df280c2" + +[[package]] +name = "core-foundation" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "core-foundation-sys" +version = "0.8.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "06ea2b9bc92be3c2baa9334a323ebca2d6f074ff852cd1d7b11064035cd3868f" + +[[package]] +name = "cpufeatures" +version = "0.2.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53fe5e26ff1b7aef8bca9c6080520cfb8d9333c7568e1829cef191a9723e5504" +dependencies = [ + "libc", +] + +[[package]] +name = "crossbeam-channel" +version = "0.5.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "33480d6946193aa8033910124896ca395333cae7e2d1113d1fef6c3272217df2" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-deque" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "613f8cc01fe9cf1a3eb3d7f488fd2fa8388403e97039e2f73692932e291a770d" +dependencies = [ + "crossbeam-epoch", + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-epoch" +version = "0.9.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-utils" +version = "0.8.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80" + +[[package]] +name = "crunchy" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7" + +[[package]] +name = "crypto-bigint" +version = "0.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0dc92fb57ca44df6db8059111ab3af99a63d5d0f8375d9972e319a379c6bab76" +dependencies = [ + "generic-array", + "rand_core 0.6.4", + "subtle", + "zeroize", +] + +[[package]] +name = "crypto-common" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3" +dependencies = [ + "generic-array", + "rand_core 0.6.4", + "typenum", +] + +[[package]] +name = "crypto-mac" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b584a330336237c1eecd3e94266efb216c56ed91225d634cb2991c5f3fd1aeab" +dependencies = [ + "generic-array", + "subtle", +] + +[[package]] +name = "ct-codecs" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f3b7eb4404b8195a9abb6356f4ac07d8ba267045c8d6d220ac4dc992e6cc75df" + +[[package]] +name = "ctr" +version = "0.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0369ee1ad671834580515889b80f2ea915f23b8be8d0daa4bbaf2ac5c7590835" +dependencies = [ + "cipher", +] + +[[package]] +name = "curve25519-dalek-ng" +version = "4.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1c359b7249347e46fb28804470d071c921156ad62b3eef5d34e2ba867533dec8" +dependencies = [ + "byteorder", + "digest 0.9.0", + "rand_core 0.6.4", + "subtle-ng", + "zeroize", +] + +[[package]] +name = "data-encoding" +version = "2.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e8566979429cf69b49a5c740c60791108e86440e8be149bbea4fe54d2c32d6e2" + +[[package]] +name = "der" +version = "0.7.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f55bf8e7b65898637379c1b74eb1551107c8294ed26d855ceb9fd1a09cfc9bc0" +dependencies = [ + "const-oid", + "zeroize", +] + +[[package]] +name = "deranged" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b42b6fa04a440b495c8b04d0e71b707c585f83cb9cb28cf8cd0d976c315e31b4" +dependencies = [ + "powerfmt", +] + +[[package]] +name = "derivation-path" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e5c37193a1db1d8ed868c03ec7b152175f26160a5b740e5e484143877e0adf0" + +[[package]] +name = "derivative" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fcc3dd5e9e9c0b295d6e1e4d811fb6f157d5ffd784b8d202fc62eac8035a770b" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "derive_more" +version = "0.99.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f33878137e4dafd7fa914ad4e259e18a4e8e532b9617a2d0150262bf53abfce" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "digest" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066" +dependencies = [ + "generic-array", +] + +[[package]] +name = "digest" +version = "0.10.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" +dependencies = [ + "block-buffer 0.10.4", + "const-oid", + "crypto-common", + "subtle", +] + +[[package]] +name = "directories" +version = "4.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f51c5d4ddabd36886dd3e1438cb358cdcb0d7c499cb99cb4ac2e38e18b5cb210" +dependencies = [ + "dirs-sys", +] + +[[package]] +name = "dirs-sys" +version = "0.3.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b1d1d91c932ef41c0f2663aa8b0ca0342d444d842c06914aa0a7e352d0bada6" +dependencies = [ + "libc", + "redox_users", + "winapi", +] + +[[package]] +name = "displaydoc" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "dunce" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "56ce8c6da7551ec6c462cbaf3bfbc75131ebbfa1c944aeaa9dab51ca1c5f0c3b" + +[[package]] +name = "duration-str" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c1a2e028bbf7921549873b291ddc0cfe08b673d9489da81ac28898cd5a0e6e0" +dependencies = [ + "chrono", + "rust_decimal", + "serde", + "thiserror", + "time", + "winnow 0.6.8", +] + +[[package]] +name = "dyn-clone" +version = "1.0.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0d6ef0072f8a535281e4876be788938b528e9a1d43900b82c2569af7da799125" + +[[package]] +name = "ecdsa" +version = "0.16.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee27f32b5c5292967d2d4a9d7f1e0b0aed2c15daded5a60300e4abb9d8020bca" +dependencies = [ + "der", + "digest 0.10.7", + "elliptic-curve", + "rfc6979", + "serdect", + "signature", + "spki", +] + +[[package]] +name = "ed25519" +version = "2.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "115531babc129696a58c64a4fef0a8bf9e9698629fb97e9e40767d235cfbcd53" +dependencies = [ + "pkcs8", + "signature", +] + +[[package]] +name = "ed25519-consensus" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c8465edc8ee7436ffea81d21a019b16676ee3db267aa8d5a8d729581ecf998b" +dependencies = [ + "curve25519-dalek-ng", + "hex", + "rand_core 0.6.4", + "serde", + "sha2 0.9.9", + "thiserror", + "zeroize", +] + +[[package]] +name = "either" +version = "1.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3dca9240753cf90908d7e4aac30f630662b02aebaa1b58a3cadabdb23385b58b" + +[[package]] +name = "elliptic-curve" +version = "0.13.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b5e6043086bf7973472e0c7dff2142ea0b680d30e18d9cc40f267efbf222bd47" +dependencies = [ + "base16ct", + "crypto-bigint", + "digest 0.10.7", + "ff", + "generic-array", + "group", + "pkcs8", + "rand_core 0.6.4", + "sec1", + "serdect", + "subtle", + "zeroize", +] + +[[package]] +name = "encoding_rs" +version = "0.8.34" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b45de904aa0b010bce2ab45264d0631681847fa7b6f2eaa7dab7619943bc4f59" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "enr" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2a3d8dc56e02f954cac8eb489772c552c473346fc34f67412bb6244fd647f7e4" +dependencies = [ + "base64 0.21.7", + "bytes", + "hex", + "k256", + "log", + "rand 0.8.5", + "rlp", + "serde", + "sha3", + "zeroize", +] + +[[package]] +name = "equivalent" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" + +[[package]] +name = "errno" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "534c5cf6194dfab3db3242765c03bbe257cf92f22b38f6bc0c58d59108a820ba" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "eth-keystore" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fda3bf123be441da5260717e0661c25a2fd9cb2b2c1d20bf2e05580047158ab" +dependencies = [ + "aes", + "ctr", + "digest 0.10.7", + "hex", + "hmac 0.12.1", + "pbkdf2 0.11.0", + "rand 0.8.5", + "scrypt", + "serde", + "serde_json", + "sha2 0.10.8", + "sha3", + "thiserror", + "uuid 0.8.2", +] + +[[package]] +name = "ethabi" +version = "18.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7413c5f74cc903ea37386a8965a936cbeb334bd270862fdece542c1b2dcbc898" +dependencies = [ + "ethereum-types", + "hex", + "once_cell", + "regex", + "serde", + "serde_json", + "sha3", + "thiserror", + "uint", +] + +[[package]] +name = "ethbloom" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c22d4b5885b6aa2fe5e8b9329fb8d232bf739e434e6b87347c63bdd00c120f60" +dependencies = [ + "crunchy", + "fixed-hash", + "impl-codec", + "impl-rlp", + "impl-serde", + "scale-info", + "tiny-keccak", +] + +[[package]] +name = "ethbridge-bridge-contract" +version = "0.24.0" +source = "git+https://github.com/heliaxdev/ethbridge-rs?tag=v0.24.0#d66708bb8a734111988b9eaf08c7473bd7020c00" +dependencies = [ + "ethbridge-bridge-events", + "ethbridge-structs", + "ethers", + "ethers-contract", +] + +[[package]] +name = "ethbridge-bridge-events" +version = "0.24.0" +source = "git+https://github.com/heliaxdev/ethbridge-rs?tag=v0.24.0#d66708bb8a734111988b9eaf08c7473bd7020c00" +dependencies = [ + "ethabi", + "ethbridge-structs", + "ethers", + "ethers-contract", +] + +[[package]] +name = "ethbridge-structs" +version = "0.24.0" +source = "git+https://github.com/heliaxdev/ethbridge-rs?tag=v0.24.0#d66708bb8a734111988b9eaf08c7473bd7020c00" +dependencies = [ + "ethabi", + "ethers", + "ethers-contract", +] + +[[package]] +name = "ethereum-types" +version = "0.14.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02d215cbf040552efcbe99a38372fe80ab9d00268e20012b79fcd0f073edd8ee" +dependencies = [ + "ethbloom", + "fixed-hash", + "impl-codec", + "impl-rlp", + "impl-serde", + "primitive-types", + "scale-info", + "uint", +] + +[[package]] +name = "ethers" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "839a392641e746a1ff365ef7c901238410b5c6285d240cf2409ffaaa7df9a78a" +dependencies = [ + "ethers-addressbook", + "ethers-contract", + "ethers-core", + "ethers-etherscan", + "ethers-middleware", + "ethers-providers", + "ethers-signers", +] + +[[package]] +name = "ethers-addressbook" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5495afd16b4faa556c3bba1f21b98b4983e53c1755022377051a975c3b021759" +dependencies = [ + "ethers-core", + "once_cell", + "serde", + "serde_json", +] + +[[package]] +name = "ethers-contract" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6fceafa3578c836eeb874af87abacfb041f92b4da0a78a5edd042564b8ecdaaa" +dependencies = [ + "const-hex", + "ethers-contract-abigen", + "ethers-contract-derive", + "ethers-core", + "ethers-providers", + "futures-util", + "once_cell", + "pin-project", + "serde", + "serde_json", + "thiserror", +] + +[[package]] +name = "ethers-contract-abigen" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "04ba01fbc2331a38c429eb95d4a570166781f14290ef9fdb144278a90b5a739b" +dependencies = [ + "Inflector", + "const-hex", + "dunce", + "ethers-core", + "eyre", + "prettyplease", + "proc-macro2", + "quote", + "regex", + "serde", + "serde_json", + "syn 2.0.65", + "toml 0.8.13", + "walkdir", +] + +[[package]] +name = "ethers-contract-derive" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87689dcabc0051cde10caaade298f9e9093d65f6125c14575db3fd8c669a168f" +dependencies = [ + "Inflector", + "const-hex", + "ethers-contract-abigen", + "ethers-core", + "proc-macro2", + "quote", + "serde_json", + "syn 2.0.65", +] + +[[package]] +name = "ethers-core" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "82d80cc6ad30b14a48ab786523af33b37f28a8623fc06afd55324816ef18fb1f" +dependencies = [ + "arrayvec", + "bytes", + "cargo_metadata", + "chrono", + "const-hex", + "elliptic-curve", + "ethabi", + "generic-array", + "k256", + "num_enum", + "once_cell", + "open-fastrlp", + "rand 0.8.5", + "rlp", + "serde", + "serde_json", + "strum", + "syn 2.0.65", + "tempfile", + "thiserror", + "tiny-keccak", + "unicode-xid", +] + +[[package]] +name = "ethers-etherscan" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e79e5973c26d4baf0ce55520bd732314328cabe53193286671b47144145b9649" +dependencies = [ + "chrono", + "ethers-core", + "reqwest", + "semver 1.0.23", + "serde", + "serde_json", + "thiserror", + "tracing", +] + +[[package]] +name = "ethers-middleware" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "48f9fdf09aec667c099909d91908d5eaf9be1bd0e2500ba4172c1d28bfaa43de" +dependencies = [ + "async-trait", + "auto_impl", + "ethers-contract", + "ethers-core", + "ethers-providers", + "ethers-signers", + "futures-channel", + "futures-locks", + "futures-util", + "instant", + "reqwest", + "serde", + "serde_json", + "thiserror", + "tokio", + "tracing", + "tracing-futures", + "url", +] + +[[package]] +name = "ethers-providers" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6434c9a33891f1effc9c75472e12666db2fa5a0fec4b29af6221680a6fe83ab2" +dependencies = [ + "async-trait", + "auto_impl", + "base64 0.21.7", + "bytes", + "const-hex", + "enr", + "ethers-core", + "futures-core", + "futures-timer", + "futures-util", + "hashers", + "http 0.2.12", + "instant", + "jsonwebtoken", + "once_cell", + "pin-project", + "reqwest", + "serde", + "serde_json", + "thiserror", + "tokio", + "tracing", + "tracing-futures", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "ws_stream_wasm", +] + +[[package]] +name = "ethers-signers" +version = "2.0.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "228875491c782ad851773b652dd8ecac62cda8571d3bc32a5853644dd26766c2" +dependencies = [ + "async-trait", + "coins-bip32", + "coins-bip39", + "const-hex", + "elliptic-curve", + "eth-keystore", + "ethers-core", + "rand 0.8.5", + "sha2 0.10.8", + "thiserror", + "tracing", +] + +[[package]] +name = "eyre" +version = "0.6.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7cd915d99f24784cdc19fd37ef22b97e3ff0ae756c7e492e9fbfe897d61e2aec" +dependencies = [ + "indenter", + "once_cell", +] + +[[package]] +name = "fastrand" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9fc0510504f03c51ada170672ac806f1f105a88aa97a5281117e1ddc3368e51a" + +[[package]] +name = "ff" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ded41244b729663b1e574f1b4fb731469f69f79c17667b5d776b16cda0479449" +dependencies = [ + "bitvec", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "fixed-hash" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "835c052cb0c08c1acf6ffd71c022172e18723949c8282f2b9f27efbc51e64534" +dependencies = [ + "byteorder", + "rand 0.8.5", + "rustc-hex", + "static_assertions", +] + +[[package]] +name = "fixedbitset" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ce7134b9999ecaf8bcd65542e436736ef32ddca1b3e06094cb6ec5755203b80" + +[[package]] +name = "flex-error" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c606d892c9de11507fa0dcffc116434f94e105d0bbdc4e405b61519464c49d7b" +dependencies = [ + "paste", +] + +[[package]] +name = "flume" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da0e4dd2a88388a1f4ccc7c9ce104604dab68d9f408dc34cd45823d5a9069095" +dependencies = [ + "futures-core", + "futures-sink", + "nanorand", + "spin 0.9.8", +] + +[[package]] +name = "fnv" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" + +[[package]] +name = "foreign-types" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f6f339eb8adc052cd2ca78910fda869aefa38d22d5cb648e6485e4d3fc06f3b1" +dependencies = [ + "foreign-types-shared", +] + +[[package]] +name = "foreign-types-shared" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b" + +[[package]] +name = "form_urlencoded" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e13624c2627564efccf4934284bdd98cbaa14e79b0b5a141218e507b3a823456" +dependencies = [ + "percent-encoding", +] + +[[package]] +name = "fpe" +version = "0.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26c4b37de5ae15812a764c958297cfc50f5c010438f60c6ce75d11b802abd404" +dependencies = [ + "cbc", + "cipher", + "libm", + "num-bigint", + "num-integer", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "funty" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6d5a32815ae3f33302d95fdcb2ce17862f8c65363dcfd29360480ba1001fc9c" + +[[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-executor", + "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-executor" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a576fc72ae164fca6b9db127eaa9a9dda0d61316034f33a0a0d4eda41f02b01d" +dependencies = [ + "futures-core", + "futures-task", + "futures-util", +] + +[[package]] +name = "futures-io" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" + +[[package]] +name = "futures-locks" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "45ec6fe3675af967e67c5536c0b9d44e34e6c52f86bedc4ea49c5317b8e94d06" +dependencies = [ + "futures-channel", + "futures-task", +] + +[[package]] +name = "futures-macro" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[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-timer" +version = "3.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f288b0a4f20f9a56b5d1da57e2227c661b7b16168e2f72365f57b63326e29b24" +dependencies = [ + "gloo-timers", + "send_wrapper 0.4.0", +] + +[[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-macro", + "futures-sink", + "futures-task", + "memchr", + "pin-project-lite", + "pin-utils", + "slab", +] + +[[package]] +name = "fxhash" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c31b6d751ae2c7f11320402d34e41349dd1016f8d5d45e48c4312bc8625af50c" +dependencies = [ + "byteorder", +] + +[[package]] +name = "generic-array" +version = "0.14.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a" +dependencies = [ + "typenum", + "version_check", + "zeroize", +] + +[[package]] +name = "getrandom" +version = "0.1.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce" +dependencies = [ + "cfg-if", + "js-sys", + "libc", + "wasi 0.9.0+wasi-snapshot-preview1", + "wasm-bindgen", +] + +[[package]] +name = "getrandom" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7" +dependencies = [ + "cfg-if", + "js-sys", + "libc", + "wasi 0.11.0+wasi-snapshot-preview1", + "wasm-bindgen", +] + +[[package]] +name = "ghash" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0d8a4362ccb29cb0b265253fb0a2728f592895ee6854fd9bc13f2ffda266ff1" +dependencies = [ + "opaque-debug", + "polyval", +] + +[[package]] +name = "gimli" +version = "0.28.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4271d37baee1b8c7e4b708028c57d816cf9d2434acb33a549475f78c181f6253" + +[[package]] +name = "gloo-timers" +version = "0.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b995a66bb87bebce9a0f4a95aed01daca4872c050bfcb21653361c03bc35e5c" +dependencies = [ + "futures-channel", + "futures-core", + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "gloo-utils" +version = "0.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "037fcb07216cb3a30f7292bd0176b050b7b9a052ba830ef7d5d65f6dc64ba58e" +dependencies = [ + "js-sys", + "serde", + "serde_json", + "wasm-bindgen", + "web-sys", +] + +[[package]] +name = "group" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0f9ef7462f7c099f518d754361858f86d8a07af53ba9af0fe635bbccb151a63" +dependencies = [ + "ff", + "memuse", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "h2" +version = "0.3.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "81fe527a889e1532da5c525686d96d4c2e74cdd345badf8dfef9f6b39dd5f5e8" +dependencies = [ + "bytes", + "fnv", + "futures-core", + "futures-sink", + "futures-util", + "http 0.2.12", + "indexmap 2.2.6", + "slab", + "tokio", + "tokio-util", + "tracing", +] + +[[package]] +name = "hashbrown" +version = "0.12.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" + +[[package]] +name = "hashbrown" +version = "0.14.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1" + +[[package]] +name = "hashers" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b2bca93b15ea5a746f220e56587f71e73c6165eab783df9e26590069953e3c30" +dependencies = [ + "fxhash", +] + +[[package]] +name = "heck" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8" + +[[package]] +name = "heck" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2304e00983f87ffb38b55b444b5e3b60a884b5d30c0fca7d82fe33449bbe55ea" + +[[package]] +name = "hermit-abi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024" + +[[package]] +name = "hex" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" + +[[package]] +name = "hmac" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "126888268dcc288495a26bf004b38c5fdbb31682f992c84ceb046a1f0fe38840" +dependencies = [ + "crypto-mac", + "digest 0.9.0", +] + +[[package]] +name = "hmac" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e" +dependencies = [ + "digest 0.10.7", +] + +[[package]] +name = "hmac-sha512" +version = "0.1.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77e806677ce663d0a199541030c816847b36e8dc095f70dae4a4f4ad63da5383" + +[[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" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21b9ddb458710bc376481b842f5da65cdf31522de232c1ca8146abce2a358258" +dependencies = [ + "bytes", + "fnv", + "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 = "httparse" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d897f394bad6a705d5f4104762e116a75639e470d80901eed05a860a95cb1904" + +[[package]] +name = "httpdate" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9" + +[[package]] +name = "hyper" +version = "0.14.28" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf96e135eb83a2a8ddf766e426a841d8ddd7449d5f00d34ea02b41d2f19eef80" +dependencies = [ + "bytes", + "futures-channel", + "futures-core", + "futures-util", + "h2", + "http 0.2.12", + "http-body", + "httparse", + "httpdate", + "itoa", + "pin-project-lite", + "socket2", + "tokio", + "tower-service", + "tracing", + "want", +] + +[[package]] +name = "hyper-tls" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d6183ddfa99b85da61a140bea0efc93fdf56ceaa041b37d553518030827f9905" +dependencies = [ + "bytes", + "hyper", + "native-tls", + "tokio", + "tokio-native-tls", +] + +[[package]] +name = "iana-time-zone" +version = "0.1.60" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e7ffbb5a1b541ea2561f8c41c087286cc091e21e556a4f09a8f6cbf17b69b141" +dependencies = [ + "android_system_properties", + "core-foundation-sys", + "iana-time-zone-haiku", + "js-sys", + "wasm-bindgen", + "windows-core", +] + +[[package]] +name = "iana-time-zone-haiku" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f" +dependencies = [ + "cc", +] + +[[package]] +name = "ibc" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-apps", + "ibc-clients", + "ibc-core", + "ibc-core-host-cosmos", + "ibc-derive", + "ibc-primitives", +] + +[[package]] +name = "ibc-app-nft-transfer" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-app-nft-transfer-types", + "ibc-core", + "serde-json-wasm", +] + +[[package]] +name = "ibc-app-nft-transfer-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "base64 0.22.1", + "borsh", + "derive_more", + "displaydoc", + "http 1.1.0", + "ibc-app-transfer-types", + "ibc-core", + "ibc-proto", + "mime", + "parity-scale-codec", + "scale-info", + "schemars", + "serde", + "serde-json-wasm", +] + +[[package]] +name = "ibc-app-transfer" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-app-transfer-types", + "ibc-core", + "serde-json-wasm", +] + +[[package]] +name = "ibc-app-transfer-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-core", + "ibc-proto", + "parity-scale-codec", + "primitive-types", + "scale-info", + "schemars", + "serde", + "uint", +] + +[[package]] +name = "ibc-apps" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-app-nft-transfer", + "ibc-app-transfer", +] + +[[package]] +name = "ibc-client-tendermint" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "derive_more", + "ibc-client-tendermint-types", + "ibc-core-client", + "ibc-core-commitment-types", + "ibc-core-handler-types", + "ibc-core-host", + "ibc-primitives", + "serde", + "tendermint 0.38.1", + "tendermint-light-client-verifier", +] + +[[package]] +name = "ibc-client-tendermint-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "displaydoc", + "ibc-core-client-types", + "ibc-core-commitment-types", + "ibc-core-host-types", + "ibc-primitives", + "ibc-proto", + "serde", + "tendermint 0.38.1", + "tendermint-light-client-verifier", + "tendermint-proto 0.38.1", +] + +[[package]] +name = "ibc-client-wasm-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "base64 0.22.1", + "displaydoc", + "ibc-core-client", + "ibc-core-host-types", + "ibc-primitives", + "ibc-proto", + "serde", +] + +[[package]] +name = "ibc-clients" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-client-tendermint", + "ibc-client-wasm-types", +] + +[[package]] +name = "ibc-core" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-core-channel", + "ibc-core-client", + "ibc-core-commitment-types", + "ibc-core-connection", + "ibc-core-handler", + "ibc-core-host", + "ibc-core-router", + "ibc-derive", + "ibc-primitives", +] + +[[package]] +name = "ibc-core-channel" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-core-channel-types", + "ibc-core-client", + "ibc-core-commitment-types", + "ibc-core-connection", + "ibc-core-handler-types", + "ibc-core-host", + "ibc-core-router", + "ibc-primitives", +] + +[[package]] +name = "ibc-core-channel-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-core-client-types", + "ibc-core-commitment-types", + "ibc-core-connection-types", + "ibc-core-host-types", + "ibc-primitives", + "ibc-proto", + "parity-scale-codec", + "scale-info", + "schemars", + "serde", + "sha2 0.10.8", + "subtle-encoding", + "tendermint 0.38.1", +] + +[[package]] +name = "ibc-core-client" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-core-client-context", + "ibc-core-client-types", + "ibc-core-commitment-types", + "ibc-core-handler-types", + "ibc-core-host", + "ibc-primitives", +] + +[[package]] +name = "ibc-core-client-context" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "derive_more", + "displaydoc", + "ibc-core-client-types", + "ibc-core-commitment-types", + "ibc-core-handler-types", + "ibc-core-host-types", + "ibc-primitives", + "subtle-encoding", + "tendermint 0.38.1", +] + +[[package]] +name = "ibc-core-client-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-core-commitment-types", + "ibc-core-host-types", + "ibc-primitives", + "ibc-proto", + "parity-scale-codec", + "scale-info", + "schemars", + "serde", + "subtle-encoding", + "tendermint 0.38.1", +] + +[[package]] +name = "ibc-core-commitment-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-core-host-types", + "ibc-primitives", + "ibc-proto", + "ics23", + "parity-scale-codec", + "scale-info", + "schemars", + "serde", + "subtle-encoding", +] + +[[package]] +name = "ibc-core-connection" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-client-wasm-types", + "ibc-core-client", + "ibc-core-connection-types", + "ibc-core-handler-types", + "ibc-core-host", + "ibc-primitives", + "prost 0.13.2", +] + +[[package]] +name = "ibc-core-connection-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-core-client-types", + "ibc-core-commitment-types", + "ibc-core-host-types", + "ibc-primitives", + "ibc-proto", + "parity-scale-codec", + "scale-info", + "schemars", + "serde", + "subtle-encoding", + "tendermint 0.38.1", +] + +[[package]] +name = "ibc-core-handler" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "ibc-core-channel", + "ibc-core-client", + "ibc-core-commitment-types", + "ibc-core-connection", + "ibc-core-handler-types", + "ibc-core-host", + "ibc-core-router", + "ibc-primitives", +] + +[[package]] +name = "ibc-core-handler-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-core-channel-types", + "ibc-core-client-types", + "ibc-core-commitment-types", + "ibc-core-connection-types", + "ibc-core-host-types", + "ibc-core-router-types", + "ibc-primitives", + "ibc-proto", + "parity-scale-codec", + "scale-info", + "schemars", + "serde", + "subtle-encoding", + "tendermint 0.38.1", +] + +[[package]] +name = "ibc-core-host" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "derive_more", + "displaydoc", + "ibc-core-channel-types", + "ibc-core-client-context", + "ibc-core-client-types", + "ibc-core-commitment-types", + "ibc-core-connection-types", + "ibc-core-handler-types", + "ibc-core-host-types", + "ibc-primitives", + "subtle-encoding", +] + +[[package]] +name = "ibc-core-host-cosmos" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "derive_more", + "displaydoc", + "ibc-app-transfer-types", + "ibc-client-tendermint", + "ibc-core-client-context", + "ibc-core-client-types", + "ibc-core-commitment-types", + "ibc-core-connection-types", + "ibc-core-handler-types", + "ibc-core-host-types", + "ibc-primitives", + "ibc-proto", + "serde", + "sha2 0.10.8", + "subtle-encoding", + "tendermint 0.38.1", +] + +[[package]] +name = "ibc-core-host-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-primitives", + "parity-scale-codec", + "scale-info", + "schemars", + "serde", +] + +[[package]] +name = "ibc-core-router" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "derive_more", + "displaydoc", + "ibc-core-channel-types", + "ibc-core-host-types", + "ibc-core-router-types", + "ibc-primitives", + "subtle-encoding", +] + +[[package]] +name = "ibc-core-router-types" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-core-host-types", + "ibc-primitives", + "ibc-proto", + "parity-scale-codec", + "scale-info", + "schemars", + "serde", + "subtle-encoding", + "tendermint 0.38.1", +] + +[[package]] +name = "ibc-derive" +version = "0.8.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "ibc-primitives" +version = "0.54.0" +source = "git+https://github.com/heliaxdev/cosmos-ibc-rs?rev=38bd2a32f35117d4d9165a3c68c64ccd87ad56dd#38bd2a32f35117d4d9165a3c68c64ccd87ad56dd" +dependencies = [ + "borsh", + "derive_more", + "displaydoc", + "ibc-proto", + "parity-scale-codec", + "prost 0.13.2", + "scale-info", + "schemars", + "serde", + "tendermint 0.38.1", + "time", +] + +[[package]] +name = "ibc-proto" +version = "0.47.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c852d22b782d2d793f4a646f968de419be635e02bc8798d5d74a6e44eef27733" +dependencies = [ + "base64 0.22.1", + "borsh", + "bytes", + "flex-error", + "ics23", + "informalsystems-pbjson", + "parity-scale-codec", + "prost 0.13.2", + "scale-info", + "schemars", + "serde", + "subtle-encoding", + "tendermint-proto 0.38.1", +] + +[[package]] +name = "ics23" +version = "0.12.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73b17f1a5bd7d12ad30a21445cfa5f52fd7651cb3243ba866f9916b1ec112f12" +dependencies = [ + "anyhow", + "blake2", + "blake3", + "bytes", + "hex", + "informalsystems-pbjson", + "prost 0.13.2", + "ripemd", + "serde", + "sha2 0.10.8", + "sha3", +] + +[[package]] +name = "idna" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" +dependencies = [ + "unicode-bidi", + "unicode-normalization", +] + +[[package]] +name = "impl-codec" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba6a270039626615617f3f36d15fc827041df3b78c439da2cadfa47455a77f2f" +dependencies = [ + "parity-scale-codec", +] + +[[package]] +name = "impl-num-traits" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "951641f13f873bff03d4bf19ae8bec531935ac0ac2cc775f84d7edfdcfed3f17" +dependencies = [ + "integer-sqrt", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "uint", +] + +[[package]] +name = "impl-rlp" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f28220f89297a075ddc7245cd538076ee98b01f2a9c23a53a4f1105d5a322808" +dependencies = [ + "rlp", +] + +[[package]] +name = "impl-serde" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ebc88fc67028ae3db0c853baa36269d398d5f45b6982f95549ff5def78c935cd" +dependencies = [ + "serde", +] + +[[package]] +name = "impl-trait-for-tuples" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11d7a9f6330b71fea57921c9b61c47ee6e84f72d394754eff6163ae67e7395eb" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "incrementalmerkletree" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d5ad43a3f5795945459d577f6589cf62a476e92c79b75e70cd954364e14ce17b" +dependencies = [ + "serde", +] + +[[package]] +name = "indenter" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce23b50ad8242c51a442f3ff322d56b02f08852c77e4c0b4d3fd684abc89c683" + +[[package]] +name = "index-set" +version = "0.8.0" +source = "git+https://github.com/heliaxdev/index-set?tag=v0.8.1#b0d928f83cf0d465ccda299d131e8df2859b5184" +dependencies = [ + "borsh", + "serde", +] + +[[package]] +name = "indexmap" +version = "1.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd070e393353796e801d209ad339e89596eb4c8d430d18ede6a1cced8fafbd99" +dependencies = [ + "autocfg", + "hashbrown 0.12.3", +] + +[[package]] +name = "indexmap" +version = "2.2.4" +source = "git+https://github.com/heliaxdev/indexmap?tag=2.2.4-heliax-1#b5b5b547bd6ab04bbb16e060326a50ddaeb6c909" +dependencies = [ + "borsh", + "equivalent", + "hashbrown 0.14.5", + "serde", +] + +[[package]] +name = "indexmap" +version = "2.2.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "168fb715dda47215e360912c096649d23d58bf392ac62f73919e831745e40f26" +dependencies = [ + "equivalent", + "hashbrown 0.14.5", +] + +[[package]] +name = "informalsystems-pbjson" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9aa4a0980c8379295100d70854354e78df2ee1c6ca0f96ffe89afeb3140e3a3d" +dependencies = [ + "base64 0.21.7", + "serde", +] + +[[package]] +name = "init-once" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e0863329819ed5ecf33446da6cb9104d2f8943ff8530d2b6c51adbc6be4f0632" +dependencies = [ + "portable-atomic", +] + +[[package]] +name = "inout" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a0c10553d664a4d0bcff9f4215d0aac67a639cc68ef660840afe309b807bc9f5" +dependencies = [ + "generic-array", +] + +[[package]] +name = "instant" +version = "0.1.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e0242819d153cba4b4b05a5a8f2a7e9bbf97b6055b2a002b395c96b5ff3c0222" +dependencies = [ + "cfg-if", +] + +[[package]] +name = "integer-sqrt" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "276ec31bcb4a9ee45f58bec6f9ec700ae4cf4f4f8f2fa7e06cb406bd5ffdd770" +dependencies = [ + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "ipnet" +version = "2.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f518f335dce6725a761382244631d86cf0ccb2863413590b31338feb467f9c3" + +[[package]] +name = "itertools" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1c173a5686ce8bfa551b3563d0c2170bf24ca44da99c7ca4bfdab5418c3fe57" +dependencies = [ + "either", +] + +[[package]] +name = "itertools" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba291022dbbd398a455acf126c1e341954079855bc60dfdda641363bd6922569" +dependencies = [ + "either", +] + +[[package]] +name = "itoa" +version = "1.0.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49f1f14873335454500d59611f1cf4a4b0f786f9ac11f4312a78e4cf2566695b" + +[[package]] +name = "js-sys" +version = "0.3.69" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29c15563dc2726973df627357ce0c9ddddbea194836909d655df6a75d2cf296d" +dependencies = [ + "wasm-bindgen", +] + +[[package]] +name = "jsonwebtoken" +version = "8.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6971da4d9c3aa03c3d8f3ff0f4155b534aad021292003895a469716b2a230378" +dependencies = [ + "base64 0.21.7", + "pem", + "ring", + "serde", + "serde_json", + "simple_asn1", +] + +[[package]] +name = "jubjub" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8499f7a74008aafbecb2a2e608a3e13e4dd3e84df198b604451efe93f2de6e61" +dependencies = [ + "bitvec", + "bls12_381 0.8.0 (registry+https://github.com/rust-lang/crates.io-index)", + "ff", + "group", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "jubjub" +version = "0.10.0" +source = "git+https://github.com/heliaxdev/jubjub.git?rev=a373686962f4e9d0edb3b4716f86ff6bbd9aa86c#a373686962f4e9d0edb3b4716f86ff6bbd9aa86c" +dependencies = [ + "bitvec", + "bls12_381 0.8.0 (git+https://github.com/heliaxdev/bls12_381.git?rev=d3ebe9dd6488fac1923db120a7498079e55dd838)", + "ff", + "group", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "k256" +version = "0.13.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "956ff9b67e26e1a6a866cb758f12c6f8746208489e3e4a4b5580802f2f0a587b" +dependencies = [ + "cfg-if", + "ecdsa", + "elliptic-curve", + "once_cell", + "serdect", + "sha2 0.10.8", + "signature", +] + +[[package]] +name = "kdam" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "526586ea01a9a132b5f8d3a60f6d6b41b411550236f5ee057795f20b37316957" +dependencies = [ + "terminal_size", + "windows-sys 0.52.0", +] + +[[package]] +name = "keccak" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ecc2af9a1119c51f12a14607e783cb977bde58bc069ff0c3da1095e635d70654" +dependencies = [ + "cpufeatures", +] + +[[package]] +name = "konst" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "50a0ba6de5f7af397afff922f22c149ff605c766cd3269cf6c1cd5e466dbe3b9" +dependencies = [ + "const_panic", + "konst_kernel", + "typewit", +] + +[[package]] +name = "konst_kernel" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be0a455a1719220fd6adf756088e1c69a85bf14b6a9e24537a5cc04f503edb2b" +dependencies = [ + "typewit", +] + +[[package]] +name = "lazy_static" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" + +[[package]] +name = "libc" +version = "0.2.155" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97b3888a4aecf77e811145cadf6eef5901f4782c53886191b2f693f24761847c" + +[[package]] +name = "libm" +version = "0.2.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ec2a862134d2a7d32d7983ddcdd1c4923530833c9f2ea1a44fc5fa473989058" + +[[package]] +name = "libredox" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c0ff37bd590ca25063e35af745c343cb7a0271906fb7b37e4813e8f79f00268d" +dependencies = [ + "bitflags 2.5.0", + "libc", +] + +[[package]] +name = "linux-raw-sys" +version = "0.4.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78b3ae25bc7c8c38cec158d1f2757ee79e9b3740fbc7ccf0e59e4b08d793fa89" + +[[package]] +name = "lock_api" +version = "0.4.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07af8b9cdd281b7915f413fa73f29ebd5d55d0d3f0155584dade1ff18cea1b17" +dependencies = [ + "autocfg", + "scopeguard", +] + +[[package]] +name = "log" +version = "0.4.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90ed8c1e510134f979dbc4f070f87d4313098b704861a105fe34231c70a3901c" + +[[package]] +name = "masp_note_encryption" +version = "1.0.0" +source = "git+https://github.com/anoma/masp?tag=v1.1.0#f24691c0eb76909e3c15ae03aef294dccebd2df3" +dependencies = [ + "borsh", + "chacha20", + "chacha20poly1305", + "cipher", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "masp_note_encryption" +version = "1.0.0" +source = "git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe#0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe" +dependencies = [ + "borsh", + "chacha20", + "chacha20poly1305", + "cipher", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "masp_primitives" +version = "1.0.0" +source = "git+https://github.com/anoma/masp?tag=v1.1.0#f24691c0eb76909e3c15ae03aef294dccebd2df3" +dependencies = [ + "aes", + "bip0039 0.10.1", + "bitvec", + "blake2b_simd", + "blake2s_simd", + "bls12_381 0.8.0 (registry+https://github.com/rust-lang/crates.io-index)", + "borsh", + "byteorder", + "ff", + "fpe", + "group", + "hex", + "incrementalmerkletree", + "jubjub 0.10.0 (registry+https://github.com/rust-lang/crates.io-index)", + "lazy_static", + "masp_note_encryption 1.0.0 (git+https://github.com/anoma/masp?tag=v1.1.0)", + "memuse", + "nonempty", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.8.5", + "rand_core 0.6.4", + "sha2 0.10.8", + "subtle", + "zcash_encoding", +] + +[[package]] +name = "masp_primitives" +version = "1.0.0" +source = "git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe#0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe" +dependencies = [ + "aes", + "bip0039 0.12.0", + "bitvec", + "blake2b_simd", + "blake2s_simd", + "bls12_381 0.8.0 (git+https://github.com/heliaxdev/bls12_381.git?rev=d3ebe9dd6488fac1923db120a7498079e55dd838)", + "borsh", + "byteorder", + "ff", + "fpe", + "group", + "hex", + "incrementalmerkletree", + "jubjub 0.10.0 (git+https://github.com/heliaxdev/jubjub.git?rev=a373686962f4e9d0edb3b4716f86ff6bbd9aa86c)", + "lazy_static", + "masp_note_encryption 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "memuse", + "nonempty", + "num-traits 0.2.19 (git+https://github.com/heliaxdev/num-traits?rev=3f3657caa34b8e116fdf3f8a3519c4ac29f012fe)", + "rand 0.8.5", + "rand_core 0.6.4", + "sha2 0.10.8", + "subtle", + "zcash_encoding", +] + +[[package]] +name = "masp_proofs" +version = "1.0.0" +source = "git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe#0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe" +dependencies = [ + "bellman", + "blake2b_simd", + "bls12_381 0.8.0 (git+https://github.com/heliaxdev/bls12_381.git?rev=d3ebe9dd6488fac1923db120a7498079e55dd838)", + "directories", + "getrandom 0.2.15", + "group", + "itertools 0.11.0", + "jubjub 0.10.0 (git+https://github.com/heliaxdev/jubjub.git?rev=a373686962f4e9d0edb3b4716f86ff6bbd9aa86c)", + "lazy_static", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "rand_core 0.6.4", + "redjubjub", + "tracing", +] + +[[package]] +name = "memchr" +version = "2.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c8640c5d730cb13ebd907d8d04b52f55ac9a2eec55b440c8892f40d56c76c1d" + +[[package]] +name = "memuse" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2145869435ace5ea6ea3d35f59be559317ec9a0d04e1812d5f185a87b6d36f1a" + +[[package]] +name = "memzero" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93c0d11ac30a033511ae414355d80f70d9f29a44a49140face477117a1ee90db" + +[[package]] +name = "mime" +version = "0.3.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a" + +[[package]] +name = "miniz_oxide" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87dfd01fe195c66b572b37921ad8803d010623c0aca821bea2302239d155cdae" +dependencies = [ + "adler", +] + +[[package]] +name = "mio" +version = "0.8.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4a650543ca06a924e8b371db273b2756685faae30f8487da1b56505a8f78b0c" +dependencies = [ + "libc", + "wasi 0.11.0+wasi-snapshot-preview1", + "windows-sys 0.48.0", +] + +[[package]] +name = "multimap" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "defc4c55412d89136f966bbb339008b474350e5e6e78d2714439c386b3137a03" + +[[package]] +name = "namada-sdk-wasm" +version = "1.0.0" +dependencies = [ + "aes-gcm", + "argon2", + "async-trait", + "borsh-ext", + "chrono", + "console_error_panic_hook", + "getrandom 0.2.15", + "gloo-utils", + "hex", + "js-sys", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?tag=v1.1.0)", + "namada_sdk", + "namada_tx", + "password-hash 0.3.2", + "rand 0.8.5", + "rayon", + "reqwest", + "rexie", + "serde", + "serde_json", + "slip10_ed25519", + "subtle-encoding", + "tendermint-config 0.34.1", + "thiserror", + "tiny-bip39 1.0.0", + "tokio", + "wasm-bindgen", + "wasm-bindgen-futures", + "wasm-bindgen-rayon", + "wasm-bindgen-test", + "web-sys", + "zeroize", +] + +[[package]] +name = "namada_account" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "namada_core", + "namada_macros", + "namada_storage", + "serde", +] + +[[package]] +name = "namada_controller" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "namada_core", + "smooth-operator", + "thiserror", +] + +[[package]] +name = "namada_core" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "bech32 0.8.1", + "borsh", + "borsh-ext", + "chrono", + "data-encoding", + "ed25519-consensus", + "ethabi", + "ethbridge-structs", + "eyre", + "ibc", + "ics23", + "impl-num-traits", + "index-set", + "indexmap 2.2.4", + "k256", + "lazy_static", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "namada_macros", + "num-integer", + "num-rational", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "num256", + "num_enum", + "primitive-types", + "prost-types 0.13.2", + "rand 0.8.5", + "rand_core 0.6.4", + "rayon", + "ripemd", + "serde", + "serde_json", + "sha2 0.9.9", + "smooth-operator", + "sparse-merkle-tree", + "tendermint 0.38.1", + "tendermint-proto 0.38.1", + "thiserror", + "tiny-keccak", + "tokio", + "tracing", + "uint", + "wasmtimer", + "zeroize", +] + +[[package]] +name = "namada_ethereum_bridge" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "ethers", + "eyre", + "itertools 0.12.1", + "konst", + "namada_core", + "namada_events", + "namada_macros", + "namada_parameters", + "namada_proof_of_stake", + "namada_state", + "namada_storage", + "namada_systems", + "namada_trans_token", + "namada_tx", + "namada_vote_ext", + "namada_vp_env", + "serde", + "smooth-operator", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_events" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "namada_core", + "namada_macros", + "serde", + "serde_json", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_gas" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "namada_core", + "namada_events", + "namada_macros", + "serde", + "thiserror", +] + +[[package]] +name = "namada_governance" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "itertools 0.12.1", + "konst", + "namada_account", + "namada_core", + "namada_events", + "namada_macros", + "namada_state", + "namada_systems", + "namada_tx", + "namada_vp_env", + "serde", + "serde_json", + "smooth-operator", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_ibc" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "data-encoding", + "ibc", + "ibc-derive", + "ics23", + "konst", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "namada_core", + "namada_events", + "namada_gas", + "namada_macros", + "namada_state", + "namada_systems", + "namada_tx", + "namada_vp", + "primitive-types", + "prost 0.13.2", + "serde", + "serde_json", + "sha2 0.9.9", + "smooth-operator", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_io" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "async-trait", + "kdam", + "namada_core", + "tendermint-rpc", + "thiserror", + "tokio", +] + +[[package]] +name = "namada_macros" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "data-encoding", + "proc-macro2", + "quote", + "sha2 0.9.9", + "syn 1.0.109", +] + +[[package]] +name = "namada_merkle_tree" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "eyre", + "ics23", + "namada_core", + "namada_macros", + "prost 0.13.2", + "sparse-merkle-tree", + "thiserror", +] + +[[package]] +name = "namada_parameters" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "namada_core", + "namada_macros", + "namada_state", + "namada_systems", + "namada_tx", + "namada_vp_env", + "smooth-operator", + "thiserror", +] + +[[package]] +name = "namada_proof_of_stake" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "itertools 0.12.1", + "konst", + "namada_account", + "namada_controller", + "namada_core", + "namada_events", + "namada_macros", + "namada_state", + "namada_systems", + "namada_tx", + "namada_vp_env", + "once_cell", + "serde", + "smooth-operator", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_replay_protection" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "namada_core", +] + +[[package]] +name = "namada_sdk" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "async-trait", + "bimap", + "borsh", + "borsh-ext", + "circular-queue", + "clap", + "data-encoding", + "duration-str", + "either", + "ethbridge-bridge-contract", + "ethers", + "eyre", + "futures", + "init-once", + "itertools 0.12.1", + "lazy_static", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "masp_proofs", + "namada_account", + "namada_core", + "namada_ethereum_bridge", + "namada_events", + "namada_gas", + "namada_governance", + "namada_ibc", + "namada_io", + "namada_macros", + "namada_parameters", + "namada_proof_of_stake", + "namada_state", + "namada_storage", + "namada_token", + "namada_tx", + "namada_vm", + "namada_vote_ext", + "namada_vp", + "namada_wallet", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "num256", + "owo-colors", + "paste", + "patricia_tree", + "prost 0.13.2", + "rand 0.8.5", + "rand_core 0.6.4", + "rayon", + "regex", + "reqwest", + "rustversion", + "serde", + "serde_json", + "sha2 0.9.9", + "smooth-operator", + "tempfile", + "tendermint-rpc", + "thiserror", + "tiny-bip39 0.8.2", + "tokio", + "toml 0.5.11", + "tracing", + "xorf", + "zeroize", +] + +[[package]] +name = "namada_shielded_token" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "async-trait", + "borsh", + "eyre", + "flume", + "futures", + "itertools 0.12.1", + "lazy_static", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "masp_proofs", + "namada_account", + "namada_controller", + "namada_core", + "namada_events", + "namada_gas", + "namada_io", + "namada_macros", + "namada_state", + "namada_systems", + "namada_tx", + "namada_vp_env", + "namada_wallet", + "rand 0.8.5", + "rand_core 0.6.4", + "rayon", + "ripemd", + "serde", + "serde_json", + "sha2 0.9.9", + "smooth-operator", + "tempfile", + "thiserror", + "tracing", + "typed-builder", + "xorf", +] + +[[package]] +name = "namada_state" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "clru", + "itertools 0.12.1", + "namada_core", + "namada_events", + "namada_gas", + "namada_macros", + "namada_merkle_tree", + "namada_replay_protection", + "namada_storage", + "namada_systems", + "namada_tx", + "patricia_tree", + "smooth-operator", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_storage" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "itertools 0.12.1", + "namada_core", + "namada_gas", + "namada_macros", + "namada_merkle_tree", + "namada_replay_protection", + "regex", + "serde", + "smooth-operator", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_systems" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "namada_core", + "namada_events", + "namada_storage", +] + +[[package]] +name = "namada_token" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "namada_core", + "namada_events", + "namada_macros", + "namada_shielded_token", + "namada_storage", + "namada_systems", + "namada_trans_token", + "namada_tx", + "namada_tx_env", + "serde", +] + +[[package]] +name = "namada_trans_token" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "konst", + "namada_core", + "namada_events", + "namada_state", + "namada_systems", + "namada_tx", + "namada_tx_env", + "namada_vp_env", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_tx" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "ark-bls12-381", + "bitflags 2.5.0", + "borsh", + "data-encoding", + "either", + "konst", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "namada_account", + "namada_core", + "namada_events", + "namada_gas", + "namada_macros", + "num-derive 0.4.2", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "prost 0.13.2", + "prost-types 0.13.2", + "rand_core 0.6.4", + "serde", + "serde_json", + "sha2 0.9.9", + "thiserror", + "tonic-build", +] + +[[package]] +name = "namada_tx_env" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "namada_core", + "namada_events", + "namada_storage", +] + +[[package]] +name = "namada_vm" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "clru", + "namada_account", + "namada_core", + "namada_events", + "namada_gas", + "namada_parameters", + "namada_state", + "namada_token", + "namada_tx", + "namada_vp", + "smooth-operator", + "thiserror", + "tracing", + "wasmparser", +] + +[[package]] +name = "namada_vote_ext" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "borsh", + "namada_core", + "namada_macros", + "namada_tx", + "serde", +] + +[[package]] +name = "namada_vp" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "namada_core", + "namada_events", + "namada_gas", + "namada_state", + "namada_tx", + "namada_vp_env", + "smooth-operator", + "thiserror", + "tracing", +] + +[[package]] +name = "namada_vp_env" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "derivative", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "namada_core", + "namada_events", + "namada_gas", + "namada_storage", + "namada_tx", + "smooth-operator", +] + +[[package]] +name = "namada_wallet" +version = "0.46.0" +source = "git+https://github.com/anoma/namada?rev=49a4a5d3260423df19ead14df82d18a51fa9b157#49a4a5d3260423df19ead14df82d18a51fa9b157" +dependencies = [ + "bimap", + "borsh", + "borsh-ext", + "data-encoding", + "derivation-path", + "itertools 0.12.1", + "masp_primitives 1.0.0 (git+https://github.com/anoma/masp?rev=0d0da3507a6f9ad135f00fd8201dc54c2f1d9efe)", + "namada_core", + "namada_ibc", + "namada_macros", + "orion", + "rand 0.8.5", + "rand_core 0.6.4", + "serde", + "slip10_ed25519", + "smooth-operator", + "thiserror", + "tiny-bip39 0.8.2", + "tiny-hderive", + "toml 0.5.11", + "zeroize", +] + +[[package]] +name = "nanorand" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a51313c5820b0b02bd422f4b44776fbf47961755c74ce64afc73bfad10226c3" +dependencies = [ + "getrandom 0.2.15", +] + +[[package]] +name = "native-tls" +version = "0.2.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "07226173c32f2926027b63cce4bcd8076c3552846cbe7925f3aaffeac0a3b92e" +dependencies = [ + "lazy_static", + "libc", + "log", + "openssl", + "openssl-probe", + "openssl-sys", + "schannel", + "security-framework", + "security-framework-sys", + "tempfile", +] + +[[package]] +name = "nonempty" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e9e591e719385e6ebaeb5ce5d3887f7d5676fceca6411d1925ccc95745f3d6f7" + +[[package]] +name = "num" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "35bd024e8b2ff75562e5f34e7f4905839deb4b22955ef5e73d2fea1b9813cb23" +dependencies = [ + "num-bigint", + "num-complex", + "num-integer", + "num-iter", + "num-rational", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "num-bigint" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c165a9ab64cf766f73521c0dd2cfdff64f488b8f0b3e621face3462d3db536d7" +dependencies = [ + "num-integer", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "num-complex" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "73f88a1307638156682bada9d7604135552957b7818057dcef22705b4d509495" +dependencies = [ + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "num-conv" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "51d515d32fb182ee37cda2ccdcb92950d6a3c2893aa280e540671c2cd0f3b1d9" + +[[package]] +name = "num-derive" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "876a53fff98e03a936a674b29568b0e605f06b29372c2489ff4de23f1949743d" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "num-derive" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed3955f1a9c7c0c15e092f9c887db08b1fc683305fdf6eb6684f22555355e202" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "num-integer" +version = "0.1.46" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7969661fd2958a5cb096e56c8e1ad0444ac2bbcd0061bd28660485a44879858f" +dependencies = [ + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "num-iter" +version = "0.1.45" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1429034a0490724d0075ebb2bc9e875d6503c3cf69e235a8941aa757d83ef5bf" +dependencies = [ + "autocfg", + "num-integer", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "num-rational" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f83d14da390562dca69fc84082e73e548e1ad308d24accdedd2720017cb37824" +dependencies = [ + "num-bigint", + "num-integer", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "num-traits" +version = "0.2.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841" +dependencies = [ + "autocfg", + "libm", +] + +[[package]] +name = "num-traits" +version = "0.2.19" +source = "git+https://github.com/heliaxdev/num-traits?rev=3f3657caa34b8e116fdf3f8a3519c4ac29f012fe#3f3657caa34b8e116fdf3f8a3519c4ac29f012fe" +dependencies = [ + "autocfg", +] + +[[package]] +name = "num256" +version = "0.3.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aa9b5179e82f0867b23e0b9b822493821f9345561f271364f409c8e4a058367d" +dependencies = [ + "lazy_static", + "num", + "num-derive 0.3.3", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "serde", + "serde_derive", +] + +[[package]] +name = "num_cpus" +version = "1.16.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" +dependencies = [ + "hermit-abi", + "libc", +] + +[[package]] +name = "num_enum" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02339744ee7253741199f897151b38e72257d13802d4ee837285cc2990a90845" +dependencies = [ + "num_enum_derive", +] + +[[package]] +name = "num_enum_derive" +version = "0.7.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "681030a937600a36906c185595136d26abfebb4aa9c65701cefcaf8578bb982b" +dependencies = [ + "proc-macro-crate", + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "object" +version = "0.32.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a6a622008b6e321afc04970976f62ee297fdbaa6f95318ca343e3eebb9648441" +dependencies = [ + "memchr", +] + +[[package]] +name = "once_cell" +version = "1.19.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92" + +[[package]] +name = "opaque-debug" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c08d65885ee38876c4f86fa503fb49d7b507c2b62552df7c70b2fce627e06381" + +[[package]] +name = "open-fastrlp" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "786393f80485445794f6043fd3138854dd109cc6c4bd1a6383db304c9ce9b9ce" +dependencies = [ + "arrayvec", + "auto_impl", + "bytes", + "ethereum-types", + "open-fastrlp-derive", +] + +[[package]] +name = "open-fastrlp-derive" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "003b2be5c6c53c1cfeb0a238b8a1c3915cd410feb684457a36c10038f764bb1c" +dependencies = [ + "bytes", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "openssl" +version = "0.10.64" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "95a0481286a310808298130d22dd1fef0fa571e05a8f44ec801801e84b216b1f" +dependencies = [ + "bitflags 2.5.0", + "cfg-if", + "foreign-types", + "libc", + "once_cell", + "openssl-macros", + "openssl-sys", +] + +[[package]] +name = "openssl-macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "openssl-probe" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" + +[[package]] +name = "openssl-sys" +version = "0.9.102" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c597637d56fbc83893a35eb0dd04b2b8e7a50c91e64e9493e398b5df4fb45fa2" +dependencies = [ + "cc", + "libc", + "pkg-config", + "vcpkg", +] + +[[package]] +name = "orion" +version = "0.16.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6624905ddd92e460ff0685567539ed1ac985b2dee4c92c7edcd64fce905b00c" +dependencies = [ + "ct-codecs", + "getrandom 0.2.15", + "subtle", + "zeroize", +] + +[[package]] +name = "owo-colors" +version = "3.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c1b04fb49957986fdce4d6ee7a65027d55d4b6d2265e5848bbb507b58ccfdb6f" + +[[package]] +name = "pairing" +version = "0.23.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "81fec4625e73cf41ef4bb6846cafa6d44736525f442ba45e407c4a000a13996f" +dependencies = [ + "group", +] + +[[package]] +name = "parity-scale-codec" +version = "3.6.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "306800abfa29c7f16596b5970a588435e3d5b3149683d00c12b699cc19f895ee" +dependencies = [ + "arrayvec", + "bitvec", + "byte-slice-cast", + "impl-trait-for-tuples", + "parity-scale-codec-derive", + "serde", +] + +[[package]] +name = "parity-scale-codec-derive" +version = "3.6.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d830939c76d294956402033aee57a6da7b438f2294eb94864c37b0569053a42c" +dependencies = [ + "proc-macro-crate", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "parking_lot" +version = "0.12.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7e4af0ca4f6caed20e900d564c242b8e5d4903fdacf31d3daf527b66fe6f42fb" +dependencies = [ + "lock_api", + "parking_lot_core", +] + +[[package]] +name = "parking_lot_core" +version = "0.9.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" +dependencies = [ + "cfg-if", + "libc", + "redox_syscall", + "smallvec", + "windows-targets 0.52.5", +] + +[[package]] +name = "password-hash" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d791538a6dcc1e7cb7fe6f6b58aca40e7f79403c45b2bc274008b5e647af1d8" +dependencies = [ + "base64ct", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "password-hash" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7676374caaee8a325c9e7a2ae557f216c5563a171d6997b0ef8a65af35147700" +dependencies = [ + "base64ct", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "password-hash" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "346f04948ba92c43e8469c1ee6736c7563d71012b17d40745260fe106aac2166" +dependencies = [ + "base64ct", + "rand_core 0.6.4", + "subtle", +] + +[[package]] +name = "pasta_curves" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3e57598f73cc7e1b2ac63c79c517b31a0877cd7c402cdcaa311b5208de7a095" +dependencies = [ + "ff", + "group", + "rand 0.8.5", + "static_assertions", + "subtle", +] + +[[package]] +name = "paste" +version = "1.0.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a" + +[[package]] +name = "patricia_tree" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31f2f4539bffe53fc4b4da301df49d114b845b077bd5727b7fe2bd9d8df2ae68" +dependencies = [ + "bitflags 2.5.0", +] + +[[package]] +name = "pbkdf2" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "216eaa586a190f0a738f2f918511eecfa90f13295abec0e457cdebcceda80cbd" +dependencies = [ + "crypto-mac", +] + +[[package]] +name = "pbkdf2" +version = "0.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "271779f35b581956db91a3e55737327a03aa051e90b1c47aeb189508533adfd7" +dependencies = [ + "digest 0.10.7", + "password-hash 0.3.2", +] + +[[package]] +name = "pbkdf2" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "83a0692ec44e4cf1ef28ca317f14f8f07da2d95ec3fa01f86e4467b725e60917" +dependencies = [ + "digest 0.10.7", +] + +[[package]] +name = "pbkdf2" +version = "0.12.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8ed6a7761f76e3b9f92dfb0a60a6a6477c61024b775147ff0973a02653abaf2" +dependencies = [ + "digest 0.10.7", + "hmac 0.12.1", + "password-hash 0.5.0", +] + +[[package]] +name = "peg" +version = "0.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a625d12ad770914cbf7eff6f9314c3ef803bfe364a1b20bc36ddf56673e71e5" +dependencies = [ + "peg-macros", + "peg-runtime", +] + +[[package]] +name = "peg-macros" +version = "0.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f241d42067ed3ab6a4fece1db720838e1418f36d868585a27931f95d6bc03582" +dependencies = [ + "peg-runtime", + "proc-macro2", + "quote", +] + +[[package]] +name = "peg-runtime" +version = "0.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3aeb8f54c078314c2065ee649a7241f46b9d8e418e1a9581ba0546657d7aa3a" + +[[package]] +name = "pem" +version = "1.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8835c273a76a90455d7344889b0964598e3316e2a79ede8e36f16bdcf2228b8" +dependencies = [ + "base64 0.13.1", +] + +[[package]] +name = "percent-encoding" +version = "2.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" + +[[package]] +name = "pest" +version = "2.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "560131c633294438da9f7c4b08189194b20946c8274c6b9e38881a7874dc8ee8" +dependencies = [ + "memchr", + "thiserror", + "ucd-trie", +] + +[[package]] +name = "petgraph" +version = "0.6.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b4c5cc86750666a3ed20bdaf5ca2a0344f9c67674cae0515bec2da16fbaa47db" +dependencies = [ + "fixedbitset", + "indexmap 2.2.6", +] + +[[package]] +name = "pharos" +version = "0.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e9567389417feee6ce15dd6527a8a1ecac205ef62c2932bcf3d9f6fc5b78b414" +dependencies = [ + "futures", + "rustc_version 0.4.0", +] + +[[package]] +name = "pin-project" +version = "1.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6bf43b791c5b9e34c3d182969b4abb522f9343702850a2e57f460d00d09b4b3" +dependencies = [ + "pin-project-internal", +] + +[[package]] +name = "pin-project-internal" +version = "1.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2f38a4412a78282e09a2cf38d195ea5420d15ba0602cb375210efbc877243965" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "pin-project-lite" +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 = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" + +[[package]] +name = "pkcs8" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f950b2377845cebe5cf8b5165cb3cc1a5e0fa5cfa3e1f7f55707d8fd82e0a7b7" +dependencies = [ + "der", + "spki", +] + +[[package]] +name = "pkg-config" +version = "0.3.30" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d231b230927b5e4ad203db57bbcbee2802f6bce620b1e4a9024a07d94e2907ec" + +[[package]] +name = "poly1305" +version = "0.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8159bd90725d2df49889a078b54f4f79e87f1f8a8444194cdca81d38f5393abf" +dependencies = [ + "cpufeatures", + "opaque-debug", + "universal-hash", +] + +[[package]] +name = "polyval" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d1fe60d06143b2430aa532c94cfe9e29783047f06c0d7fd359a9a51b729fa25" +dependencies = [ + "cfg-if", + "cpufeatures", + "opaque-debug", + "universal-hash", +] + +[[package]] +name = "portable-atomic" +version = "1.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da544ee218f0d287a911e9c99a39a8c9bc8fcad3cb8db5959940044ecfc67265" + +[[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.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b40af805b3121feab8a3c29f04d8ad262fa8e0561883e7653e024ae4479e6de" + +[[package]] +name = "prettyplease" +version = "0.2.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f12335488a2f3b0a83b14edad48dca9879ce89b2edd10e80237e4e852dd645e" +dependencies = [ + "proc-macro2", + "syn 2.0.65", +] + +[[package]] +name = "primitive-types" +version = "0.12.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b34d9fd68ae0b74a41b21c03c2f62847aa0ffea044eee893b4c140b37e244e2" +dependencies = [ + "fixed-hash", + "impl-codec", + "impl-rlp", + "impl-serde", + "scale-info", + "uint", +] + +[[package]] +name = "proc-macro-crate" +version = "3.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6d37c51ca738a55da99dc0c4a34860fd675453b8b36209178c2249bb13651284" +dependencies = [ + "toml_edit 0.21.1", +] + +[[package]] +name = "proc-macro-error" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "da25490ff9892aab3fcf7c36f08cfb902dd3e71ca0f9f9517bea02a73a5ce38c" +dependencies = [ + "proc-macro-error-attr", + "proc-macro2", + "quote", + "version_check", +] + +[[package]] +name = "proc-macro-error-attr" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1be40180e52ecc98ad80b184934baf3d0d29f979574e439af5a55274b35f869" +dependencies = [ + "proc-macro2", + "quote", + "version_check", +] + +[[package]] +name = "proc-macro2" +version = "1.0.83" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b33eb56c327dec362a9e55b3ad14f9d2f0904fb5a5b03b513ab5465399e9f43" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "proptest" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31b476131c3c86cb68032fdc5cb6d5a1045e3e42d96b69fa599fd77701e1f5bf" +dependencies = [ + "bitflags 2.5.0", + "lazy_static", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "rand 0.8.5", + "rand_chacha 0.3.1", + "rand_xorshift", + "regex-syntax", + "unarray", +] + +[[package]] +name = "prost" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "deb1435c188b76130da55f17a466d252ff7b1418b2ad3e037d127b94e3411f29" +dependencies = [ + "bytes", + "prost-derive 0.12.6", +] + +[[package]] +name = "prost" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3b2ecbe40f08db5c006b5764a2645f7f3f141ce756412ac9e1dd6087e6d32995" +dependencies = [ + "bytes", + "prost-derive 0.13.2", +] + +[[package]] +name = "prost-build" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22505a5c94da8e3b7c2996394d1c933236c4d743e81a410bcca4e6989fc066a4" +dependencies = [ + "bytes", + "heck 0.5.0", + "itertools 0.12.1", + "log", + "multimap", + "once_cell", + "petgraph", + "prettyplease", + "prost 0.12.6", + "prost-types 0.12.6", + "regex", + "syn 2.0.65", + "tempfile", +] + +[[package]] +name = "prost-derive" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "81bddcdb20abf9501610992b6759a4c888aef7d1a7247ef75e2404275ac24af1" +dependencies = [ + "anyhow", + "itertools 0.12.1", + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "prost-derive" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "acf0c195eebb4af52c752bec4f52f645da98b6e92077a04110c7f349477ae5ac" +dependencies = [ + "anyhow", + "itertools 0.12.1", + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "prost-types" +version = "0.12.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9091c90b0a32608e984ff2fa4091273cbdd755d54935c51d520887f4a1dbd5b0" +dependencies = [ + "prost 0.12.6", +] + +[[package]] +name = "prost-types" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "60caa6738c7369b940c3d49246a8d1749323674c65cb13010134f5c9bad5b519" +dependencies = [ + "prost 0.13.2", +] + +[[package]] +name = "quote" +version = "1.0.36" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0fa76aaf39101c457836aec0ce2316dbdc3ab723cdda1c6bd4e6ad4208acaca7" +dependencies = [ + "proc-macro2", +] + +[[package]] +name = "radium" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc33ff2d4973d518d823d61aa239014831e521c75da58e3df4840d3f47749d09" + +[[package]] +name = "rand" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03" +dependencies = [ + "getrandom 0.1.16", + "libc", + "rand_chacha 0.2.2", + "rand_core 0.5.1", + "rand_hc", +] + +[[package]] +name = "rand" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" +dependencies = [ + "libc", + "rand_chacha 0.3.1", + "rand_core 0.6.4", +] + +[[package]] +name = "rand_chacha" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402" +dependencies = [ + "ppv-lite86", + "rand_core 0.5.1", +] + +[[package]] +name = "rand_chacha" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" +dependencies = [ + "ppv-lite86", + "rand_core 0.6.4", +] + +[[package]] +name = "rand_core" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19" +dependencies = [ + "getrandom 0.1.16", +] + +[[package]] +name = "rand_core" +version = "0.6.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" +dependencies = [ + "getrandom 0.2.15", +] + +[[package]] +name = "rand_hc" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c" +dependencies = [ + "rand_core 0.5.1", +] + +[[package]] +name = "rand_xorshift" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d25bf25ec5ae4a3f1b92f929810509a2f53d7dca2f50b794ff57e3face536c8f" +dependencies = [ + "rand_core 0.6.4", +] + +[[package]] +name = "rayon" +version = "1.5.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd99e5772ead8baa5215278c9b15bf92087709e9c1b2d1f97cdb5a183c933a7d" +dependencies = [ + "autocfg", + "crossbeam-deque", + "either", + "rayon-core", +] + +[[package]] +name = "rayon-core" +version = "1.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1465873a3dfdaa8ae7cb14b4383657caab0b3e8a0aa9ae8e04b044854c8dfce2" +dependencies = [ + "crossbeam-deque", + "crossbeam-utils", + "wasm_sync", +] + +[[package]] +name = "reddsa" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78a5191930e84973293aa5f532b513404460cd2216c1cfb76d08748c15b40b02" +dependencies = [ + "blake2b_simd", + "byteorder", + "group", + "hex", + "jubjub 0.10.0 (registry+https://github.com/rust-lang/crates.io-index)", + "pasta_curves", + "rand_core 0.6.4", + "serde", + "thiserror", + "zeroize", +] + +[[package]] +name = "redjubjub" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a60db2c3bc9c6fd1e8631fee75abc008841d27144be744951d6b9b75f9b569c" +dependencies = [ + "rand_core 0.6.4", + "reddsa", + "serde", + "thiserror", + "zeroize", +] + +[[package]] +name = "redox_syscall" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "469052894dcb553421e483e4209ee581a45100d31b4018de03e5a7ad86374a7e" +dependencies = [ + "bitflags 2.5.0", +] + +[[package]] +name = "redox_users" +version = "0.4.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd283d9651eeda4b2a83a43c1c91b266c40fd76ecd39a50a8c630ae69dc72891" +dependencies = [ + "getrandom 0.2.15", + "libredox", + "thiserror", +] + +[[package]] +name = "regex" +version = "1.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c117dbdfde9c8308975b6a18d71f3f385c89461f7b3fb054288ecf2a2058ba4c" +dependencies = [ + "aho-corasick", + "memchr", + "regex-automata", + "regex-syntax", +] + +[[package]] +name = "regex-automata" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "86b83b8b9847f9bf95ef68afb0b8e6cdb80f498442f5179a29fad448fcc1eaea" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax", +] + +[[package]] +name = "regex-syntax" +version = "0.8.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "adad44e29e4c806119491a7f06f03de4d1af22c3a680dd47f1e6e179439d1f56" + +[[package]] +name = "reqwest" +version = "0.11.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd67538700a17451e7cba03ac727fb961abb7607553461627b97de0b89cf4a62" +dependencies = [ + "base64 0.21.7", + "bytes", + "encoding_rs", + "futures-core", + "futures-util", + "h2", + "http 0.2.12", + "http-body", + "hyper", + "hyper-tls", + "ipnet", + "js-sys", + "log", + "mime", + "native-tls", + "once_cell", + "percent-encoding", + "pin-project-lite", + "rustls-pemfile", + "serde", + "serde_json", + "serde_urlencoded", + "sync_wrapper", + "system-configuration", + "tokio", + "tokio-native-tls", + "tower-service", + "url", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", + "winreg", +] + +[[package]] +name = "rexie" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d34618a20b14cebebc63f8b1c5228116e7379765e8f1b7a9f66b5920babd8595" +dependencies = [ + "js-sys", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "thiserror", + "tokio", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + +[[package]] +name = "rfc6979" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8dd2a808d456c4a54e300a23e9f5a67e122c3024119acbfd73e3bf664491cb2" +dependencies = [ + "hmac 0.12.1", + "subtle", +] + +[[package]] +name = "ring" +version = "0.16.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3053cf52e236a3ed746dfc745aa9cacf1b791d846bdaf412f60a8d7d6e17c8fc" +dependencies = [ + "cc", + "libc", + "once_cell", + "spin 0.5.2", + "untrusted", + "web-sys", + "winapi", +] + +[[package]] +name = "ripemd" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bd124222d17ad93a644ed9d011a40f4fb64aa54275c08cc216524a9ea82fb09f" +dependencies = [ + "digest 0.10.7", +] + +[[package]] +name = "rlp" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bb919243f34364b6bd2fc10ef797edbfa75f33c252e7998527479c6d6b47e1ec" +dependencies = [ + "bytes", + "rlp-derive", + "rustc-hex", +] + +[[package]] +name = "rlp-derive" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e33d7b2abe0c340d8797fe2907d3f20d3b5ea5908683618bfe80df7f621f672a" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "rust_decimal" +version = "1.35.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1790d1c4c0ca81211399e0e0af16333276f375209e71a37b67698a373db5b47a" +dependencies = [ + "arrayvec", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", +] + +[[package]] +name = "rustc-demangle" +version = "0.1.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "719b953e2095829ee67db738b3bfa9fa368c94900df327b3f07fe6e794d2fe1f" + +[[package]] +name = "rustc-hash" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2" + +[[package]] +name = "rustc-hex" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3e75f6a532d0fd9f7f13144f392b6ad56a32696bfcd9c78f797f16bbb6f072d6" + +[[package]] +name = "rustc_version" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f0dfe2087c51c460008730de8b57e6a320782fbfb312e1f4d520e6c6fae155ee" +dependencies = [ + "semver 0.11.0", +] + +[[package]] +name = "rustc_version" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa0f585226d2e68097d4f95d113b15b83a82e819ab25717ec0590d9584ef366" +dependencies = [ + "semver 1.0.23", +] + +[[package]] +name = "rustix" +version = "0.38.34" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70dc5ec042f7a43c4a73241207cecc9873a06d45debb38b329f8541d85c2730f" +dependencies = [ + "bitflags 2.5.0", + "errno", + "libc", + "linux-raw-sys", + "windows-sys 0.52.0", +] + +[[package]] +name = "rustls-pemfile" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1c74cae0a4cf6ccbbf5f359f08efdf8ee7e1dc532573bf0db71968cb56b1448c" +dependencies = [ + "base64 0.21.7", +] + +[[package]] +name = "rustversion" +version = "1.0.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "955d28af4278de8121b7ebeb796b6a45735dc01436d898801014aced2773a3d6" + +[[package]] +name = "ryu" +version = "1.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f" + +[[package]] +name = "salsa20" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97a22f5af31f73a954c10289c93e8a50cc23d971e80ee446f1f6f7137a088213" +dependencies = [ + "cipher", +] + +[[package]] +name = "same-file" +version = "1.0.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93fc1dc3aaa9bfed95e02e6eadabb4baf7e3078b0bd1b4d7b6b0b68378900502" +dependencies = [ + "winapi-util", +] + +[[package]] +name = "scale-info" +version = "2.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eca070c12893629e2cc820a9761bedf6ce1dcddc9852984d1dc734b8bd9bd024" +dependencies = [ + "cfg-if", + "derive_more", + "parity-scale-codec", + "scale-info-derive", +] + +[[package]] +name = "scale-info-derive" +version = "2.11.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2d35494501194174bda522a32605929eefc9ecf7e0a326c26db1fdd85881eb62" +dependencies = [ + "proc-macro-crate", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "schannel" +version = "0.1.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fbc91545643bcf3a0bbb6569265615222618bdf33ce4ffbbd13c4bbd4c093534" +dependencies = [ + "windows-sys 0.52.0", +] + +[[package]] +name = "schemars" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09c024468a378b7e36765cd36702b7a90cc3cba11654f6685c8f233408e89e92" +dependencies = [ + "dyn-clone", + "schemars_derive", + "serde", + "serde_json", +] + +[[package]] +name = "schemars_derive" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1eee588578aff73f856ab961cd2f79e36bc45d7ded33a7562adba4667aecc0e" +dependencies = [ + "proc-macro2", + "quote", + "serde_derive_internals", + "syn 2.0.65", +] + +[[package]] +name = "scoped-tls" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e1cf6437eb19a8f4a6cc0f7dca544973b0b78843adbfeb3683d1a94a0024a294" + +[[package]] +name = "scopeguard" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49" + +[[package]] +name = "scrypt" +version = "0.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f9e24d2b632954ded8ab2ef9fea0a0c769ea56ea98bddbafbad22caeeadf45d" +dependencies = [ + "hmac 0.12.1", + "pbkdf2 0.11.0", + "salsa20", + "sha2 0.10.8", +] + +[[package]] +name = "sec1" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3e97a565f76233a6003f9f5c54be1d9c5bdfa3eccfb189469f11ec4901c47dc" +dependencies = [ + "base16ct", + "der", + "generic-array", + "pkcs8", + "serdect", + "subtle", + "zeroize", +] + +[[package]] +name = "security-framework" +version = "2.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c627723fd09706bacdb5cf41499e95098555af3c3c29d014dc3c458ef6be11c0" +dependencies = [ + "bitflags 2.5.0", + "core-foundation", + "core-foundation-sys", + "libc", + "security-framework-sys", +] + +[[package]] +name = "security-framework-sys" +version = "2.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "317936bbbd05227752583946b9e66d7ce3b489f84e11a94a510b4437fef407d7" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "semver" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f301af10236f6df4160f7c3f04eec6dbc70ace82d23326abad5edee88801c6b6" +dependencies = [ + "semver-parser", +] + +[[package]] +name = "semver" +version = "1.0.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61697e0a1c7e512e84a621326239844a24d8207b4669b41bc18b32ea5cbf988b" +dependencies = [ + "serde", +] + +[[package]] +name = "semver-parser" +version = "0.10.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "00b0bef5b7f9e0df16536d3961cfb6e84331c065b4066afb39768d0e319411f7" +dependencies = [ + "pest", +] + +[[package]] +name = "send_wrapper" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f638d531eccd6e23b980caf34876660d38e265409d8e99b397ab71eb3612fad0" + +[[package]] +name = "send_wrapper" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cd0b0ec5f1c1ca621c432a25813d8d60c88abe6d3e08a3eb9cf37d97a0fe3d73" + +[[package]] +name = "serde" +version = "1.0.202" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "226b61a0d411b2ba5ff6d7f73a476ac4f8bb900373459cd00fab8512828ba395" +dependencies = [ + "serde_derive", +] + +[[package]] +name = "serde-json-wasm" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f05da0d153dd4595bdffd5099dc0e9ce425b205ee648eb93437ff7302af8c9a5" +dependencies = [ + "serde", +] + +[[package]] +name = "serde_bytes" +version = "0.11.14" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b8497c313fd43ab992087548117643f6fcd935cbf36f176ffda0aacf9591734" +dependencies = [ + "serde", +] + +[[package]] +name = "serde_derive" +version = "1.0.202" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6048858004bcff69094cd972ed40a32500f153bd3be9f716b2eed2e8217c4838" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "serde_derive_internals" +version = "0.29.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "18d26a20a969b9e3fdf2fc2d9f21eda6c40e2de84c9408bb5d3b05d499aae711" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "serde_json" +version = "1.0.117" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "455182ea6142b14f93f4bc5320a2b31c1f266b66a4a5c858b013302a5d8cbfc3" +dependencies = [ + "itoa", + "ryu", + "serde", +] + +[[package]] +name = "serde_repr" +version = "0.1.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c64451ba24fc7a6a2d60fc75dd9c83c90903b19028d4eff35e88fc1e86564e9" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "serde_spanned" +version = "0.6.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "79e674e01f999af37c49f70a6ede167a8a60b2503e56c5599532a65baa5969a0" +dependencies = [ + "serde", +] + +[[package]] +name = "serde_urlencoded" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd" +dependencies = [ + "form_urlencoded", + "itoa", + "ryu", + "serde", +] + +[[package]] +name = "serdect" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a84f14a19e9a014bb9f4512488d9829a68e04ecabffb0f9904cd1ace94598177" +dependencies = [ + "base16ct", + "serde", +] + +[[package]] +name = "sha2" +version = "0.9.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800" +dependencies = [ + "block-buffer 0.9.0", + "cfg-if", + "cpufeatures", + "digest 0.9.0", + "opaque-debug", +] + +[[package]] +name = "sha2" +version = "0.10.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest 0.10.7", +] + +[[package]] +name = "sha3" +version = "0.10.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75872d278a8f37ef87fa0ddbda7802605cb18344497949862c0d4dcb291eba60" +dependencies = [ + "digest 0.10.7", + "keccak", +] + +[[package]] +name = "signal-hook-registry" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a9e9e0b4211b72e7b8b6e85c807d36c212bdb33ea8587f7569562a84df5465b1" +dependencies = [ + "libc", +] + +[[package]] +name = "signature" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77549399552de45a898a580c1b41d445bf730df867cc44e6c0233bbc4b8329de" +dependencies = [ + "digest 0.10.7", + "rand_core 0.6.4", +] + +[[package]] +name = "simple_asn1" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "adc4e5204eb1910f40f9cfa375f6f05b68c3abac4b6fd879c8ff5e7ae8a0a085" +dependencies = [ + "num-bigint", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "thiserror", + "time", +] + +[[package]] +name = "slab" +version = "0.4.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f92a496fb766b417c996b9c5e57daf2f7ad3b0bebe1ccfca4856390e3d3bb67" +dependencies = [ + "autocfg", +] + +[[package]] +name = "slip10_ed25519" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4be0ff28bf14f9610a342169084e87a4f435ad798ec528dc7579a3678fa9dc9a" +dependencies = [ + "hmac-sha512", +] + +[[package]] +name = "smallvec" +version = "1.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" + +[[package]] +name = "smooth-operator" +version = "0.7.0" +source = "git+https://github.com/heliaxdev/smooth-operator?tag=v0.7.0#0e182707f5e5bb9c6e0efa2d235dc9efd715d0a1" +dependencies = [ + "smooth-operator-impl", +] + +[[package]] +name = "smooth-operator-impl" +version = "0.7.0" +source = "git+https://github.com/heliaxdev/smooth-operator?tag=v0.7.0#0e182707f5e5bb9c6e0efa2d235dc9efd715d0a1" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "socket2" +version = "0.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce305eb0b4296696835b71df73eb912e0f1ffd2556a501fcede6e0c50349191c" +dependencies = [ + "libc", + "windows-sys 0.52.0", +] + +[[package]] +name = "sparse-merkle-tree" +version = "0.3.1-pre" +source = "git+https://github.com/heliaxdev/sparse-merkle-tree?rev=a93c55ccd47840ee0967eee237e47d9245478594#a93c55ccd47840ee0967eee237e47d9245478594" +dependencies = [ + "borsh", + "cfg-if", + "ics23", + "itertools 0.12.1", + "sha2 0.9.9", +] + +[[package]] +name = "spin" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e63cff320ae2c57904679ba7cb63280a3dc4613885beafb148ee7bf9aa9042d" + +[[package]] +name = "spin" +version = "0.9.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67" +dependencies = [ + "lock_api", +] + +[[package]] +name = "spki" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d91ed6c858b01f942cd56b37a94b3e0a1798290327d1236e4d9cf4eaca44d29d" +dependencies = [ + "base64ct", + "der", +] + +[[package]] +name = "static_assertions" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" + +[[package]] +name = "strum" +version = "0.26.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d8cec3501a5194c432b2b7976db6b7d10ec95c253208b45f83f7136aa985e29" +dependencies = [ + "strum_macros", +] + +[[package]] +name = "strum_macros" +version = "0.26.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6cf59daf282c0a494ba14fd21610a0325f9f90ec9d1231dea26bcb1d696c946" +dependencies = [ + "heck 0.4.1", + "proc-macro2", + "quote", + "rustversion", + "syn 2.0.65", +] + +[[package]] +name = "subtle" +version = "2.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6bdef32e8150c2a081110b42772ffe7d7c9032b606bc226c8260fd97e0976601" + +[[package]] +name = "subtle-encoding" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7dcb1ed7b8330c5eed5441052651dd7a12c75e2ed88f2ec024ae1fa3a5e59945" +dependencies = [ + "zeroize", +] + +[[package]] +name = "subtle-ng" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "734676eb262c623cec13c3155096e08d1f8f29adce39ba17948b18dad1e54142" + +[[package]] +name = "syn" +version = "1.0.109" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "syn" +version = "2.0.65" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d2863d96a84c6439701d7a38f9de935ec562c8832cc55d1dde0f513b52fad106" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "syn_derive" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1329189c02ff984e9736652b1631330da25eaa6bc639089ed4915d25446cbe7b" +dependencies = [ + "proc-macro-error", + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "sync_wrapper" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160" + +[[package]] +name = "system-configuration" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba3a3adc5c275d719af8cb4272ea1c4a6d668a777f37e115f6d11ddbc1c8e0e7" +dependencies = [ + "bitflags 1.3.2", + "core-foundation", + "system-configuration-sys", +] + +[[package]] +name = "system-configuration-sys" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75fb188eb626b924683e3b95e3a48e63551fcfb51949de2f06a9d91dbee93c9" +dependencies = [ + "core-foundation-sys", + "libc", +] + +[[package]] +name = "tap" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "55937e1799185b12863d447f42597ed69d9928686b8d88a1df17376a097d8369" + +[[package]] +name = "tempfile" +version = "3.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85b77fafb263dd9d05cbeac119526425676db3784113aa9295c88498cbf8bff1" +dependencies = [ + "cfg-if", + "fastrand", + "rustix", + "windows-sys 0.52.0", +] + +[[package]] +name = "tendermint" +version = "0.34.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15ab8f0a25d0d2ad49ac615da054d6a76aa6603ff95f7d18bafdd34450a1a04b" +dependencies = [ + "bytes", + "digest 0.10.7", + "ed25519", + "ed25519-consensus", + "flex-error", + "futures", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "once_cell", + "prost 0.12.6", + "prost-types 0.12.6", + "serde", + "serde_bytes", + "serde_json", + "serde_repr", + "sha2 0.10.8", + "signature", + "subtle", + "subtle-encoding", + "tendermint-proto 0.34.1", + "time", + "zeroize", +] + +[[package]] +name = "tendermint" +version = "0.38.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "505d9d6ffeb83b1de47c307c6e0d2dff56c6256989299010ad03cd80a8491e97" +dependencies = [ + "bytes", + "digest 0.10.7", + "ed25519", + "ed25519-consensus", + "flex-error", + "futures", + "k256", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "once_cell", + "prost 0.13.2", + "prost-types 0.13.2", + "ripemd", + "serde", + "serde_bytes", + "serde_json", + "serde_repr", + "sha2 0.10.8", + "signature", + "subtle", + "subtle-encoding", + "tendermint-proto 0.38.1", + "time", + "zeroize", +] + +[[package]] +name = "tendermint-config" +version = "0.34.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e1a02da769166e2052cd537b1a97c78017632c2d9e19266367b27e73910434fc" +dependencies = [ + "flex-error", + "serde", + "serde_json", + "tendermint 0.34.1", + "toml 0.5.11", + "url", +] + +[[package]] +name = "tendermint-config" +version = "0.38.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9de111ea653b2adaef627ac2452b463c77aa615c256eaaddf279ec5a1cf9775f" +dependencies = [ + "flex-error", + "serde", + "serde_json", + "tendermint 0.38.1", + "toml 0.8.13", + "url", +] + +[[package]] +name = "tendermint-light-client-verifier" +version = "0.38.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a2674adbf0dc51aa0c8eaf8462c7d6692ec79502713e50ed5432a442002be90" +dependencies = [ + "derive_more", + "flex-error", + "serde", + "tendermint 0.38.1", + "time", +] + +[[package]] +name = "tendermint-proto" +version = "0.34.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b797dd3d2beaaee91d2f065e7bdf239dc8d80bba4a183a288bc1279dd5a69a1e" +dependencies = [ + "bytes", + "flex-error", + "num-derive 0.3.3", + "num-traits 0.2.19 (registry+https://github.com/rust-lang/crates.io-index)", + "prost 0.12.6", + "prost-types 0.12.6", + "serde", + "serde_bytes", + "subtle-encoding", + "time", +] + +[[package]] +name = "tendermint-proto" +version = "0.38.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ed14abe3b0502a3afe21ca74ca5cdd6c7e8d326d982c26f98a394445eb31d6e" +dependencies = [ + "bytes", + "flex-error", + "prost 0.13.2", + "prost-types 0.13.2", + "serde", + "serde_bytes", + "subtle-encoding", + "time", +] + +[[package]] +name = "tendermint-rpc" +version = "0.38.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02f96a2b8a0d3d0b59e4024b1a6bdc1589efc6af4709d08a480a20cc4ba90f63" +dependencies = [ + "async-trait", + "bytes", + "flex-error", + "getrandom 0.2.15", + "peg", + "pin-project", + "rand 0.8.5", + "semver 1.0.23", + "serde", + "serde_bytes", + "serde_json", + "subtle", + "subtle-encoding", + "tendermint 0.38.1", + "tendermint-config 0.38.1", + "tendermint-proto 0.38.1", + "thiserror", + "time", + "url", + "uuid 1.8.0", + "walkdir", +] + +[[package]] +name = "terminal_size" +version = "0.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "21bebf2b7c9e0a515f6e0f8c51dc0f8e4696391e6f1ff30379559f8365fb0df7" +dependencies = [ + "rustix", + "windows-sys 0.48.0", +] + +[[package]] +name = "thiserror" +version = "1.0.61" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c546c80d6be4bc6a00c0f01730c08df82eaa7a7a61f11d656526506112cc1709" +dependencies = [ + "thiserror-impl", +] + +[[package]] +name = "thiserror-impl" +version = "1.0.61" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "46c3384250002a6d5af4d114f2845d37b57521033f30d5c3f46c4d70e1197533" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "time" +version = "0.3.36" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5dfd88e563464686c916c7e46e623e520ddc6d79fa6641390f2e3fa86e83e885" +dependencies = [ + "deranged", + "itoa", + "num-conv", + "powerfmt", + "serde", + "time-core", + "time-macros", +] + +[[package]] +name = "time-core" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef927ca75afb808a4d64dd374f00a2adf8d0fcff8e7b184af886c3c87ec4a3f3" + +[[package]] +name = "time-macros" +version = "0.2.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f252a68540fde3a3877aeea552b832b40ab9a69e318efd078774a01ddee1ccf" +dependencies = [ + "num-conv", + "time-core", +] + +[[package]] +name = "tiny-bip39" +version = "0.8.2" +source = "git+https://github.com/anoma/tiny-bip39.git?rev=bf0f6d8713589b83af7a917366ec31f5275c0e57#bf0f6d8713589b83af7a917366ec31f5275c0e57" +dependencies = [ + "anyhow", + "hmac 0.8.1", + "once_cell", + "pbkdf2 0.4.0", + "rand 0.7.3", + "rustc-hash", + "sha2 0.9.9", + "thiserror", + "unicode-normalization", + "wasm-bindgen", + "zeroize", +] + +[[package]] +name = "tiny-bip39" +version = "1.0.0" +source = "git+https://github.com/anoma/tiny-bip39?rev=743d537349c8deab14409ce726b868dcde90fd8e#743d537349c8deab14409ce726b868dcde90fd8e" +dependencies = [ + "anyhow", + "hmac 0.12.1", + "once_cell", + "pbkdf2 0.11.0", + "rand 0.8.5", + "rustc-hash", + "sha2 0.10.8", + "thiserror", + "unicode-normalization", + "wasm-bindgen", + "zeroize", +] + +[[package]] +name = "tiny-hderive" +version = "0.3.0" +source = "git+https://github.com/heliaxdev/tiny-hderive.git?rev=173ae03abed0cd25d88a5a13efac00af96b75b87#173ae03abed0cd25d88a5a13efac00af96b75b87" +dependencies = [ + "base58", + "hmac 0.12.1", + "k256", + "memzero", + "sha2 0.10.8", +] + +[[package]] +name = "tiny-keccak" +version = "2.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2c9d3793400a45f954c52e73d068316d76b6f4e36977e3fcebb13a2721e80237" +dependencies = [ + "crunchy", +] + +[[package]] +name = "tinyvec" +version = "1.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50" +dependencies = [ + "tinyvec_macros", +] + +[[package]] +name = "tinyvec_macros" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" + +[[package]] +name = "tokio" +version = "1.37.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1adbebffeca75fcfd058afa480fb6c0b81e165a0323f9c9d39c9697e37c46787" +dependencies = [ + "backtrace", + "bytes", + "libc", + "mio", + "num_cpus", + "parking_lot", + "pin-project-lite", + "signal-hook-registry", + "socket2", + "tokio-macros", + "windows-sys 0.48.0", +] + +[[package]] +name = "tokio-macros" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b8a1e28f2deaa14e508979454cb3a223b10b938b45af148bc0986de36f1923b" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "tokio-native-tls" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbae76ab933c85776efabc971569dd6119c580d8f5d448769dec1764bf796ef2" +dependencies = [ + "native-tls", + "tokio", +] + +[[package]] +name = "tokio-util" +version = "0.7.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9cf6b47b3771c49ac75ad09a6162f53ad4b8088b76ac60e8ec1455b31a189fe1" +dependencies = [ + "bytes", + "futures-core", + "futures-sink", + "pin-project-lite", + "tokio", +] + +[[package]] +name = "toml" +version = "0.5.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f4f7f0dd8d50a853a531c426359045b1998f04219d88799810762cd4ad314234" +dependencies = [ + "serde", +] + +[[package]] +name = "toml" +version = "0.8.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a4e43f8cc456c9704c851ae29c67e17ef65d2c30017c17a9765b89c382dc8bba" +dependencies = [ + "serde", + "serde_spanned", + "toml_datetime", + "toml_edit 0.22.13", +] + +[[package]] +name = "toml_datetime" +version = "0.6.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4badfd56924ae69bcc9039335b2e017639ce3f9b001c393c1b2d1ef846ce2cbf" +dependencies = [ + "serde", +] + +[[package]] +name = "toml_edit" +version = "0.21.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6a8534fd7f78b5405e860340ad6575217ce99f38d4d5c8f2442cb5ecb50090e1" +dependencies = [ + "indexmap 2.2.6", + "toml_datetime", + "winnow 0.5.40", +] + +[[package]] +name = "toml_edit" +version = "0.22.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c127785850e8c20836d49732ae6abfa47616e60bf9d9f57c43c250361a9db96c" +dependencies = [ + "indexmap 2.2.6", + "serde", + "serde_spanned", + "toml_datetime", + "winnow 0.6.8", +] + +[[package]] +name = "tonic-build" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be4ef6dd70a610078cb4e338a0f79d06bc759ff1b22d2120c2ff02ae264ba9c2" +dependencies = [ + "prettyplease", + "proc-macro2", + "prost-build", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "tower-service" +version = "0.3.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6bc1c9ce2b5135ac7f93c72918fc37feb872bdc6a5533a8b85eb4b86bfdae52" + +[[package]] +name = "tracing" +version = "0.1.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3523ab5a71916ccf420eebdf5521fcef02141234bbc0b8a49f2fdc4544364ef" +dependencies = [ + "pin-project-lite", + "tracing-attributes", + "tracing-core", +] + +[[package]] +name = "tracing-attributes" +version = "0.1.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "tracing-core" +version = "0.1.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c06d3da6113f116aaee68e4d601191614c9053067f9ab7f6edbcb161237daa54" +dependencies = [ + "once_cell", +] + +[[package]] +name = "tracing-futures" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97d095ae15e245a057c8e8451bab9b3ee1e1f68e9ba2b4fbc18d0ac5237835f2" +dependencies = [ + "pin-project", + "tracing", +] + +[[package]] +name = "try-lock" +version = "0.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" + +[[package]] +name = "typed-builder" +version = "0.19.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a06fbd5b8de54c5f7c91f6fe4cebb949be2125d7758e630bb58b1d831dbce600" +dependencies = [ + "typed-builder-macro", +] + +[[package]] +name = "typed-builder-macro" +version = "0.19.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f9534daa9fd3ed0bd911d462a37f172228077e7abf18c18a5f67199d959205f8" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "typenum" +version = "1.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" + +[[package]] +name = "typewit" +version = "1.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c6fb9ae6a3cafaf0a5d14c2302ca525f9ae8e07a0f0e6949de88d882c37a6e24" +dependencies = [ + "typewit_proc_macros", +] + +[[package]] +name = "typewit_proc_macros" +version = "1.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e36a83ea2b3c704935a01b4642946aadd445cea40b10935e3f8bd8052b8193d6" + +[[package]] +name = "ucd-trie" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed646292ffc8188ef8ea4d1e0e0150fb15a5c2e12ad9b8fc191ae7a8a7f3c4b9" + +[[package]] +name = "uint" +version = "0.9.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76f64bba2c53b04fcab63c01a7d7427eadc821e3bc48c34dc9ba29c501164b52" +dependencies = [ + "byteorder", + "crunchy", + "hex", + "static_assertions", +] + +[[package]] +name = "unarray" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eaea85b334db583fe3274d12b4cd1880032beab409c0d774be044d4480ab9a94" + +[[package]] +name = "unicode-bidi" +version = "0.3.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08f95100a766bf4f8f28f90d77e0a5461bbdb219042e7679bebe79004fed8d75" + +[[package]] +name = "unicode-ident" +version = "1.0.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b" + +[[package]] +name = "unicode-normalization" +version = "0.1.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a56d1686db2308d901306f92a263857ef59ea39678a5458e7cb17f01415101f5" +dependencies = [ + "tinyvec", +] + +[[package]] +name = "unicode-xid" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f962df74c8c05a667b5ee8bcf162993134c104e96440b663c8daa176dc772d8c" + +[[package]] +name = "universal-hash" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fc1de2c688dc15305988b563c3854064043356019f97a4b46276fe734c4f07ea" +dependencies = [ + "crypto-common", + "subtle", +] + +[[package]] +name = "untrusted" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a156c684c91ea7d62626509bce3cb4e1d9ed5c4d978f7b4352658f96a4c26b4a" + +[[package]] +name = "url" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "31e6302e3bb753d46e83516cae55ae196fc0c309407cf11ab35cc51a4c2a4633" +dependencies = [ + "form_urlencoded", + "idna", + "percent-encoding", +] + +[[package]] +name = "uuid" +version = "0.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bc5cf98d8186244414c848017f0e2676b3fcb46807f6668a97dfe67359a3c4b7" +dependencies = [ + "getrandom 0.2.15", + "serde", +] + +[[package]] +name = "uuid" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a183cf7feeba97b4dd1c0d46788634f6221d87fa961b305bed08c851829efcc0" + +[[package]] +name = "vcpkg" +version = "0.2.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" + +[[package]] +name = "version_check" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f" + +[[package]] +name = "walkdir" +version = "2.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b" +dependencies = [ + "same-file", + "winapi-util", +] + +[[package]] +name = "want" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e" +dependencies = [ + "try-lock", +] + +[[package]] +name = "wasi" +version = "0.9.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519" + +[[package]] +name = "wasi" +version = "0.11.0+wasi-snapshot-preview1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" + +[[package]] +name = "wasm-bindgen" +version = "0.2.92" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4be2531df63900aeb2bca0daaaddec08491ee64ceecbee5076636a3b026795a8" +dependencies = [ + "cfg-if", + "wasm-bindgen-macro", +] + +[[package]] +name = "wasm-bindgen-backend" +version = "0.2.92" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "614d787b966d3989fa7bb98a654e369c762374fd3213d212cfc0251257e747da" +dependencies = [ + "bumpalo", + "log", + "once_cell", + "proc-macro2", + "quote", + "syn 2.0.65", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-futures" +version = "0.4.42" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76bc14366121efc8dbb487ab05bcc9d346b3b5ec0eaa76e46594cabbe51762c0" +dependencies = [ + "cfg-if", + "js-sys", + "wasm-bindgen", + "web-sys", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.92" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1f8823de937b71b9460c0c34e25f3da88250760bec0ebac694b49997550d726" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.92" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e94f17b526d0a461a191c78ea52bbce64071ed5c04c9ffe424dcb38f74171bb7" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", + "wasm-bindgen-backend", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-rayon" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a9beda8dfdfaf2e0ec0b47e130a0794d18188fba4da8a2155dcc3bbeb7e0d454" +dependencies = [ + "crossbeam-channel", + "js-sys", + "rayon-core", + "wasm-bindgen", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.92" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af190c94f2773fdb3729c55b007a722abb5384da03bc0986df4c289bf5567e96" + +[[package]] +name = "wasm-bindgen-test" +version = "0.3.42" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9bf62a58e0780af3e852044583deee40983e5886da43a271dd772379987667b" +dependencies = [ + "console_error_panic_hook", + "js-sys", + "scoped-tls", + "wasm-bindgen", + "wasm-bindgen-futures", + "wasm-bindgen-test-macro", +] + +[[package]] +name = "wasm-bindgen-test-macro" +version = "0.3.42" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b7f89739351a2e03cb94beb799d47fb2cac01759b40ec441f7de39b00cbf7ef0" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] + +[[package]] +name = "wasm_sync" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cff360cade7fec41ff0e9d2cda57fe58258c5f16def0e21302394659e6bbb0ea" +dependencies = [ + "js-sys", + "wasm-bindgen", + "web-sys", +] + +[[package]] +name = "wasmparser" +version = "0.107.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "29e3ac9b780c7dda0cac7a52a5d6d2d6707cc6e3451c9db209b6c758f40d7acb" +dependencies = [ + "indexmap 1.9.3", + "semver 1.0.23", +] + +[[package]] +name = "wasmtimer" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f656cd8858a5164932d8a90f936700860976ec21eb00e0fe2aa8cab13f6b4cf" +dependencies = [ + "futures", + "js-sys", + "parking_lot", + "pin-utils", + "slab", + "wasm-bindgen", +] + +[[package]] +name = "web-sys" +version = "0.3.69" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77afa9a11836342370f4817622a2f0f418b134426d91a82dfb48f532d2ec13ef" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "winapi" +version = "0.3.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" +dependencies = [ + "winapi-i686-pc-windows-gnu", + "winapi-x86_64-pc-windows-gnu", +] + +[[package]] +name = "winapi-i686-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" + +[[package]] +name = "winapi-util" +version = "0.1.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4d4cc384e1e73b93bafa6fb4f1df8c41695c8a91cf9c4c64358067d15a7b6c6b" +dependencies = [ + "windows-sys 0.52.0", +] + +[[package]] +name = "winapi-x86_64-pc-windows-gnu" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" + +[[package]] +name = "windows-core" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "33ab640c8d7e35bf8ba19b884ba838ceb4fba93a4e8c65a9059d08afcfc683d9" +dependencies = [ + "windows-targets 0.52.5", +] + +[[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.5", +] + +[[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", +] + +[[package]] +name = "windows-targets" +version = "0.52.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6f0713a46559409d202e70e28227288446bf7841d3211583a4b53e3f6d96e7eb" +dependencies = [ + "windows_aarch64_gnullvm 0.52.5", + "windows_aarch64_msvc 0.52.5", + "windows_i686_gnu 0.52.5", + "windows_i686_gnullvm", + "windows_i686_msvc 0.52.5", + "windows_x86_64_gnu 0.52.5", + "windows_x86_64_gnullvm 0.52.5", + "windows_x86_64_msvc 0.52.5", +] + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.52.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7088eed71e8b8dda258ecc8bac5fb1153c5cffaf2578fc8ff5d61e23578d3263" + +[[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.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9985fd1504e250c615ca5f281c3f7a6da76213ebd5ccc9561496568a2752afb6" + +[[package]] +name = "windows_i686_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" + +[[package]] +name = "windows_i686_gnu" +version = "0.52.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "88ba073cf16d5372720ec942a8ccbf61626074c6d4dd2e745299726ce8b89670" + +[[package]] +name = "windows_i686_gnullvm" +version = "0.52.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "87f4261229030a858f36b459e748ae97545d6f1ec60e5e0d6a3d32e0dc232ee9" + +[[package]] +name = "windows_i686_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" + +[[package]] +name = "windows_i686_msvc" +version = "0.52.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db3c2bf3d13d5b658be73463284eaf12830ac9a26a90c717b7f771dfe97487bf" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.52.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4e4246f76bdeff09eb48875a0fd3e2af6aada79d409d33011886d3e1581517d9" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.52.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "852298e482cd67c356ddd9570386e2862b5673c85bd5f88df9ab6802b334c596" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538" + +[[package]] +name = "windows_x86_64_msvc" +version = "0.52.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bec47e5bfd1bff0eeaf6d8b485cc1074891a197ab4225d504cb7a1ab88b02bf0" + +[[package]] +name = "winnow" +version = "0.5.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f593a95398737aeed53e489c785df13f3618e41dbcd6718c6addbf1395aa6876" +dependencies = [ + "memchr", +] + +[[package]] +name = "winnow" +version = "0.6.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3c52e9c97a68071b23e836c9380edae937f17b9c4667bd021973efc689f618d" +dependencies = [ + "memchr", +] + +[[package]] +name = "winreg" +version = "0.50.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "524e57b2c537c0f9b1e69f1965311ec12182b4122e45035b1508cd24d2adadb1" +dependencies = [ + "cfg-if", + "windows-sys 0.48.0", +] + +[[package]] +name = "ws_stream_wasm" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7999f5f4217fe3818726b66257a4475f71e74ffd190776ad053fa159e50737f5" +dependencies = [ + "async_io_stream", + "futures", + "js-sys", + "log", + "pharos", + "rustc_version 0.4.0", + "send_wrapper 0.6.0", + "thiserror", + "wasm-bindgen", + "wasm-bindgen-futures", + "web-sys", +] + +[[package]] +name = "wyz" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05f360fc0b24296329c78fda852a1e9ae82de9cf7b27dae4b7f62f118f77b9ed" +dependencies = [ + "tap", +] + +[[package]] +name = "xorf" +version = "0.11.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cf24c008fe464f5d8f58b8d16a1ab7e930bd73b2a6933ff8704c414b2bed7f92" +dependencies = [ + "libm", + "rand 0.8.5", + "serde", +] + +[[package]] +name = "zcash_encoding" +version = "0.2.0" +source = "git+https://github.com/zcash/librustzcash?rev=bd7f9d7#bd7f9d7c3ce5cfd14af169ffe0e1c5c903162f46" +dependencies = [ + "byteorder", + "nonempty", +] + +[[package]] +name = "zeroize" +version = "1.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "525b4ec142c6b68a2d10f01f7bbf6755599ca3f81ea53b8431b7dd348f5fdb2d" +dependencies = [ + "zeroize_derive", +] + +[[package]] +name = "zeroize_derive" +version = "1.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.65", +] diff --git a/packages/sdk/lib/Cargo.toml b/packages/sdk/lib/Cargo.toml new file mode 100644 index 0000000000..bb3c90a1d9 --- /dev/null +++ b/packages/sdk/lib/Cargo.toml @@ -0,0 +1,98 @@ +[package] +name = "namada-sdk-wasm" +authors = ["Heliax AG "] +version = "1.0.0" +edition = "2021" +repository = "https://github.com/anoma/namada-interface/" +description = "Shared functionality from Namada protocol" +license = "MIT" + +[lib] +crate-type = ["cdylib", "rlib"] + +[features] +default = [] +dev = [] +multicore = ["rayon", "wasm-bindgen-rayon", "namada_sdk/multicore"] +nodejs = [] +web = [] + +[build-dependencies] +namada_tx = { git = "https://github.com/anoma/namada", rev = "49a4a5d3260423df19ead14df82d18a51fa9b157" } + +[dependencies] +async-trait = {version = "0.1.51"} +# tiny-bip39 = "0.8.2" +tiny-bip39 = { git = "https://github.com/anoma/tiny-bip39", rev = "743d537349c8deab14409ce726b868dcde90fd8e" } +chrono = "0.4.22" +getrandom = { version = "0.2.7", features = ["js"] } +gloo-utils = { version = "0.1.5", features = ["serde"] } +js-sys = "0.3.60" +namada_sdk = { git = "https://github.com/anoma/namada", rev="49a4a5d3260423df19ead14df82d18a51fa9b157", default-features = false } +rand = "0.8.5" +rayon = { version = "1.5.3", optional = true } +rexie = "0.5" +serde = "^1.0.181" +serde_json = "1.0" +tendermint-config = "0.34.0" +tokio = {version = "1.8.2", features = ["rt"]} +thiserror = "^1" +wasm-bindgen = "0.2.86" +wasm-bindgen-futures = "0.4.33" +wasm-bindgen-rayon = { version = "1.0", optional = true } +console_error_panic_hook = "0.1.6" +zeroize = "1.6.0" +hex = "0.4.3" +reqwest = "0.11.25" +subtle-encoding = "0.5.1" +aes-gcm = "0.10.1" +argon2 = "0.4.1" +slip10_ed25519 = "0.1.3" +password-hash = "0.3.2" +masp_primitives = { git = "https://github.com/anoma/masp", tag = "v1.1.0" } +borsh-ext = { git = "https://github.com/heliaxdev/borsh-ext", tag = "v1.2.0" } + +[dependencies.web-sys] +version = "0.3.4" +features = [ + 'console', + 'Document', + 'Event', + 'EventTarget', + 'CustomEvent', + 'CustomEventInit', + 'Headers', + 'Request', + 'RequestInit', + 'RequestMode', + 'Response', + 'Window', + 'WorkerGlobalScope' +] + +[dev-dependencies] +wasm-bindgen-test = "0.3.13" + +# https://doc.rust-lang.org/cargo/reference/profiles.html +[profile.release] +lto = true + +[profile.dev] +opt-level = 3 +lto = true + +# wasm-pack specific configuration +[package.metadata.wasm-pack.profile.release] +# https://docs.rs/wasm-opt/latest/wasm_opt/ +wasm-opt = ['-O4'] + +[package.metadata.wasm-pack.profile.dev] +wasm-opt = false + +[package.metadata.wasm-pack.profile.dev.wasm-bindgen] +omit-default-module-path = true +# We set it to false as it checks if return type from setTimout is a number which is not true in the nodejs environment +debug-js-glue = false + +[package.metadata.wasm-pack.profile.release.wasm-bindgen] +omit-default-module-path = true diff --git a/packages/sdk/lib/LICENSE_MIT b/packages/sdk/lib/LICENSE_MIT new file mode 100644 index 0000000000..c306d4c0fa --- /dev/null +++ b/packages/sdk/lib/LICENSE_MIT @@ -0,0 +1,25 @@ +Copyright (c) 2018 Heliax Dev + +Permission is hereby granted, free of charge, to any +person obtaining a copy of this software and associated +documentation files (the "Software"), to deal in the +Software without restriction, including without +limitation the rights to use, copy, modify, merge, +publish, distribute, sublicense, and/or sell copies of +the Software, and to permit persons to whom the Software +is furnished to do so, subject to the following +conditions: + +The above copyright notice and this permission notice +shall be included in all copies or substantial portions +of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF +ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED +TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A +PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT +SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY +CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION +OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR +IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER +DEALINGS IN THE SOFTWARE. diff --git a/packages/sdk/lib/README.md b/packages/sdk/lib/README.md new file mode 100644 index 0000000000..b0f32c4909 --- /dev/null +++ b/packages/sdk/lib/README.md @@ -0,0 +1,25 @@ +# shared lib + +Rust library for wrapping Namada types and functionality and compiling to wasm, as used by Namada Interface and the wallet extension. + +## Usage + +```bash +# Install wasm-bindgen-cli +cargo install -f wasm-bindgen-cli + +# Build wasm +../scripts/build.sh + +# Build wasm to a NodeJS target (for testing) +../scripts/build-test.sh +``` + +## Testing + +```bash +cargo test + +# Test wasm-specific features +wasm-pack test --node +``` diff --git a/packages/sdk/lib/rust-toolchain.toml b/packages/sdk/lib/rust-toolchain.toml new file mode 100644 index 0000000000..35d88f3d92 --- /dev/null +++ b/packages/sdk/lib/rust-toolchain.toml @@ -0,0 +1,4 @@ +[toolchain] +channel = "nightly-2024-09-08" +components = ["rustc", "cargo", "rust-std", "rust-docs", "rls", "rust-src", "rust-analysis"] +targets = ['wasm32-unknown-unknown'] diff --git a/packages/sdk/lib/src/crypto/aes.rs b/packages/sdk/lib/src/crypto/aes.rs new file mode 100644 index 0000000000..05c2f2a033 --- /dev/null +++ b/packages/sdk/lib/src/crypto/aes.rs @@ -0,0 +1,99 @@ +use crate::crypto::pointer_types::VecU8Pointer; +use aes_gcm::{ + aead::{generic_array::GenericArray, Aead, KeyInit}, + Aes256Gcm, Nonce, +}; +use thiserror::Error; +use wasm_bindgen::prelude::*; +use zeroize::Zeroize; + +#[derive(Debug, Error)] +pub enum AESError { + #[error("Invalid key size! Minimum key size is 32.")] + KeyLengthError, + #[error("Invalid IV! Expected 96 bits (12 bytes)")] + IVSizeError, +} + +#[wasm_bindgen] +pub struct AES { + cipher: Aes256Gcm, + iv: [u8; 12], +} + +#[wasm_bindgen] +impl AES { + #[wasm_bindgen(constructor)] + pub fn new(key: VecU8Pointer, iv: Vec) -> Result { + if key.length < 32 { + return Err(format!( + "{} Received {}", + AESError::KeyLengthError, + key.length + )); + } + let mut key = GenericArray::from_iter(key.vec.clone().into_iter()); + let iv: [u8; 12] = match iv.try_into() { + Ok(iv) => iv, + Err(_) => { + key.zeroize(); + return Err(AESError::IVSizeError.to_string()); + } + }; + + let aes = AES { + cipher: Aes256Gcm::new(&key), + iv, + }; + key.zeroize(); + Ok(aes) + } + + pub fn encrypt(&self, mut text: String) -> Result, String> { + let nonce = Nonce::from_slice(&self.iv); + let result = self + .cipher + .encrypt(nonce, text.as_ref()) + .map_err(|err| err.to_string()); + text.zeroize(); + result + } + + pub fn decrypt(&self, ciphertext: Vec) -> Result { + let nonce = Nonce::from_slice(&self.iv); + let plaintext = self + .cipher + .decrypt(nonce, ciphertext.as_ref()) + .map_err(|err| err.to_string())?; + + Ok(VecU8Pointer::new(plaintext)) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::crypto::rng::{ByteSize, Rng}; + use wasm_bindgen_test::*; + + #[wasm_bindgen_test] + fn can_encrypt_and_decrypt() { + let key = Rng::generate_bytes(Some(ByteSize::N32)) + .expect("Generating random bytes should not fail"); + let iv = Rng::generate_bytes(Some(ByteSize::N12)) + .expect("Generating random bytes should not fail"); + let aes = AES::new(VecU8Pointer::new(key), iv).unwrap(); + let plaintext = "my secret message"; + let encrypted = aes + .encrypt(String::from(plaintext)) + .expect("AES should not fail encrypting plaintext"); + + let decrypted: &[u8] = &aes + .decrypt(encrypted) + .expect("AES should not fail decrypting ciphertext") + .vec; + let decrypted = std::str::from_utf8(decrypted).expect("Should parse as string"); + + assert_eq!(decrypted, plaintext); + } +} diff --git a/packages/sdk/lib/src/crypto/argon2.rs b/packages/sdk/lib/src/crypto/argon2.rs new file mode 100644 index 0000000000..98cf0b4c4b --- /dev/null +++ b/packages/sdk/lib/src/crypto/argon2.rs @@ -0,0 +1,214 @@ +use crate::crypto::pointer_types::VecU8Pointer; +use password_hash::{rand_core::OsRng, PasswordHash, PasswordHasher, PasswordVerifier, SaltString}; +use wasm_bindgen::prelude::*; +use zeroize::{Zeroize, ZeroizeOnDrop}; + +#[wasm_bindgen] +pub struct Argon2Params { + m_cost: u32, + t_cost: u32, + p_cost: u32, +} + +#[wasm_bindgen] +impl Argon2Params { + #[wasm_bindgen(constructor)] + pub fn new(m_cost: u32, t_cost: u32, p_cost: u32) -> Self { + Self { + m_cost, + t_cost, + p_cost, + } + } + + #[wasm_bindgen(getter)] + pub fn m_cost(&self) -> u32 { + self.m_cost + } + + #[wasm_bindgen(getter)] + pub fn t_cost(&self) -> u32 { + self.t_cost + } + + #[wasm_bindgen(getter)] + pub fn p_cost(&self) -> u32 { + self.p_cost + } +} + +#[wasm_bindgen] +#[derive(ZeroizeOnDrop)] +pub struct Argon2 { + #[zeroize(skip)] + salt: SaltString, + password: Vec, + #[zeroize(skip)] + params: argon2::Params, +} + +/// Argon2 password hashing +#[wasm_bindgen] +impl Argon2 { + #[wasm_bindgen(constructor)] + pub fn new( + password: String, + salt: Option, + params: Option, + ) -> Result { + let password = Vec::from(password.as_bytes()); + let default_params = argon2::Params::default(); + + let salt = match salt { + Some(salt) => SaltString::new(&salt).map_err(|err| err.to_string())?, + None => SaltString::generate(&mut OsRng), + }; + + let params = match params { + Some(params) => argon2::Params::new(params.m_cost, params.t_cost, params.p_cost, None) + .map_err(|err| err.to_string())?, + None => default_params, + }; + + Ok(Argon2 { + salt, + password, + params, + }) + } + + pub fn to_hash(&self) -> Result { + let argon2 = argon2::Argon2::default(); + let bytes: &[u8] = &self.password; + let params = &self.params; + + // Hash password to PHC string ($argon2id$v=19$...) + let password_hash = argon2 + .hash_password_customized( + bytes, + None, // Default alg_id = Argon2id + None, // Default ver = v19 + params.to_owned(), + &self.salt, + ) + .map_err(|err| err.to_string())? + .to_string(); + + Ok(password_hash) + } + + pub fn verify(&self, hash: String) -> Result<(), String> { + let argon2 = argon2::Argon2::default(); + let bytes: &[u8] = &self.password; + let parsed_hash = PasswordHash::new(&hash).map_err(|err| err.to_string())?; + + match argon2.verify_password(bytes, &parsed_hash) { + Ok(_) => Ok(()), + Err(err) => Err(err.to_string()), + } + } + + pub fn params(&self) -> Argon2Params { + Argon2Params::new( + self.params.m_cost(), + self.params.t_cost(), + self.params.p_cost(), + ) + } + + /// Convert PHC string to serialized key + pub fn key(&self) -> Result { + let mut hash = self.to_hash()?; + let split = hash.split('$'); + let items: Vec<&str> = split.collect(); + + let key = items[items.len() - 1]; + let vec = Vec::from(key.as_bytes()); + hash.zeroize(); + + Ok(VecU8Pointer::new(vec)) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use wasm_bindgen_test::*; + + #[wasm_bindgen_test] + fn can_hash_password() { + let password = "unhackable"; + let argon2 = Argon2::new(password.into(), None, None) + .expect("Creating instance with default params should not fail"); + let hash = argon2 + .to_hash() + .expect("Hashing password with Argon2 should not fail!"); + + assert!(argon2.verify(hash).is_ok()); + } + + #[wasm_bindgen_test] + fn can_hash_password_with_custom_params() { + // Memory cost + let m_cost = 2048; + // Iterations/Time cost: + let t_cost = 2; + // Degree of parallelism: + let p_cost = 2; + let params = Argon2Params::new(m_cost, t_cost, p_cost); + let password = "unhackable"; + let argon2 = Argon2::new(password.into(), None, Some(params)) + .expect("Creating instance with custom params should not fail"); + + let hash = argon2 + .to_hash() + .expect("Hashing password with Argon2 should not fail!"); + assert!(argon2.verify(hash).is_ok()); + } + + #[wasm_bindgen_test] + fn can_verify_stored_hash() { + let password = "unhackable"; + let argon2 = Argon2::new(password.into(), None, None) + .expect("Creating instance with default params should not fail"); + let stored_hash = "$argon2id$v=19$m=4096,t=3,p=1$0UUjc4ZBOJJLTPrS1mQr1w$orbgGGRzWC0GvplgJuteaDORldnQiJfVumhXSuwO3UE"; + + // With randomly generated salt, this should not create + // an equivalent hash: + assert_ne!(argon2.to_hash().unwrap(), stored_hash); + assert!(argon2.verify(stored_hash.to_string()).is_ok()); + } + + #[wasm_bindgen_test] + fn can_verify_stored_hash_with_custom_salt() { + let password = "unhackable"; + let salt = String::from("41oVKhMIBZ+oF4efwq7e0A"); + let argon2 = Argon2::new(password.into(), Some(salt), None) + .expect("Creating instance with default params should not fail"); + let stored_hash = "$argon2id$v=19$m=4096,t=3,p=1$41oVKhMIBZ+oF4efwq7e0A$ec9kY153e/S6z9awayWdUTLdaQowoAxrdo7ZkTjhBl4"; + + // Providing salt, this should create an equivalent hash: + assert_eq!(argon2.to_hash().unwrap(), stored_hash); + assert!(argon2.verify(stored_hash.to_string()).is_ok()); + } + + #[wasm_bindgen_test] + fn can_get_key_and_params() { + let password = "unhackable"; + let argon2 = Argon2::new(password.into(), None, None) + .expect("Creating instance with default params should not fail"); + let hash = argon2 + .to_hash() + .expect("Hashing password with Argon2 should not fail!"); + + assert!(argon2.verify(hash).is_ok()); + + let params = argon2.params(); + let key = argon2.key().expect("Creating key should not fail"); + + assert_eq!(params.m_cost(), 4096); + assert_eq!(params.t_cost(), 3); + assert_eq!(params.p_cost(), 1); + assert_eq!(key.vec.len(), 43); + } +} diff --git a/packages/sdk/lib/src/crypto/bip32.rs b/packages/sdk/lib/src/crypto/bip32.rs new file mode 100644 index 0000000000..7cd4e5ddd6 --- /dev/null +++ b/packages/sdk/lib/src/crypto/bip32.rs @@ -0,0 +1,137 @@ +use crate::crypto::pointer_types::{StringPointer, VecU8Pointer}; +use rand::{rngs::OsRng, RngCore}; +use slip10_ed25519; +use thiserror::Error; +use wasm_bindgen::prelude::*; +use zeroize::{Zeroize, ZeroizeOnDrop}; + +#[derive(Debug, Error)] +pub enum HDWalletError { + #[error("Unable to derive keys from path")] + DerivationError, + #[error("Invalid key size")] + InvalidKeySize, + #[error("Invalid seed")] + InvalidSeed, +} + +#[wasm_bindgen] +#[derive(Zeroize)] +pub struct Key { + bytes: [u8; 32], +} + +/// A 32 byte ed25519 key +#[wasm_bindgen] +impl Key { + #[wasm_bindgen(constructor)] + pub fn new(bytes: Vec) -> Result { + let bytes: [u8; 32] = match bytes.try_into() { + Ok(bytes) => bytes, + Err(err) => return Err(format!("{}: {:?}", HDWalletError::InvalidKeySize, err)), + }; + + Ok(Key { bytes }) + } + + pub fn to_bytes(&self) -> Vec { + Vec::from(self.bytes) + } + + pub fn to_hex(&self) -> StringPointer { + let bytes: &[u8] = &self.bytes; + let string = hex::encode(&bytes); + StringPointer::new(string) + } +} + +#[wasm_bindgen] +#[derive(ZeroizeOnDrop)] +pub struct HDWallet { + seed: [u8; 64], +} + +/// A set of methods to derive keys from a BIP32/BIP44 path +#[wasm_bindgen] +impl HDWallet { + #[wasm_bindgen(constructor)] + pub fn new(seed_ptr: VecU8Pointer) -> Result { + let seed: [u8; 64] = match seed_ptr.vec.clone().try_into() { + Ok(seed) => seed, + Err(err) => return Err(format!("{}: {:?}", HDWalletError::InvalidSeed, err)), + }; + + Ok(HDWallet { seed }) + } + + pub fn from_seed(seed: Vec) -> Result { + let seed: [u8; 64] = match seed.try_into() { + Ok(seed) => seed, + Err(err) => return Err(format!("{}: {:?}", HDWalletError::InvalidSeed, err)), + }; + + Ok(HDWallet { seed }) + } + + /// Derive account from a seed and a path + pub fn derive(&self, path: Vec) -> Result { + let key = slip10_ed25519::derive_ed25519_private_key(&self.seed, &path); + let private = Key::new(Vec::from(key)) + .map_err(|err| format!("{}: {:?}", HDWalletError::DerivationError, err))?; + + Ok(private) + } + + pub fn disposable_keypair() -> Result { + let path = vec![44, 877, 0, 0, 0]; + let mut key = [0u8; 32]; + OsRng.fill_bytes(&mut key); + + let key = slip10_ed25519::derive_ed25519_private_key(&key, &path); + + Key::new(Vec::from(key)) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::crypto::bip39::Mnemonic; + use wasm_bindgen_test::*; + + #[wasm_bindgen_test] + fn can_derive_keys_from_path() { + let phrase = "caught pig embody hip goose like become worry face oval manual flame \ + pizza steel viable proud eternal speed chapter sunny boat because view bullet"; + let mnemonic = + Mnemonic::from_phrase(phrase.into()).expect("Should not fail with a valid phrase!"); + let seed = mnemonic.to_seed(None).unwrap(); + let bip44: HDWallet = HDWallet::new(seed).unwrap(); + let path = vec![44, 877, 0, 0, 0]; + + let key = bip44.derive(path).expect("Should derive keys from a path"); + + assert_eq!( + key.to_bytes(), + [ + 228, 104, 14, 30, 58, 200, 239, 116, 140, 154, 151, 251, 162, 132, 183, 188, 107, + 0, 45, 182, 36, 48, 46, 39, 113, 29, 252, 73, 44, 242, 125, 30 + ] + ); + } + + // TODO: we use test instead of wasm_bindgen_test because we want to catch the panic + #[wasm_bindgen_test] + fn invalid_seed_should_panic() { + let res = HDWallet::new(VecU8Pointer::new(vec![0, 1, 2, 3, 4])); + + assert!(res.is_err()); + } + + #[wasm_bindgen_test] + fn invalid_key_should_panic() { + let res = Key::new(vec![0, 1, 2, 3, 4]); + + assert!(res.is_err()); + } +} diff --git a/packages/sdk/lib/src/crypto/bip39.rs b/packages/sdk/lib/src/crypto/bip39.rs new file mode 100644 index 0000000000..168738c92b --- /dev/null +++ b/packages/sdk/lib/src/crypto/bip39.rs @@ -0,0 +1,158 @@ +use crate::crypto::pointer_types::{ + new_vec_string_pointer, StringPointer, VecStringPointer, VecU8Pointer, +}; +use bip39::{Language, Mnemonic as M, MnemonicType, Seed}; +use thiserror::Error; +use wasm_bindgen::prelude::*; +use zeroize::Zeroize; + +#[derive(Debug, Error)] +pub enum Bip39Error { + #[error("Invalid phrase")] + InvalidPhrase, +} + +#[wasm_bindgen] +#[derive(Copy, Clone)] +pub enum PhraseSize { + N12 = 12, + N24 = 24, +} + +#[wasm_bindgen] +pub struct Mnemonic { + mnemonic: M, +} + +#[wasm_bindgen] +impl Mnemonic { + #[wasm_bindgen(constructor)] + pub fn new(size: PhraseSize) -> Mnemonic { + let mnemonic_type = match size { + PhraseSize::N12 => MnemonicType::Words12, + PhraseSize::N24 => MnemonicType::Words24, + }; + + let mnemonic = M::new(mnemonic_type, Language::English); + + Mnemonic { mnemonic } + } + + pub fn validate(phrase: &str) -> bool { + M::validate(phrase, Language::English).is_ok() + } + + pub fn from_phrase(mut phrase: String) -> Result { + // First validate phrase, provide error to client if this fails + M::validate(&phrase, Language::English).map_err(|e| format!("{}", e))?; + + let mnemonic = M::from_phrase(&phrase, Language::English) + .map_err(|e| format!("{}: {:?}", Bip39Error::InvalidPhrase, e))?; + + phrase.zeroize(); + + Ok(Mnemonic { mnemonic }) + } + + pub fn to_seed(&self, passphrase: Option) -> Result { + let mut passphrase = match passphrase { + Some(passphrase) => passphrase.string.clone(), + None => "".into(), + }; + + let seed = Seed::new(&self.mnemonic, &passphrase); + passphrase.zeroize(); + + Ok(VecU8Pointer::new(Vec::from(seed.as_bytes()))) + } + + pub fn to_words(&self) -> Result { + let words: Vec = self + .mnemonic + .phrase() + .split(' ') + .map(String::from) + .collect(); + Ok(new_vec_string_pointer(words)) + } + + pub fn phrase(&self) -> String { + String::from(self.mnemonic.phrase()) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use wasm_bindgen_test::*; + + #[wasm_bindgen_test] + fn can_generate_mnemonic_from_size() { + let mnemonic = Mnemonic::new(PhraseSize::N12); + let phrase = mnemonic.phrase(); + let words: Vec<&str> = phrase.split(' ').collect(); + + assert_eq!(words.iter().len(), 12); + + let mnemonic = Mnemonic::new(PhraseSize::N24); + let phrase = mnemonic.phrase(); + let words: Vec<&str> = phrase.split(' ').collect(); + + assert_eq!(words.iter().len(), 24); + } + + #[wasm_bindgen_test] + fn can_generate_seed_from_phrase() { + let phrase = "caught pig embody hip goose like become worry face oval manual flame \ + pizza steel viable proud eternal speed chapter sunny boat because view bullet"; + let mnemonic = Mnemonic::from_phrase(phrase.into()).unwrap(); + let seed = mnemonic + .to_seed(None) + .expect("Should return seed from mnemonic phrase"); + + assert_eq!(seed.vec.len(), 64); + } + + #[wasm_bindgen_test] + fn can_restore_seed_from_phrase() { + let phrase = "caught pig embody hip goose like become worry face oval manual flame \ + pizza steel viable proud eternal speed chapter sunny boat because view bullet"; + let seed_bytes = vec![ + 178, 64, 160, 168, 33, 68, 84, 63, 0, 137, 121, 29, 66, 47, 123, 36, 64, 38, 160, 236, + 93, 38, 53, 157, 169, 119, 42, 153, 188, 80, 209, 149, 51, 92, 251, 168, 150, 220, 70, + 78, 230, 16, 152, 160, 85, 248, 115, 82, 183, 126, 96, 112, 58, 238, 230, 63, 89, 239, + 0, 250, 163, 169, 166, 174, + ]; + let mnemonic = Mnemonic::from_phrase(phrase.into()).unwrap(); + let seed = mnemonic + .to_seed(None) + .expect("Should return seed from mnemonic phrase"); + + assert_eq!(seed.vec, seed_bytes); + } + + #[wasm_bindgen_test] + fn invalid_phrase_should_panic() { + let bad_phrase = "caught pig embody hip goose like become"; + let res = Mnemonic::from_phrase(bad_phrase.into()); + + assert!(res.is_err()); + } + + #[wasm_bindgen_test] + fn can_generate_word_list_from_mnemonic() { + let mnemonic = Mnemonic::new(PhraseSize::N12); + let words = mnemonic + .to_words() + .expect("Should return a VecStringPointer containing the words"); + + assert_eq!(words.strings.len(), 12); + + let mnemonic = Mnemonic::new(PhraseSize::N24); + let words = mnemonic + .to_words() + .expect("Should return a VecStringPointer containing the words"); + + assert_eq!(words.strings.len(), 24); + } +} diff --git a/packages/sdk/lib/src/crypto/mod.rs b/packages/sdk/lib/src/crypto/mod.rs new file mode 100644 index 0000000000..603532b31b --- /dev/null +++ b/packages/sdk/lib/src/crypto/mod.rs @@ -0,0 +1,8 @@ +pub mod aes; +// pub mod argon2; +pub mod bip32; +pub mod bip39; +pub mod pointer_types; +pub mod rng; +// pub mod salt; +// pub mod zip32; diff --git a/packages/sdk/lib/src/crypto/pointer_types.rs b/packages/sdk/lib/src/crypto/pointer_types.rs new file mode 100644 index 0000000000..2d556a7a1b --- /dev/null +++ b/packages/sdk/lib/src/crypto/pointer_types.rs @@ -0,0 +1,98 @@ +//! Types for wrapping sensitive data. +//! +//! Wrapped values will not automatically be accessible to JavaScript, so we can +//! avoid loading sensitive data into browser memory. However, values can be +//! accessed in JavaScript by using the pointer and length fields to read +//! directly from WASM memory if required. +//! +//! These types will also zeroize sensitive data when dropped. +use wasm_bindgen::prelude::*; +use zeroize::ZeroizeOnDrop; + +#[wasm_bindgen] +#[derive(ZeroizeOnDrop)] +pub struct VecU8Pointer { + #[zeroize(skip)] + pub pointer: *const u8, + #[zeroize(skip)] + pub length: usize, + #[wasm_bindgen(skip)] + pub vec: Vec, +} + +#[wasm_bindgen] +impl VecU8Pointer { + #[wasm_bindgen(constructor)] + pub fn new(vec: Vec) -> VecU8Pointer { + VecU8Pointer { + pointer: vec.as_ptr(), + length: vec.len(), + vec, + } + } + + pub fn clone(&self) -> VecU8Pointer { + VecU8Pointer::new(self.vec.clone()) + } +} + +#[wasm_bindgen] +#[derive(ZeroizeOnDrop)] +pub struct StringPointer { + #[zeroize(skip)] + pub pointer: *const u8, + #[zeroize(skip)] + pub length: usize, + #[wasm_bindgen(skip)] + pub string: String, +} + +#[wasm_bindgen] +impl StringPointer { + #[wasm_bindgen(constructor)] + pub fn new(string: String) -> StringPointer { + StringPointer { + pointer: string.as_ptr(), + length: string.len(), + string, + } + } + + pub fn clone(&self) -> StringPointer { + StringPointer::new(self.string.clone()) + } +} + +#[wasm_bindgen] +#[derive(ZeroizeOnDrop)] +pub struct VecStringPointer { + #[zeroize(skip)] + pointers: Vec, + #[zeroize(skip)] + lengths: Vec, + #[wasm_bindgen(skip)] + pub strings: Vec, +} + +#[wasm_bindgen] +impl VecStringPointer { + #[wasm_bindgen(getter)] + pub fn pointers(&self) -> Vec { + self.pointers.clone() + } + + #[wasm_bindgen(getter)] + pub fn lengths(&self) -> Vec { + self.lengths.clone() + } +} + +pub fn new_vec_string_pointer(strings: Vec) -> VecStringPointer { + let pointers = strings.iter().map(|str| str.as_ptr() as usize).collect(); + let lengths = strings.iter().map(|str| str.len()).collect(); + VecStringPointer { + pointers, + lengths, + strings, + } +} diff --git a/packages/sdk/lib/src/crypto/rng.rs b/packages/sdk/lib/src/crypto/rng.rs new file mode 100644 index 0000000000..775a139b44 --- /dev/null +++ b/packages/sdk/lib/src/crypto/rng.rs @@ -0,0 +1,56 @@ +use serde::{Deserialize, Serialize}; +use wasm_bindgen::prelude::*; + +#[wasm_bindgen] +#[derive(Serialize, Deserialize)] +pub enum ByteSize { + N12 = 12, + N24 = 24, + N32 = 32, +} + +#[wasm_bindgen] +pub struct Rng; + +#[wasm_bindgen] +impl Rng { + pub fn generate_bytes(size: Option) -> Result, String> { + let size = match size { + Some(ByteSize::N12) => 12, + Some(ByteSize::N24) => 24, + Some(ByteSize::N32) => 32, + None => 32, + }; + + let mut buf = [0u8; 32]; + getrandom::getrandom(&mut buf).map_err(|err| err.to_string())?; + + let buf = Vec::from(buf); + + Ok(Vec::from(&buf[0..size])) + } +} + +#[cfg(test)] +mod test { + use super::*; + use wasm_bindgen_test::*; + + #[wasm_bindgen_test] + fn can_generate_bytes() { + let bytes = + Rng::generate_bytes(Some(ByteSize::N12)).expect("Generating 12 bytes should not fail"); + + assert_eq!(bytes.len(), 12); + + let bytes = + Rng::generate_bytes(Some(ByteSize::N24)).expect("Generating 24 bytes should not fail"); + + assert_eq!(bytes.len(), 24); + + let bytes = + Rng::generate_bytes(Some(ByteSize::N32)).expect("Generating 32 bytes should not fail"); + + assert_eq!(bytes.len(), 32); + } +} diff --git a/packages/sdk/lib/src/crypto/salt.rs b/packages/sdk/lib/src/crypto/salt.rs new file mode 100644 index 0000000000..1ddd4b674c --- /dev/null +++ b/packages/sdk/lib/src/crypto/salt.rs @@ -0,0 +1,61 @@ +use password_hash::{rand_core::OsRng, SaltString}; +use wasm_bindgen::prelude::*; + +#[wasm_bindgen] +pub struct Salt { + salt: SaltString, +} + +#[wasm_bindgen] +impl Salt { + #[wasm_bindgen(constructor)] + pub fn new(salt: String) -> Result { + let salt = SaltString::new(&salt).map_err(|err| err.to_string())?; + + Ok(Salt { salt }) + } + + pub fn generate() -> Self { + Self { + salt: SaltString::generate(&mut OsRng), + } + } + + pub fn to_bytes(&self) -> Result, String> { + let salt_string = &self.salt.to_string(); + let salt = argon2::password_hash::Salt::new(salt_string).map_err(|err| err.to_string())?; + let bytes: &[u8] = salt.as_bytes(); + Ok(Vec::from(bytes)) + } + + pub fn as_string(&self) -> String { + self.salt.to_string() + } +} + +#[cfg(test)] +mod tests { + use super::*; + use wasm_bindgen_test::*; + + #[wasm_bindgen_test] + fn can_generate_salt_from_string() { + let salt_string = String::from("41oVKhMIBZ+oF4efwq7e0A"); + let salt = + Salt::new(salt_string.clone()).expect("Creating instance of Salt should not fail!"); + + assert_eq!(salt_string, salt.as_string()); + } + + #[wasm_bindgen_test] + fn can_generate_salt_bytes_from_string() { + let salt = String::from("41oVKhMIBZ+oF4efwq7e0A"); + let salt = Salt::new(salt).expect("Creating salt from string should not fail"); + let expected_bytes = vec![ + 52, 49, 111, 86, 75, 104, 77, 73, 66, 90, 43, 111, 70, 52, 101, 102, 119, 113, 55, 101, + 48, 65, + ]; + let bytes = salt.to_bytes().expect("Returning to bytes should not fail"); + assert_eq!(bytes, expected_bytes); + } +} diff --git a/packages/sdk/lib/src/crypto/zip32.rs b/packages/sdk/lib/src/crypto/zip32.rs new file mode 100644 index 0000000000..b0be45872b --- /dev/null +++ b/packages/sdk/lib/src/crypto/zip32.rs @@ -0,0 +1,185 @@ +//! ShieldedHDWallet - Provide wasm_bindgen bindings for zip32 HD wallets +//! Imports from masp_primitives::zip32, instead of zcash_primitives::zip32, as +//! the value for constant ZIP32_SAPLING_MASTER_PERSONALIZATION is different! +//! Otherwise, these implementations should be equivalent. +use borsh_ext::BorshSerializeExt; +use masp_primitives::{ + sapling::PaymentAddress, + zip32::{sapling, ChildIndex, ExtendedFullViewingKey, ExtendedSpendingKey}, +}; +use namada_sdk::borsh::BorshDeserialize; +use wasm_bindgen::prelude::*; +use zeroize::{Zeroize, ZeroizeOnDrop}; + +#[wasm_bindgen] +#[derive(Zeroize)] +pub struct DerivationResult { + xsk: Vec, + xfvk: Vec, + payment_address: Vec, +} + +#[wasm_bindgen] +impl DerivationResult { + pub fn xsk(&self) -> Vec { + self.xsk.clone() + } + + pub fn xfvk(&self) -> Vec { + self.xfvk.clone() + } + + pub fn payment_address(&self) -> Vec { + self.payment_address.clone() + } +} + +#[wasm_bindgen] +#[derive(ZeroizeOnDrop)] +pub struct ShieldedHDWallet { + seed: [u8; 32], +} + +#[wasm_bindgen] +impl ShieldedHDWallet { + #[wasm_bindgen(constructor)] + pub fn new(seed: JsValue, path: Vec) -> Result { + let seed = js_sys::Uint8Array::from(seed).to_vec(); + let sk = slip10_ed25519::derive_ed25519_private_key(&seed, &path); + + Ok(ShieldedHDWallet { seed: sk }) + } + + pub fn new_from_sk(sk_bytes: Vec) -> Result { + let sk: [u8; 32] = match sk_bytes.try_into() { + Ok(bytes) => bytes, + Err(err) => return Err(format!("Invalid Private Key! {:?}", err)), + }; + + Ok(ShieldedHDWallet { seed: sk }) + } + + pub fn derive( + &self, + path: Vec, + diversifier: Option>, + ) -> Result { + let master_spend_key = sapling::ExtendedSpendingKey::master(&self.seed); + + let purpose = path.first().expect("zip32 purpose is required!"); + let coin_type = path.get(1).expect("zip32 coin_type is required!"); + let account = path.get(2).expect("zip32 account is required!"); + + // Optional address index + let address_index = path.get(3); + + let mut zip32_path: Vec = vec![purpose, coin_type, account] + .iter() + .map(|i| ChildIndex::Hardened(**i)) + .collect(); + + if address_index.is_some() { + zip32_path.push(ChildIndex::NonHardened(*address_index.unwrap())); + } + + let xsk: ExtendedSpendingKey = + ExtendedSpendingKey::from_path(&master_spend_key, &zip32_path); + + let xfvk = ExtendedFullViewingKey::from(&xsk); + + // We either use passed diversifier or the default payment_address + let payment_address: PaymentAddress = match diversifier { + Some(d) => { + let diversifier = BorshDeserialize::try_from_slice(&d).unwrap(); + xfvk.fvk.vk.to_payment_address(diversifier).unwrap() + } + None => xfvk.default_address().1, + }; + + Ok(DerivationResult { + xsk: xsk.serialize_to_vec(), + xfvk: xfvk.serialize_to_vec(), + payment_address: payment_address.serialize_to_vec(), + }) + } +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::crypto::bip39; + use masp_primitives::sapling::PaymentAddress; + use wasm_bindgen_test::*; + + const KEY_SIZE: usize = 96; + + #[wasm_bindgen_test] + fn can_instantiate_from_seed() { + let seed = JsValue::from(js_sys::Uint8Array::new_with_length(64)); + let path = vec![44, 877, 0, 0, 0]; + let shielded_wallet = ShieldedHDWallet::new(seed, path); + + assert!(shielded_wallet.is_ok()); + } + + #[wasm_bindgen_test] + fn can_derive_shielded_key_to_serialized() { + let seed = JsValue::from(js_sys::Uint8Array::new_with_length(64)); + let path = vec![44, 877, 0, 0, 0]; + let shielded_wallet = ShieldedHDWallet::new(seed, path) + .expect("Instantiating ShieldedHDWallet should not fail"); + + let DerivationResult { + ref payment_address, + ref xsk, + ref xfvk, + } = shielded_wallet + .derive(vec![32, 877, 0], None) + .expect("Deriving from ExtendedKeys should not fail"); + + let payment_address: PaymentAddress = + borsh::BorshDeserialize::try_from_slice(payment_address) + .expect("Should be able to deserialize payment address!"); + let xsk: ExtendedSpendingKey = borsh::BorshDeserialize::try_from_slice(xsk) + .expect("Should be able to deserialize extended spending key!"); + let xfvk: ExtendedFullViewingKey = borsh::BorshDeserialize::try_from_slice(xfvk) + .expect("Should be able to deserialize full viewing key!"); + + assert_eq!(payment_address.to_bytes().len(), 43); + assert_eq!(xsk.expsk.to_bytes().len(), KEY_SIZE); + assert_eq!(xfvk.fvk.to_bytes().len(), KEY_SIZE); + } + + #[wasm_bindgen_test] + fn can_restore_shielded_keys_from_mnemonic() { + let phrase = "great sphere inmate december menu warrior adjust glass flat heavy act mail"; + let mnemonic = bip39::Mnemonic::from_phrase(phrase.into()).unwrap(); + let seed = mnemonic + .to_seed(None) + .expect("Should return seed from mnemonic phrase"); + let path = vec![44, 877, 0, 0, 0]; + + let shielded_wallet = ShieldedHDWallet::new(JsValue::from(seed), path) + .expect("Instantiating ShieldedHDWallet should not fail"); + + let shielded_account = shielded_wallet + .derive(vec![32, 877, 0], None) + .expect("Deriving from ExtendedKeys should not fail"); + + let payment_address = PaymentAddress::try_from_slice(&shielded_account.payment_address()) + .expect("should instantiate from serialized bytes"); + let xfvk = ExtendedFullViewingKey::try_from_slice(&shielded_account.xfvk()) + .expect("should instantiate from serialized bytes"); + + assert_eq!(payment_address.to_string(), "efad0a092281f049a04250b91b84a8454cec0c5da75821ef7fd2deb684201cc83dd7bb287c241b11cd88d9"); + assert_eq!( + xfvk.fvk.to_string(), + format!( + "{}{}{}", + "a654d32c7b361f77a774a3f80c7dcd053a9e904f0c3bab1e9e207ed4e01434103fa", + "d5db7d3784841e0dd5f1b931b515186da3058562c103eaf11dc665c9da19f12ea71", + "19818ed1f124bd0573f15a82e97893664b7bc3e80b19ed96ba4f52eef3", + ) + ); + } +} diff --git a/packages/sdk/lib/src/lib.rs b/packages/sdk/lib/src/lib.rs new file mode 100644 index 0000000000..83d15cb37e --- /dev/null +++ b/packages/sdk/lib/src/lib.rs @@ -0,0 +1,23 @@ +//! # shared +//! +//! A library of functions to integrate shared functionality from the Namada ecosystem + +pub mod crypto; +pub mod query; +pub mod rpc_client; +pub mod sdk; +pub mod types; +mod utils; + +#[cfg(feature = "multicore")] +pub use wasm_bindgen_rayon::init_thread_pool; + +// Empty function for non-multicore builds +// Simplifies imports in js code +#[cfg(not(feature = "multicore"))] +use wasm_bindgen::prelude::wasm_bindgen; + +#[cfg(not(feature = "multicore"))] +#[allow(non_snake_case)] +#[wasm_bindgen] +pub async fn initThreadPool(_threads: u8) {} diff --git a/packages/sdk/lib/src/query.rs b/packages/sdk/lib/src/query.rs new file mode 100644 index 0000000000..e815c2a9f9 --- /dev/null +++ b/packages/sdk/lib/src/query.rs @@ -0,0 +1,863 @@ +use gloo_utils::format::JsValueSerdeExt; +use js_sys::Uint8Array; +use namada_sdk::address::Address; +use namada_sdk::borsh::BorshSerialize; +use namada_sdk::collections::{HashMap, HashSet}; +use namada_sdk::eth_bridge::bridge_pool::query_signed_bridge_pool; +use namada_sdk::eth_bridge_pool::TransferToEthereum; +use namada_sdk::governance::storage::keys as governance_storage; +use namada_sdk::governance::utils::{ + compute_proposal_result, ProposalVotes, TallyResult, TallyType, VotePower, +}; +use namada_sdk::governance::{ProposalType, ProposalVote}; +use namada_sdk::hash::Hash; +use namada_sdk::masp::shielded_wallet::ShieldedApi; +use namada_sdk::masp::utils::MaspClient as NamadaMaspClient; +use namada_sdk::masp::utils::RetryStrategy; +use namada_sdk::masp::{IndexerMaspClient, LedgerMaspClient}; +use namada_sdk::masp::{ShieldedContext, ShieldedSyncConfig}; +use namada_sdk::masp_primitives::asset_type::AssetType; +use namada_sdk::masp_primitives::sapling::ViewingKey; +use namada_sdk::masp_primitives::transaction::components::ValueSum; +use namada_sdk::masp_primitives::zip32::ExtendedFullViewingKey; +use namada_sdk::parameters::storage; +use namada_sdk::proof_of_stake::Epoch; +use namada_sdk::queries::RPC; +use namada_sdk::rpc::{ + self, get_public_key_at, get_token_balance, get_total_staked_tokens, is_steward, query_epoch, + query_masp_epoch, query_native_token, query_proposal_by_id, query_proposal_votes, + query_storage_value, +}; +use namada_sdk::state::BlockHeight; +use namada_sdk::state::Key; +use namada_sdk::token; +use namada_sdk::tx::{ + TX_BOND_WASM, TX_CLAIM_REWARDS_WASM, TX_IBC_WASM, TX_REDELEGATE_WASM, TX_REVEAL_PK, + TX_TRANSFER_WASM, TX_UNBOND_WASM, TX_VOTE_PROPOSAL, TX_WITHDRAW_WASM, +}; +use namada_sdk::uint::I256; +use namada_sdk::wallet::DatedKeypair; +use namada_sdk::ExtendedSpendingKey; +use namada_sdk::ExtendedViewingKey; +use std::collections::BTreeMap; +use std::str::FromStr; +use std::time::Duration; +use wasm_bindgen::prelude::*; +use wasm_bindgen::JsError; + +use crate::rpc_client::HttpClient; +use crate::sdk::{ + io::WebIo, + masp::{sync, JSShieldedUtils}, +}; +use crate::types::query::{ProposalInfo, WasmHash}; +use crate::utils::{set_panic_hook, to_js_result}; + +/// Progress bar names +pub const SDK_SCANNED_PROGRESS_BAR: &str = "namada_sdk::progress_bar::scanned"; +pub const SDK_FETCHED_PROGRESS_BAR: &str = "namada_sdk::progress_bar::fetched"; +pub const SDK_APPLIED_PROGRESS_BAR: &str = "namada_sdk::progress_bar::applied"; + +#[wasm_bindgen] +pub struct ProgressBarNames {} + +#[wasm_bindgen] +impl ProgressBarNames { + #[allow(non_snake_case)] + #[wasm_bindgen(getter)] + pub fn Scanned() -> String { + SDK_SCANNED_PROGRESS_BAR.to_string() + } + + #[allow(non_snake_case)] + #[wasm_bindgen(getter)] + pub fn Fetched() -> String { + SDK_FETCHED_PROGRESS_BAR.to_string() + } + + #[allow(non_snake_case)] + #[wasm_bindgen(getter)] + pub fn Applied() -> String { + SDK_APPLIED_PROGRESS_BAR.to_string() + } +} + +enum MaspClient { + Ledger(LedgerMaspClient), + Indexer(IndexerMaspClient), +} + +#[wasm_bindgen] +/// Represents an API for querying the ledger +pub struct Query { + client: HttpClient, + masp_client: MaspClient, +} + +#[wasm_bindgen] +impl Query { + #[wasm_bindgen(constructor)] + pub fn new(url: String, masp_url: Option) -> Query { + set_panic_hook(); + let client = HttpClient::new(url); + + let masp_client = if let Some(url) = masp_url { + let client = reqwest::Client::builder().build().unwrap(); + // TODO: for now we just concatenate the v1 api path + let url = reqwest::Url::parse(&format!("{}/api/v1", url)).unwrap(); + + MaspClient::Indexer(IndexerMaspClient::new(client, url, true, 10)) + } else { + MaspClient::Ledger(LedgerMaspClient::new( + client.clone(), + // Using one does not break the progress indicators + 1, + Duration::from_millis(5), + )) + }; + + Query { + client, + masp_client, + } + } + + /// Gets current epoch + /// + /// # Errors + /// + /// Returns an error if the RPC call fails + pub async fn query_epoch(&self) -> Result { + let epoch = RPC.shell().epoch(&self.client).await?; + + Ok(epoch.0) + } + + /// Gets all active validator addresses + /// + /// # Errors + /// + /// Returns an error if the RPC call fails + pub async fn query_all_validator_addresses(&self) -> Result { + let validator_addresses = RPC + .vp() + .pos() + .validator_addresses(&self.client, &None) + .await?; + + to_js_result(validator_addresses) + } + + /// Gets total bonds by validator address + /// + /// # Errors + /// + /// Returns an error if the RPC call fails + pub async fn query_total_bonds(&self, address: String) -> Result { + let address = Address::from_str(&address)?; + + let total_bonds = RPC + .vp() + .pos() + .validator_stake(&self.client, &address, &None) + .await?; + + to_js_result(total_bonds) + } + + /// Gets all delegations for every provided address. + /// Returns a tuple of: + /// (owner_address, validator_address, total_bonds, total_unbonds, withdrawable) + /// + /// # Arguments + /// + /// * `owner_addresses` - Account address in form of bech32, base64 encoded string + /// + /// # Errors + /// + /// Panics if address can't be deserialized + pub async fn query_my_validators( + &self, + owner_addresses: Box<[JsValue]>, + ) -> Result { + let owner_addresses: Vec
= owner_addresses + .iter() + .map(|address| { + address + .as_string() + .and_then(|address_str| Address::from_str(&address_str).ok()) + // TODO: we unwrap but we could also filter_map with warning or map to error + .unwrap() + }) + .collect(); + + let mut validators_per_address: HashMap> = HashMap::new(); + + for address in owner_addresses.into_iter() { + let validators = RPC + .vp() + .pos() + .delegation_validators(&self.client, &address, &None) + .await?; + + validators_per_address.insert(address, validators); + } + + let mut result: Vec<(Address, Address, String, String, String)> = Vec::new(); + + let epoch = query_epoch(&self.client).await?; + for (owner, validators) in validators_per_address.into_iter() { + for validator in validators.into_iter() { + let owner_option = &Some(owner.clone()); + let validator_option = &Some(validator.clone()); + + let enriched = RPC + .vp() + .pos() + .enriched_bonds_and_unbonds(&self.client, epoch, owner_option, validator_option) + .await?; + + result.push(( + owner.clone(), + validator, + enriched.bonds_total.to_string_native(), + enriched.unbonds_total.to_string_native(), + enriched.total_withdrawable.to_string_native(), + )); + } + } + + to_js_result(result) + } + + fn get_decoded_balance( + decoded_balance: (ValueSum, ValueSum), + ) -> Vec<(Address, token::Amount)> { + let mut result = Vec::new(); + + for (token_addr, amount) in decoded_balance.0.components() { + let amount = token::Amount::from_change(*amount); + result.push((token_addr.clone(), amount)); + } + + result + } + + pub async fn query_staking_positions( + &self, + owner_addresses: Box<[JsValue]>, + ) -> Result { + let owner_addresses: Vec
= owner_addresses + .iter() + .filter_map(|address| address.as_string()) + .filter_map(|address| Address::from_str(&address).ok()) + .collect(); + + let mut validators_per_address: HashMap> = HashMap::new(); + + for address in owner_addresses.into_iter() { + let validators = RPC + .vp() + .pos() + .delegation_validators(&self.client, &address, &None) + .await?; + + validators_per_address.insert(address, validators); + } + + let mut bonds = vec![]; + let mut unbonds = vec![]; + + let epoch = query_epoch(&self.client).await?; + for (owner, validators) in validators_per_address.into_iter() { + for validator in validators.into_iter() { + let owner_option = &Some(owner.clone()); + let validator_option = &Some(validator.clone()); + + let enriched = RPC + .vp() + .pos() + .enriched_bonds_and_unbonds(&self.client, epoch, owner_option, validator_option) + .await?; + + for (bond_id, details) in &enriched.data { + for bond in &details.data.bonds { + bonds.push(( + bond_id.source.clone(), + bond_id.validator.clone(), + bond.amount.to_string_native(), + bond.start.to_string(), + )); + } + + for unbond in &details.data.unbonds { + unbonds.push(( + bond_id.source.clone(), + bond_id.validator.clone(), + unbond.amount.to_string_native(), + unbond.start.to_string(), + unbond.withdraw.to_string(), + )); + } + } + } + } + + to_js_result((bonds, unbonds)) + } + + /// Queries transparent balance for a given address + /// + /// # Arguments + /// + /// * `owner` - Account address in form of bech32, base64 encoded string + async fn query_transparent_balance( + &self, + owner: Address, + tokens: Vec
, + ) -> Result, JsError> { + let mut result = vec![]; + for token in tokens { + let balances = get_token_balance(&self.client, &token, &owner, None).await?; + result.push((token, balances)); + } + + Ok(result) + } + + pub async fn shielded_sync( + &self, + vks: Box<[JsValue]>, + sks: Box<[JsValue]>, + ) -> Result<(), JsError> { + let vks: Vec = vks + .iter() + .filter_map(|owner| owner.as_string()) + .map(|o| { + ExtendedFullViewingKey::from(ExtendedViewingKey::from_str(&o).unwrap()) + .fvk + .vk + }) + .collect(); + + let sks = sks + .iter() + .filter_map(|owner| owner.as_string()) + .map(|sk| ExtendedSpendingKey::from_str(&sk).unwrap()) + .collect::>(); + + let dated_keypairs = vks + .into_iter() + .map(|vk| DatedKeypair { + key: vk, + birthday: BlockHeight::from(0), + }) + .collect::>(); + + let dated_sks = sks + .into_iter() + .map(|sk| DatedKeypair { + key: sk, + birthday: BlockHeight::from(0), + }) + .collect::>(); + + match &self.masp_client { + MaspClient::Indexer(client) => { + web_sys::console::log_1(&"Syncing using IndexerMaspClient".into()); + self.sync(client.clone(), dated_keypairs, dated_sks).await? + } + MaspClient::Ledger(client) => { + web_sys::console::log_1(&"Syncing using LedgerMaspClient".into()); + self.sync(client.clone(), dated_keypairs, dated_sks).await? + } + }; + + Ok(()) + } + + async fn sync( + &self, + client: C, + dated_keypairs: Vec>, + dated_sks: Vec>, + ) -> Result<(), JsError> + where + C: NamadaMaspClient + Send + Sync + Unpin + 'static, + { + let progress_bar_scanned = sync::ProgressBarWeb::new(SDK_SCANNED_PROGRESS_BAR); + let progress_bar_fetched = sync::ProgressBarWeb::new(SDK_FETCHED_PROGRESS_BAR); + let progress_bar_applied = sync::ProgressBarWeb::new(SDK_APPLIED_PROGRESS_BAR); + let shutdown_signal_web = sync::ShutdownSignalWeb {}; + // batch size does not matter for masp ledger client, and if we set to sth else than 1 it breaks + // progress bar + let batch_size = match self.masp_client { + MaspClient::Ledger(_) => 1, + MaspClient::Indexer(_) => 100, + }; + + let config = ShieldedSyncConfig::builder() + .client(client) + .scanned_tracker(progress_bar_scanned) + .fetched_tracker(progress_bar_fetched) + .applied_tracker(progress_bar_applied) + .shutdown_signal(shutdown_signal_web) + .block_batch_size(batch_size) + .wait_for_last_query_height(true) + .retry_strategy(RetryStrategy::Times(10)) + .build(); + + let env = sync::TaskEnvWeb::new(); + + let mut shielded_context: ShieldedContext = ShieldedContext::default(); + + shielded_context + .sync( + env, + config, + None, + dated_sks.as_slice(), + dated_keypairs.as_slice(), + ) + .await + .map_err(|e| JsError::new(&format!("{:?}", e)))?; + + Ok(()) + } + + /// Queries shielded balance for a given extended viewing key + /// + /// # Arguments + /// + /// * `xvk` - Extended viewing key + async fn query_shielded_balance( + &self, + xvk: ExtendedViewingKey, + tokens: Vec
, + ) -> Result, JsError> { + let viewing_key = ExtendedFullViewingKey::from(xvk).fvk.vk; + + // We are recreating shielded context to avoid multiple mutable borrows + let mut shielded: ShieldedContext = ShieldedContext::default(); + shielded.load().await?; + shielded + .precompute_asset_types(&self.client, tokens.iter().collect()) + .await + .map_err(|e| JsError::new(&format!("{:?}", e)))?; + + let epoch = query_masp_epoch(&self.client).await?; + let balance = shielded + .compute_exchanged_balance(&self.client, &WebIo, &viewing_key, epoch) + .await + .map_err(|e| JsError::new(&format!("{:?}", e)))?; + + let res = match balance { + Some(balance) => { + let decoded_balance = shielded + .decode_combine_sum_to_epoch(&self.client, balance, epoch) + .await; + + Self::get_decoded_balance(decoded_balance) + } + None => vec![], + }; + + Ok(res) + } + + pub async fn query_balance( + &self, + owner: String, + tokens: Box<[JsValue]>, + ) -> Result { + let tokens: Vec
= tokens + .iter() + .map(|address| { + let address_str = address.as_string().unwrap(); + Address::from_str(&address_str).unwrap() + }) + .collect(); + + let result = match Address::from_str(&owner) { + Ok(addr) => self.query_transparent_balance(addr, tokens).await, + Err(e1) => match ExtendedViewingKey::from_str(&owner) { + Ok(xvk) => self.query_shielded_balance(xvk, tokens).await, + Err(e2) => return Err(JsError::new(&format!("{} {}", e1, e2))), + }, + }?; + + let mut mapped_result: Vec<(Address, String)> = vec![]; + for (token, amount) in result { + mapped_result.push((token.clone(), amount.to_string())) + } + + to_js_result(mapped_result) + } + + pub async fn query_public_key(&self, address: &str) -> Result { + let addr = Address::from_str(address).map_err(JsError::from)?; + let pk = get_public_key_at(&self.client, &addr, 0).await?; + + let result = pk.map(|v| v.to_string()); + + to_js_result(result) + } + + pub async fn query_signed_bridge_pool( + &self, + owner_addresses: Box<[JsValue]>, + ) -> Result { + let bridge_pool = query_signed_bridge_pool(&self.client, &WebIo).await?; + + let owner_addresses: Vec
= owner_addresses + .iter() + .filter_map(|address| address.as_string()) + .filter_map(|address| Address::from_str(&address).ok()) + .collect(); + + let result: Vec = bridge_pool + .into_iter() + .filter_map(|(_hash, pending_transfer)| { + if owner_addresses.contains(&pending_transfer.transfer.sender) { + Some(pending_transfer.transfer) + } else { + None + } + }) + .collect(); + + to_js_result(result) + } + + pub async fn query_total_staked_tokens(&self, epoch: u64) -> Result { + let total_staked_tokens = get_total_staked_tokens(&self.client, Epoch(epoch)).await?; + + to_js_result(total_staked_tokens) + } + + pub async fn query_proposal_counter(&self) -> Result { + let proposal_counter_key = governance_storage::get_counter_key(); + let proposal_counter = + query_storage_value::(&self.client, &proposal_counter_key) + .await + .unwrap(); + + to_js_result(proposal_counter) + } + + pub async fn query_proposal_by_id(&self, id: u64) -> Result { + let proposal = query_proposal_by_id(&self.client, id) + .await + .unwrap() + .expect("Proposal should be written to storage."); + + let content = serde_json::to_string(&proposal.content)?; + + let is_steward = is_steward(&self.client, &proposal.author).await; + let tally_type = proposal.get_tally_type(is_steward); + let tally_type_string = match tally_type { + // TODO: Change in interface + TallyType::TwoFifths => "two-fifths", + TallyType::OneHalfOverOneThird => "one-half-over-one-third", + TallyType::LessOneHalfOverOneThirdNay => "less-one-half-over-one-third-nay", + }; + + let (proposal_type, data) = match proposal.r#type { + ProposalType::Default => ("default", None), + ProposalType::DefaultWithWasm(hash) => ("default", Some(hash.to_string())), + ProposalType::PGFSteward(data) => { + let data_string = serde_json::to_string(&data)?; + ("pgf_steward", Some(data_string)) + } + ProposalType::PGFPayment(data) => { + let data_string = serde_json::to_string(&data)?; + ("pgf_payment", Some(data_string)) + } + }; + + let proposal_info = ProposalInfo { + id: proposal.id, + author: proposal.author.to_string(), + start_epoch: proposal.voting_start_epoch.0, + end_epoch: proposal.voting_end_epoch.0, + grace_epoch: proposal.activation_epoch.0, + content, + tally_type: String::from(tally_type_string), + proposal_type: String::from(proposal_type), + data, + }; + + let mut writer = vec![]; + BorshSerialize::serialize(&proposal_info, &mut writer)?; + + Ok(Uint8Array::from(writer.as_slice())) + } + + #[allow(clippy::type_complexity)] + pub async fn query_proposal_votes( + &self, + proposal_id: u64, + epoch: u64, + ) -> Result { + let votes = compute_proposal_votes(&self.client, proposal_id, Epoch(epoch)).await; + + let validator_votes: Vec<(Address, String, token::Amount)> = + Vec::from_iter(votes.validators_vote.iter().map(|(address, vote)| { + let vote = match vote { + ProposalVote::Yay => "yay", + ProposalVote::Nay => "nay", + ProposalVote::Abstain => "abstain", + }; + + let voting_power = *votes + .validator_voting_power + .get(address) + .expect("validator has voting power entry"); + + (address.clone(), String::from(vote), voting_power) + })); + + // TODO: refactor this to fix type_complexity clippy warning + let delegator_votes: Vec<(Address, String, Vec<(Address, token::Amount)>)> = + Vec::from_iter(votes.delegators_vote.iter().map(|(address, vote)| { + let vote = match vote { + ProposalVote::Yay => "yay", + ProposalVote::Nay => "nay", + ProposalVote::Abstain => "abstain", + }; + + let voting_power = votes + .delegator_voting_power + .get(address) + .expect("delegator has voting power entry") + .clone() + .into_iter() + .collect(); + + (address.clone(), String::from(vote), voting_power) + })); + + to_js_result((validator_votes, delegator_votes)) + } + + pub async fn query_proposal_result( + &self, + proposal_id: u64, + epoch: u64, + ) -> Result { + let epoch = Epoch(epoch); + + let votes = compute_proposal_votes(&self.client, proposal_id, epoch).await; + + let total_voting_power = get_total_staked_tokens(&self.client, epoch).await?; + + let proposal = query_proposal_by_id(&self.client, proposal_id) + .await + .unwrap() + .expect("Proposal should be written to storage."); + let is_steward = is_steward(&self.client, &proposal.author).await; + let tally_type = proposal.get_tally_type(is_steward); + + let proposal_result = compute_proposal_result(votes, total_voting_power, tally_type) + .expect("could compute proposal result"); + + let passed = match proposal_result.result { + TallyResult::Passed => true, + TallyResult::Rejected => false, + }; + + to_js_result(( + passed, + proposal_result.total_yay_power, + proposal_result.total_nay_power, + proposal_result.total_abstain_power, + proposal_result.total_voting_power, + )) + } + + pub async fn query_proposal_code(&self, proposal_id: u64) -> Result { + let proposal_code_key = governance_storage::get_proposal_code_key(proposal_id); + let code = + query_storage_value::>(&self.client, &proposal_code_key).await?; + + Ok(Uint8Array::from(code.as_slice())) + } + + /// Returns a list of all delegations for given addresses and epoch + /// + /// # Arguments + /// + /// * `addresses` - delegators addresses + /// * `epoch` - epoch in which we want to query delegations + pub async fn get_total_delegations( + &self, + addresses: Box<[JsValue]>, + epoch: Option, + ) -> Result { + let addresses: Vec
= addresses + .iter() + .filter_map(|address| address.as_string()) + .filter_map(|address| Address::from_str(&address).ok()) + .collect(); + + let epoch = epoch.map(Epoch); + + let mut delegations: HashMap = HashMap::new(); + + for address in addresses.into_iter() { + let validators: HashMap = RPC + .vp() + .pos() + .delegations(&self.client, &address, &epoch) + .await?; + let sum_of_delegations = validators + .into_values() + .fold(token::Amount::zero(), |acc, curr| { + acc.checked_add(curr).expect("Amount overflow") + }); + + delegations.insert(address, sum_of_delegations); + } + + to_js_result(delegations) + } + + /// Returns list of delegators that already voted on a proposal + /// + /// # Arguments + /// + /// * `proposal_id` - id of proposal to get delegators votes from + pub async fn delegators_votes(&self, proposal_id: u64) -> Result { + let votes = query_proposal_votes(&self.client, proposal_id).await?; + let res: Vec<(Address, bool)> = votes + .into_iter() + .map(|vote| { + let is_yay = match vote.data { + ProposalVote::Yay => true, + ProposalVote::Nay => false, + ProposalVote::Abstain => false, + }; + (vote.delegator, is_yay) + }) + .collect(); + + to_js_result(res) + } + + pub async fn query_gas_costs(&self) -> Result { + let key = storage::get_gas_cost_key(); + let gas_cost_table = + query_storage_value::>(&self.client, &key) + .await + .expect("Parameter should be defined."); + + let mut result: Vec<(String, String)> = Vec::new(); + + for (token, gas_cost) in gas_cost_table { + result.push((token.to_string(), gas_cost.to_string_native())); + } + + to_js_result(result) + } + + pub async fn query_native_token(&self) -> Result { + let address = query_native_token(&self.client).await?; + to_js_result(address) + } + + // Vec of code paths of supported transactions + pub fn code_paths() -> Vec { + vec![ + TX_TRANSFER_WASM.to_string(), + TX_BOND_WASM.to_string(), + TX_REDELEGATE_WASM.to_string(), + TX_UNBOND_WASM.to_string(), + TX_WITHDRAW_WASM.to_string(), + TX_CLAIM_REWARDS_WASM.to_string(), + TX_REVEAL_PK.to_string(), + TX_VOTE_PROPOSAL.to_string(), + TX_IBC_WASM.to_string(), + ] + } + + // Query supported wasm code-paths, and return a serialized vec of object containing path and hash + pub async fn query_wasm_hashes(&self) -> Result { + let mut results: Vec = vec![]; + let code_paths = Query::code_paths(); + + for path in code_paths { + let hash = self.query_wasm_hash(&path).await; + + if hash.is_some() { + let wasm_hash = WasmHash::new(path, hash.unwrap()); + results.push(wasm_hash); + } + } + + Ok(JsValue::from_serde(&results).unwrap()) + } + + // Query hash of wasm code on chain + pub async fn query_wasm_hash(&self, tx_code_path: &str) -> Option { + let hash_key = Key::wasm_hash(tx_code_path); + let (tx_code_res, _) = rpc::query_storage_value_bytes(&self.client, &hash_key, None, false) + .await + .ok()?; + if let Some(tx_code_bytes) = tx_code_res { + let tx_code = Hash::try_from(&tx_code_bytes[..]).expect("Invalid code hash"); + Some(tx_code.to_string()) + } else { + None + } + } +} + +//TODO: remove after moving this fn from apps to shared +pub async fn compute_proposal_votes( + client: &HttpClient, + proposal_id: u64, + epoch: Epoch, +) -> ProposalVotes { + let votes = query_proposal_votes(client, proposal_id).await.unwrap(); + + let mut validators_vote: HashMap = HashMap::default(); + let mut validator_voting_power: HashMap = HashMap::default(); + let mut delegators_vote: HashMap = HashMap::default(); + let mut delegator_voting_power: HashMap> = + HashMap::default(); + + for vote in votes { + if vote.is_validator() { + let validator_stake = RPC + .vp() + .pos() + .validator_stake(client, &vote.validator.clone(), &Some(epoch)) + .await + .expect("Validator stake should be present") + .unwrap_or_default(); + + validators_vote.insert(vote.validator.clone(), vote.data); + validator_voting_power.insert(vote.validator, validator_stake); + } else { + let delegator_stake = RPC + .vp() + .pos() + .bond_with_slashing(client, &vote.delegator, &vote.validator, &Some(epoch)) + .await + .expect("Delegator stake should be present"); + + delegators_vote.insert(vote.delegator.clone(), vote.data); + delegator_voting_power + .entry(vote.delegator.clone()) + .or_default() + .insert(vote.validator, delegator_stake); + } + } + + ProposalVotes { + validators_vote, + validator_voting_power, + delegators_vote, + delegator_voting_power, + } +} diff --git a/packages/sdk/lib/src/rpc_client.rs b/packages/sdk/lib/src/rpc_client.rs new file mode 100644 index 0000000000..4327224e3b --- /dev/null +++ b/packages/sdk/lib/src/rpc_client.rs @@ -0,0 +1,156 @@ +use js_sys::JSON::stringify; +use namada_sdk::storage::BlockHeight; +use std::fmt::Debug; +use std::fmt::Display; +use thiserror::Error; +use wasm_bindgen::prelude::*; +use wasm_bindgen::JsCast; +use wasm_bindgen_futures::JsFuture; +use web_sys::Response; + +use namada_sdk::io::Client; +use namada_sdk::queries::EncodedResponseQuery; +use namada_sdk::tendermint::{self, abci::Code}; +use namada_sdk::tendermint_rpc::{ + error::Error as TendermintRpcError, Response as RpcResponse, SimpleRequest, +}; + +#[wasm_bindgen(module = "/src/rpc_client.js")] +extern "C" { + #[wasm_bindgen(catch, js_name = "wasmFetch")] + async fn wasmFetch(url: JsValue, method: JsValue, body: JsValue) -> Result; +} + +#[derive(Clone, Error, Debug)] +pub struct RpcError(String); + +impl RpcError { + pub fn new(msg: &str) -> Self { + RpcError(String::from(msg)) + } +} + +impl Display for RpcError { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + Display::fmt(&self.0, f) + } +} + +impl From for RpcError { + fn from(error: std::io::Error) -> Self { + RpcError::new(&error.to_string()) + } +} + +impl From for RpcError { + fn from(error: namada_sdk::tendermint::Error) -> Self { + RpcError::new(&error.to_string()) + } +} + +impl From for RpcError { + fn from(error: namada_sdk::tendermint_rpc::Error) -> Self { + RpcError::new(&error.to_string()) + } +} + +#[derive(Clone)] +pub struct HttpClient { + url: String, +} + +/// HttpClient implementation using `window.fetch` API. +impl HttpClient { + pub fn new(url: String) -> HttpClient { + HttpClient { url } + } + + async fn fetch(&self, url: &str, method: &str, body: &str) -> Result { + let resp_value = wasmFetch( + JsValue::from_str(url), + JsValue::from_str(method), + JsValue::from_str(body), + ) + .await?; + + let resp: Response = resp_value.dyn_into()?; + JsFuture::from(resp.json().unwrap()).await + } +} + +#[async_trait::async_trait(?Send)] +impl Client for HttpClient { + /// Implementation of the `Client` trait for the `HttpClient` struct. + /// It's used by the Sdk to perform queries to the blockchain. + type Error = RpcError; + + /// Wrapper for tendermint specific abci_query. + /// + /// # Arguments + /// + /// * `path` - path of the resource in the storage. + /// * `data` - query params in the form of bytearray. + /// * `height` - height of the specific blockchain block that we are querying information from. + /// 0 means the latest block. + /// * `prove` - include proofs of the transactions inclusion in the block + async fn request( + &self, + path: String, + data: Option>, + height: Option, + prove: bool, + ) -> Result { + let data = data.unwrap_or_default(); + let height = height + .map(|height| tendermint::block::Height::try_from(height.0)) + .transpose()?; + + let response = self + .abci_query( + Some(std::str::FromStr::from_str(&path).unwrap()), + data, + height, + prove, + ) + .await?; + let response = response.clone(); + + match response.code { + Code::Ok => Ok(EncodedResponseQuery { + data: response.value, + info: response.info, + proof: response.proof, + height: response.height.into(), + }), + Code::Err(code) => Err(RpcError::new(&format!("Error code {}", code))), + } + } + + /// Performs request using fetch API. Maps returned JS object to the `RpcResponse` struct. + /// + /// # Arguments + /// + /// * `request` - request type to be performed. Check `Client` trait for available requests. + async fn perform(&self, request: R) -> Result + where + R: SimpleRequest, + { + let request_body = request.into_json(); + + let response = self + .fetch(&self.url[..], "POST", &request_body) + .await + .map_err(|e| { + let e = stringify(&e).expect("Error to be serializable"); + let e_str: String = e.into(); + // There is no "generic" RpcError, so we have to pick + // one with error msg as an argument. + TendermintRpcError::server(e_str) + })?; + let response_json: String = stringify(&response) + .expect("JS object to be serializable") + .into(); + + Ok(R::Response::from_string(&response_json).unwrap().into()) + } +} diff --git a/packages/sdk/lib/src/rpc_client.ts b/packages/sdk/lib/src/rpc_client.ts new file mode 100644 index 0000000000..77f7a01894 --- /dev/null +++ b/packages/sdk/lib/src/rpc_client.ts @@ -0,0 +1,12 @@ +/* eslint-disable @typescript-eslint/explicit-function-return-type */ +/** + * Small wrapper for fetch to make it easier to pass props + * Called wasmFetch to avoid naming conflict + */ +export async function wasmFetch(url: string, method: string, body: string) { + const res = await fetch(url, { + method, + body, + }); + return res; +} diff --git a/packages/sdk/lib/src/sdk/args.rs b/packages/sdk/lib/src/sdk/args.rs new file mode 100644 index 0000000000..d3cb120f39 --- /dev/null +++ b/packages/sdk/lib/src/sdk/args.rs @@ -0,0 +1,1073 @@ +use std::ops::Deref; +use std::{path::PathBuf, str::FromStr}; + +use namada_sdk::borsh::{BorshDeserialize, BorshSerialize}; +use namada_sdk::collections::HashMap; +use namada_sdk::ibc::core::host::types::identifiers::{ChannelId, PortId}; +use namada_sdk::ibc::IbcShieldingData; +use namada_sdk::masp::partial_deauthorize; +use namada_sdk::masp_primitives::sapling::redjubjub::PrivateKey; +use namada_sdk::masp_primitives::sapling::spend_sig; +use namada_sdk::masp_primitives::transaction::components::sapling; +use namada_sdk::masp_primitives::transaction::components::sapling::builder::{ + BuildParams as BuildParamsTrait, RngBuildParams, StoredBuildParams, +}; +use namada_sdk::masp_primitives::transaction::sighash::{signature_hash, SignableInput}; +use namada_sdk::masp_primitives::transaction::txid::TxIdDigester; +use namada_sdk::masp_primitives::zip32; +use namada_sdk::signing::SigningTxData; +use namada_sdk::tx::data::GasLimit; +use namada_sdk::tx::{Section, Tx}; +use namada_sdk::{ + address::Address, + args::{self, InputAmount, TxExpiration}, + chain::ChainId, + ethereum_events::EthAddress, + key::common::PublicKey, + token::{Amount, DenominatedAmount, NATIVE_MAX_DECIMAL_PLACES}, + TransferSource, +}; +use namada_sdk::{error, masp_primitives, tendermint_rpc}; +use namada_sdk::{ExtendedSpendingKey, PaymentAddress}; +use rand::rngs::OsRng; +use wasm_bindgen::JsError; + +use crate::types::masp::PseudoExtendedKey; + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct RevealPkMsg { + public_key: String, +} + +impl RevealPkMsg { + pub fn new(public_key: String) -> RevealPkMsg { + RevealPkMsg { public_key } + } +} + +#[derive(BorshSerialize, BorshDeserialize, Clone, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct WrapperTxMsg { + token: String, + fee_amount: String, + gas_limit: String, + chain_id: String, + public_key: Option, + memo: Option, + force: Option, +} + +impl WrapperTxMsg { + pub fn new( + token: String, + fee_amount: String, + gas_limit: String, + chain_id: String, + public_key: Option, + memo: Option, + force: Option, + ) -> WrapperTxMsg { + WrapperTxMsg { + token, + fee_amount, + gas_limit, + chain_id, + public_key, + memo, + force, + } + } +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct BondMsg { + source: String, + validator: String, + amount: String, +} + +impl BondMsg { + pub fn new(source: String, validator: String, amount: String) -> BondMsg { + BondMsg { + source, + validator, + amount, + } + } +} + +/// Maps serialized tx_msg into BondTx args. +/// +/// # Arguments +/// +/// * `bond_msg` - Borsh serialized bond_msg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn bond_tx_args(bond_msg: &[u8], tx_msg: &[u8]) -> Result { + let bond_msg = BondMsg::try_from_slice(bond_msg)?; + + let BondMsg { + source, + validator, + amount, + } = bond_msg; + + let source = Address::from_str(&source)?; + let validator = Address::from_str(&validator)?; + let amount = Amount::from_str(&amount, NATIVE_MAX_DECIMAL_PLACES)?; + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::Bond { + tx, + validator, + amount, + source: Some(source), + tx_code_path: PathBuf::from("tx_bond.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct UnbondMsg { + source: String, + validator: String, + amount: String, +} + +impl UnbondMsg { + pub fn new(source: String, validator: String, amount: String) -> UnbondMsg { + UnbondMsg { + source, + validator, + amount, + } + } +} + +/// Maps serialized tx_msg into UnbondTx args. +/// +/// # Arguments +/// +/// * `unbond_msg` - Borsh serialized unbond_msg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn unbond_tx_args(unbond_msg: &[u8], tx_msg: &[u8]) -> Result { + let unbond_msg = UnbondMsg::try_from_slice(unbond_msg)?; + + let UnbondMsg { + source, + validator, + amount, + } = unbond_msg; + + let source = Address::from_str(&source)?; + let validator = Address::from_str(&validator)?; + + let amount = Amount::from_str(&amount, NATIVE_MAX_DECIMAL_PLACES)?; + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::Unbond { + tx, + validator, + amount, + source: Some(source), + tx_code_path: PathBuf::from("tx_unbond.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct WithdrawMsg { + source: String, + validator: String, +} + +impl WithdrawMsg { + pub fn new(source: String, validator: String) -> WithdrawMsg { + WithdrawMsg { source, validator } + } +} + +/// Maps serialized tx_msg into WithdrawTx args. +/// +/// # Arguments +/// +/// * `withdraw_msg` - Borsh serialized withdraw_msg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn withdraw_tx_args(withdraw_msg: &[u8], tx_msg: &[u8]) -> Result { + let withdraw_msg = WithdrawMsg::try_from_slice(withdraw_msg)?; + + let WithdrawMsg { source, validator } = withdraw_msg; + + let source = Address::from_str(&source)?; + let validator = Address::from_str(&validator)?; + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::Withdraw { + tx, + validator, + source: Some(source), + tx_code_path: PathBuf::from("tx_withdraw.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct RedelegateMsg { + owner: String, + source_validator: String, + destination_validator: String, + amount: String, +} + +impl RedelegateMsg { + pub fn new( + owner: String, + source_validator: String, + destination_validator: String, + amount: String, + ) -> RedelegateMsg { + RedelegateMsg { + owner, + source_validator, + destination_validator, + amount, + } + } +} + +/// Maps serialized tx_msg into RedelgationTx args. +/// +/// # Arguments +/// +/// * `redelegate_msg` - Borsh serialized redelegation_msg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn redelegate_tx_args( + redelegate_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let redelegate_msg = RedelegateMsg::try_from_slice(redelegate_msg)?; + + let RedelegateMsg { + owner, + source_validator, + destination_validator, + amount, + } = redelegate_msg; + + let owner = Address::from_str(&owner)?; + let src_validator = Address::from_str(&source_validator)?; + let dest_validator = Address::from_str(&destination_validator)?; + let amount = Amount::from_str(&amount, NATIVE_MAX_DECIMAL_PLACES)?; + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::Redelegate { + tx, + src_validator, + dest_validator, + amount, + owner, + tx_code_path: PathBuf::from("tx_redelegate.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct VoteProposalMsg { + signer: String, + proposal_id: u64, + vote: String, +} + +impl VoteProposalMsg { + pub fn new(signer: String, proposal_id: u64, vote: String) -> VoteProposalMsg { + VoteProposalMsg { + signer, + proposal_id, + vote, + } + } +} + +/// Maps serialized tx_msg into VoteProposalTx args. +/// +/// # Arguments +/// +/// * `vote_proposal_msg` - Borsh serialized vote_proposal_msg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn vote_proposal_tx_args( + vote_proposal_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let vote_proposal_msg = VoteProposalMsg::try_from_slice(vote_proposal_msg)?; + + let VoteProposalMsg { + signer, + proposal_id, + vote, + } = vote_proposal_msg; + let tx = tx_msg_into_args(tx_msg)?; + let voter_address = Address::from_str(&signer)?; + + let args = args::VoteProposal { + tx, + proposal_id, + vote, + voter_address, + tx_code_path: PathBuf::from("tx_vote_proposal.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct ClaimRewardsMsg { + validator: String, + source: Option, +} + +impl ClaimRewardsMsg { + pub fn new(validator: String, source: Option) -> ClaimRewardsMsg { + ClaimRewardsMsg { validator, source } + } +} + +/// Maps serialized tx_msg into ClaimRewardsTx args. +/// +/// # Arguments +/// +/// * `claim_rewards_msg` - Borsh serialized claim_rewards_msg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn claim_rewards_tx_args( + claim_rewards_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let claim_rewards_msg = ClaimRewardsMsg::try_from_slice(claim_rewards_msg)?; + + let ClaimRewardsMsg { validator, source } = claim_rewards_msg; + let tx = tx_msg_into_args(tx_msg)?; + + let validator_address = Address::from_str(&validator)?; + let source_address = source.map(|str| Address::from_str(&str).expect("valid address")); + + let args = args::ClaimRewards { + tx, + validator: validator_address, + source: source_address, + tx_code_path: PathBuf::from("tx_claim_rewards.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct TransferDataMsg { + owner: String, + token: String, + amount: String, +} + +impl TransferDataMsg { + pub fn new(owner: String, token: String, amount: String) -> TransferDataMsg { + TransferDataMsg { + owner, + token, + amount, + } + } +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct TransferMsg { + sources: Vec, + targets: Vec, + shielded_section_hash: Option>, +} + +impl TransferMsg { + pub fn new( + sources: Vec, + targets: Vec, + shielded_section_hash: Option>, + ) -> TransferMsg { + TransferMsg { + sources, + targets, + shielded_section_hash, + } + } +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct TransparentTransferDataMsg { + source: String, + target: String, + token: String, + amount: String, +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct TransparentTransferMsg { + data: Vec, +} + +/// Maps serialized tx_msg into TransferTx args. +/// +/// # Arguments +/// +/// * `transfer_msg` - Borsh serialized TransparentTransferMsg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn transparent_transfer_tx_args( + transfer_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let transfer_msg = TransparentTransferMsg::try_from_slice(transfer_msg)?; + let TransparentTransferMsg { data } = transfer_msg; + + let mut transfer_data: Vec = vec![]; + + for transfer in data { + let source = Address::from_str(&transfer.source)?; + let target = Address::from_str(&transfer.target)?; + let token = Address::from_str(&transfer.token)?; + let denom_amount = + DenominatedAmount::from_str(&transfer.amount).expect("Amount to be valid."); + let amount = InputAmount::Unvalidated(denom_amount); + + transfer_data.push(args::TxTransparentTransferData { + source, + target, + token, + amount, + }); + } + + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::TxTransparentTransfer { + tx, + data: transfer_data, + tx_code_path: PathBuf::from("tx_transfer.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct ShieldedTransferDataMsg { + source: String, + target: String, + token: String, + amount: String, +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct ShieldedTransferMsg { + data: Vec, + gas_spending_key: Option, +} + +/// Maps serialized tx_msg into TxShieldedTransfer args. +/// +/// # Arguments +/// +/// * `shielding_transfer_msg` - Borsh serialized ShieldingTransferMsg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn shielded_transfer_tx_args( + shielded_transfer_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let shielded_transfer_msg = ShieldedTransferMsg::try_from_slice(shielded_transfer_msg)?; + let ShieldedTransferMsg { + data, + gas_spending_key, + } = shielded_transfer_msg; + + let gas_spending_key = gas_spending_key.map(|v| PseudoExtendedKey::decode(v).0); + + let mut shielded_transfer_data: Vec = vec![]; + + for shielded_transfer in data { + let source = PseudoExtendedKey::decode(shielded_transfer.source).0; + let target = PaymentAddress::from_str(&shielded_transfer.target)?; + let token = Address::from_str(&shielded_transfer.token)?; + let denom_amount = + DenominatedAmount::from_str(&shielded_transfer.amount).expect("Amount to be valid."); + let amount = InputAmount::Unvalidated(denom_amount); + + shielded_transfer_data.push(args::TxShieldedTransferData { + source, + target, + token, + amount, + }); + } + + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::TxShieldedTransfer { + data: shielded_transfer_data, + tx, + tx_code_path: PathBuf::from("tx_transfer.wasm"), + // false, we do this manually + disposable_signing_key: false, + gas_spending_key, + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct ShieldingTransferDataMsg { + source: String, + token: String, + amount: String, +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct ShieldingTransferMsg { + target: String, + data: Vec, +} + +/// Maps serialized tx_msg into TxShieldingTransfer args. +/// +/// # Arguments +/// +/// * `shielding_transfer_msg` - Borsh serialized ShieldingTransferMsg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn shielding_transfer_tx_args( + shielding_transfer_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let shielding_transfer_msg = ShieldingTransferMsg::try_from_slice(shielding_transfer_msg)?; + let ShieldingTransferMsg { target, data } = shielding_transfer_msg; + let target = PaymentAddress::from_str(&target)?; + + let mut shielding_transfer_data: Vec = vec![]; + + for shielding_transfer in data { + let source = Address::from_str(&shielding_transfer.source)?; + let token = Address::from_str(&shielding_transfer.token)?; + let denom_amount = + DenominatedAmount::from_str(&shielding_transfer.amount).expect("Amount to be valid."); + let amount = InputAmount::Unvalidated(denom_amount); + + shielding_transfer_data.push(args::TxShieldingTransferData { + source, + token, + amount, + }); + } + + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::TxShieldingTransfer { + data: shielding_transfer_data, + target, + tx, + tx_code_path: PathBuf::from("tx_transfer.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct UnshieldingTransferDataMsg { + target: String, + token: String, + amount: String, +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct UnshieldingTransferMsg { + source: String, + data: Vec, + gas_spending_key: Option, +} + +/// Maps serialized tx_msg into TxUnshieldingTransfer args. +/// +/// # Arguments +/// +/// * `shielding_transfer_msg` - Borsh serialized UnshieldingTransferMsg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn unshielding_transfer_tx_args( + unshielding_transfer_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let unshielding_transfer_msg = + UnshieldingTransferMsg::try_from_slice(unshielding_transfer_msg)?; + let UnshieldingTransferMsg { + source, + data, + gas_spending_key, + } = unshielding_transfer_msg; + let source = PseudoExtendedKey::decode(source).0; + let gas_spending_key = gas_spending_key.map(|v| PseudoExtendedKey::decode(v).0); + let mut unshielding_transfer_data: Vec = vec![]; + + for unshielding_transfer in data { + let target = Address::from_str(&unshielding_transfer.target)?; + let token = Address::from_str(&unshielding_transfer.token)?; + let denom_amount = + DenominatedAmount::from_str(&unshielding_transfer.amount).expect("Amount to be valid."); + let amount = InputAmount::Unvalidated(denom_amount); + + unshielding_transfer_data.push(args::TxUnshieldingTransferData { + target, + token, + amount, + }); + } + + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::TxUnshieldingTransfer { + data: unshielding_transfer_data, + source, + tx, + gas_spending_key, + // false, we do this manually + disposable_signing_key: false, + tx_code_path: PathBuf::from("tx_transfer.wasm"), + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct IbcTransferMsg { + source: String, + receiver: String, + token: String, + amount_in_base_denom: String, + port_id: String, + channel_id: String, + timeout_height: Option, + timeout_sec_offset: Option, + memo: Option, + shielding_data: Option>, +} + +impl IbcTransferMsg { + pub fn new( + source: String, + receiver: String, + token: String, + amount_in_base_denom: String, + port_id: String, + channel_id: String, + timeout_height: Option, + timeout_sec_offset: Option, + memo: Option, + shielding_data: Option>, + ) -> IbcTransferMsg { + IbcTransferMsg { + source, + receiver, + token, + amount_in_base_denom, + port_id, + channel_id, + timeout_height, + timeout_sec_offset, + memo, + shielding_data, + } + } +} + +/// Maps serialized tx_msg into IbcTransferTx args. +/// +/// # Arguments +/// +/// * `ibc_transfer_msg` - Borsh serialized ibc_transfer_msg. +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if the tx_msg can't be deserialized or +/// Rust structs can't be created. +pub fn ibc_transfer_tx_args( + ibc_transfer_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let ibc_transfer_msg = IbcTransferMsg::try_from_slice(ibc_transfer_msg)?; + let IbcTransferMsg { + source, + receiver, + token, + amount_in_base_denom, + port_id, + channel_id, + timeout_height, + timeout_sec_offset, + memo, + shielding_data, + } = ibc_transfer_msg; + + let source_address = Address::from_str(&source)?; + let source = TransferSource::Address(source_address); + let token = Address::from_str(&token)?; + let amount = Amount::from_str(&amount_in_base_denom, 0u8).expect("Amount to be valid."); + // Using InputAmount::Validated because the amount is already in the base + // denom. If Unvalidated is used, the SDK will change the denom based on the + // token address, which complicates knowing which amount to pass to this function. + let amount = InputAmount::Validated(amount.into()); + let port_id = PortId::from_str(&port_id).expect("Port id to be valid"); + let channel_id = ChannelId::from_str(&channel_id).expect("Channel id to be valid"); + let ibc_shielding_data = match shielding_data { + Some(v) => Some(IbcShieldingData::try_from_slice(&v)?), + None => None, + }; + + let tx = tx_msg_into_args(tx_msg)?; + + let args = args::TxIbcTransfer { + tx, + ibc_memo: memo, + ibc_shielding_data, + source, + receiver, + token, + amount, + port_id, + channel_id, + timeout_height, + timeout_sec_offset, + // TODO: false for now + disposable_signing_key: false, + tx_code_path: PathBuf::from("tx_ibc.wasm"), + refund_target: None, + // We do not support ibc unshielding for now + gas_spending_key: None, + }; + + Ok(args) +} + +#[derive(BorshSerialize, BorshDeserialize, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct EthBridgeTransferMsg { + nut: bool, + asset: String, + recipient: String, + sender: String, + amount: String, + fee_amount: String, + fee_payer: Option, + fee_token: String, +} + +pub fn eth_bridge_transfer_tx_args( + eth_bridge_transfer_msg: &[u8], + tx_msg: &[u8], +) -> Result { + let eth_bridge_transfer_msg = EthBridgeTransferMsg::try_from_slice(eth_bridge_transfer_msg)?; + let EthBridgeTransferMsg { + nut, + asset, + recipient, + sender, + amount, + fee_amount, + fee_payer, + fee_token, + } = eth_bridge_transfer_msg; + + let tx = tx_msg_into_args(tx_msg)?; + let asset = EthAddress::from_str(&asset).map_err(|e| JsError::new(&format!("{}", e)))?; + let recipient = + EthAddress::from_str(&recipient).map_err(|e| JsError::new(&format!("{}", e)))?; + let sender = Address::from_str(&sender)?; + let denom_amount = DenominatedAmount::from_str(&amount).expect("Amount to be valid."); + let amount = InputAmount::Unvalidated(denom_amount); + let denom_amount = DenominatedAmount::from_str(&fee_amount).expect("Amount to be valid."); + let fee_amount = InputAmount::Unvalidated(denom_amount); + let fee_payer = fee_payer.map(|v| Address::from_str(&v)).transpose()?; + let fee_token = Address::from_str(&fee_token)?; + let code_path = PathBuf::from("tx_bridge_pool.wasm"); + + let args = args::EthereumBridgePool { + nut, + tx, + asset, + recipient, + sender, + amount, + fee_amount, + fee_payer, + fee_token, + code_path, + }; + + Ok(args) +} + +pub fn tx_args_from_slice(tx_msg_bytes: &[u8]) -> Result { + let args = tx_msg_into_args(tx_msg_bytes)?; + + Ok(args) +} + +/// Maps serialized tx_msg into Tx args. +/// This is common for all tx types. +/// +/// # Arguments +/// +/// * `tx_msg` - Borsh serialized tx_msg. +/// +/// # Errors +/// +/// Returns JsError if token address is invalid. +fn tx_msg_into_args(tx_msg: &[u8]) -> Result { + let tx_msg = WrapperTxMsg::try_from_slice(tx_msg)?; + let WrapperTxMsg { + token, + fee_amount, + gas_limit, + chain_id, + public_key, + memo, + force, + } = tx_msg; + + let token = Address::from_str(&token)?; + + let fee_amount = DenominatedAmount::from_str(&fee_amount) + .unwrap_or_else(|_| panic!("Fee amount has to be valid. Received {}", fee_amount)); + let fee_input_amount = InputAmount::Unvalidated(fee_amount); + + let public_key = match public_key { + Some(v) => { + let pk = PublicKey::from_str(&v)?; + Some(pk) + } + _ => None, + }; + + let signing_keys: Vec = match public_key { + Some(v) => vec![v.clone()], + _ => vec![], + }; + + // Ledger address is not used in the SDK. + // We can leave it as whatever as long as it's valid url. + let ledger_address = tendermint_rpc::Url::from_str("http://notinuse:13337").unwrap(); + + let memo = memo.map(|v| v.as_bytes().to_vec()); + + let force = force.unwrap_or(false); + + let args = args::Tx { + dry_run: false, + dry_run_wrapper: false, + dump_tx: false, + dump_wrapper_tx: false, + force, + broadcast_only: false, + ledger_address, + wallet_alias_force: false, + initialized_account_alias: None, + fee_amount: Some(fee_input_amount), + fee_token: token.clone(), + gas_limit: GasLimit::from_str(&gas_limit).expect("Gas limit to be valid"), + wrapper_fee_payer: None, + output_folder: None, + expiration: TxExpiration::Default, + chain_id: Some(ChainId(chain_id)), + signatures: vec![], + wrapper_signature: None, + signing_keys, + tx_reveal_code_path: PathBuf::from("tx_reveal_pk.wasm"), + use_device: false, + password: None, + memo, + device_transport: Default::default(), + }; + + Ok(args) +} + +pub enum BuildParams { + RngBuildParams(RngBuildParams), + // TODO: HD Wallet support + #[allow(dead_code)] + StoredBuildParams(StoredBuildParams), +} + +pub async fn generate_masp_build_params( + // TODO: those will be needed for HD Wallet support + _spend_len: usize, + _convert_len: usize, + _output_len: usize, + args: &args::Tx, +) -> Result { + // Construct the build parameters that parameterized the Transaction + // authorizations + if args.use_device { + // HD Wallet support + Err(error::Error::Other("Device not supported".into())) + } else { + Ok(BuildParams::RngBuildParams(RngBuildParams::new(OsRng))) + } +} + +// Sign the given transaction's MASP component using real signatures +pub async fn masp_sign( + tx: &mut Tx, + signing_data: &SigningTxData, + mut bparams: T, + xsk: ExtendedSpendingKey, +) -> Result<(), error::Error> +where + T: BuildParamsTrait, +{ + // Get the MASP section that is the target of our signing + if let Some(shielded_hash) = signing_data.shielded_hash { + let mut masp_tx = tx + .get_masp_section(&shielded_hash) + .expect("Expected to find the indicated MASP Transaction") + .clone(); + let masp_builder = tx + .get_masp_builder(&shielded_hash) + .expect("Expected to find the indicated MASP Builder"); + + // Reverse the spend metadata to enable looking up construction + // material + let sapling_inputs = masp_builder.builder.sapling_inputs(); + let mut descriptor_map = vec![0; sapling_inputs.len()]; + for i in 0.. { + if let Some(pos) = masp_builder.metadata.spend_index(i) { + descriptor_map[pos] = i; + } else { + break; + }; + } + + let tx_data = masp_tx.deref(); + + let unauth_tx_data = partial_deauthorize(tx_data).unwrap(); + + let txid_parts = unauth_tx_data.digest(TxIdDigester); + let sighash = signature_hash(&unauth_tx_data, &SignableInput::Shielded, &txid_parts); + + let mut authorizations = HashMap::new(); + for (tx_pos, _) in descriptor_map.iter().enumerate() { + let pk = PrivateKey(zip32::ExtendedSpendingKey::from(xsk).expsk.ask); + let mut rng = OsRng; + + let sig = spend_sig(pk, bparams.spend_alpha(tx_pos), sighash.as_ref(), &mut rng); + + authorizations.insert(tx_pos, sig); + } + + masp_tx = (*masp_tx) + .clone() + .map_authorization::( + (), + MapSaplingSigAuth(authorizations), + ) + .freeze() + .unwrap(); + + tx.remove_masp_section(&shielded_hash); + tx.add_section(Section::MaspTx(masp_tx)); + } + Ok(()) +} + +struct MapSaplingSigAuth(HashMap::AuthSig>); + +impl sapling::MapAuth for MapSaplingSigAuth { + fn map_proof( + &self, + p: ::Proof, + _pos: usize, + ) -> ::Proof { + p + } + + fn map_auth_sig( + &self, + s: ::AuthSig, + pos: usize, + ) -> ::AuthSig { + self.0.get(&pos).cloned().unwrap_or(s) + } + + fn map_authorization(&self, a: sapling::Authorized) -> sapling::Authorized { + a + } +} diff --git a/packages/sdk/lib/src/sdk/events.rs b/packages/sdk/lib/src/sdk/events.rs new file mode 100644 index 0000000000..e1b6ddb91e --- /dev/null +++ b/packages/sdk/lib/src/sdk/events.rs @@ -0,0 +1,151 @@ +use crate::utils::to_js_result; +use serde::Serialize; +use wasm_bindgen::prelude::*; +use web_sys::{CustomEvent, CustomEventInit, WorkerGlobalScope}; + +#[wasm_bindgen] +#[derive(Debug, Serialize)] +pub struct ProgressStart { + name: String, +} + +impl ProgressStart { + pub fn to_json(&self) -> JsValue { + let json = serde_json::to_value(&self).unwrap(); + JsValue::from_str(&json.to_string()) + } +} + +#[wasm_bindgen] +#[derive(Debug, Serialize)] +pub struct ProgressFinish { + name: String, +} + +impl ProgressFinish { + pub fn to_json(&self) -> JsValue { + let json = serde_json::to_value(&self).unwrap(); + JsValue::from_str(&json.to_string()) + } +} + +#[wasm_bindgen] +#[derive(Debug, Serialize)] +pub struct ProgressIncrement { + name: String, + current: usize, + total: usize, +} + +impl ProgressIncrement { + pub fn to_json(&self) -> JsValue { + let json = serde_json::to_value(&self).unwrap(); + JsValue::from_str(&json.to_string()) + } +} + +/// Event constants for use in Rust +pub const SDK_EVENT_PROGRESS_BAR_STARTED: &str = "namada_sdk::progress_bar::started"; +pub const SDK_EVENT_PROGRESS_BAR_INCREMENTED: &str = "namada_sdk::progress_bar::incremented"; +pub const SDK_EVENT_PROGRESS_BAR_FINISHED: &str = "namada_sdk::progress_bar::finished"; + +// This will generate proper enum in TypeScript, the downisde is that we need to copy the values. +// Unfortunately we can't use macros here. +#[wasm_bindgen(typescript_custom_section)] +const SDK_EVENTS: &'static str = r#" +export enum SdkEvents { + ProgressBarStarted = "namada_sdk::progress_bar::started", + ProgressBarIncremented = "namada_sdk::progress_bar::incremented", + ProgressBarFinished = "namada_sdk::progress_bar::finished", +} +"#; + +#[wasm_bindgen] +pub struct SdkEvents {} + +#[wasm_bindgen] +impl SdkEvents { + #[allow(non_snake_case)] + #[wasm_bindgen(getter)] + pub fn ProgressBarStarted() -> String { + SDK_EVENT_PROGRESS_BAR_STARTED.to_string() + } + + #[allow(non_snake_case)] + #[wasm_bindgen(getter)] + pub fn ProgressBarIncremented() -> String { + SDK_EVENT_PROGRESS_BAR_INCREMENTED.to_string() + } + + #[allow(non_snake_case)] + #[wasm_bindgen(getter)] + pub fn ProgressBarFinished() -> String { + SDK_EVENT_PROGRESS_BAR_FINISHED.to_string() + } +} + +pub struct EventDispatcher { + scope: WorkerGlobalScope, +} + +impl EventDispatcher { + pub fn new() -> Self { + let global = js_sys::global(); + // We assume that the scope is always worker global scope + let scope = global.unchecked_into::(); + + Self { scope } + } + + fn dispatch_custom_event(&self, custom_event: CustomEvent) -> Result { + to_js_result( + self.scope + .dispatch_event(&custom_event) + .map_err(|err| JsError::new(&format!("Error dispatching: {:?}", err)))?, + ) + } + + pub fn progress_bar_started(&self, name: String) -> Result { + let start = ProgressStart { name }; + let mut options = CustomEventInit::new(); + options.detail(&start.to_json()); + + let event = CustomEvent::new_with_event_init_dict(SDK_EVENT_PROGRESS_BAR_STARTED, &options) + .unwrap(); + + self.dispatch_custom_event(event) + } + + pub fn progress_bar_incremented( + &self, + name: String, + current: usize, + total: usize, + ) -> Result { + let increment = ProgressIncrement { + name, + current, + total, + }; + let mut options = CustomEventInit::new(); + options.detail(&increment.to_json()); + + let event = + CustomEvent::new_with_event_init_dict(SDK_EVENT_PROGRESS_BAR_INCREMENTED, &options) + .unwrap(); + + self.dispatch_custom_event(event) + } + + pub fn progress_bar_finished(&self, name: String) -> Result { + let finish = ProgressFinish { name }; + let mut options = CustomEventInit::new(); + options.detail(&finish.to_json()); + + let event = + CustomEvent::new_with_event_init_dict(SDK_EVENT_PROGRESS_BAR_FINISHED, &options) + .unwrap(); + + self.dispatch_custom_event(event) + } +} diff --git a/packages/sdk/lib/src/sdk/io.rs b/packages/sdk/lib/src/sdk/io.rs new file mode 100644 index 0000000000..eaa7f7b1ef --- /dev/null +++ b/packages/sdk/lib/src/sdk/io.rs @@ -0,0 +1,67 @@ +use namada_sdk::io::Io; +use wasm_bindgen::JsValue; + +fn read(question: Option<&str>) -> std::io::Result { + match web_sys::window() { + Some(w) => { + let question = question.unwrap_or("Input: "); + let input = w + .prompt_with_message(question) + .expect("Prompt to be defined"); + + input.ok_or(std::io::Error::new( + std::io::ErrorKind::Other, + "Input is null", + )) + } + None => Err(std::io::Error::new(std::io::ErrorKind::Other, "No window")), + } +} + +pub struct WebIo; + +#[async_trait::async_trait(?Send)] +impl Io for WebIo { + /// Print to the console.log + /// + /// # Arguments + /// + /// * `output` - The output to print + fn print(&self, output: impl AsRef) { + web_sys::console::log_1(&output.as_ref().into()); + } + + /// Print to the console.log with newline + /// + /// # Arguments + /// + /// * `output` - The output to print + fn println(&self, output: impl AsRef) { + let js_output: JsValue = format!("{}\n", output.as_ref()).into(); + web_sys::console::log_1(&js_output); + } + + /// Print to the console.error with newline + /// + /// # Arguments + /// + /// * `output` - The output to print + fn eprintln(&self, output: impl AsRef) { + let js_output: JsValue = format!("{}\n", output.as_ref()).into(); + web_sys::console::error_1(&js_output); + } + + /// Read from the prompt + /// + /// **Note**: Works only in the context of a document(WebWorkers and ServiceWorkers are not supported) + async fn read(&self) -> std::io::Result { + read(None) + } + + /// Read from the prompt with a question + /// + /// **Note**: Works only in the context of a document(WebWorkers and ServiceWorkers are not supported) + async fn prompt(&self, question: impl AsRef) -> String { + read(Some(question.as_ref())).unwrap_or(String::from("")) + } +} diff --git a/packages/sdk/lib/src/sdk/masp/masp_node.rs b/packages/sdk/lib/src/sdk/masp/masp_node.rs new file mode 100644 index 0000000000..821f72dc62 --- /dev/null +++ b/packages/sdk/lib/src/sdk/masp/masp_node.rs @@ -0,0 +1,196 @@ +use async_trait::async_trait; +use namada_sdk::{ + borsh::{BorshDeserialize, BorshSerialize}, + masp::{ContextSyncStatus, DispatcherCache, ShieldedUtils}, + masp_proofs::prover::LocalTxProver, + ShieldedWallet, +}; +use wasm_bindgen::{prelude::wasm_bindgen, JsValue}; + +use std::path::PathBuf; + +use crate::utils::to_bytes; + +/// Spend circuit name +pub const SPEND_NAME: &str = "masp-spend.params"; +/// Output circuit name +pub const OUTPUT_NAME: &str = "masp-output.params"; +/// Convert circuit name +pub const CONVERT_NAME: &str = "masp-convert.params"; + +const FILE_NAME: &str = "shielded.dat"; +const TMP_FILE_NAME: &str = "shielded.tmp"; +const SPECULATIVE_FILE_NAME: &str = "speculative_shielded.dat"; +const SPECULATIVE_TMP_FILE_NAME: &str = "speculative_shielded.tmp"; +const CACHE_FILE_NAME: &str = "shielded_sync.cache"; +const CACHE_FILE_TMP_PREFIX: &str = "shielded_sync.cache.tmp"; + +/// Mostly copied from the Namada CLI + +#[derive(Default, Debug, BorshSerialize, BorshDeserialize, Clone)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct NodeShieldedUtils { + #[borsh(skip)] + context_dir: PathBuf, +} + +impl NodeShieldedUtils { + pub async fn new(context_dir: &str) -> ShieldedWallet { + let context_dir = PathBuf::from(context_dir); + + let spend_path = context_dir.join(SPEND_NAME); + let convert_path = context_dir.join(CONVERT_NAME); + let output_path = context_dir.join(OUTPUT_NAME); + + if !(file_exists(spend_path.clone()) + && file_exists(convert_path.clone()) + && file_exists(output_path.clone())) + { + Self::fetch_params(spend_path, SPEND_NAME).await; + Self::fetch_params(convert_path, CONVERT_NAME).await; + Self::fetch_params(output_path, OUTPUT_NAME).await; + } + + let sync_status = if std::fs::read(context_dir.join(SPECULATIVE_FILE_NAME)).is_ok() { + ContextSyncStatus::Speculative + } else { + ContextSyncStatus::Confirmed + }; + + let utils = Self { context_dir }; + + ShieldedWallet { + utils, + sync_status, + ..Default::default() + } + } + + async fn fetch_params(path: PathBuf, name: &str) { + let path = path.to_str().unwrap(); + let response = reqwest::get(format!( + "https://github.com/anoma/masp-mpc/releases/download/namada-trusted-setup/{}", + name + )) + .await + .unwrap(); + let content: Vec = response.bytes().await.unwrap().into(); + let uint8_array = js_sys::Uint8Array::from(&content[..]); + write_file_sync(JsValue::from_str(path), uint8_array.into()).unwrap(); + } +} + +#[async_trait(?Send)] +impl ShieldedUtils for NodeShieldedUtils { + fn local_tx_prover(&self) -> LocalTxProver { + LocalTxProver::with_default_location().expect("unable to load MASP Parameters") + } + + async fn load( + &self, + ctx: &mut ShieldedWallet, + force_confirmed: bool, + ) -> std::io::Result<()> { + let file_name = if force_confirmed { + FILE_NAME + } else { + match ctx.sync_status { + ContextSyncStatus::Confirmed => FILE_NAME, + ContextSyncStatus::Speculative => SPECULATIVE_FILE_NAME, + } + }; + + let path = path_buf_to_js_value(self.context_dir.join(file_name)); + //TODO: change to_bytes to sth more descripive, add "from_bytes" + let bytes = to_bytes(read_file_sync(path).unwrap().into()); + + *ctx = ShieldedWallet { + utils: ctx.utils.clone(), + ..ShieldedWallet::::deserialize(&mut &bytes[..])? + }; + Ok(()) + } + + async fn save(&self, ctx: &ShieldedWallet) -> std::io::Result<()> { + let (tmp_file_name, file_name) = match ctx.sync_status { + ContextSyncStatus::Confirmed => (TMP_FILE_NAME, FILE_NAME), + ContextSyncStatus::Speculative => (SPECULATIVE_TMP_FILE_NAME, SPECULATIVE_FILE_NAME), + }; + + let tmp_path = path_buf_to_js_value(self.context_dir.join(tmp_file_name)); + { + let mut bytes = Vec::new(); + ctx.serialize(&mut bytes) + .expect("cannot serialize shielded context"); + let uint8_array = js_sys::Uint8Array::from(&bytes[..]); + + write_file_sync(tmp_path.clone(), uint8_array.into()).unwrap(); + } + + let new_path = path_buf_to_js_value(self.context_dir.join(file_name)); + renameSync(tmp_path, new_path).unwrap(); + + if let ContextSyncStatus::Confirmed = ctx.sync_status { + unlinkSync(path_buf_to_js_value( + self.context_dir.join(SPECULATIVE_FILE_NAME), + )) + .unwrap(); + } + + Ok(()) + } + + /// Save a cache of data as part of shielded sync if that + /// process gets interrupted. + async fn cache_save(&self, cache: &DispatcherCache) -> std::io::Result<()> { + let tmp_path = path_buf_to_js_value(self.context_dir.join(CACHE_FILE_TMP_PREFIX)); + { + let mut bytes = Vec::new(); + cache.serialize(&mut bytes).expect("cannot serialize cache"); + let uint8_array = js_sys::Uint8Array::from(&bytes[..]); + + write_file_sync(tmp_path.clone(), uint8_array.into()).unwrap(); + } + + let new_path = path_buf_to_js_value(self.context_dir.join(CACHE_FILE_NAME)); + renameSync(tmp_path, new_path).unwrap(); + + Ok(()) + } + + /// Load a cache of data as part of shielded sync if that + /// process gets interrupted. + async fn cache_load(&self) -> std::io::Result { + // TODO: + todo!() + } +} + +fn path_buf_to_js_value(path: PathBuf) -> JsValue { + JsValue::from_str(path.to_str().unwrap()) +} + +fn file_exists(path: PathBuf) -> bool { + exists_sync(path_buf_to_js_value(path)) + .unwrap() + .as_bool() + .unwrap() +} + +#[wasm_bindgen(module = "/src/sdk/masp/masp.node.js")] +extern "C" { + #[wasm_bindgen(catch, js_name = "writeFileSync")] + fn write_file_sync(path: JsValue, content: JsValue) -> Result; + + #[wasm_bindgen(catch, js_name = "readFileSync")] + fn read_file_sync(path: JsValue) -> Result; + + #[wasm_bindgen(catch, js_name = "renameSync")] + fn renameSync(pathA: JsValue, pathB: JsValue) -> Result; + + #[wasm_bindgen(catch, js_name = "unlinkSync")] + fn unlinkSync(path: JsValue) -> Result; + + #[wasm_bindgen(catch, js_name = "existsSync")] + fn exists_sync(path: JsValue) -> Result; +} diff --git a/packages/sdk/lib/src/sdk/masp/masp_web.rs b/packages/sdk/lib/src/sdk/masp/masp_web.rs new file mode 100644 index 0000000000..1c9a060891 --- /dev/null +++ b/packages/sdk/lib/src/sdk/masp/masp_web.rs @@ -0,0 +1,213 @@ +use async_trait::async_trait; +use gloo_utils::format::JsValueSerdeExt; +use namada_sdk::borsh::{BorshDeserialize, BorshSerialize}; +use namada_sdk::masp::{ContextSyncStatus, DispatcherCache, ShieldedUtils}; +use namada_sdk::masp_proofs::prover::LocalTxProver; +use namada_sdk::ShieldedWallet; +use rexie::{Error, ObjectStore, Rexie, TransactionMode}; +use wasm_bindgen::{JsError, JsValue}; + +use crate::utils::to_bytes; + +const DB_PREFIX: &str = "namada_sdk::MASP"; +const SHIELDED_CONTEXT_TABLE: &str = "ShieldedContext"; +const SHIELDED_CONTEXT_KEY_CONFIRMED: &str = "shielded-context-confirmed"; +const SHIELDED_CONTEXT_KEY_SPECULATIVE: &str = "shielded-context-speculative"; +const SHIELDED_CONTEXT_KEY_TEMP: &str = "shielded-context-temp"; + +#[derive(Default, Debug, BorshSerialize, BorshDeserialize, Clone)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct WebShieldedUtils { + spend_param_bytes: Vec, + output_param_bytes: Vec, + convert_param_bytes: Vec, +} + +impl WebShieldedUtils { + pub async fn new( + spend_param_bytes: Vec, + output_param_bytes: Vec, + convert_param_bytes: Vec, + ) -> Result, JsError> { + let utils = Self { + spend_param_bytes, + output_param_bytes, + convert_param_bytes, + }; + + let db = Self::build_database().await?; + + let sync_status = if Self::get_context(&db, false, false).await.is_ok() { + ContextSyncStatus::Speculative + } else { + ContextSyncStatus::Confirmed + }; + + Ok(ShieldedWallet { + utils, + sync_status, + ..Default::default() + }) + } + + fn to_io_err(e: Error) -> std::io::Error { + std::io::Error::new(std::io::ErrorKind::Other, e.to_string()) + } + + pub async fn build_database() -> Result { + let rexie = Rexie::builder(DB_PREFIX) + .version(1) + .add_object_store(ObjectStore::new(SHIELDED_CONTEXT_TABLE)) + .build() + .await?; + + Ok(rexie) + } + + pub async fn set_context( + rexie: &Rexie, + context: JsValue, + confirmed: bool, + cache: bool, + ) -> Result<(), Error> { + //TODO: add readwriteflush + let transaction = + rexie.transaction(&[SHIELDED_CONTEXT_TABLE], TransactionMode::ReadWrite)?; + + let context_store = transaction.store(SHIELDED_CONTEXT_TABLE)?; + + let key = Self::get_key(confirmed, cache); + + context_store + .put(&context, Some(&JsValue::from_str(key))) + .await?; + + Ok(()) + } + + async fn get_context(rexie: &Rexie, confirmed: bool, cache: bool) -> Result { + let transaction = + rexie.transaction(&[SHIELDED_CONTEXT_TABLE], TransactionMode::ReadOnly)?; + + let context_store = transaction.store(SHIELDED_CONTEXT_TABLE)?; + + let key = Self::get_key(confirmed, cache); + + let context = context_store.get(&JsValue::from_str(key)).await?; + + Ok(context) + } + + async fn remove_speculative_context(rexie: &Rexie) -> Result<(), Error> { + let transaction = + rexie.transaction(&[SHIELDED_CONTEXT_TABLE], TransactionMode::ReadWrite)?; + + let context_store = transaction.store(SHIELDED_CONTEXT_TABLE)?; + + context_store + .delete(&JsValue::from_str(SHIELDED_CONTEXT_KEY_SPECULATIVE)) + .await?; + + Ok(()) + } + + fn get_key(force_confirmed: bool, cache: bool) -> &'static str { + if cache { + SHIELDED_CONTEXT_KEY_TEMP + } else if force_confirmed { + SHIELDED_CONTEXT_KEY_CONFIRMED + } else { + SHIELDED_CONTEXT_KEY_SPECULATIVE + } + } +} + +fn get_confirmed(status: &ContextSyncStatus) -> bool { + match status { + ContextSyncStatus::Confirmed => true, + ContextSyncStatus::Speculative => false, + } +} + +#[async_trait(?Send)] +impl ShieldedUtils for WebShieldedUtils { + fn local_tx_prover(&self) -> LocalTxProver { + LocalTxProver::from_bytes( + &self.spend_param_bytes, + &self.output_param_bytes, + &self.convert_param_bytes, + ) + } + + async fn load( + &self, + ctx: &mut ShieldedWallet, + force_confirmed: bool, + ) -> std::io::Result<()> { + let db = Self::build_database().await.map_err(Self::to_io_err)?; + let confirmed = force_confirmed || get_confirmed(&ctx.sync_status); + + let stored_ctx = Self::get_context(&db, confirmed, false) + .await + .map_err(Self::to_io_err)?; + let stored_ctx_bytes = to_bytes(stored_ctx); + + let context: ShieldedWallet = if stored_ctx_bytes.is_empty() { + ShieldedWallet::default() + } else { + ShieldedWallet::deserialize(&mut &stored_ctx_bytes[..])? + }; + + *ctx = ShieldedWallet { + utils: ctx.utils.clone(), + ..context + }; + + Ok(()) + } + + async fn save(&self, ctx: &ShieldedWallet) -> std::io::Result<()> { + let mut bytes = Vec::new(); + ctx.serialize(&mut bytes) + .expect("cannot serialize shielded context"); + let db = Self::build_database().await.map_err(Self::to_io_err)?; + let confirmed = get_confirmed(&ctx.sync_status); + + Self::set_context(&db, JsValue::from_serde(&bytes).unwrap(), confirmed, false) + .await + .map_err(Self::to_io_err)?; + + if let ContextSyncStatus::Confirmed = ctx.sync_status { + Self::remove_speculative_context(&db) + .await + .map_err(Self::to_io_err)?; + } + + Ok(()) + } + + /// Save a cache of data as part of shielded sync if that + /// process gets interrupted. + async fn cache_save(&self, cache: &DispatcherCache) -> std::io::Result<()> { + let mut bytes = Vec::new(); + cache.serialize(&mut bytes).expect("cannot serialize cache"); + + let db = Self::build_database().await.map_err(Self::to_io_err)?; + Self::set_context(&db, JsValue::from_serde(&bytes).unwrap(), false, true) + .await + .map_err(Self::to_io_err) + } + + /// Load a cache of data as part of shielded sync if that + /// process gets interrupted. + async fn cache_load(&self) -> std::io::Result { + let db = Self::build_database().await.map_err(Self::to_io_err)?; + + let stored_cache = Self::get_context(&db, false, true) + .await + .map_err(Self::to_io_err)?; + let stored_cache_bytes = to_bytes(stored_cache); + + DispatcherCache::deserialize(&mut &stored_cache_bytes[..]) + } +} diff --git a/packages/sdk/lib/src/sdk/masp/mod.rs b/packages/sdk/lib/src/sdk/masp/mod.rs new file mode 100644 index 0000000000..4db35e53f6 --- /dev/null +++ b/packages/sdk/lib/src/sdk/masp/mod.rs @@ -0,0 +1,13 @@ +#[cfg(feature = "web")] +mod masp_web; + +#[cfg(feature = "web")] +pub use masp_web::WebShieldedUtils as JSShieldedUtils; + +#[cfg(feature = "nodejs")] +mod masp_node; + +#[cfg(feature = "nodejs")] +pub use masp_node::NodeShieldedUtils as JSShieldedUtils; + +pub mod sync; diff --git a/packages/sdk/lib/src/sdk/masp/sync.rs b/packages/sdk/lib/src/sdk/masp/sync.rs new file mode 100644 index 0000000000..42ca4bc9fa --- /dev/null +++ b/packages/sdk/lib/src/sdk/masp/sync.rs @@ -0,0 +1,139 @@ +use crate::sdk::events::EventDispatcher; +use namada_sdk::control_flow::ShutdownSignal; +use namada_sdk::io::ProgressBar; +use namada_sdk::task_env::{TaskEnvironment, TaskSpawner}; +use tokio::task::LocalSet; + +#[cfg(not(feature = "multicore"))] +mod spawner { + use super::*; + pub struct TaskSpawnerWeb {} + + impl TaskSpawner for TaskSpawnerWeb { + fn spawn_async(&self, fut: F) + where + F: std::future::Future + 'static, + { + tokio::task::spawn_local(fut); + } + + fn spawn_sync(&self, job: F) + where + F: FnOnce() + Send + 'static, + { + job(); + } + } +} + +#[cfg(feature = "multicore")] +mod spawner { + use super::*; + use rayon; + use std::future::Future; + + pub struct TaskSpawnerWeb {} + + impl TaskSpawner for TaskSpawnerWeb { + #[inline] + fn spawn_async(&self, fut: F) + where + F: Future + 'static, + { + tokio::task::spawn_local(fut); + } + + #[inline] + fn spawn_sync(&self, job: F) + where + F: FnOnce() + Send + 'static, + { + rayon::spawn(job); + } + } +} + +pub struct TaskEnvWeb {} + +impl Default for TaskEnvWeb { + fn default() -> Self { + Self::new() + } +} + +impl TaskEnvWeb { + pub fn new() -> Self { + Self {} + } +} + +impl TaskEnvironment for TaskEnvWeb { + type Spawner = spawner::TaskSpawnerWeb; + + async fn run(self, main: M) -> R + where + M: FnOnce(Self::Spawner) -> F, + F: std::future::Future, + { + LocalSet::new() + .run_until(main(spawner::TaskSpawnerWeb {})) + .await + } +} + +// We can't use the real shutdown signal in the browser, so we just use a dummy one. +pub struct ShutdownSignalWeb {} + +impl ShutdownSignal for ShutdownSignalWeb { + async fn wait_for_shutdown(&mut self) {} + + fn received(&mut self) -> bool { + false + } +} + +pub struct ProgressBarWeb { + pub name: String, + pub total: usize, + pub current: usize, +} + +impl ProgressBarWeb { + pub fn new(name: &str) -> Self { + Self { + name: name.to_string(), + total: 0, + current: 0, + } + } +} + +impl ProgressBar for ProgressBarWeb { + fn upper_limit(&self) -> u64 { + self.total as u64 + } + + fn set_upper_limit(&mut self, limit: u64) { + self.total = limit as usize; + let _ = EventDispatcher::new() + .progress_bar_started(self.name.clone()) + .is_ok(); + } + + fn increment_by(&mut self, amount: u64) { + self.current += amount as usize; + let _ = EventDispatcher::new() + .progress_bar_incremented(self.name.clone(), self.current, self.total) + .is_ok(); + } + + fn message(&mut self, message: String) { + web_sys::console::log_1(&message.into()); + } + + fn finish(&mut self) { + let _ = EventDispatcher::new() + .progress_bar_finished(self.name.clone()) + .is_ok(); + } +} diff --git a/packages/sdk/lib/src/sdk/mod.rs b/packages/sdk/lib/src/sdk/mod.rs new file mode 100644 index 0000000000..69493b1ed5 --- /dev/null +++ b/packages/sdk/lib/src/sdk/mod.rs @@ -0,0 +1,752 @@ +mod args; +pub mod events; +pub mod io; +pub mod masp; +mod signature; +mod transaction; +mod tx; +mod wallet; + +use self::io::WebIo; +use crate::rpc_client::HttpClient; +use crate::utils::set_panic_hook; +#[cfg(feature = "web")] +use crate::utils::to_bytes; +use crate::utils::to_js_result; +use args::{generate_masp_build_params, masp_sign, BuildParams}; +use gloo_utils::format::JsValueSerdeExt; +use namada_sdk::address::{Address, MASP}; +use namada_sdk::args::{GenIbcShieldingTransfer, InputAmount, Query, TxExpiration}; +use namada_sdk::borsh::{self, BorshDeserialize}; +use namada_sdk::eth_bridge::bridge_pool::build_bridge_pool_tx; +use namada_sdk::hash::Hash; +use namada_sdk::ibc::convert_masp_tx_to_ibc_memo; +use namada_sdk::ibc::core::host::types::identifiers::{ChannelId, PortId}; +use namada_sdk::io::NamadaIo; +use namada_sdk::key::{common, ed25519, SigScheme}; +use namada_sdk::masp::ShieldedContext; +use namada_sdk::masp_primitives::transaction::components::sapling::fees::InputView; +use namada_sdk::masp_primitives::zip32::{ExtendedFullViewingKey, ExtendedKey}; +use namada_sdk::rpc::{query_epoch, InnerTxResult}; +use namada_sdk::signing::SigningTxData; +use namada_sdk::string_encoding::Format; +use namada_sdk::tendermint_rpc::Url; +use namada_sdk::token::DenominatedAmount; +use namada_sdk::token::{MaspTxId, OptionExt}; +use namada_sdk::tx::{ + build_batch, build_bond, build_claim_rewards, build_ibc_transfer, build_redelegation, + build_reveal_pk, build_shielded_transfer, build_shielding_transfer, build_transparent_transfer, + build_unbond, build_unshielding_transfer, build_vote_proposal, build_withdraw, + data::compute_inner_tx_hash, either::Either, gen_ibc_shielding_transfer, process_tx, + ProcessTxResponse, Tx, +}; +use namada_sdk::wallet::{Store, Wallet}; +use namada_sdk::{Namada, NamadaImpl, PaymentAddress, TransferTarget}; +use std::collections::BTreeMap; +use std::str::FromStr; +use tx::MaspSigningData; +use wasm_bindgen::{prelude::wasm_bindgen, JsError, JsValue}; + +// Maximum number of spend description randomness parameters that can be +// generated on the hardware wallet. It is hard to compute the exact required +// number because a given MASP source could be distributed amongst several +// notes. +const MAX_HW_SPEND: usize = 15; +// Maximum number of convert description randomness parameters that can be +// generated on the hardware wallet. It is hard to compute the exact required +// number because the number of conversions that are used depends on the +// protocol's current state. +const MAX_HW_CONVERT: usize = 15; +// Maximum number of output description randomness parameters that can be +// generated on the hardware wallet. It is hard to compute the exact required +// number because the number of outputs depends on the number of dummy outputs +// introduced. +const MAX_HW_OUTPUT: usize = 15; + +/// Represents the Sdk public API. +#[wasm_bindgen] +pub struct Sdk { + namada: NamadaImpl, + rpc_url: String, +} + +#[wasm_bindgen] +/// Sdk mostly wraps the logic of the Sdk struct members, making it a part of public API. +/// For more details, navigate to the corresponding modules. +impl Sdk { + #[wasm_bindgen(constructor)] + pub fn new(url: String, native_token: String, path_or_db_name: String) -> Self { + set_panic_hook(); + let client: HttpClient = HttpClient::new(url.clone()); + let wallet: Wallet = Wallet::new( + wallet::JSWalletUtils::new_utils(&path_or_db_name), + Store::default(), + ); + let shielded_ctx: ShieldedContext = ShieldedContext::default(); + + let namada = NamadaImpl::native_new( + client, + wallet, + shielded_ctx.into(), + WebIo, + //NAM address + Address::from_str(&native_token).unwrap(), + ); + + Sdk { + namada, + rpc_url: url, + } + } + + pub async fn has_masp_params() -> Result { + let has = has_masp_params().await?; + + Ok(js_sys::Boolean::from(has.as_bool().unwrap()).into()) + } + + pub async fn fetch_and_store_masp_params(url: Option) -> Result<(), JsValue> { + fetch_and_store_masp_params(url).await?; + Ok(()) + } + + #[cfg(feature = "web")] + pub async fn load_masp_params(&self, _db_name: JsValue) -> Result<(), JsValue> { + // _dn_name is not used in the web version for a time being + let params = get_masp_params().await?; + let params_iter = js_sys::try_iter(¶ms)?.ok_or("Can't iterate over JsValue")?; + let mut params_bytes = params_iter.map(|p| to_bytes(p.unwrap())); + + let spend = params_bytes.next().unwrap(); + let output = params_bytes.next().unwrap(); + let convert = params_bytes.next().unwrap(); + + // We are making sure that there are no more params left + assert_eq!(params_bytes.next(), None); + + let mut shielded = self.namada.shielded_mut().await; + *shielded = ShieldedContext::new(masp::JSShieldedUtils::new(spend, output, convert).await?); + + Ok(()) + } + + #[cfg(feature = "nodejs")] + pub async fn load_masp_params(&self, context_dir: JsValue) -> Result<(), JsValue> { + let context_dir = context_dir.as_string().unwrap(); + + let mut shielded = self.namada.shielded_mut().await; + *shielded = ShieldedContext::new(masp::JSShieldedUtils::new(&context_dir).await); + + Ok(()) + } + + pub async fn add_spending_key(&self, xsk: String, alias: String) { + let mut wallet = self.namada.wallet_mut().await; + wallet::add_spending_key(&mut wallet, xsk, alias) + } + + pub async fn add_viewing_key(&self, xvk: String, alias: String) { + let mut wallet = self.namada.wallet_mut().await; + wallet::add_viewing_key(&mut wallet, xvk, alias) + } + + pub async fn add_payment_address(&self, pa: String, alias: String) { + let mut wallet = self.namada.wallet_mut().await; + wallet::add_payment_address(&mut wallet, pa, alias) + } + + pub async fn add_default_payment_address(&self, xvk: String, alias: String) { + let mut wallet = self.namada.wallet_mut().await; + wallet::add_default_payment_address(&mut wallet, xvk, alias) + } + + pub async fn add_keypair(&self, secret_key: String, alias: String, password: Option) { + let mut wallet = self.namada.wallet_mut().await; + wallet::add_keypair(&mut wallet, secret_key, alias, password) + } + + pub async fn save_wallet(&self) -> Result<(), JsValue> { + let wallet = self.namada.wallet_mut().await; + wallet.save().map_err(JsError::from)?; + + Ok(()) + } + + pub async fn load_wallet(&self) -> Result<(), JsValue> { + let mut wallet = self.namada.wallet_mut().await; + wallet.load().map_err(JsError::from)?; + + Ok(()) + } + + pub async fn sign_masp(&self, xsks: Box<[String]>, tx: Vec) -> Result { + let tx: tx::Tx = borsh::from_slice(&tx)?; + let mut namada_tx: Tx = borsh::from_slice(&tx.tx_bytes())?; + + // Use keys_map to easily map xfvk to xsk + let mut keys_map = BTreeMap::new(); + for xsk_s in xsks.iter() { + let xsk = namada_sdk::ExtendedSpendingKey::from_str(xsk_s)?; + let xvk = xsk.to_viewing_key(); + let xfvk = ExtendedFullViewingKey::from(xvk); + + keys_map.insert(xfvk, xsk); + } + + for signing_data in tx.signing_data() { + if let Some(masp_signing_data) = signing_data.masp() { + let masp_signing_data = + borsh::from_slice::(&masp_signing_data).ok(); + + if let Some(masp_signing_data) = masp_signing_data { + let signing_tx_data = signing_data.to_signing_tx_data()?; + let bparams = masp_signing_data.bparams(); + + for xfvk in masp_signing_data.xfvks() { + let xsk = keys_map.get(&xfvk).ok_or_err_msg("Can't map xfvk to xsk")?; + masp_sign(&mut namada_tx, &signing_tx_data, bparams.clone(), *xsk).await?; + } + } + } + } + + let signing_data = tx + .signing_tx_data()? + .iter() + .cloned() + .map(|std| (std, None)) + .collect::)>>(); + + // Recreate the tx with the new signatures, we can pass None for masp_signing_data as it + // was already used + let tx = tx::Tx::new(namada_tx, &borsh::to_vec(&tx.args())?, signing_data)?; + + to_js_result(borsh::to_vec(&tx)?) + } + + pub async fn sign_tx( + &self, + tx: Vec, + private_key: Option, + chain_id: Option, + ) -> Result { + let tx: tx::Tx = borsh::from_slice(&tx)?; + let mut namada_tx: Tx = borsh::from_slice(&tx.tx_bytes())?; + + // If chain_id is provided, validate this against value in Tx header + if let Some(c) = chain_id { + if c != namada_tx.header.chain_id.to_string() { + return Err(JsError::new(&format!( + "chain_id {} does not match Tx header chain_id {}", + &c, + namada_tx.header.chain_id.as_str() + ))); + } + } + + let signing_keys = match private_key.clone() { + Some(private_key) => vec![common::SecretKey::Ed25519(ed25519::SecretKey::from_str( + &private_key, + )?)], + // If no private key is provided, we assume masp source and return empty vec + None => vec![], + }; + + for signing_tx_data in tx.signing_tx_data()? { + if let Some(account_public_keys_map) = signing_tx_data.account_public_keys_map.clone() { + // We only sign the raw header for transfers from transparent source + if !signing_keys.is_empty() { + // Sign the raw header + namada_tx.sign_raw( + signing_keys.clone(), + account_public_keys_map, + signing_tx_data.owner.clone(), + ); + } + } + } + + // The key is either passed private key for transparent sources or the disposable signing + // key for shielded sources + let key = signing_keys[0].clone(); + + // Sign the fee header + namada_tx.sign_wrapper(key); + + to_js_result(borsh::to_vec(&namada_tx)?) + } + + // Broadcast Tx + pub async fn process_tx(&self, tx_bytes: &[u8], tx_msg: &[u8]) -> Result { + let args = args::tx_args_from_slice(tx_msg)?; + let tx = Tx::try_from_slice(tx_bytes)?; + let cmts = tx.commitments().clone(); + let wrapper_hash = tx.wrapper_hash(); + let resp = process_tx(&self.namada, &args, tx.clone()).await?; + + let mut batch_tx_results: Vec = vec![]; + + // Collect results and return + match resp { + ProcessTxResponse::Applied(tx_response) => { + let code = tx_response.code.to_string(); + let gas_used = tx_response.gas_used.to_string(); + let height = tx_response.height.to_string(); + let info = tx_response.info.to_string(); + let log = tx_response.log.to_string(); + + for cmt in cmts { + let hash = compute_inner_tx_hash(wrapper_hash.as_ref(), Either::Right(&cmt)); + + if let Some(InnerTxResult::Success(_)) = tx_response.batch_result().get(&hash) { + batch_tx_results.push(tx::BatchTxResult::new(hash.to_string(), true)); + } else { + batch_tx_results.push(tx::BatchTxResult::new(hash.to_string(), false)); + } + } + + let response = tx::TxResponse::new( + code, + batch_tx_results, + gas_used, + wrapper_hash.unwrap().to_string(), + height, + info, + log, + ); + to_js_result(borsh::to_vec(&response)?) + } + _ => Err(JsError::new(&format!( + "Tx not applied: {}", + &wrapper_hash.unwrap().to_string() + ))), + } + } + + /// Build a batch Tx from built transactions and return the bytes + pub fn build_batch(txs: JsValue) -> Result { + let mut built_txs: Vec = vec![]; + let built_txs_bytes: Vec> = txs.into_serde().unwrap(); + + for bytes in built_txs_bytes.iter() { + let tx: tx::Tx = borsh::from_slice(bytes)?; + built_txs.push(tx); + } + + // Get wrapper args + let first_tx = built_txs + .first() + .expect("At least one Tx is required for building batches!"); + + let args = first_tx.args(); + + let mut txs: Vec<(Tx, SigningTxData)> = vec![]; + let mut masp_map: BTreeMap = BTreeMap::new(); + + // Iterate through provided tx::Tx and deserialize bytes to Namada Tx + for built_tx in built_txs.into_iter() { + let tx_bytes = built_tx.tx_bytes(); + let signing_data = built_tx.signing_data(); + let first_signing_data = signing_data + .first() + .expect("At least one signing data should be present on a Tx"); + + let signing_tx_data = first_signing_data.to_signing_tx_data()?; + + if let Some(sh) = signing_tx_data.shielded_hash { + let masp_signing_data = first_signing_data.masp(); + + // We do not need to insert masp_signing_data when we shield + if let Some(masp_signing_data) = masp_signing_data { + let masp_signing_data = ::try_from_slice(&masp_signing_data)?; + masp_map.insert(sh, masp_signing_data); + } + } + + let tx: Tx = Tx::try_from_slice(&tx_bytes)?; + + txs.push((tx, signing_tx_data.to_owned())); + } + + let (tx, signing_data) = build_batch(txs.clone())?; + + let signing_data = signing_data + .iter() + .cloned() + .map(|sd| { + if let Some(sh) = sd.shielded_hash { + (sd, masp_map.get(&sh).cloned()) + } else { + (sd, None) + } + }) + .collect::>(); + + to_js_result(borsh::to_vec(&tx::Tx::new( + tx, + &borsh::to_vec(&args)?, + signing_data, + )?)?) + } + + // Append signatures and return tx bytes + pub fn append_signature( + &self, + tx_bytes: &[u8], + sig_msg_bytes: &[u8], + ) -> Result { + let mut tx: Tx = Tx::try_from_slice(tx_bytes)?; + let signature::SignatureMsg { + pubkey, + raw_indices, + raw_signature, + wrapper_indices, + wrapper_signature, + } = signature::SignatureMsg::try_from_slice(sig_msg_bytes)?; + + let raw_sig_section = + signature::construct_signature_section(&pubkey, &raw_indices, &raw_signature, &tx)?; + tx.add_section(raw_sig_section); + + let wrapper_sig_section = signature::construct_signature_section( + &pubkey, + &wrapper_indices, + &wrapper_signature, + &tx, + )?; + tx.add_section(wrapper_sig_section); + + tx.protocol_filter(); + + to_js_result(borsh::to_vec(&tx)?) + } + + pub async fn build_transparent_transfer( + &self, + transfer_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let mut args = args::transparent_transfer_tx_args(transfer_msg, wrapper_tx_msg)?; + let (tx, signing_data) = build_transparent_transfer(&self.namada, &mut args).await?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_shielded_transfer( + &self, + shielded_transfer_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let mut args = args::shielded_transfer_tx_args(shielded_transfer_msg, wrapper_tx_msg)?; + let bparams = + generate_masp_build_params(MAX_HW_SPEND, MAX_HW_CONVERT, MAX_HW_OUTPUT, &args.tx) + .await?; + + let _ = &self.namada.shielded_mut().await.load().await?; + + let xfvks = args + .data + .iter() + .map(|data| data.source.to_viewing_key()) + .collect::>(); + + let ((tx, signing_data), masp_signing_data) = match bparams { + BuildParams::RngBuildParams(mut bparams) => { + let tx = build_shielded_transfer(&self.namada, &mut args, &mut bparams).await?; + let masp_signing_data = MaspSigningData::new( + bparams + .to_stored() + .ok_or_err_msg("Cannot convert bparams to stored")?, + xfvks, + ); + + (tx, masp_signing_data) + } + BuildParams::StoredBuildParams(mut bparams) => { + let tx = build_shielded_transfer(&self.namada, &mut args, &mut bparams).await?; + let masp_signing_data = MaspSigningData::new(bparams, xfvks); + + (tx, masp_signing_data) + } + }; + + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, Some(masp_signing_data)) + } + + pub async fn build_unshielding_transfer( + &self, + unshielding_transfer_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let mut args = + args::unshielding_transfer_tx_args(unshielding_transfer_msg, wrapper_tx_msg)?; + let bparams = + generate_masp_build_params(MAX_HW_SPEND, MAX_HW_CONVERT, MAX_HW_OUTPUT, &args.tx) + .await?; + + let _ = &self.namada.shielded_mut().await.load().await?; + + let xfvks = vec![args.source.to_viewing_key()]; + + let ((tx, signing_data), masp_signing_data) = match bparams { + BuildParams::RngBuildParams(mut bparams) => { + let tx = build_unshielding_transfer(&self.namada, &mut args, &mut bparams).await?; + let masp_signing_data = MaspSigningData::new( + bparams + .to_stored() + .ok_or_err_msg("Cannot convert bparams to stored")?, + xfvks, + ); + + (tx, masp_signing_data) + } + BuildParams::StoredBuildParams(mut bparams) => { + let tx = build_unshielding_transfer(&self.namada, &mut args, &mut bparams).await?; + let masp_signing_data = MaspSigningData::new(bparams, xfvks); + + (tx, masp_signing_data) + } + }; + + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, Some(masp_signing_data)) + } + + pub async fn build_shielding_transfer( + &self, + shielding_transfer_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let mut args = args::shielding_transfer_tx_args(shielding_transfer_msg, wrapper_tx_msg)?; + let bparams = + generate_masp_build_params(MAX_HW_SPEND, MAX_HW_CONVERT, MAX_HW_OUTPUT, &args.tx) + .await?; + + let (tx, signing_data, _) = match bparams { + BuildParams::RngBuildParams(mut bparams) => { + build_shielding_transfer(&self.namada, &mut args, &mut bparams).await? + } + BuildParams::StoredBuildParams(mut bparams) => { + build_shielding_transfer(&self.namada, &mut args, &mut bparams).await? + } + }; + + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_ibc_transfer( + &self, + ibc_transfer_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let args = args::ibc_transfer_tx_args(ibc_transfer_msg, wrapper_tx_msg)?; + let bparams = + generate_masp_build_params(MAX_HW_SPEND, MAX_HW_CONVERT, MAX_HW_OUTPUT, &args.tx) + .await?; + + let ((tx, signing_data, _), bparams) = match bparams { + BuildParams::RngBuildParams(mut bparams) => { + let tx = build_ibc_transfer(&self.namada, &args, &mut bparams).await?; + let bparams = bparams + .to_stored() + .ok_or_err_msg("Cannot convert bparams to stored")?; + + (tx, bparams) + } + BuildParams::StoredBuildParams(mut bparams) => { + let tx = build_ibc_transfer(&self.namada, &args, &mut bparams).await?; + + (tx, bparams) + } + }; + + // As we can't get ExtendedFullViewingKeys from the tx args we need to get them from the + // MASP Builder section of transaction + let masp_signing_data = if let Some(shielded_hash) = signing_data.shielded_hash { + let masp_builder = tx + .get_masp_builder(&shielded_hash) + .ok_or_err_msg("Expected to find the indicated MASP Builder")?; + let xfvks = masp_builder + .builder + .sapling_inputs() + .iter() + .map(|input| input.key()) + .cloned() + .collect::>(); + + let masp_signing_data = MaspSigningData::new(bparams, xfvks); + + Some(masp_signing_data) + } else { + None + }; + + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, masp_signing_data) + } + + pub async fn build_eth_bridge_transfer( + &self, + eth_bridge_transfer_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let args = args::eth_bridge_transfer_tx_args(eth_bridge_transfer_msg, wrapper_tx_msg)?; + let (tx, signing_data) = build_bridge_pool_tx(&self.namada, args.clone()).await?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_vote_proposal( + &self, + vote_proposal_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let args = args::vote_proposal_tx_args(vote_proposal_msg, wrapper_tx_msg)?; + let epoch = query_epoch(self.namada.client()).await?; + let (tx, signing_data) = build_vote_proposal(&self.namada, &args, epoch) + .await + .map_err(JsError::from)?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_claim_rewards( + &self, + claim_rewards_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let args = args::claim_rewards_tx_args(claim_rewards_msg, wrapper_tx_msg)?; + let (tx, signing_data) = build_claim_rewards(&self.namada, &args) + .await + .map_err(JsError::from)?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_bond( + &self, + bond_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let args = args::bond_tx_args(bond_msg, wrapper_tx_msg)?; + let (tx, signing_data) = build_bond(&self.namada, &args).await?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_unbond( + &self, + unbond_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let args = args::unbond_tx_args(unbond_msg, wrapper_tx_msg)?; + let (tx, signing_data, _) = build_unbond(&self.namada, &args).await?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_withdraw( + &self, + withdraw_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let args = args::withdraw_tx_args(withdraw_msg, wrapper_tx_msg)?; + let (tx, signing_data) = build_withdraw(&self.namada, &args).await?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_redelegate( + &self, + redelegate_msg: &[u8], + wrapper_tx_msg: &[u8], + ) -> Result { + let args = args::redelegate_tx_args(redelegate_msg, wrapper_tx_msg)?; + let (tx, signing_data) = build_redelegation(&self.namada, &args).await?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + pub async fn build_reveal_pk(&self, wrapper_tx_msg: &[u8]) -> Result { + let args = args::tx_args_from_slice(wrapper_tx_msg)?; + let public_key = args.signing_keys[0].clone(); + let (tx, signing_data) = build_reveal_pk(&self.namada, &args.clone(), &public_key).await?; + self.serialize_tx_result(tx, wrapper_tx_msg, signing_data, None) + } + + // Sign arbitrary data with the provided signing key + pub fn sign_arbitrary(&self, signing_key: String, data: String) -> Result { + let hash = Hash::sha256(data); + let secret = common::SecretKey::Ed25519(ed25519::SecretKey::from_str(&signing_key)?); + let signature = common::SigScheme::sign(&secret, hash); + let sig_bytes = signature.to_bytes(); + + to_js_result((hash.to_string().to_lowercase(), hex::encode(sig_bytes))) + } + + // Verify signed arbitrary data + pub fn verify_arbitrary( + &self, + public_key: String, + signed_hash: String, + signature: String, + ) -> Result<(), JsError> { + let public_key = common::PublicKey::from_str(&public_key)?; + let sig = common::Signature::try_from_slice(&hex::decode(signature)?)?; + let signed_hash = Hash::from_str(&signed_hash)?; + + common::SigScheme::verify_signature(&public_key, &signed_hash, &sig).map_err(JsError::from) + } + + pub async fn generate_ibc_shielding_memo( + &self, + target: &str, + token: String, + amount: &str, + channel_id: &str, + ) -> Result { + let ledger_address = Url::from_str(&self.rpc_url).expect("RPC URL is a valid URL"); + let target = TransferTarget::PaymentAddress( + PaymentAddress::from_str(target).expect("target is a valid shielded address"), + ); + let amount = + InputAmount::Unvalidated(DenominatedAmount::from_str(amount).expect("amount is valid")); + let channel_id = ChannelId::from_str(channel_id).expect("channel ID is valid"); + + let args = GenIbcShieldingTransfer { + query: Query { ledger_address }, + output_folder: None, + target, + token, + amount, + port_id: PortId::transfer(), + channel_id, + expiration: TxExpiration::Default, + }; + + if let Some(masp_tx) = gen_ibc_shielding_transfer(&self.namada, args).await? { + let memo = convert_masp_tx_to_ibc_memo(&masp_tx); + to_js_result(memo) + } else { + Err(JsError::new( + "Generating ibc shielding transfer generated nothing", + )) + } + } + + pub fn masp_address(&self) -> String { + MASP.to_string() + } + + fn serialize_tx_result( + &self, + tx: Tx, + wrapper_tx_msg: &[u8], + signing_data: SigningTxData, + masp_signing_data: Option, + ) -> Result { + let tx = tx::Tx::new(tx, wrapper_tx_msg, vec![(signing_data, masp_signing_data)])?; + + to_js_result(borsh::to_vec(&tx)?) + } +} + +#[wasm_bindgen(module = "/src/sdk/mod.js")] +extern "C" { + #[wasm_bindgen(catch, js_name = "getMaspParams")] + async fn get_masp_params() -> Result; + #[wasm_bindgen(catch, js_name = "hasMaspParams")] + async fn has_masp_params() -> Result; + #[wasm_bindgen(catch, js_name = "fetchAndStoreMaspParams")] + async fn fetch_and_store_masp_params(url: Option) -> Result; +} diff --git a/packages/sdk/lib/src/sdk/mod.ts b/packages/sdk/lib/src/sdk/mod.ts new file mode 100644 index 0000000000..9388d215dc --- /dev/null +++ b/packages/sdk/lib/src/sdk/mod.ts @@ -0,0 +1,226 @@ +const PREFIX = "Namada::SDK"; +const MASP_MPC_RELEASE_URL = + "https://github.com/anoma/masp-mpc/releases/download/namada-trusted-setup/"; + +const sha256Hash = async (msg: Uint8Array): Promise => { + const hashBuffer = await crypto.subtle.digest("SHA-256", msg); + const hashArray = Array.from(new Uint8Array(hashBuffer)); + // Return hash as hex + return hashArray.map((byte) => byte.toString(16).padStart(2, "0")).join(""); +}; + +enum MaspParam { + Output = "masp-output.params", + Convert = "masp-convert.params", + Spend = "masp-spend.params", +} + +type MaspParamBytes = { + param: MaspParam; + bytes: Uint8Array; +}; + +/** + * The following sha256 digests where produced by downloading the following: + * https://github.com/anoma/masp-mpc/releases/download/namada-trusted-setup/masp-convert.params + * https://github.com/anoma/masp-mpc/releases/download/namada-trusted-setup/masp-spend.params + * https://github.com/anoma/masp-mpc/releases/download/namada-trusted-setup/masp-output.params + * + * And running "sha256sum" against each file: + * + * > sha256sum masp-convert.params + * 8e049c905e0e46f27662c7577a4e3480c0047ee1171f7f6d9c5b0de757bf71f1 masp-convert.params + * + * > sha256sum masp-spend.params + * 62b3c60ca54bd99eb390198e949660624612f7db7942db84595fa9f1b4a29fd8 masp-spend.params + * + * > sha256sum masp-output.params + * ed8b5d354017d808cfaf7b31eca5c511936e65ef6d276770251f5234ec5328b8 masp-output.params + * + * Length is specified in bytes, and can be retrieved with: + * + * > wc -c < masp-convert.params + * 22570940 + * > wc -c < masp-spend.params + * 49848572 + * > wc -c < masp-output.params + * 16398620 + */ +const MASP_PARAM_ATTR: Record< + MaspParam, + { length: number; sha256sum: string } +> = { + [MaspParam.Output]: { + length: 16398620, + sha256sum: + "ed8b5d354017d808cfaf7b31eca5c511936e65ef6d276770251f5234ec5328b8", + }, + [MaspParam.Spend]: { + length: 49848572, + sha256sum: + "62b3c60ca54bd99eb390198e949660624612f7db7942db84595fa9f1b4a29fd8", + }, + [MaspParam.Convert]: { + length: 22570940, + sha256sum: + "8e049c905e0e46f27662c7577a4e3480c0047ee1171f7f6d9c5b0de757bf71f1", + }, +}; + +const validateMaspParamBytes = async ({ + param, + bytes, +}: MaspParamBytes): Promise => { + const { length, sha256sum } = MASP_PARAM_ATTR[param]; + + // Reject if invalid length (incomplete download or invalid) + console.info(`Validating data length for ${param}, expecting ${length}...`); + + if (length !== bytes.length) { + return Promise.reject( + `[${param}]: Invalid data length! Expected ${length}, received ${bytes.length}!` + ); + } + + // Reject if invalid hash (otherwise invalid data) + console.info(`Validating sha256sum for ${param}, expecting ${sha256sum}...`); + const hash = await sha256Hash(bytes); + + if (hash !== sha256sum) { + return Promise.reject( + `[${param}]: Invalid sha256sum! Expected ${sha256sum}, received ${hash}!` + ); + } + + return bytes; +}; + +export async function hasMaspParams(): Promise { + return ( + (await has(MaspParam.Spend)) && + (await has(MaspParam.Output)) && + (await has(MaspParam.Convert)) + ); +} + +export async function fetchAndStoreMaspParams( + url?: string +): Promise<[void, void, void]> { + return Promise.all([ + fetchAndStore(MaspParam.Spend, url), + fetchAndStore(MaspParam.Output, url), + fetchAndStore(MaspParam.Convert, url), + ]); +} + +export async function getMaspParams(): Promise<[unknown, unknown, unknown]> { + return Promise.all([ + get(MaspParam.Spend), + get(MaspParam.Output), + get(MaspParam.Convert), + ]); +} + +export async function fetchAndStore( + param: MaspParam, + url?: string +): Promise { + return await fetchParams(param, url) + .then((data) => set(param, data)) + .catch((e) => { + return Promise.reject(`Encountered errors fetching ${param}: ${e}`); + }); +} + +export async function fetchParams( + param: MaspParam, + url: string = MASP_MPC_RELEASE_URL +): Promise { + return fetch(`${url}${param}`) + .then((response) => response.arrayBuffer()) + .then((ab) => { + const bytes = new Uint8Array(ab); + return validateMaspParamBytes({ param, bytes }); + }); +} + +function getDB(): Promise { + return new Promise((resolve, reject) => { + const request = indexedDB.open(PREFIX); + request.onerror = (event) => { + event.stopPropagation(); + reject(event.target); + }; + + request.onupgradeneeded = (event) => { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const db = (event.target as any).result; + + db.createObjectStore(PREFIX, { keyPath: "key" }); + }; + + request.onsuccess = () => { + resolve(request.result); + }; + }); +} + +export async function get(key: string): Promise { + const tx = (await getDB()).transaction(PREFIX, "readonly"); + const store = tx.objectStore(PREFIX); + + return new Promise((resolve, reject) => { + const request = store.get(key); + request.onerror = (event) => { + event.stopPropagation(); + + reject(event.target); + }; + request.onsuccess = () => { + if (!request.result) { + resolve(undefined); + } else { + resolve(request.result.data); + } + }; + }); +} + +export async function has(key: string): Promise { + const tx = (await getDB()).transaction(PREFIX, "readonly"); + const store = tx.objectStore(PREFIX); + + return new Promise((resolve, reject) => { + const request = store.openCursor(key); + request.onerror = (event) => { + event.stopPropagation(); + + reject(event.target); + }; + request.onsuccess = (e) => { + // eslint-disable-next-line @typescript-eslint/no-explicit-any + const cursor = (e.target as any).result; + resolve(!!cursor); + }; + }); +} + +export async function set(key: string, data: unknown): Promise { + const tx = (await getDB()).transaction(PREFIX, "readwrite"); + const store = tx.objectStore(PREFIX); + + return new Promise((resolve, reject) => { + const request = store.put({ + key, + data, + }); + request.onerror = (event) => { + event.stopPropagation(); + + reject(event.target); + }; + request.onsuccess = () => { + resolve(); + }; + }); +} diff --git a/packages/sdk/lib/src/sdk/signature.rs b/packages/sdk/lib/src/sdk/signature.rs new file mode 100644 index 0000000000..498de142bb --- /dev/null +++ b/packages/sdk/lib/src/sdk/signature.rs @@ -0,0 +1,48 @@ +use namada_sdk::borsh::{BorshDeserialize, BorshSerialize}; +use namada_sdk::{ + key::common::{PublicKey, Signature}, + tx::{CompressedAuthorization, Section, Signer, Tx}, +}; +use std::collections::BTreeMap; +use wasm_bindgen::JsError; + +#[derive(BorshSerialize, BorshDeserialize)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct SignatureMsg { + pub pubkey: Vec, + pub raw_indices: Vec, + pub raw_signature: Vec, + pub wrapper_indices: Vec, + pub wrapper_signature: Vec, +} + +/// Reconstructs a proto::Section signature using the provided indices to retrieve hashes +/// from Tx +/// +/// # Arguments +/// +/// * `pubkey` - Public key bytes +/// * `sec_indices` - Indices indicating hash location +/// * `signature` - Signature bytes +/// * `tx` - A proto::Tx +/// +/// # Errors +/// +/// Returns JsError if the sig_msg can't be deserialized or +/// Rust structs can't be created. +pub fn construct_signature_section( + pubkey: &[u8], + sec_indices: &[u8], + signature: &[u8], + tx: &Tx, +) -> Result { + let signatures = BTreeMap::from([(0, Signature::try_from_slice(signature)?)]); + + let compressed_signature = CompressedAuthorization { + targets: sec_indices.to_vec(), + signer: Signer::PubKeys(vec![PublicKey::try_from_slice(pubkey)?]), + signatures, + }; + + Ok(Section::Authorization(compressed_signature.expand(tx))) +} diff --git a/packages/sdk/lib/src/sdk/transaction.rs b/packages/sdk/lib/src/sdk/transaction.rs new file mode 100644 index 0000000000..ff088178b4 --- /dev/null +++ b/packages/sdk/lib/src/sdk/transaction.rs @@ -0,0 +1,215 @@ +use namada_sdk::borsh::BorshSerializeExt; +use namada_sdk::token::Transfer; + +use namada_sdk::governance::VoteProposalData; +use namada_sdk::ibc::core::channel::types::timeout::TimeoutHeight; +use namada_sdk::ibc::MsgTransfer; +use namada_sdk::tx::data::pos::{Bond, ClaimRewards, Redelegation, Unbond, Withdraw}; +use namada_sdk::{ + borsh::{self, BorshDeserialize}, + key::common::PublicKey, +}; +use wasm_bindgen::JsError; + +use crate::sdk::{ + args::{ + BondMsg, ClaimRewardsMsg, IbcTransferMsg, RedelegateMsg, RevealPkMsg, TransferDataMsg, + TransferMsg, UnbondMsg, VoteProposalMsg, WithdrawMsg, + }, + tx::TxType, +}; + +#[derive(Debug, Clone)] +pub enum TransactionKind { + Transfer(Transfer), + Bond(Bond), + Redelegation(Redelegation), + Unbond(Unbond), + Withdraw(Withdraw), + ProposalVote(VoteProposalData), + ClaimRewards(ClaimRewards), + RevealPk(PublicKey), + IbcTransfer(MsgTransfer), + Unknown, +} + +impl TransactionKind { + pub fn from(tx_type: TxType, data: &[u8]) -> Self { + match tx_type { + TxType::Transfer => TransactionKind::Transfer( + Transfer::try_from_slice(data).expect("Cannot deserialize TransparentTransfer"), + ), + TxType::Bond => { + TransactionKind::Bond(Bond::try_from_slice(data).expect("Cannot deserialize Bond")) + } + TxType::Redelegate => TransactionKind::Redelegation( + Redelegation::try_from_slice(data).expect("Cannot deserialize Redelegation"), + ), + TxType::Unbond => TransactionKind::Unbond( + Unbond::try_from_slice(data).expect("Cannot deserialize Unbond"), + ), + TxType::Withdraw => TransactionKind::Withdraw( + Withdraw::try_from_slice(data).expect("Cannot deserialize Withdraw"), + ), + TxType::VoteProposal => TransactionKind::ProposalVote( + VoteProposalData::try_from_slice(data).expect("Cannot deserialize VoteProposal"), + ), + TxType::ClaimRewards => TransactionKind::ClaimRewards( + ClaimRewards::try_from_slice(data).expect("Cannot deserialize ClaimRewards"), + ), + TxType::RevealPK => TransactionKind::RevealPk( + PublicKey::try_from_slice(data).expect("Cannot deserialize PublicKey"), + ), + TxType::IBCTransfer => TransactionKind::IbcTransfer( + MsgTransfer::try_from_slice(data).expect("Cannot deserialize MsgTransfer"), + ), + _ => TransactionKind::Unknown, + } + } + + // Returns vec of borsh-serialized arguments bytes based on transaction type + pub fn to_bytes(&self) -> Result, JsError> { + let bytes: Vec = match self { + TransactionKind::Bond(bond) => { + let Bond { + amount, + validator, + source, + } = bond; + + if !source.is_some() { + return Err(JsError::new("Bond source must be defined!")); + } + + let bond = BondMsg::new( + source.clone().unwrap().to_string(), + validator.to_string(), + amount.native_denominated().to_string(), + ); + borsh::to_vec(&bond)? + } + TransactionKind::Unbond(unbond) => { + let Unbond { + amount, + validator, + source, + } = unbond; + + if !source.is_some() { + return Err(JsError::new("Unbond source must be defined!")); + } + + let unbond = UnbondMsg::new( + source.clone().unwrap().to_string(), + validator.to_string(), + amount.native_denominated().to_string(), + ); + borsh::to_vec(&unbond)? + } + TransactionKind::Withdraw(withdraw) => { + let Withdraw { validator, source } = withdraw; + + if !source.is_some() { + return Err(JsError::new("Withdraw source must be defined!")); + } + + let withdraw = + WithdrawMsg::new(source.clone().unwrap().to_string(), validator.to_string()); + borsh::to_vec(&withdraw)? + } + TransactionKind::Redelegation(redelegation) => { + let Redelegation { + owner, + src_validator, + dest_validator, + amount, + } = redelegation; + + let redelegation = RedelegateMsg::new( + owner.to_string(), + src_validator.to_string(), + dest_validator.to_string(), + amount.native_denominated().to_string(), + ); + borsh::to_vec(&redelegation)? + } + TransactionKind::RevealPk(public_key) => { + let reveal_pk = RevealPkMsg::new(public_key.to_string()); + borsh::to_vec(&reveal_pk)? + } + TransactionKind::Transfer(transfer) => { + let Transfer { + sources, + targets, + shielded_section_hash, + } = transfer; + + let ssh = match shielded_section_hash { + Some(masp_tx_id) => { + // Serialize and return bytes + let bytes = masp_tx_id.serialize_to_vec(); + Some(bytes) + } + None => None, + }; + + let mut sources_data: Vec = vec![]; + let mut targets_data: Vec = vec![]; + + for (source, amount) in sources { + let owner = source.owner.to_string(); + let token = source.token.to_string(); + let amount = amount.amount().native_denominated().to_string(); + sources_data.push(TransferDataMsg::new(owner, token, amount)) + } + + for (target, amount) in targets { + let owner = target.owner.to_string(); + let token = target.token.to_string(); + let amount = amount.amount().native_denominated().to_string(); + targets_data.push(TransferDataMsg::new(owner, token, amount)) + } + + borsh::to_vec(&TransferMsg::new(sources_data, targets_data, ssh))? + } + TransactionKind::ProposalVote(vote_proposal) => { + let VoteProposalData { id, vote, voter } = vote_proposal; + let vote_proposal = VoteProposalMsg::new(voter.to_string(), *id, vote.to_string()); + borsh::to_vec(&vote_proposal)? + } + TransactionKind::ClaimRewards(claim_rewards) => { + let ClaimRewards { validator, source } = claim_rewards; + let claim_rewards = ClaimRewardsMsg::new( + validator.to_string(), + source.clone().map(|addr| addr.to_string()), + ); + borsh::to_vec(&claim_rewards)? + } + TransactionKind::IbcTransfer(msg_transfer) => { + let MsgTransfer { message, .. } = msg_transfer; + + let timeout_height = match message.timeout_height_on_b { + TimeoutHeight::At(height) => Some(height.revision_height()), + TimeoutHeight::Never => None, + }; + + let ibc_transfer_msg = IbcTransferMsg::new( + message.packet_data.sender.to_string(), + message.packet_data.receiver.to_string(), + message.packet_data.token.denom.base_denom.to_string(), + message.packet_data.token.amount.to_string(), + message.port_id_on_a.to_string(), + message.chan_id_on_a.to_string(), + timeout_height, + None, + Some(message.packet_data.memo.to_string()), + None, + ); + borsh::to_vec(&ibc_transfer_msg)? + } + _ => panic!("Unsupported Tx provided, cannot serialize"), + }; + + Ok(bytes) + } +} diff --git a/packages/sdk/lib/src/sdk/tx.rs b/packages/sdk/lib/src/sdk/tx.rs new file mode 100644 index 0000000000..4bd5ecf670 --- /dev/null +++ b/packages/sdk/lib/src/sdk/tx.rs @@ -0,0 +1,382 @@ +use std::collections::HashMap; +use std::str::FromStr; + +use gloo_utils::format::JsValueSerdeExt; +use namada_sdk::borsh::{self, BorshDeserialize, BorshSerialize}; +use namada_sdk::masp_primitives::transaction::components::sapling::builder::StoredBuildParams; +use namada_sdk::masp_primitives::zip32::ExtendedFullViewingKey; +use namada_sdk::signing::SigningTxData; +use namada_sdk::tx::data::compute_inner_tx_hash; +use namada_sdk::tx::either::Either; +use namada_sdk::tx::{ + self, TX_BOND_WASM, TX_CLAIM_REWARDS_WASM, TX_IBC_WASM, TX_REDELEGATE_WASM, TX_REVEAL_PK, + TX_TRANSFER_WASM, TX_UNBOND_WASM, TX_VOTE_PROPOSAL, TX_WITHDRAW_WASM, +}; +use namada_sdk::uint::Uint; +use namada_sdk::{address::Address, key::common::PublicKey}; +use wasm_bindgen::{prelude::wasm_bindgen, JsError, JsValue}; + +use super::args::WrapperTxMsg; +use crate::sdk::transaction; +use crate::types::query::WasmHash; + +#[wasm_bindgen] +#[derive(BorshSerialize, BorshDeserialize, Copy, Clone, Debug)] +#[borsh(crate = "namada_sdk::borsh", use_discriminant = true)] +pub enum TxType { + Bond = 1, + Unbond = 2, + Withdraw = 3, + Transfer = 4, + IBCTransfer = 5, + EthBridgeTransfer = 6, + RevealPK = 7, + VoteProposal = 8, + Redelegate = 9, + Batch = 10, + ClaimRewards = 11, +} + +#[derive(BorshSerialize, BorshDeserialize, Clone, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct SigningData { + owner: Option, + public_keys: Vec, + threshold: u8, + account_public_keys_map: Option>, + fee_payer: String, + shielded_hash: Option>, + masp: Option>, +} + +impl SigningData { + // Create serializable struct from Namada type + pub fn from_signing_tx_data( + signing_tx_data: SigningTxData, + masp_signing_data: Option, + ) -> Result { + let owner: Option = signing_tx_data.owner.map(|addr| addr.to_string()); + let public_keys = signing_tx_data + .public_keys + .into_iter() + .map(|pk| pk.to_string()) + .collect(); + + let account_public_keys_map = match signing_tx_data.account_public_keys_map { + Some(pk_map) => Some(borsh::to_vec(&pk_map)?), + None => None, + }; + + let fee_payer = signing_tx_data.fee_payer.to_string(); + let threshold = signing_tx_data.threshold; + let shielded_hash = match signing_tx_data.shielded_hash { + Some(v) => Some(borsh::to_vec(&v)?), + None => None, + }; + let masp_signing_data = match masp_signing_data { + Some(v) => Some(borsh::to_vec(&v)?), + None => None, + }; + + Ok(SigningData { + owner, + public_keys, + threshold, + account_public_keys_map, + fee_payer, + shielded_hash, + masp: masp_signing_data, + }) + } + + // Create Namada type from this struct + pub fn to_signing_tx_data(&self) -> Result { + let owner: Option
= match &self.owner { + Some(addr) => Some(Address::from_str(addr)?), + None => None, + }; + + let mut public_keys: Vec = vec![]; + for pk in self.public_keys.clone() { + let pk = PublicKey::from_str(&pk)?; + public_keys.push(pk); + } + + let fee_payer = PublicKey::from_str(&self.fee_payer)?; + let threshold = self.threshold; + let account_public_keys_map = match &self.account_public_keys_map { + Some(pk_map) => Some(borsh::from_slice(pk_map)?), + None => None, + }; + let shielded_hash = match &self.shielded_hash { + Some(v) => Some(borsh::from_slice(v)?), + None => None, + }; + + Ok(SigningTxData { + owner, + public_keys, + fee_payer, + threshold, + account_public_keys_map, + shielded_hash, + }) + } + + pub fn masp(&self) -> Option> { + self.masp.clone() + } +} + +#[derive(BorshSerialize, BorshDeserialize, Clone, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct MaspSigningData { + pub bparams: StoredBuildParams, + pub xfvks: Vec, +} + +impl MaspSigningData { + pub fn new(bparams: StoredBuildParams, xfvks: Vec) -> MaspSigningData { + MaspSigningData { bparams, xfvks } + } + + pub fn xfvks(&self) -> Vec { + self.xfvks.clone() + } + + pub fn bparams(&self) -> StoredBuildParams { + self.bparams.clone() + } +} + +/// Serializable Tx for exported build functions +#[derive(BorshSerialize, BorshDeserialize, Clone, Debug)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct Tx { + args: WrapperTxMsg, + hash: String, + bytes: Vec, + pub signing_data: Vec, +} + +impl Tx { + pub fn new( + tx: tx::Tx, + args: &[u8], + signing_tx_data: Vec<(SigningTxData, Option)>, + ) -> Result { + let args: WrapperTxMsg = borsh::from_slice(args)?; + let mut signing_data: Vec = vec![]; + for (sd, msd) in signing_tx_data.into_iter() { + let sd = SigningData::from_signing_tx_data(sd, msd)?; + signing_data.push(sd); + } + let hash = tx.wrapper_hash(); + let bytes: Vec = borsh::to_vec(&tx)?; + + Ok(Tx { + args, + hash: hash.unwrap().to_string(), + bytes, + signing_data, + }) + } + + pub fn tx_bytes(&self) -> Vec { + self.bytes.clone() + } + + pub fn signing_tx_data(&self) -> Result, JsError> { + let mut signing_tx_data: Vec = vec![]; + for sd in self.signing_data.clone().iter() { + signing_tx_data.push(sd.to_signing_tx_data()?); + } + + Ok(signing_tx_data) + } + + pub fn signing_data(&self) -> Vec { + self.signing_data.clone() + } + + pub fn args(&self) -> WrapperTxMsg { + self.args.clone() + } +} + +// Given the bytes of a Namada Tx, return all inner Tx hashes +#[wasm_bindgen] +pub fn get_inner_tx_hashes(tx_bytes: &[u8]) -> Result, JsError> { + let nam_tx: tx::Tx = borsh::from_slice(tx_bytes)?; + let hash = nam_tx.wrapper_hash(); + let cmts = nam_tx.commitments(); + let mut inner_tx_hashes: Vec = vec![]; + + for cmt in cmts { + let inner_tx_hash = compute_inner_tx_hash(hash.as_ref(), Either::Right(cmt)); + inner_tx_hashes.push(inner_tx_hash.to_string()); + } + + Ok(inner_tx_hashes) +} + +pub fn wasm_hash_to_tx_type(wasm_hash: &str, wasm_hashes: &Vec) -> Option { + let type_map: HashMap = HashMap::from([ + (TX_TRANSFER_WASM.to_string(), TxType::Transfer), + (TX_BOND_WASM.to_string(), TxType::Bond), + (TX_REDELEGATE_WASM.to_string(), TxType::Redelegate), + (TX_UNBOND_WASM.to_string(), TxType::Unbond), + (TX_WITHDRAW_WASM.to_string(), TxType::Withdraw), + (TX_CLAIM_REWARDS_WASM.to_string(), TxType::ClaimRewards), + (TX_REVEAL_PK.to_string(), TxType::RevealPK), + (TX_VOTE_PROPOSAL.to_string(), TxType::VoteProposal), + (TX_IBC_WASM.to_string(), TxType::IBCTransfer), + ]); + + for wh in wasm_hashes { + if wh.hash() == wasm_hash { + let tx_type = type_map.get(&wh.path()); + + if let Some(tx_type) = tx_type { + return Some(*tx_type); + } + } + } + + None +} + +// Deserialize Tx commitments into Borsh-serialized struct +#[wasm_bindgen] +pub fn deserialize_tx(tx_bytes: Vec, wasm_hashes: JsValue) -> Result, JsError> { + let tx = TxDetails::from_bytes(tx_bytes, wasm_hashes)?; + Ok(borsh::to_vec(&tx)?) +} + +#[derive(BorshSerialize, BorshDeserialize)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct Commitment { + tx_type: TxType, + hash: String, + tx_code_id: String, + data: Vec, + memo: Option, +} + +#[derive(BorshSerialize, BorshDeserialize)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct TxDetails { + wrapper_tx: WrapperTxMsg, + commitments: Vec, +} + +impl TxDetails { + pub fn from_bytes(tx_bytes: Vec, wasm_hashes: JsValue) -> Result { + let tx: tx::Tx = borsh::from_slice(&tx_bytes)?; + let chain_id = tx.header().chain_id.to_string(); + + match tx.header().tx_type { + tx::data::TxType::Wrapper(wrapper) => { + let fee_amount = wrapper.fee.amount_per_gas_unit.to_string(); + let gas_limit = Uint::from(wrapper.gas_limit).to_string(); + let token = wrapper.fee.token.to_string(); + + let wrapper_tx = + WrapperTxMsg::new(token, fee_amount, gas_limit, chain_id, None, None, None); + let mut commitments: Vec = vec![]; + let wasm_hashes: Vec = wasm_hashes.into_serde().unwrap(); + + for cmt in tx.commitments() { + let memo = tx + .memo(cmt) + .map(|memo_bytes| String::from_utf8_lossy(&memo_bytes).to_string()); + + let hash = cmt.get_hash().to_string(); + let tx_code_id = tx + .get_section(cmt.code_sechash()) + .and_then(|s| s.code_sec()) + .map(|s| s.code.hash().0) + .map(|bytes| { + String::from_utf8(subtle_encoding::hex::encode(bytes)).unwrap() + }); + + if tx_code_id.is_some() { + let tx_code_id = tx_code_id.unwrap(); + let tx_type = wasm_hash_to_tx_type(&tx_code_id, &wasm_hashes); + + if tx_type.is_some() { + let tx_type = tx_type.unwrap(); + let tx_data = tx.data(cmt).unwrap_or_default(); + let tx_kind = transaction::TransactionKind::from(tx_type, &tx_data); + let data = tx_kind.to_bytes()?; + + commitments.push(Commitment { + tx_type, + hash, + tx_code_id, + data, + memo, + }); + } + } + } + + Ok(TxDetails { + wrapper_tx, + commitments, + }) + } + _ => Err(JsError::new("Invalid transaction type!")), + } + } +} + +#[wasm_bindgen] +#[derive(BorshSerialize, BorshDeserialize)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct BatchTxResult { + hash: String, + is_applied: bool, +} + +impl BatchTxResult { + pub fn new(hash: String, is_applied: bool) -> BatchTxResult { + BatchTxResult { hash, is_applied } + } +} + +/// Serializable response for process_tx calls +#[wasm_bindgen] +#[derive(BorshSerialize, BorshDeserialize)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct TxResponse { + code: String, + commitments: Vec, + gas_used: String, + hash: String, + height: String, + info: String, + log: String, +} + +impl TxResponse { + pub fn new( + code: String, + commitments: Vec, + gas_used: String, + hash: String, + height: String, + info: String, + log: String, + ) -> TxResponse { + TxResponse { + code, + commitments, + gas_used, + hash, + height, + info, + log, + } + } +} diff --git a/packages/sdk/lib/src/sdk/wallet/mod.rs b/packages/sdk/lib/src/sdk/wallet/mod.rs new file mode 100644 index 0000000000..4346c991e0 --- /dev/null +++ b/packages/sdk/lib/src/sdk/wallet/mod.rs @@ -0,0 +1,103 @@ +use namada_sdk::{ + key::common::SecretKey, + masp_primitives::zip32::ExtendedFullViewingKey, + wallet::{alias::Alias, Wallet, WalletIo}, + ExtendedSpendingKey, ExtendedViewingKey, PaymentAddress, +}; +use std::str::FromStr; +use zeroize::Zeroizing; + +#[cfg(feature = "web")] +mod wallet_web; + +#[cfg(feature = "web")] +pub use wallet_web::BrowserWalletUtils as JSWalletUtils; + +#[cfg(feature = "nodejs")] +mod wallet_node; + +#[cfg(feature = "nodejs")] +pub use wallet_node::NodeWalletUtils as JSWalletUtils; + +/// Adds spending key to the wallet. +/// It's needed because we create addresses without using the Sdk. +/// Panics if inserting spending key is impossible. +/// +/// # Arguments +/// +/// * `wallet` - Instance of a wallet struct. +/// * `xsk` - String representing serialized ExtendedSpendingKey. +/// * `alias` - Spending key alias. +pub fn add_spending_key(wallet: &mut Wallet, xsk: String, alias: String) { + let xsk = ExtendedSpendingKey::from_str(&xsk).expect("XSK deserialization failed."); + let alias = Alias::from(alias); + + // xsk is decrypted outside of this wallet instance, so we specify None below + if wallet + .store_mut() + .insert_spending_key::(alias.clone(), xsk, None, None, None, true) + .is_none() + { + panic!("Action cancelled, no changes persisted."); + } +} + +pub fn add_viewing_key(wallet: &mut Wallet, xvk: String, alias: String) { + let xvk = ExtendedViewingKey::from_str(&xvk).expect("XVK deserialization failed."); + let alias = Alias::from(alias); + + if wallet + .store_mut() + .insert_viewing_key::(alias.clone(), xvk, None, None, true) + .is_none() + { + panic!("Action cancelled, no changes persisted."); + } +} + +pub fn add_payment_address(wallet: &mut Wallet, pa: String, alias: String) { + let pa = PaymentAddress::from_str(&pa).expect("Payment address deserialization failed."); + let alias = Alias::from(alias); + + if wallet + .store_mut() + .insert_payment_addr::(alias.clone(), pa, true) + .is_none() + { + panic!("Action cancelled, no changes persisted."); + } +} + +pub fn add_default_payment_address( + wallet: &mut Wallet, + xvk: String, + alias: String, +) { + let xfvk: ExtendedFullViewingKey = ExtendedViewingKey::from_str(&xvk) + .expect("XVK deserialization failed.") + .into(); + let alias = Alias::from(alias); + let pa: PaymentAddress = xfvk.default_address().1.into(); + + if wallet + .store_mut() + .insert_payment_addr::(alias.clone(), pa, true) + .is_none() + { + panic!("Action cancelled, no changes persisted."); + } +} + +pub fn add_keypair( + wallet: &mut Wallet, + secret_key: String, + alias: String, + password: Option, +) { + let alias = Alias::from(alias); + let secret_key = SecretKey::from_str(&secret_key).expect("SecretKey deserialization failed."); + let password = password.map(Zeroizing::new); + wallet + .store_mut() + .insert_keypair::(alias.clone(), secret_key, password, None, None, true); +} diff --git a/packages/sdk/lib/src/sdk/wallet/wallet_node.rs b/packages/sdk/lib/src/sdk/wallet/wallet_node.rs new file mode 100644 index 0000000000..4d6361d202 --- /dev/null +++ b/packages/sdk/lib/src/sdk/wallet/wallet_node.rs @@ -0,0 +1,90 @@ +use std::path::PathBuf; + +use namada_sdk::{ + borsh::{BorshDeserialize, BorshSerialize}, + wallet::{LoadStoreError, Store, Wallet, WalletIo, WalletStorage}, +}; +use rand::rngs::OsRng; +use wasm_bindgen::{prelude::wasm_bindgen, JsValue}; + +use crate::utils::to_bytes; + +#[derive(Debug, BorshSerialize, BorshDeserialize, Clone)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct NodeWalletUtils { + #[borsh(skip)] + store_dir: PathBuf, +} + +impl NodeWalletUtils { + pub fn new_utils(store_dir: &str) -> Self { + let store_dir = PathBuf::from(store_dir); + Self { store_dir } + } +} + +impl WalletIo for NodeWalletUtils { + type Rng = OsRng; +} + +pub trait NodeWalletStorage: Clone { + fn store_dir(&self) -> &PathBuf; +} + +impl NodeWalletStorage for NodeWalletUtils { + fn store_dir(&self) -> &PathBuf { + &self.store_dir + } +} + +const FILE_NAME: &str = "wallet.toml"; + +impl WalletStorage for NodeWalletUtils { + fn save(&self, wallet: &Wallet) -> Result<(), namada_sdk::wallet::LoadStoreError> { + let data = wallet.store().encode(); + + let wallet_path = self.store_dir().join(FILE_NAME); + // Make sure the dir exists + let wallet_dir = wallet_path.parent().unwrap(); + let uint8_array = js_sys::Uint8Array::from(&data[..]); + + write_file_sync( + JsValue::from_str(wallet_dir.to_str().unwrap()), + uint8_array.into(), + ) + .unwrap(); + + Ok(()) + } + + fn load(&self, wallet: &mut Wallet) -> Result<(), namada_sdk::wallet::LoadStoreError> { + let wallet_file = self.store_dir().join(FILE_NAME); + + let stored_data: Vec = + to_bytes(read_file_sync(JsValue::from_str(wallet_file.to_str().unwrap())).unwrap()); + + let store = wallet.store_mut(); + + *store = Store::decode(stored_data).map_err(LoadStoreError::Decode)?; + + Ok(()) + } +} + +#[wasm_bindgen(module = "/src/sdk/masp/masp.node.js")] +extern "C" { + #[wasm_bindgen(catch, js_name = "writeFileSync")] + fn write_file_sync(path: JsValue, content: JsValue) -> Result; + + #[wasm_bindgen(catch, js_name = "readFileSync")] + fn read_file_sync(path: JsValue) -> Result; + + #[wasm_bindgen(catch, js_name = "renameSync")] + fn renameSync(pathA: JsValue, pathB: JsValue) -> Result; + + #[wasm_bindgen(catch, js_name = "unlinkSync")] + fn unlinkSync(path: JsValue) -> Result; + + #[wasm_bindgen(catch, js_name = "existsSync")] + fn exists_sync(path: JsValue) -> Result; +} diff --git a/packages/sdk/lib/src/sdk/wallet/wallet_web.rs b/packages/sdk/lib/src/sdk/wallet/wallet_web.rs new file mode 100644 index 0000000000..5ae274d8cc --- /dev/null +++ b/packages/sdk/lib/src/sdk/wallet/wallet_web.rs @@ -0,0 +1,35 @@ +use namada_sdk::{ + borsh::{BorshDeserialize, BorshSerialize}, + wallet::{Wallet, WalletIo, WalletStorage}, +}; +use rand::rngs::OsRng; + +#[derive(Debug, BorshSerialize, BorshDeserialize, Clone)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct BrowserWalletUtils { + #[borsh(skip)] + _name: String, +} + +impl BrowserWalletUtils { + pub fn new_utils(name: &str) -> Self { + Self { + _name: name.to_string(), + } + } +} + +impl WalletIo for BrowserWalletUtils { + type Rng = OsRng; +} + +//TODO: We can't implement it until namada changes trait to be async +impl WalletStorage for BrowserWalletUtils { + fn save(&self, _wallet: &Wallet) -> Result<(), namada_sdk::wallet::LoadStoreError> { + todo!() + } + + fn load(&self, _wallet: &mut Wallet) -> Result<(), namada_sdk::wallet::LoadStoreError> { + todo!() + } +} diff --git a/packages/sdk/lib/src/types/address.rs b/packages/sdk/lib/src/types/address.rs new file mode 100644 index 0000000000..85f48490b5 --- /dev/null +++ b/packages/sdk/lib/src/types/address.rs @@ -0,0 +1,102 @@ +use namada_sdk::borsh::BorshDeserialize; +use namada_sdk::{ + address, + key::{ + self, + common::{PublicKey, SecretKey}, + PublicKeyHash, RefTo, + }, +}; +use std::str::FromStr; +use wasm_bindgen::prelude::*; + +/// Helper function to bech32 encode a public key from bytes +#[wasm_bindgen] +pub fn public_key_to_bech32(bytes: Vec) -> Result { + let public_key = PublicKey::try_from_slice(&bytes)?; + + Ok(public_key.to_string()) +} + +#[wasm_bindgen] +pub struct Address { + implicit: address::Address, + public: PublicKey, + hash: PublicKeyHash, +} + +#[wasm_bindgen] +impl Address { + /// Address helpers for wasm_bindgen + #[wasm_bindgen(constructor)] + pub fn new(secret: String) -> Address { + let private = SecretKey::Ed25519( + key::ed25519::SecretKey::from_str(&secret).expect("ed25519 encoding should not fail"), + ); + + #[allow(clippy::useless_conversion)] + let public = PublicKey::from(private.ref_to()); + let hash = PublicKeyHash::from(&public); + let implicit = address::Address::Implicit(address::ImplicitAddress::from(&public)); + + Address { + implicit, + public, + hash, + } + } + + pub fn implicit(&self) -> String { + self.implicit.encode() + } + + pub fn public(&self) -> String { + self.public.to_string() + } + + pub fn hash(&self) -> String { + self.hash.to_string() + } +} + +#[cfg(test)] +mod tests { + use super::*; + use wasm_bindgen_test::*; + + #[wasm_bindgen_test] + fn can_generate_implicit_address() { + let secret = + String::from("1498b5467a63dffa2dc9d9e069caf075d16fc33fdd4c3b01bfadae6433767d93"); + let address = Address::new(secret); + let implicit = address.implicit(); + + assert_eq!(implicit, "tnam1qpgk927uh2aqjs92yhycsh08n5yggvltn5nk92zp"); + assert_eq!(implicit.len(), address::ADDRESS_LEN); + } + + #[wasm_bindgen_test] + fn can_return_correct_public_key() { + let secret = + String::from("1498b5467a63dffa2dc9d9e069caf075d16fc33fdd4c3b01bfadae6433767d93"); + let address = Address::new(secret); + let public = address.public(); + + assert_eq!( + public, + "tpknam1qzm68sfdcryvwj9tqaf9kuq3y2ugh4u0vqx8vdpdyle9uhujg3xduf408cn" + ); + assert_eq!(public.len(), 66); + } + + #[wasm_bindgen_test] + fn can_return_correct_public_key_hash() { + let secret = + String::from("1498b5467a63dffa2dc9d9e069caf075d16fc33fdd4c3b01bfadae6433767d93"); + let address = Address::new(secret.clone()); + let hash = address.hash(); + + assert_eq!("5162ABDCBABA0940AA25C9885DE79D088433EB9D", hash); + assert_eq!(hash.len(), 40); + } +} diff --git a/packages/sdk/lib/src/types/masp.rs b/packages/sdk/lib/src/types/masp.rs new file mode 100644 index 0000000000..9d279d801f --- /dev/null +++ b/packages/sdk/lib/src/types/masp.rs @@ -0,0 +1,245 @@ +//! PaymentAddress - Provide wasm_bindgen bindings for shielded addresses +//! See @namada/crypto for zip32 HD wallet functionality. +use js_sys::Uint8Array; +use namada_sdk::borsh::{self, BorshDeserialize}; +use namada_sdk::masp_primitives::zip32::ExtendedKey; +use namada_sdk::masp_primitives::{sapling, zip32}; +use namada_sdk::masp_proofs::jubjub; +use namada_sdk::{ + ExtendedSpendingKey as NamadaExtendedSpendingKey, + ExtendedViewingKey as NamadaExtendedViewingKey, PaymentAddress as NamadaPaymentAddress, +}; +use thiserror::Error; +use wasm_bindgen::prelude::*; + +#[derive(Debug, Error)] +pub enum MaspError { + #[error("PaymentAddress from_str failed!")] + PaymentAddress, + #[error("BorshDeserialize failed!")] + BorshDeserialize, +} + +/// Wrap ExtendedViewingKey +#[wasm_bindgen] +pub struct ExtendedViewingKey(pub(crate) NamadaExtendedViewingKey); + +/// wasm_bindgen bindings for ExtendedViewingKey +#[wasm_bindgen] +impl ExtendedViewingKey { + /// Instantiate ExtendedViewingKey from serialized vector + #[wasm_bindgen(constructor)] + pub fn new(key: &[u8]) -> Result { + let xfvk: zip32::ExtendedFullViewingKey = BorshDeserialize::try_from_slice(key) + .map_err(|err| format!("{}: {:?}", MaspError::BorshDeserialize, err))?; + + let vk = NamadaExtendedViewingKey::from(xfvk); + + Ok(ExtendedViewingKey(vk)) + } + + /// Return ExtendedViewingKey as Bech32-encoded String + pub fn encode(&self) -> String { + self.0.to_string() + } +} + +#[wasm_bindgen] +pub struct ProofGenerationKey(pub(crate) sapling::ProofGenerationKey); + +#[wasm_bindgen] +impl ProofGenerationKey { + pub fn encode(&self) -> String { + hex::encode( + borsh::to_vec(&self.0).expect("Serializing ProofGenerationKey should not fail!"), + ) + } + pub fn decode(encoded: String) -> ProofGenerationKey { + let decoded = hex::decode(encoded).expect("Decoding ProofGenerationKey should not fail!"); + + ProofGenerationKey( + sapling::ProofGenerationKey::try_from_slice(decoded.as_slice()) + .expect("Deserializing ProofGenerationKey should not fail!"), + ) + } +} + +/// Wrap ExtendedSpendingKey +#[wasm_bindgen] +pub struct PseudoExtendedKey(pub(crate) zip32::PseudoExtendedKey); + +#[wasm_bindgen] +impl PseudoExtendedKey { + pub fn encode(&self) -> String { + hex::encode(borsh::to_vec(&self.0).expect("Serializing PseudoExtendedKey should not fail!")) + } + pub fn decode(encoded: String) -> PseudoExtendedKey { + let decoded = hex::decode(encoded).expect("Decoding PsuedoExtendedKey should not fail!"); + + PseudoExtendedKey( + zip32::PseudoExtendedKey::try_from_slice(decoded.as_slice()) + .expect("Deserializing ProofGenerationKey should not fail!"), + ) + } +} + +/// Wrap ExtendedSpendingKey +#[wasm_bindgen] +pub struct ExtendedSpendingKey(pub(crate) NamadaExtendedSpendingKey); + +/// wasm_bindgen bindings for ExtendedViewingKey +#[wasm_bindgen] +impl ExtendedSpendingKey { + /// Instantiate ExtendedSpendingKey from serialized vector + #[wasm_bindgen(constructor)] + pub fn new(key: Uint8Array) -> Result { + let xsk: zip32::ExtendedSpendingKey = + BorshDeserialize::try_from_slice(key.to_vec().as_slice()) + .map_err(|err| format!("{}: {:?}", MaspError::BorshDeserialize, err))?; + + let xsk = NamadaExtendedSpendingKey::from(xsk); + + Ok(ExtendedSpendingKey(xsk)) + } + + pub fn to_proof_generation_key(&self) -> ProofGenerationKey { + let xsk = zip32::ExtendedSpendingKey::from(self.0); + let pgk = xsk + .to_proof_generation_key() + .expect("Converting to proof generation key should not fail!"); + + ProofGenerationKey(pgk) + } + + pub fn to_pseudo_extended_key(&self) -> PseudoExtendedKey { + let xsk = zip32::ExtendedSpendingKey::from(self.0); + let mut pxk = zip32::PseudoExtendedKey::from(xsk); + pxk.augment_spend_authorizing_key_unchecked(sapling::redjubjub::PrivateKey( + jubjub::Fr::default(), + )); + + PseudoExtendedKey(pxk) + } + + /// Return ExtendedSpendingKey as Bech32-encoded String + pub fn encode(&self) -> String { + self.0.to_string() + } +} + +/// Wrap PaymentAddress +#[wasm_bindgen] +pub struct PaymentAddress(pub(crate) NamadaPaymentAddress); + +/// wasm_bindgen bindings for PaymentAddress +#[wasm_bindgen] +impl PaymentAddress { + /// Instantiate PaymentAddress from serialized vector + #[wasm_bindgen(constructor)] + pub fn new(address: &[u8]) -> Result { + let payment_address: sapling::PaymentAddress = BorshDeserialize::try_from_slice(address) + .map_err(|err| format!("{}: {:?}", MaspError::BorshDeserialize, err))?; + let payment_address = NamadaPaymentAddress::from(payment_address); + Ok(PaymentAddress(payment_address)) + } + + /// Retrieve PaymentAddress hash + pub fn hash(&self) -> String { + self.0.hash() + } + + /// Return PaymentAddress as Bech32-encoded String + pub fn encode(&self) -> String { + self.0.to_string() + } +} + +#[cfg(test)] +mod tests { + use super::*; + + use wasm_bindgen_test::*; + + #[wasm_bindgen_test] + fn can_deserialize_an_extended_spending_key() { + // BorshSerialize'd slice, generated from @namada/crypto - zip32 + let encoded_xsk: &[u8] = &[ + 1, 233, 222, 184, 155, 1, 0, 0, 0, 232, 94, 130, 41, 9, 58, 197, 35, 245, 249, 232, + 225, 222, 38, 148, 105, 204, 14, 230, 30, 241, 22, 214, 38, 221, 49, 17, 147, 255, 136, + 219, 250, 71, 230, 226, 2, 146, 75, 94, 233, 234, 254, 128, 142, 209, 73, 65, 180, 64, + 235, 159, 125, 24, 77, 12, 246, 113, 174, 41, 217, 5, 190, 215, 6, 76, 189, 55, 31, 96, + 85, 114, 22, 215, 250, 140, 98, 162, 95, 203, 154, 180, 0, 231, 40, 172, 36, 137, 30, + 142, 181, 225, 143, 180, 110, 135, 2, 213, 181, 237, 102, 55, 178, 202, 2, 123, 161, + 104, 49, 91, 37, 62, 52, 132, 72, 103, 7, 60, 110, 171, 49, 22, 100, 146, 44, 79, 205, + 112, 25, 36, 51, 226, 228, 45, 242, 201, 220, 212, 220, 58, 92, 127, 47, 214, 59, 174, + 182, 74, 90, 65, 229, 187, 76, 65, 246, 34, 237, 107, 208, 178, 243, + ]; + let xsk = ExtendedSpendingKey::new(encoded_xsk.into()) + .expect("Instantiating ExtendedSpendingKey struct should not fail!"); + + let key = xsk.encode(); + let expected_key = + format!( + "{}{}{}", + "zsknam1q85aawymqyqqqq8gt6pzjzf6c53lt70gu80zd9rfes8wv8h3zmtzdhf3zxfllzxmlfr7dcszjf94a602l6qga52fgx6yp6ul05vy6r", + "8kwxhznkg9hmtsvn9axu0kq4tjzmtl4rrz5f0uhx45qrnj3tpy3y0gad0p376xapcz6k676e3hkt9qy7apdqc4kff7xjzysec883h2kvgkvjf", + "zcn7dwqvjgvlzuskl9jwu6nwr5hrl9ltrht4kffdyredmf3qlvghdd0gt9ucw4ccj7", + ); + + assert!(key.starts_with("zsknam")); + assert_eq!(key, expected_key); + } + + #[wasm_bindgen_test] + fn can_deserialize_an_extended_viewing_key() { + // BorshSerialize'd slice, generated from @namada/crypto - zip32 + let encoded_xfvk: &[u8] = &[ + 1, 233, 222, 184, 155, 1, 0, 0, 0, 232, 94, 130, 41, 9, 58, 197, 35, 245, 249, 232, + 225, 222, 38, 148, 105, 204, 14, 230, 30, 241, 22, 214, 38, 221, 49, 17, 147, 255, 136, + 219, 250, 231, 141, 253, 33, 141, 45, 47, 253, 94, 99, 2, 58, 233, 84, 152, 142, 60, + 45, 175, 100, 10, 5, 32, 126, 133, 46, 214, 50, 136, 235, 250, 73, 125, 112, 103, 142, + 119, 204, 205, 75, 30, 208, 119, 223, 218, 19, 88, 206, 173, 185, 244, 228, 224, 32, + 104, 193, 189, 255, 9, 147, 22, 21, 240, 191, 213, 181, 237, 102, 55, 178, 202, 2, 123, + 161, 104, 49, 91, 37, 62, 52, 132, 72, 103, 7, 60, 110, 171, 49, 22, 100, 146, 44, 79, + 205, 112, 25, 36, 51, 226, 228, 45, 242, 201, 220, 212, 220, 58, 92, 127, 47, 214, 59, + 174, 182, 74, 90, 65, 229, 187, 76, 65, 246, 34, 237, 107, 208, 178, 243, + ]; + let xfvk = ExtendedViewingKey::new(encoded_xfvk) + .expect("Instantiating ExtendedViewingKey struct should not fail!"); + + let key = xfvk.encode(); + let expected_key = + format!( + "{}{}{}", + "zvknam1q85aawymqyqqqq8gt6pzjzf6c53lt70gu80zd9rfes8wv8h3zmtzdhf3zxfllzxmltncmlfp35kjll27vvpr4625nz8rctd0vs9", + "q2gr7s5hdvv5ga0ayjltsv7880nxdfv0dqa7lmgf43n4dh86wfcpqdrqmmlcfjvtptu9l6k676e3hkt9qy7apdqc4kff7xjzysec883h2k", + "vgkvjfzcn7dwqvjgvlzuskl9jwu6nwr5hrl9ltrht4kffdyredmf3qlvghdd0gt9uce7vx4v", + ); + assert!(key.starts_with("zvknam")); + assert_eq!(key, expected_key); + } + + #[wasm_bindgen_test] + fn can_deserialize_a_payment_address() { + // BorshSerialize'd slice, generated from @namada/crypto - zip32 + let encoded_payment_address: &[u8] = &[ + 100, 199, 34, 96, 93, 67, 18, 95, 86, 139, 123, 213, 141, 228, 147, 169, 218, 247, 75, + 83, 195, 72, 73, 44, 65, 232, 243, 229, 209, 63, 183, 1, 87, 87, 203, 40, 180, 242, + 103, 187, 245, 224, 36, + ]; + let payment_address = PaymentAddress::new(encoded_payment_address) + .expect("Instantiating PaymentAddress struct should not fail!"); + + let address = payment_address.encode(); + let hash = payment_address.hash(); + + let expected_address = + "znam1vnrjyczagvf9745t002cmeyn48d0wj6ncdyyjtzpare7t5flkuq4w47t9z60yeam7hszgyhdw2j"; + + let expected_hash = "4E11B97D220F336CF36A14E8DDFE15ED34BC489D"; + + assert!(address.starts_with("znam")); + assert_eq!(address, expected_address); + assert_eq!(hash, expected_hash); + } +} diff --git a/packages/sdk/lib/src/types/mod.rs b/packages/sdk/lib/src/types/mod.rs new file mode 100644 index 0000000000..29a4c18b31 --- /dev/null +++ b/packages/sdk/lib/src/types/mod.rs @@ -0,0 +1,3 @@ +pub mod address; +pub mod masp; +pub mod query; diff --git a/packages/sdk/lib/src/types/query.rs b/packages/sdk/lib/src/types/query.rs new file mode 100644 index 0000000000..a118bc8262 --- /dev/null +++ b/packages/sdk/lib/src/types/query.rs @@ -0,0 +1,36 @@ +use namada_sdk::borsh::BorshSerialize; +use serde::{Deserialize, Serialize}; + +#[derive(BorshSerialize)] +#[borsh(crate = "namada_sdk::borsh")] +pub struct ProposalInfo { + pub id: u64, + pub content: String, + pub author: String, + pub start_epoch: u64, + pub end_epoch: u64, + pub grace_epoch: u64, + pub tally_type: String, + pub proposal_type: String, + pub data: Option, +} + +#[derive(Debug, Serialize, Deserialize)] +pub struct WasmHash { + path: String, + hash: String, +} + +impl WasmHash { + pub fn new(path: String, hash: String) -> WasmHash { + WasmHash { path, hash } + } + + pub fn path(&self) -> String { + self.path.clone() + } + + pub fn hash(&self) -> String { + self.hash.clone() + } +} diff --git a/packages/sdk/lib/src/utils.rs b/packages/sdk/lib/src/utils.rs new file mode 100644 index 0000000000..3abde409d0 --- /dev/null +++ b/packages/sdk/lib/src/utils.rs @@ -0,0 +1,50 @@ +use gloo_utils::format::JsValueSerdeExt; +use js_sys::Uint8Array; +use serde::Serialize; +use std::fmt::Debug; +use wasm_bindgen::prelude::*; + +#[wasm_bindgen] +extern "C" { + #[wasm_bindgen(js_namespace = console)] + pub fn log(s: &str); +} + +#[allow(dead_code)] +pub fn console_log(string: &str) { + log(string); +} + +#[allow(dead_code)] +pub fn console_log_any(string: &T) { + log(format!("{:?}", string).as_str()); +} + +pub fn to_bytes(u_int_8_array: JsValue) -> Vec { + let array = Uint8Array::new(&u_int_8_array); + array.to_vec() +} + +/// Maps a result to a JsValue using Serde and Error into a JsError +/// +/// # Arguments +/// +/// * `result` - The result to map +pub fn to_js_result(result: T) -> Result +where + T: Serialize, +{ + match JsValue::from_serde(&result) { + Ok(v) => Ok(v), + Err(e) => Err(JsError::new(&e.to_string())), + } +} + +#[cfg(feature = "dev")] +pub fn set_panic_hook() { + web_sys::console::log_1(&"Set panic hook".into()); + console_error_panic_hook::set_once(); +} + +#[cfg(not(feature = "dev"))] +pub fn set_panic_hook() {} diff --git a/packages/sdk/package.json b/packages/sdk/package.json index 12219bdc43..2d29a2de6f 100644 --- a/packages/sdk/package.json +++ b/packages/sdk/package.json @@ -1,6 +1,6 @@ { "name": "@namada/sdk", - "version": "0.13.0", + "version": "0.14.0", "description": "Namada SDK package", "exports": { "./web": { @@ -43,21 +43,25 @@ "build:web": "yarn wasm:build && tspc -p tsconfig.web.json && yarn wasm:web:cp && yarn wasm:web:rename", "build:node": "yarn wasm:build:node && tspc -p tsconfig.node.json && yarn wasm:node:cp", "build": "yarn prebuild && yarn build:web && yarn build:node", + "build:docs": "typedoc --plugin typedoc-plugin-markdown --out docs src/index.ts", "lint": "eslint", "lint:fix": "yarn lint --fix", "lint:ci": "yarn lint --max-warnings 0", - "wasm:build": "node ./scripts/build.cjs --release", - "wasm:build:multicore": "node ./scripts/build.cjs --release --multicore", - "wasm:build:dev": "node ./scripts/build.cjs", - "wasm:build:dev:multicore": "node ./scripts/build.cjs --multicore", - "wasm:build:node": "node ./scripts/build.cjs --node --release", - "wasm:build:node:dev": "node ./scripts/build.cjs --node", - "wasm:build:node:dev:multicore": "node ./scripts/build.cjs --node --multicore", - "build:docs": "typedoc --plugin typedoc-plugin-markdown --out docs src/index.ts", + "wasm:ts:node": "tsc -p tsconfig.node.json", + "wasm:ts:web": "tsc -p tsconfig.web.json", + "wasm:build": "yarn wasm:ts:web && node ./scripts/build.js --release", + "wasm:build:multicore": "yarn wasm:ts:web && node ./scripts/build.js --release --multicore", + "wasm:build:dev": "yarn wasm:ts:web && node ./scripts/build.js", + "wasm:build:dev:multicore": "yarn wasm:ts:web && node ./scripts/build.js --multicore", + "wasm:build:node": "yarn wasm:ts:node && node ./scripts/build.js --target nodejs --release", + "wasm:build:node:multicore": "yarn wasm:ts:node && node ./scripts/build.js --target nodejs --release --multicore", + "wasm:build:node:dev": "yarn wasm:ts:node && node ./scripts/build.js --target nodejs", + "wasm:build:node:dev:multicore": "yarn wasm:ts:node && node ./scripts/build.js --target node --multicore", "test": "yarn wasm:build:node:dev && yarn jest", "test:cov": "yarn wasm:build:node:dev && yarn jest --coverage", "test:watch": "yarn wasm:build:node:dev && yarn jest --watchAll=true", - "test:ci": "jest" + "test:ci": "jest", + "test-wasm:ci": "yarn wasm:ts:node && cd ./lib && wasm-pack test --node -- --features nodejs" }, "dependencies": { "@cosmjs/encoding": "^0.29.0", diff --git a/packages/sdk/scripts/build.js b/packages/sdk/scripts/build.js new file mode 100644 index 0000000000..56ad977827 --- /dev/null +++ b/packages/sdk/scripts/build.js @@ -0,0 +1,90 @@ +const { parseArgs } = require("node:util"); +const { spawnSync, execSync } = require("child_process"); + +const targets = ["web", "nodejs"]; + +const argsOptions = { + target: { + type: "string", + short: "t", + }, + multicore: { + type: "boolean", + short: "m", + }, + release: { + type: "boolean", + short: "r", + }, +}; +const { + multicore, + release, + target: maybeTarget, +} = parseArgs({ + args: process.argv.slice(2), + options: argsOptions, +}).values; + +const mode = release ? "release" : "development"; +const multicoreLabel = multicore ? "on" : "off"; +const target = targets.includes(maybeTarget) ? maybeTarget : "web"; + +execSync("rm -rf dist"); +execSync("rm -rf src/wasm"); + +console.log( + `Building \"SDK\" in ${mode} mode for ${target} target. Multicore is ${multicoreLabel}.` +); + +const features = [target]; +let profile = "--release"; + +if (multicore) { + features.push("multicore"); +} +if (!release) { + features.push("dev"); + profile = "--dev"; +} + +const outDir = `${__dirname}/../src/wasm`; + +execSync(`rm -rf ${outDir}}`); +const { status, stderr, pid } = spawnSync( + "wasm-pack", + [ + "build", + `${__dirname}/../lib`, + profile, + `--target`, + target, + `--out-dir`, + outDir, + `--`, + ["--features", features.join(",")].flat(), + multicore ? [`-Z`, `build-std=panic_abort,std`] : [], + ].flat(), + { + stdio: "inherit", + ...(multicore && { + env: { + ...process.env, + RUSTFLAGS: "-C target-feature=+atomics,+bulk-memory,+mutable-globals", + }, + }), + } +); + +if (status !== 0) { + console.error("EXIT"); + process.exit(status); +} + +execSync("rm -rf dist && mkdir dist && mkdir dist/wasm"); + +// Remove the .gitignore so we can publish generated files +execSync(`rm -rf ${outDir}.gitignore`); + +// Manually copy wasms to dist +execSync(`cp -r ${outDir}/*.wasm ${__dirname}/../dist/wasm`); diff --git a/packages/sdk/tsconfig.web.json b/packages/sdk/tsconfig.web.json index f63b38c762..1cadc0ce71 100644 --- a/packages/sdk/tsconfig.web.json +++ b/packages/sdk/tsconfig.web.json @@ -3,26 +3,29 @@ "include": [ "src/indexWeb.ts", "src/initWeb.ts", - "../shared/src/shared/**/*", "../crypto/src/crypto/**/*" ], "exclude": [ "**/*.test.ts", "node_modules", - "scripts", - "../shared/src/shared/**/*.node.*" + "scripts" //, + // "/src/shared/**/*.node.*" ], "compilerOptions": { "declaration": true, - "outDir": "dist/web", + "outDir": "dist/wasm/web", "module": "esnext", "moduleResolution": "bundler", "plugins": [ // Transform paths in output .js files - { "transform": "typescript-transform-paths" }, - + { + "transform": "typescript-transform-paths" + }, // Transform paths in output .d.ts files (Include this line if you output declarations files) - { "transform": "typescript-transform-paths", "afterDeclarations": true } + { + "transform": "typescript-transform-paths", + "afterDeclarations": true + } ] } }