From 3f543f19a038289d9b6cbc7fd480976d7b81fdc1 Mon Sep 17 00:00:00 2001 From: dndll Date: Thu, 14 Dec 2023 13:09:03 +0000 Subject: [PATCH 1/4] feat: blinded, cached & reusable proofs --- Cargo.lock | 155 ++++- Cargo.toml | 28 +- fixtures/batch.hex | 1 + fixtures/batch.json | 1315 +++++++++++++++++++++++++++++++++++++++++++ fixtures/new.json | 277 +++++++++ fixtures/old.json | 281 +++++++++ flake.lock | 18 +- src/client/mod.rs | 167 +++++- src/client/proof.rs | 605 ++++++++++++++++++++ src/main.rs | 2 + 10 files changed, 2775 insertions(+), 74 deletions(-) create mode 100644 fixtures/batch.hex create mode 100644 fixtures/batch.json create mode 100644 fixtures/new.json create mode 100644 fixtures/old.json create mode 100644 src/client/proof.rs diff --git a/Cargo.lock b/Cargo.lock index dff20e8..68af862 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -468,6 +468,12 @@ 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 = "chrono" version = "0.4.26" @@ -609,6 +615,19 @@ dependencies = [ "crossbeam-utils", ] +[[package]] +name = "crossbeam-epoch" +version = "0.9.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ae211234986c545741a7dc064309f67ee1e5ad243d0e48335adc0484d960bcc7" +dependencies = [ + "autocfg", + "cfg-if", + "crossbeam-utils", + "memoffset", + "scopeguard", +] + [[package]] name = "crossbeam-utils" version = "0.8.16" @@ -784,6 +803,9 @@ name = "either" version = "1.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a26ae43d7bcc3b814de94796a5e736d4029efb0ee900c12e2d54c993ad1a1e07" +dependencies = [ + "serde", +] [[package]] name = "encoding_rs" @@ -928,6 +950,15 @@ dependencies = [ "percent-encoding", ] +[[package]] +name = "fs-err" +version = "2.10.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fb5fd9bcbe8b1087cbd395b51498c01bc997cef73e778a80b77a811af5e2d29f" +dependencies = [ + "autocfg", +] + [[package]] name = "fs2" version = "0.4.3" @@ -1027,6 +1058,15 @@ dependencies = [ "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" @@ -1331,6 +1371,24 @@ dependencies = [ "either", ] +[[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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "25db6b064527c5d482d0423354fcd07a89a2dfe07b67892e62411946db7f07b0" +dependencies = [ + "either", +] + [[package]] name = "itoa" version = "1.0.9" @@ -1375,12 +1433,6 @@ version = "0.2.147" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b4668fb0ea861c1df094127ac5f1da3409a82116a4ba74fca2e58ef927159bb3" -[[package]] -name = "libm" -version = "0.2.7" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f7012b1bbb0719e1097c47611d3898568c546d597c2e74d66f6087edd5233ff4" - [[package]] name = "linked-hash-map" version = "0.5.6" @@ -1409,15 +1461,6 @@ version = "0.4.19" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b06a4cde4c0f271a446782e3eff8de789548ce57dbc8eca9292c27f4a42004b4" -[[package]] -name = "lru" -version = "0.7.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e999beba7b6e8345721bd280141ed958096a2e4abdf74f67ff4ce49b4b54e47a" -dependencies = [ - "hashbrown 0.12.3", -] - [[package]] name = "matchers" version = "0.1.0" @@ -1439,6 +1482,15 @@ version = "2.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2dffe52ecf27772e601905b7522cb4ef790d2cc203488bbd0e2fe85fcb74566d" +[[package]] +name = "memoffset" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5a634b1c61a95585bd15607c6ab0c4e5b226e695ff2800ba0cdccddf208c406c" +dependencies = [ + "autocfg", +] + [[package]] name = "mime" version = "0.3.17" @@ -1654,9 +1706,11 @@ dependencies = [ "axum", "borsh", "config", + "either", "flume", "futures", "hex", + "itertools 0.12.0", "log", "near-crypto", "near-jsonrpc-client", @@ -1665,12 +1719,13 @@ dependencies = [ "pretty_env_logger", "prost 0.11.9", "rand 0.8.5", - "reed-solomon-erasure 6.0.0", + "reed-solomon-novelpoly", "reqwest", "rustc-hex", "serde", "serde_json", "sha256", + "sled", "thiserror", "tokio", ] @@ -1700,7 +1755,7 @@ dependencies = [ "once_cell", "primitive-types", "rand 0.8.5", - "reed-solomon-erasure 4.0.2", + "reed-solomon-erasure", "serde", "serde_json", "serde_with", @@ -2209,7 +2264,7 @@ checksum = "62941722fb675d463659e49c4f3fe1fe792ff24fe5bbaa9c08cd3b98a1c354f5" dependencies = [ "bytes", "heck 0.3.3", - "itertools", + "itertools 0.10.5", "lazy_static", "log", "multimap", @@ -2228,7 +2283,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f9cc1a3263e07e0bf68e96268f37665207b49560d98739662cdfaae215c720fe" dependencies = [ "anyhow", - "itertools", + "itertools 0.10.5", "proc-macro2", "quote", "syn 1.0.109", @@ -2241,7 +2296,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e5d2d8d10f3c6ded6da8b05b5fb3b8a5082514344d56c9f871412d29b4e075b4" dependencies = [ "anyhow", - "itertools", + "itertools 0.10.5", "proc-macro2", "quote", "syn 1.0.109", @@ -2377,16 +2432,16 @@ dependencies = [ ] [[package]] -name = "reed-solomon-erasure" -version = "6.0.0" +name = "reed-solomon-novelpoly" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7263373d500d4d4f505d43a2a662d475a894aa94503a1ee28e9188b5f3960d4f" +checksum = "58130877ca403ab42c864fbac74bb319a0746c07a634a92a5cfc7f54af272582" dependencies = [ - "libm", - "lru", - "parking_lot 0.11.2", - "smallvec", - "spin", + "derive_more", + "fs-err", + "itertools 0.11.0", + "static_init", + "thiserror", ] [[package]] @@ -2772,6 +2827,22 @@ dependencies = [ "autocfg", ] +[[package]] +name = "sled" +version = "0.34.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f96b4737c2ce5987354855aed3797279def4ebf734436c6aa4552cf8e169935" +dependencies = [ + "crc32fast", + "crossbeam-epoch", + "crossbeam-utils", + "fs2", + "fxhash", + "libc", + "log", + "parking_lot 0.11.2", +] + [[package]] name = "smallvec" version = "1.11.0" @@ -2814,6 +2885,34 @@ version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" +[[package]] +name = "static_init" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a2a1c578e98c1c16fc3b8ec1328f7659a500737d7a0c6d625e73e830ff9c1f6" +dependencies = [ + "bitflags 1.3.2", + "cfg_aliases", + "libc", + "parking_lot 0.11.2", + "parking_lot_core 0.8.6", + "static_init_macro", + "winapi", +] + +[[package]] +name = "static_init_macro" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "70a2595fc3aa78f2d0e45dd425b22282dd863273761cc77780914b2cf3003acf" +dependencies = [ + "cfg_aliases", + "memchr", + "proc-macro2", + "quote", + "syn 1.0.109", +] + [[package]] name = "strsim" version = "0.10.0" diff --git a/Cargo.toml b/Cargo.toml index 3d921a6..4d98a1f 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -6,13 +6,17 @@ version = "0.1.0" # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [dependencies] -anyhow = "1.0" -async-trait = "0.1" -config = "0.13" -log = "0.4" -pretty_env_logger = "0.4" -sled = "0.34" -thiserror = "1.0" +anyhow = "1.0" +async-trait = "0.1" +config = "0.13" +either = { version = "1.9", features = [ "serde" ] } +itertools = "0.12" +log = "0.4" +pretty_env_logger = "0.4" +reed-solomon-novelpoly = "1.0.1" +sha256 = "*" +sled = "0.34" +thiserror = "1.0" # Async axum = "*" @@ -30,12 +34,10 @@ serde = { version = "1.0", features = [ "derive" ] } serde_json = "1.0" # Near specific -near-crypto = "0.17.0" -near-jsonrpc-client = "0.6.0" -near-primitives = "0.17.0" -near-primitives-core = "0.17.0" -reed-solomon-novelpoly = "1.0.1" -sha256 = "*" +near-crypto = "0.17.0" +near-jsonrpc-client = "0.6.0" +near-primitives = "0.17.0" +near-primitives-core = "0.17.0" [dev-dependencies] rand = "*" diff --git a/fixtures/batch.hex b/fixtures/batch.hex new file mode 100644 index 0000000..169684d --- /dev/null +++ b/fixtures/batch.hex @@ -0,0 +1 @@ +078f987e29655eae4abcd21a325701f29e25b0d95ea2b8424f5d44aca71e8421180000002a4a110aaadae7249dc588f970ea9afd954dfef3e8a42b90ccd45d45ab3e407f3272deaa61d9bd57e4946634398a38c8beadfc7fcca913e1d674764938a6a1370100000001cb142747f40271c041b4fd44c729c16513461168e4278b61e6ffe0bf270003c9000200000001355cfe22d3d7768a502a1ce230483660c3ab7c497d5512b6f831930393686ec601011c4e49559b5166487af6d2520f13e3239858a06461342c0bf808ac7218417bdc000a00000001f7de0fac83512df04313f8b0de80c2c728d381fbc8b829248f8e597310df2cda0101e6f727987394fa64da4767430408b412edb712e37e991e90719318187b91fed90000000000000001000000000200000000030000000004000000000500000000060000000007000000192a592de69635981c500d812abcc4b01a0f5cc78761092cdaaf6091ad6f3ac21db4e0b1eb35cbd93d26dcc1769c1cdfd0138e0dd820c3c7bb319b010e159f53c2ecfd00d6ea5d13bc4178691efbefae55929ed46db1992e7b200e713e1a025c18fd16a6b9b35dad7481ded98702fff2138120313c8c0312969de7a0ad77782e9131fa931ac9af6fb74833a999815e29171350154dccd3cb3b1859386b6673f1e7667f2b4042e508bf2cd0ab5112be3e24c75bc6bfdd880c6d97c148cb8aaa400300000001bb307fb8444eec779f4c07cca199d903552579867bd0345dff5f4115ea7311630101a9c2aad17465a2b0483865cb9c7baa2aaf2927e810bae7b314a7b5cd9b88d25e0001b34c6229b4a7fafee3883c03a12dfb145c32d6f9929a8af8c6a07442483cffe4010200000001f4a35c9dc28a8348ef540f45eb90f9321d533ff2d1a039efe199ed2656e79a8e010186cb8fce87b6fc0d9f729c00cecdbad73607f6a596b48a6bbfc8b6e92df59b43000a00000001c2ecfd00d6ea5d13bc4178691efbefae55929ed46db1992e7b200e713e1a025c0101330f027c029edd47ee255b129f8f83aa887552284e52d44417c5e65ab56acb2e01000000000000010000000002000000000300000000040000000005000000000600000000070000004859f5625e88975b766f7da748a32a46fbe887eb4aa365d8445f924b771f036dfd7700f5383c5468a5d755ff86a8ffc1d269795dfefea22f60c432c1d41bcde9f691d858b09ccb116b4c2dc6aa6278dde88c451a5e39515d71b15c4d2f4203176d928a094d952b0dd9cd45ae7cc114026e333d75c8d4754623830d5f1a439c14708006bc14da54fd712deb56c5dc753b850d8ffd9a8b548d865f94494188cefd27d2e7f12f791b84990070894ff19c079e9aca19a149ab2ea638cdf508538503000000000200000001540fe1a631970a669b10a0789dcbb03b4013ff4980827be04bd5b42c14ba56ac0101a3a675fdcf550b9fa1375a5073c78b629c98ad814994fbd006b2b9227f99d62b000a00000001ea10864f815dbf2cf7b10db21a057340cc20f359642076803e89b52f6fb97eab0001a9691a36af9a6023657f7306c655b4cac06667af1ca3f1fef304c8910b0f7e9f0101cef440ebcabc01e63fbf55b674cbfea9e14e7c2108b6283b1161b6e23df6902b010001000000000200000000030000000004000000000500000000060000000007000000e747253eb1ecda9569304d0c49e864a0e887499964699ba36a68723e22895ddb383cc0541fbc7d7cfddc0c958c89a77b5d8474e5b400b6ffab7c70a362caac27ea10864f815dbf2cf7b10db21a057340cc20f359642076803e89b52f6fb97eab00d12fafff76185148b7e448dc06fe32b483d66ecb331f23369d129bf8909233d944e8ccc350accf6e3decf0223a45a0570aced7ed60212699b95072144cb934f85b558f641576f5a46838ae6ce1d625f597d62786999670ded2e9fa87c7b8b00100000001286b2489e57be281f9f4bbf78aad1e9285c240627950b6c3b733b7bf7af3661a010200000001bfcf6cb2309837c96e72105665b2b0c3ab452004cf26eff63c6bcf169cc689180101e66c726eb86615e454e93e45f8b97920d98beb7a64e1d0cb7d496b1b4e6895d2000a00000001274d338453c1edcdae9041b9b38dca1499a5fee3608bb0c9e016fe977d89e65701016de6ab7b1b27cfe0b9ba86efe68c6cacc314880e9731d6dbe4b0c5d9c33c5c410101007b40600b5b761104f1dd0dafc15478faf817ce069fdf904ea2e0a2d8b61cfc0001fd5bcf4d05d3e482ce98506024f97dc9bbfdd1cf1acdae52468d94d67d2a2c1d000008000000000900000000040000000005000000000600000000070000001d8e2f5bfcbef96b5fe1c4b7ef3525cf99a324e12c0afb0906fdd7e19b4d8824c0f7ed082d6963660b3fa664fae2b2a258ef514db298a8b1de0536b7a0da169dfa3ac9c4a34cfe3eb64f32cc1b6d38f4dd323efec66d554b5f84e9edcf8c589938a870948e94780b12a7c71b982745212e7b31f0a8116f026923453d691b2657a01fa6e94b2942e779673739fa2fbfcd987c1c4f48609d5df1c661a6abfac2e03266269a2731a918a82172be09d6128f25945b48a69b08849d59c31af966477d000000000200000001e792efe1827cd68a663e140af1c59590041c34e9f30e75d7413bab4acf6ba4b501000a0000000a00000001aa69210ac258cfb30537471f83eb239928841618c7b5e0d82f8024ea35e699d60101cfab7c39c6fe77c21ecdd753dd8074c9fe49a0b51c9592d99cee276107ae179a0001e8452f15a1033b300e9d67563046cf1c629ad52ed9317cc690b964a18a6fdccd01016d634dd7da82438b0e055b01588cab96a9527f8e23e2b7d8253fef7bc18dae21010008000000000900000000040000000005000000000600000000070000006649fbbe15972f2251e904f744abf94dbe4e4d823096e875352b2d75e2fcc3507d5d9d4142d3be0d7d817a5559003e6459018b626d2951cb415bb721aa7879abd5b044e00f6caf78c847404e38fb3831b40ca38f430a54aed277438c5a7bec00ff7b71d40ce69da46e77ea475cc4acb09a70ce604986274031971c1b32cf27c68bacc72c3f73c4d7b5a342b3c4d0a5d33224bc47e1e1a978a286a99462f4498a86c553023353ac85605bd295d5ecc128003630d8a7ab2e0624d50f681cf9f7c70000000002000000016e1a879de1845885333b417fb2de5fd96da26040566c348ad7905d4a65805e3401000a0000000a000000011f59ffc3d851bfa869135048d682604893e443d0261992f8361eb2dd4e10a1f60001ffc3300abf85de1a8bd94a0c204ff5c04d856883034c891e4cc8dea097e433d1010119f5fe1da74e48e133faddfa2218550204f761daf6cab1f8459b5488c543da9f0101249789a03f6e722f307f7cd53c3a5ea7e901eb12929dc251d61308bb8456dbac01014d5e1e082a43e4dc6f124d0bb0e6dc872f0919b50d3f4a921a496a55be7650e800000b000000000c000000000d0000000006000000000700000070069f8d319f58119f21800a0e963413d4f4756fb81b065993f54251cf23b3252c4954c5ae9c1a601cdd42e8b96f8771a0e23379c030a22bd4a07e25301b9a021f59ffc3d851bfa869135048d682604893e443d0261992f8361eb2dd4e10a1f64337ceba57fa75801e03fdc49e71dea911050610ee21b7ec2e438868f76907742755443a83f542141699218760be77f52b10076037dbbff5ea6f284175e1a3aebcee2024e760927c82f2a89b5379698825788a76de91da5a9912c35ef02be12401000000014a90f685ba1d96e1a190fad2f52be0c8ad0ce3424efccfea24b33f291f2cc563010200000001aa38f083bafb64e4c09b5e9febf77f8a691f5bbc016be7a8e9e6418e57ccfd8a0101aed2fb1f968c1211b1a22dda0ee231588ad726065a6b1f5757281663a01d2949000a00000001ad5c83c9b0deda385ee27a1449428de084871654f820b131a5b16bc71f87436a0101f1498b45c33700325d83fee3e6cfde4155c03b9567b50dcef85e920cc1064de30001dc3a30002337fa10c919bd6ac007be08b188e0b209587336c424a59b24b8915500016acaf61a111ad6805e2700c88d8cdd5ae5ada9ca0d84ea887dffa4cabcd36e5000000e000000000b000000000c000000000d00000000060000000007000000626c654e239d049b073b7fce98f77d5508d09672adb81869d710557a2350377ba1c2faf6e47db0c59573c3a5b0561cc684a9f38bbbe5758ddcb4f4b6a94acf4115682edf44f54d1e4be6678b662785cdccd970d9bdaafb082dd5e997dce630d15ebf9138eddacbd145b223563e970fcd8a25c2fdd54be598b8c9bc2250fd503ac374467a8cbb9103fac43611610297aba0733ca304e25803a3c4cf13cfd2ee2d56c538c1840dad58901f616452930337761672412567e4d87253757f3ce017a70000000002000000019411cfca9c39c3131b1e07f9fd01cc6cf90ab0a1b1ad2ca9fe2e99c013a0244d01000a0000000a000000011b0652c89f97ea23e395579ed2e07757dc6ed2f8d7476411c3b4318cd8ac513800019897da0c0a0c3c14d7b42c4cc7e05de6602e9db5bd4527ee8d9d0e97b4da6c8a0001e3768e1b278fc9e91cddc95b966944634ef90197b6e94fd847d19803849954ac01011e3ddc1d79260719f66bfb5b0175d13b8f814c9a68021f33b803629676fa234c01000e000000000b000000000c000000000d000000000600000000070000004be18cd67eb3d14a6458a61c8f230cee6e8ee51d41de4df2a435349e7379236798dfab33f10e2fc7f35207eb1be10b8c9f003f375f35e8aaffe6c19034c732ae1b0652c89f97ea23e395579ed2e07757dc6ed2f8d7476411c3b4318cd8ac51380270eec78db09649dbf0c090b85aa6727090d7dfc023131c94b9f89ff93701e35e88897da0cbe02f4c60e59f60844e17509db535ddef1a29cbe85228e26b6664daaba4d574f464d7dba8f65fe4f858d76f56abee8d33a24c6c0d9c7278273b270100000001474110327e49831d86713b3aa1710bf5a5bb68ea268ba556e1412668c532138200020000000107f007b66059074d0c86620029c5c07cab1cae7d6ba995e7def38f1e633b3d990101873fe99abe3f5852c86fab8bccae4acf65d71b0f582aa6cd37db8fa6f824441e000a00000001d4cd999dd26fa60466943e8e18b1231fbe2b8ae5894add5b66efc5cc8035e45b010149e5991f9411d245fc619424466418cc1921d6e987d6410b2cee61bf14a1b92c01011ffe071b8f1a12d5b79f5cca8fb83e2e5ab6fff571fb52d0f352fb862705ba8a0101c067cfb5eefd5c83ae5dbdbe61dd2824d0d46733d894acad46e299896fd9cb9100000f00000000100000000011000000000d000000000600000000070000002bfb903ea80287bd2805d5113f29bb3255ecc2ae36cbf79c2255d8af26dcb4b8f1787049ac6576be12745d769ea6aead1202518a833ba597cc4abe08183b12804baed8f69ca3ba6ecb3f004553f8353ba744b7faf9b086665d743260151d013ed313545583d2a25f86604841e2c7b89b4e9c3ac0ca9be326c9a5387b47a071e83386b5a952ab26254b21b665ca3eb8f1c051c81ac5ec860420b356c310e1ba04d289ea3c5c4198f7757c290894115bf94b8bc3ecd5a2753e3ba23711b23ccead000000000200000001732e664695aa27003a060df6902bf3443f7f454d4bba78a8e064effe466237eb01000a0000000a00000001b3c76593294782c7987fbcb59ec5064063e8a86e5c7eb8359625b80c2a0bcc1b0001c25b6152b687283c4e8fe4dbc05d4a142d03096b713e878611436eb60aad5067010196a8828f20a0cdb4ef4bf9d6ffcfba2ac652cdb3e0f2d7c304af556878dc2ef3000162e8ca26a27030925eb38760429786dfb4d44da357b77940a6bb7f378af171c101000f00000000100000000011000000000d000000000600000000070000008080e08806c13de92bc935e82be188c8a10eaac9ee20bffa9917605a0c95f07371d85c1d5572a1e3316b18cbe68bea90bb4a131ff1f047ac21db0cb6198f133cb3c76593294782c7987fbcb59ec5064063e8a86e5c7eb8359625b80c2a0bcc1b8b9af628875915cbdd1025dfe7cc27b4dd3844d879e5118aba1c801359035b53f5408b3ce5ed18309e51e8916a1c70a343628e0ebde1e64df2ee1000bb0f2acebceafcf859cf1c6d9aac69df24b8645c38a274e1debc08af741363cdd66df9c3030000000118c5e5b7888b44f630a5b5ec0d63b2b80f892d2f673750044280646c7bb0e920000012000000001300000002000000001400000000150000000a0000000016000000001700000000180000000019000000001a000000001b0000000011000000000d000000000600000000070000009e71316c67dc253b4dc8cc6a75ab42fba5bcedae2951ea17d7c7df2ce9d9e0905ad1de5039a9df6047881ad6e3bf4292cd182090bc72a819ee4eb4a54be3e9e90341d45406bd424caa8328692977cff2325cbd2f14b2f255e3c38346ad33771ebeac008084480c3620d87b51c96c5fb69db2bbde74de2f45de488a230d0825f0f5408b3ce5ed18309e51e8916a1c70a343628e0ebde1e64df2ee1000bb0f2ace18c5e5b7888b44f630a5b5ec0d63b2b80f892d2f673750044280646c7bb0e9200300000001bceafcf859cf1c6d9aac69df24b8645c38a274e1debc08af741363cdd66df9c3010012000000001300000002000000001400000000150000000a0000000016000000001700000000180000000019000000001a000000001b0000000011000000000d000000000600000000070000009e71316c67dc253b4dc8cc6a75ab42fba5bcedae2951ea17d7c7df2ce9d9e0905ad1de5039a9df6047881ad6e3bf4292cd182090bc72a819ee4eb4a54be3e9e90341d45406bd424caa8328692977cff2325cbd2f14b2f255e3c38346ad33771ebeac008084480c3620d87b51c96c5fb69db2bbde74de2f45de488a230d0825f0f5408b3ce5ed18309e51e8916a1c70a343628e0ebde1e64df2ee1000bb0f2ace10f66a9a48fab75fc7a4513d3ebe68548820e1ca8fc4a7120c5e567bb83c12e303000000019bd81e2992256221983e06136d87f52bd53a6077cc79f7c13f1a4048eef16e760001267cb40c01979ede567b4e1a197deea2ec1be40356595e8de5ea56d278838eda01001300000002000000001400000000150000000a0000000016000000001700000000180000000019000000001a000000001b0000000011000000000d000000000600000000070000009e71316c67dc253b4dc8cc6a75ab42fba5bcedae2951ea17d7c7df2ce9d9e0905ad1de5039a9df6047881ad6e3bf4292cd182090bc72a819ee4eb4a54be3e9e90341d45406bd424caa8328692977cff2325cbd2f14b2f255e3c38346ad33771ebeac008084480c3620d87b51c96c5fb69db2bbde74de2f45de488a230d0825f096b5504143663c3e4bfb005a7aab26b2f23166ce1d4712a573e29d47abc661d4c8380393ad43bfce1392bca206c95c0d67d5df769e441361bc5b6e19bb1996a50100000001125b4262acd2a14a95f193db711d44a4dba3eb08b9c91e13e27c725a5e0f3e9e000200000001a4a19c881fc87c1957b929ca473cb821b0a1853ff017fc16b2684abf71147d0601010182df7e1f2f8662bd298256fe4e1497663c96c9b63dd6a5a3f13a326585de17000a00000001f6bc2259f95db651fde8778ab54cea9f39851465a4ea5624f7e3fa1f4846f9b30101634799e91ca348435a8d661386fbe4f26cb76152bf177c30319bead13a4e68a101019dd50e1e0652552dc82491bf523f788ed3565fb88b0407f2884a8474cbae129f0101c9288423b063460de96e31135d21de2cbe66266bb298e6d3031e0131895fd18c01001a000000001b0000000011000000000d00000000060000000007000000994886f12ba7e022df3adecda3f6cc3fcd80f1f83a2baeec4e903291c50af2de6aa3f33ec31d2055b4f00186f75b88031e0b321d78e5f9b4d4aca40f7cb4ce5b4941492a84ac76a849ff1560b15ed3c9075a2564f5cbed6d951ee2a40d43181dea30b2417337a073b970b624c3b6d3a431fa3ca9271e2d6dbda8cba42f5ece61239c0ad436a55098aa6e562a0f190ac75d43d271d40c0d3082bb383e2044450035f92ff51e6157a087b2eaba0733d734127bbd7c8dda362c74d5eb95962b997d00000000020000000127808ae35066b0f724ee609105f0b20e0cd7078557d8cd5ba5f13b514731b04c01000a0000000a000000016a86a47a2e51fcffaa7e6f2d7911a171042b47b145a2f93911603569f12624ff00017d7a5ba381ba108b3283ef332f4e222d14dadb005d83c9e297d96b3f49470b5f010177bd881d127fae17ccbf67f8ca26e18c033474e6bd6e7efdecf7abff30eff6c600011b5899f660f55619ac652370bb07eb874de8a5c54107918a5195f9426f3ab9a800001c000000001d000000001e000000001f0000000020000000002100000052fbbe6901f6bcfd364ad7ed033a8b3b1b8ac150154012eb2594d2cd0b64ca44a98ece6ab23bcdf160f3be3868e646bc4a4d0f5d606fccda3019f9f49869546a6a86a47a2e51fcffaa7e6f2d7911a171042b47b145a2f93911603569f12624ffaeb64f08dafa846821040574529198920a10b6fa10bdb1b6f7ddfc9b2ac841fe9eac7eba37c320c05ea02fc64de05b5ce9be136fa3129dc896df2c82a1b93c3b8983ee77327ddd2e2ba0aefee0dc5a6fbe9f73c8068043cf146c92e19eecd04100000000020000000148c30d29f60c542658152524e81ff854321a7603af33e349f0c1236d3fe3be6e010171f53aa86232a27940e28c1b98ceee8667556c6dea99556167a13ef639b53e82000a000000015da9acea7619441ab489dcffcd40e7f911cf21e78069b0944d3b53af454804920101fa0f1b78a13278a2e450f23fdf2df91eb5fdf4c956e13c9c7047eb25a104cd5d000106739c3bb7480615e940075157c9ec3bb16dbc0e8bcd598b0cb4448f9f7a50390101264fcb0212bf727ed12e9d7e0e7404e9e4a345ee0a0fb9a73d1ed56d15ffcec501001c000000001d000000001e000000001f00000000200000000021000000f7a4b6c816c93268f5f9553f32f43f5dc2e9690d946a004a26fd81f3cc1607c13a5252e88202c68e35217d4b0c970fd30925b7f34e52638d02753dfe0e2c5d8b81f986e33b6558e5358cd2c4f6a38601101995646cc036e19b0938e9699e7c9e4fac304b6b380fe8525fcf6cede100a7cc9eba40185f8124044d824bb93e39a039d450188da15d3236f1a0784bfa31dafa087d9e171d9a5e09971655568606e41805b44ac6c8b29d45082d77eb6c2e7a133df8d8d446aa1a613e0ec2d1c86dc8000000000200000001c40cd88c40482c6129a40c54d53625fc0747474b577ca1fdca61b271664b6e780101aadb3623bb11efbfdd0f7d91059e458fe6261631d70b3f7f2692b43f1f08098a000a00000001d98034f8b4a08f1edb4d9e78db35d1b086558da038b1b820c87438a8416821540101a4420222b1e72b4ab750b2d831971de0b2672b502b1220ae0b5ae29027528a9a00002200000000230000000024000000001d000000001e000000001f00000000200000000021000000d182efca57d00036341f73a3302fd57885d96589292eb388df76f48e4a4f6bfb508bc071e0f08634456d294358127e14cab02af088f0dad0072abacc9e3efffe76830d10a51e79ed11ad93dc739b8bed49a5de0c04bbed1030dcd156fc9963e84a9255b30ae36aa2b2e049bd2907be6e94c61c40529066fff7ab9c5ad7c7bf6576830d10a51e79ed11ad93dc739b8bed49a5de0c04bbed1030dcd156fc9963e81086dec486a87a79399feca1803a15abd5654b055adb71b31ba77374e2cd33030200000001287e97f1a4cee6037829a2daea72cb4977e3b5a5db69773613ae3cede1821e9f0001be06c6331b15d668df186d14e32192d12cfabbcaadd08ab27513e3f17afe4b960002000000013dc5d60bfab756f922127456a9e7ff141eddbe0b975dddd119821449452e4b5e01010a0f9e6c4d921f668f4cefa5585ffcd6759ef01927c91ce946f8169d0ff117cf000a00000001bed5e23504a07b7a9c5dcdda7e29c9ca38173abea835e0e258d6721556d520250001c181afab543980803088af9ec82741d1b3976798e24c1e1a56c3a30d63d94cd501002200000000230000000024000000001d000000001e000000001f0000000020000000002100000024e68f7a366dc02ae86b68c56f94090c4a3e324b4402e2e937411023e9a28d0e52d0f1540814a0208335728946322bf1b99ca33e074480b2636f7151e6f24c36bed5e23504a07b7a9c5dcdda7e29c9ca38173abea835e0e258d6721556d52025bcff020270c3efce692ec958e4411ca982988bb37eb561672a187c3b0b5d9363aa172a401ca6f765aedaf30ebcaebf0b352a02c33c85ad06a2813e7ff8699b2d12171304e8d1b6766515a56ed7b1dabb221025eeaf3c24e5a06658e57ac5b2ce000000000200000001ac4a9c1bd90684c9418cfd9483c4af7ccb4e7779a727b72c6e6f478c0b174d2e01011e65e712d6d1db616c0d7777a90b62f3cd6d0225a8864b1ebe08d03b90cdf258000a000000016fdb2843c087f52755fa72facbe85c7d88f4dd1a47a89fa5cde0830681396c850101886c85f363d9ccdc741df5e9f3819874f8b19219233a6dad0f45442e45cf1288010123090c3d032e5fa494ae99e635c48fdcad1e38c8e1f6ffae2b8db1df9c23eafe000181c838e37e24ea477dde49057911dba7df5da3d71fbff230df79abb2a999e3be0101f1a349441d2df8cc99007892cd68132da22b5b93d6702e0611b58658af98dc75000025000000001e000000001f000000002000000000210000001d5bcb6bb6303f65c7aaacb306419cc073d68199ad54d7889725dbc6217be3d63f40dbcdf52dec28e254ef169da7f1b6501e4ad7c1374b167b9678eb30665e386ef630c794af782da8e525d8ab47e093c255638c50481dd8215149d1d7efee56a15703de1223e5db6557f902b15fa30ab7769a2069403a6452f7687bd7b526adec303379e1b1dfa33ad92c82c5a8abbe14ea13e36ced8b1a61c56329c943ff1ead1b20f29a919011485c7abf37ceab609a6c129bdd5d9f1fde1c35904b29f608000000000200000001052bb790d38593df6455520d9eb4ec1d47d032e843ee7c26326cb8f12adcb3390101589c0aaf416b67f28cb9192bf6d1768cf683849bc95441821972cbaa2be29e13000a00000001af720d4b887b079e3888a033950647237e4c9897326b0e1bdf595c8aa90f980800012197fbd4b78c713d2cc91d1a2943b1427f8ba7b990a19db1e74f19e5219ac2240001230fd37c82bf41ce117f15e077aad9c08d8525b3905fcee1f945082df755078d00014c33124bcab988c9ce036b79d7d80ab5583a5501d08d84041273ecbc9925ee810001615ec5fd999f62150b37f428e289279aea0b7cd836adb86616172c9894cf94b5010025000000001e000000001f00000000200000000021000000eb82e32ba9b7289a38d2131c11c5700d272b8ac4a7892a4dcb1b6f912433ffa74a4b6373fc0d245d543ec677ff3f2654160d74e5abd44e619560ea1c32239bb0af720d4b887b079e3888a033950647237e4c9897326b0e1bdf595c8aa90f9808469528eaa9ec66154f059d66224b38cc3f2d9640f6893b16f46a8c6fe9aa8509b231ffa3047c41ff6a9b30dd01c3fe7758c209f0832a015df68ec36306f58dea0c5d4f594c22deac65256b43ef2c2967cd4f2cfb0422f03a322d955c869776df000000000200000001fd4144e71ad9a5747a796b0111106eaea041438c6a6013aef445b7cd79e7432a0101575a61cd76c7cec0e5efc3213e52752f26ed8b993de6a68e4f84e69313852a7b000a000000014a54676268738c4adb23689059a1e0b6c1dfb5130508629f8171baf058882e5d01015770db8364498e9bb08fb107057d0a1e40d133434aeccb74151fab0c3554f05101019e57fecc827649543e14736e0da4dfb33631eecedbf9b1c47acbe7fdf6dfd3a50001751855cd525ba2eff051ce78a0b79e674017bc7228655fee35962aa926a1aabd000195124f809e8b25b703eb851846a104c74ee8b34492bdbf8878e1ec864fa1f4690000260000000027000000001f000000002000000000210000004d1cf7c99b757b440b5ce6adfe4d4cfa17ba94c51ac88bc2e4061d4e11188d8d5faff49858d1d8bae8a04d00179c7efac4ff759e46f1986d087d05caf043bbd95ce371617abeaf1590bc7c3e4ede8020c7d4a625036fb34e77d37d8fbacf4124225eb71f2385f28274b69ade09d544ee40e74208046a0f96cb754c595628e8bbf545529bc309e20dab1ff79a0e3d8a626eaf3cb921bc88b1060f9197d0466d1a9fa0035503c119415368614dc0ad696ac680dc53cfee9300b80bc7fd6259bbbf0100000001ffc0e5cbad8a64004fe88da10b42640b3d61cc76eae66742430dac3b239b28c4010200000001ba8be8765a4647a6cc06a1e1a29f358586692186adefad11b6cf1e118854b7ab0101a8dab2bf662db5e7b35391da8ae64fe234d9d7dd1c353b3831189bb1e3e7e4eb000a0000000112eedb369e85a3297415fea49f84ccd51727f11a4ead2590117f0dc552c108d4010101fd6e99f1443728ce30a487ae41c4ea8b96e1a5e735e19c3fcea956b97ad0da01011bab87b5d8c085e9e2ebd9a90a7b75814c65dc88a114aa26032a67e2e557c3ad01010399f09d3448ba190de06dfab31bbeb0d871a28b805682c402cecd2149cc23e50101cd9be89dc0978e57924013c14b3914a23de43327d1ec76cbc2b02a212a1289b70100260000000027000000001f000000002000000000210000008885ab2256f4e58b77f76cdc06f9e5ee5f759c0da79d06086976584c84b1f4ca2ebb349e41d31c4690984cb6da75b00147e1f997c2f5367685b26e56105648cdeac46d7de5280afc435e86ea60b80ec31854b80a1139455d00e8cd287fe1ccf1dae7d58b0aa8ba6d16d2d5217e499be31eba9ff42ba8d0eceec9c46c8b12711c63841217e5868e8cd46e0f2c2347820c4822615e7db2fe818b36a0ea416c5310665b0dcfa0d8a30dc936b1326cfe180aac544a416b668f51daf4ea05c2be9cfe000000000200000001a2b3196a01b93453a44a2f62788be12cdd54492fb8a30be0132480b7c510133c01000a0000000a0000000187257efd232c3c85f336581c043994412480cd837074d2cb0f5131763167c58b00015e400c499f828ca1934e25428df6489b028d7e7890d04308faf479939f7b6cdf0101188189b1fea867717f704a84a5615cb5a4e50223a7eca8ebdc91da37839640830101f30ea552ba51d88aa78410be1bd36fdc4ac3460187c7d73ff851c982d2eaf0300001f5584ac93f4f882dcc4bcf0261f7f556351bfc5c0513050cde65c6fb007d38ae0000280000000027000000001f00000000200000000021000000b636a0b6aa3c924763c839702b922301bfa744b5dfde410878aea01847595af2f00decdbec97eec13b75876d59547040ba64d2d32f2e3ea77f78d21c14a2c71587257efd232c3c85f336581c043994412480cd837074d2cb0f5131763167c58bd3edc08d9980a95f2ece5a374c11b2b3f5f2417a13a6e93a483468449eef730b3ed84c42629deb66d55061a6db293bcd81db64a71b604bd23b0d8bcc73209c65d6be9978071fdc8b3b2487f57daa478c58eb3bcce5483252b33c9e40305fc4ed00000000020000000108645fd56034f3a4b38eb24580d758a216070f410886afad92a51be4b9a663d20101dcaa4577d84110026f8bad525cb7cdc54c4ecaf7508c83e17b31ce4212848055000a00000001a63d673105b60b0648c4316370e61f43237555936ce9e399c0370c6362eec8a901010f1103ae905667796b7c411a93875344f6dd078116bd9e5071044367a226ab270101eeb4cf438c57b4d4230879c770cd4a6faf316f3ba5181388b9e61fa3be72ebe10001ed9bc14db451a4ec1fe89748489cfc8499f93f0fbc6135e81c0185b31f222db901018685348a62c01270c666d8a8f9e5222e8525a4d058d3caaa95bccddebf06c8aa0100280000000027000000001f00000000200000000021000000de72ed977767dcc8e850d2f645b8919ff72be13b5fda01e5321402516308f28652c6754728758724115782dd3fc7f4740bd3c223bdf0269ac870e4198089f3697b4be186af9884ae57cf204f332264bd14204b90361e4ddf1af9df5c1cceeecda9bf5ab429e63df9d0197fc6c5e548d55ec2557a0adfb94cb04cdd6818c9aa56100000004d765a763e8ceedc1f3afed0ceb004538924de48795cbf9472663346b20f388900bc7905ae1a2a9eff14e38556ba196b9918264640306d73f9ec10ecc7d9bad499003d34088b2ba701fa70048c5a86aa240504b3c6446dfc7a089813b7be4bbf960800f87554c7ad84118db5d9a7adde01af84525b4d4c16fbf86640a37ca9cffdf6f601c3be0a4805b286ab055a8582a62f0198b30b52af3fc8802f7473231398ae83c900f65504ed5451e324407179e1e796b1f26833f99e6619fafa9a0ddb6e51ecc1440010a66d365f0b50abe94f49c016df02f96bfda5b50b67ccd9aa51267ba22182c600fba8e9320a7cbfa17d4d32dbdd5d89edc8d4569bb8a3057b560ed2ff68087a450132de0b516af12c5929aa2bf5c852f726e34e7c1d8b2ccf758baccb98cc7e9e1401d1d9fdc932c5e2ecbcbbb378e0444d4724ed78131230cda61a11aaf272c6cf2c0174a52cfc6da44595a4aba19cc1e7e4589aeb5d179bdc3acf4a07e205795275e701d9096d1ad25df1654abf8c1769a347693da7aad0ea881dcc4173927e45bd239a01a7995b6b3524333d7415083ece669eb02faea9a618019c263be7371b234b51af0097c646a2a21dbcca6b5169257d11d75dfd334a426eeea243787db1df16ef7f8b016895c96cacc0d4582ab0cca68904211bc1d4fa71ac4fdc75a080a16030445614008e4aeae42d6aadf15ad68f3196d948afbb14c452c1165de6e63d4fc2b324748b0029000000ea0b11c305adf3f85813af630e62896b1505c4a220f211aea52081f45881b0a9007ec8d1f8d25e21dccb462a0e69bf4c7ddf09ee2dd0628127a02f5384a735625300804f45d47dc47c5d420b9f44a1bf3fbc8e42827378e72e556d160660a3a4b98a01037815a17995aee4684728d284ff3bd9440f77e5bfa6e623048dc19fc7fb4b0b009dd9647324c5a0dd094cbc350545e66a568e8798669cb92f4a29ffaeb0baf953011bba987e0be01f33295780528355b01770a413628c150e7e84c8848e4402468c00c282e2991b71259f2c2eadccc3cd66b2ba52c0780a810008058adad87a9fc77d00c1d7092d49a0ec04fa9254f8a8e2e1252769a71fc5441c8397179ea3d2c37c2500879433c823007f17c7577544cb2e3ca4d9697bc2681366aae1ce03ab9fa63ede00de0630ad61559b114c09ec68ef5cd073924d619c98b5efc9fe84d53b4b47124b012eeb74a6177f588d80c0c752b99556902ddf9682d0b906f5aa2adbaf8466a4e9004a5e2bd105e115d3f1fbb9f264ba8d5f6db9cc9989b3e65b3f1734ebb153e302002799443039641ef4d984ffa0979987f5c309e37f47a5e42f113b467cb754c58b00a7d48feaa40912238810fb5f0f821e4b1f794d1b82a03add0f8cc1e2357a4cee01c604c992e4f9f22a73f18570843104a2ce2d5dc2443d55c3c905e2cb5853982201bba54331bea70f5208d667d1b13204cd903e7c0a903e18f0fb37f0a287e81b7c00c8d7066f513b2648d80fcbb3c2da048fedd225faf631960f5c6442fba87673b0000274bf81093c4ad2644ab653ff21031b2cd63c10dc08a67364551f9c4d10578501e2aba29595c179f139d24ef558ad394a4154b4d4f13322776a45b82012d7024c004b17bc4bcbadde665511f19da157d7a1fe215105997c5ad22cc58e33d8a98d120144fd15bfae257059198c02c069fc72d3d5d12333f0f31766e573771b0877ea5a010a3bd046dbffa362cf0736a4f5163ccb55b217f18d1b3f6fc783f032839e232d000341d45406bd424caa8328692977cff2325cbd2f14b2f255e3c38346ad33771e005df697c41510ccaf073add075e47beb552b5f263a70acb7c0edae65d41b6c9010082984a1dacc24799b1f49f0f8b86e8a81ce501c64db4cae13381a8fc3a1be12d0034b2f501736bd5b9c7be10bb3e77253287b9145fb93533375302036d5345d93c00784bc822de1b13f5f8bf28cf0ca18a193404ae02d7b5c92f9926c20ea1568a1a007ba497afe56840ff887eb8ab24d9a2dc64df92c7f4379d90ce253c9f1b0274ef01fdd045810310774b359bc80cb4db2acf2465fa5468ea258ef993bbb99762ad2200d426cc1a273b7a5a7e51e54587d0fedbfc056f709f313e3659b3a906fc946db800b59d60d7f35b077300f4be9e61aac492c92a8f3be2d88c591bd664684b2651ed00522aa6c3ce000440172ad05ad545f2c0d22d6a670c1dfc1c8c3fa522bc721cf10195535da287737366ddf3af0eeef02e72f09a6915b2884c26e5c76e1aa9e76f3c00002028e3d536ae1b9c17977afbcf93079dbf9c989d67e99c3f56fbf0805e167301b85a776bde956dda109fa5921d12eb55317bb2791cbfce807d4c9d940c57d84b0003d950c6fb46a0bea02a91a8a4dc52e6881302923391c6b1731bb91bee0019c40160f863ac47bf0fb5704e7659c793b0e4c8e8314549407ed3bacd22f37ee4a553019cfae1a8402ba656c123b69c2d7696a40e9ea7e382b243278e6fd9c36c98caf6011ed210af81928ff21bfa8b4c8e6ec1a497b21e38b630fed9bcfb90d341e3251a00d7a71cc7e86d2825dbda87f6f022914bbaf1e4b85c1ebaaf502406352bb0dea501906ed4051da8f1c1cc20e8e117e0ec17819431174570ce75677186a4300945e501 \ No newline at end of file diff --git a/fixtures/batch.json b/fixtures/batch.json new file mode 100644 index 0000000..1c9e4c1 --- /dev/null +++ b/fixtures/batch.json @@ -0,0 +1,1315 @@ +{ + "client_block_merkle_root": "WWrLWbWHwSmjtTn5oBZPYgRCuCYn6fkYVa4yhPWNK4L", + "batch": [ + { + "outcome_proof_block_hash": "3r5dS7bXYG2w35qLXx78nHjPPs8psdYZUdStUYkvHr8N", + "outcome_hash": "4PvydJrXBTeqk39TF8J9qYTm9apcKS6Ed6kQYcZZiTuY", + "outcome_proof": [ + { + "hash": "Efjb1pjAnEszsRUne8yQb1w4eHU4WXmDBmJqMKMHRkzL", + "direction": "Left" + } + ], + "outcome_root_proof": [ + { + "hash": "4bJrNUwoYufbGY7qZTEHFUvaoQkuP6FbeWn1SWL7wExD", + "direction": "Right" + }, + { + "hash": "2uVeqwzcwpn7e1KKDhgkihK92dUV564hLUDXuNaigqes", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "Hga6RGaSK1TQ1v1Bq3ZJztqzVTgz3jSeM6DZveJvfQgh", + "direction": "Right" + }, + { + "hash": "GYbMcKnQdmkh7HemcMTTbQNLRndbjMUR9kEk8qiA8R8U", + "direction": "Left" + }, + 0, + 1, + 2, + 3, + 4, + 5, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "2hEeSce4tTwiaCQ8AHprkndeg6jXv6rvg3rn7xw8m66V", + "inner_rest_hash": "2zxnxeLSfGTFyYJjASAY9gVPVx3VXdL9424FpHCw1UHg", + "prev_block_hash": "E7ugr7jwqLMBGHZf3Dszcmf4y9B96W4isaJraZJSrxUj", + "outcome_root": "2gYcpDDNTGXZdzS68ne9tAWSvrbJ2fJMeNnhibAEWTVw" + } + }, + { + "outcome_proof_block_hash": "AmnKRhJpAvL4o6yk19gRJApkjGE7yrG7jq3p5WmKFXWQ", + "outcome_hash": "GaHpxTjVPSaru5X2ZRWoLR5xVgVDyo6icwVdYHTkbZEF", + "outcome_proof": [ + { + "hash": "Dbi8faqs4TsatmmZMjYm7LogyLgtKrbELwhNHC9D9LqQ", + "direction": "Right" + }, + { + "hash": "CRg4onaW6BACVxxxCcptZ3W4ZgBy9Wp8QeL35JRyZwPB", + "direction": "Left" + }, + { + "hash": "D4uXr1FfbtE9VLG397b5DNFquUDZDgXYkbVZ3GrxHzGj", + "direction": "Right" + } + ], + "outcome_root_proof": [ + { + "hash": "HTxxToJm6JDW84wbBhbRVmU5v96Vck1xEt5EiZB3kCG5", + "direction": "Right" + }, + { + "hash": "A5BfRnNu917uKoUfegwCKALfGTEFPUThpjbYfc1pbD7x", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "E7ugr7jwqLMBGHZf3Dszcmf4y9B96W4isaJraZJSrxUj", + "direction": "Right" + }, + { + "hash": "4SK4uDGQf2HNE96efRM4gK5kncLCok4vQEmrR9Bwd4iy", + "direction": "Right" + }, + 0, + 1, + 2, + 3, + 4, + 5, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "5sRv1mRJMEtuG4T7jdQyrP15wWNEJfX6miskUS42pLHS", + "inner_rest_hash": "J4RPuwENpZmdkiMfqhqyYMbQ4MwR7idcaDBpP8431gMe", + "prev_block_hash": "HbWHEEwcNZrChQxGJL3STkWNiB1UcRtLgAEzUtgkYp2S", + "outcome_root": "8Nj3GSGjKD7DHZC9oemufj9ezAaPyBTW8Ln2qFY6huL7" + } + }, + { + "outcome_proof_block_hash": "8a9taVBYYTNHj4aLqVuB8TqBaFn3KR614dv1Amj9E97z", + "outcome_hash": "3gTRjypFxk3J2m4ksbrhvdZynXAVDDeokWMhM4R7ktux", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "6f9Hx6WiARFDdMnmyXeoXKwDu9cvB1LykMtbRa2gupaf", + "direction": "Right" + }, + { + "hash": "C1pg4zw5GZpXr7CNk9zbfaPaFYJ6CHZCQQPMSEunnoML", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "Gkh1v29dTQeZHUBsSmp37KFR9ZnV86nX4TX3fTsKuUbk", + "direction": "Left" + }, + { + "hash": "CQJrXuSXFfLnthXpxeyfUCAnyjsEJS5XP7GVaZL2xPoL", + "direction": "Right" + }, + { + "hash": "Evs1HM41TLrjga9j36qFRWgJYszpzgWkW1Ydsmw3xRna", + "direction": "Right" + }, + 1, + 2, + 3, + 4, + 5, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "GZp6n6hEkDtfsuEfYFrXD1ZCUKuGwkjGKBGFjmMsS6f4", + "inner_rest_hash": "4nXZUEBPSyFk3EM9ovfnTVX8vEh4TiqRpp2p55WwrErJ", + "prev_block_hash": "Gkh1v29dTQeZHUBsSmp37KFR9ZnV86nX4TX3fTsKuUbk", + "outcome_root": "14C1KBk1nyuwHdSjE9t7JXjzxMatHuEosDhRtKTiC7kN" + } + }, + { + "outcome_proof_block_hash": "Fd8R2RB7eW4TbLxfAd5qr25bYgjUN8APbkwGbH2SQH6B", + "outcome_hash": "HiUtMnyW4vpw2X9nQDDWVLZuz3sub8gPemVkhCGcFNQ7", + "outcome_proof": [ + { + "hash": "3in4oHZqxuGdwbmiqzcU1NNKmMrAHb92krPHJzRN13Hb", + "direction": "Right" + } + ], + "outcome_root_proof": [ + { + "hash": "DukKRXN5Heb7SsuPQ1fhubLP1TT8yyry6MyFtXqgKdN7", + "direction": "Right" + }, + { + "hash": "GWUgXnCEjbbshR8nXDmJGAkJeQ1w5DZjm5XWWFw9gFeZ", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "3eRBHhk53TcTdqNpJqujy7fVdkbba6ZL39iqBQCHxHJ6", + "direction": "Right" + }, + { + "hash": "8Q1SnmFKxwbHiosq584mDKPTHtxmXDUmmDZi5bLgs4he", + "direction": "Right" + }, + { + "hash": "12t1FCYy8VHnF6DX8i1chbjyNMCp4GxtN3kGvjYgo9NT", + "direction": "Left" + }, + { + "hash": "J41Lz6YbXnV8Dc9e3cHxe8uek8APCXmJPNsPx6EAtM9N", + "direction": "Left" + }, + 8, + 9, + 4, + 5, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "2zNaCSmvYcxriCjGxNnLFPsrqG6nxuterVC6y46kabCf", + "inner_rest_hash": "DzGYc3CDLZXLn8EfADzZud6RPiuaaDj4E5o5Snmt2hQp", + "prev_block_hash": "HqnvBjc9vvVNiNkGXtpuCEccbSR34J3uaMtX3Kao82dv", + "outcome_root": "4pAoQPyEovYCZxPJg8WvSzwELHvCrUnBSXNTwqKcEUAA" + } + }, + { + "outcome_proof_block_hash": "Bn4E3KmVthrkzEN26285FYTCjNseqDFTgKnHwGn977pK", + "outcome_hash": "4PjjCXXYozNYaT4SXbG25SvhbWu2yPjHY7BZUmtkU63E", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "Gay8YTcTjCC2nB4BGK55NJHb142fttFk6besLKqPoNKW", + "direction": "Right" + }, + 10 + ], + "block_proof": [ + { + "hash": "CUDHYktaDykajBBphmow4grQcpd6LvWVxp2uwrRfBnBF", + "direction": "Right" + }, + { + "hash": "Eyf4EwHz37kdKEfx9dfNMehHMtNjAN9p4ghiQ5FX52HP", + "direction": "Left" + }, + { + "hash": "GdgmofkrCU5pKUZqNXkAidzi6FKVWyuzjR8gobwoJbKi", + "direction": "Right" + }, + { + "hash": "8N1GKNovpVqo7WnBiwnuzibubGpABt1JJg7gaQcJwcjJ", + "direction": "Right" + }, + 8, + 9, + 4, + 5, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "7tJ1xJLMcEoD1RqmVfbBt7d9wVmdC4rzzqXd3AaYAoaf", + "inner_rest_hash": "9SNkoN3PaoQ3RzHnVoSetTTRA1yH4TovHZrPCeLZuKQn", + "prev_block_hash": "FP9jWLcURrjeZq9Nn4GPRuw1BrS3B4viE3v7RP2MzV4T", + "outcome_root": "JCJ91DiJ7onkzNfcS4eaUewJvTbtpUBhVdEDdDWaNmr5" + } + }, + { + "outcome_proof_block_hash": "AQEUbQV7w1L5uEkeWJm77f9wuJwFzycR1SD9oNupXdr1", + "outcome_hash": "A569UNq88YJTnob5StqcyzCipyA9WQynt8pK35LXeU2z", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "8QoJyB1pW3eg8DxPSd1cJanbuECT53S8w5Z6rFz4gsnK", + "direction": "Right" + }, + 10 + ], + "block_proof": [ + { + "hash": "37PEb6siLYmuKtv1S1msEj3VmsVgrJhvwKbSN1ZyGABP", + "direction": "Left" + }, + { + "hash": "JDPb6f8UxgC37gkK6Wv8J9vWNxqXnzwDJF2Hhu8SSXpU", + "direction": "Right" + }, + { + "hash": "2kLkUbR64hjc1vN8R3XcARXEkt8LGCgBibwERSN6Joft", + "direction": "Right" + }, + { + "hash": "3TqhUCu9SGuyokb2vaRRrsPEMtP2yren9WENNX8LXnh9", + "direction": "Right" + }, + { + "hash": "6D1daBxJE5SZXVt5FgTmnZchFciUJR9TYg44ppmPDycT", + "direction": "Left" + }, + 11, + 12, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "8YJX9Z1N1u4KJfxeSZoxLuNgwYUKmZoTafCaaYxxfTx8", + "inner_rest_hash": "3yso176TyBauYDM3xQZF8yDAc3wYJMaATSaLvjud17wo", + "prev_block_hash": "37PEb6siLYmuKtv1S1msEj3VmsVgrJhvwKbSN1ZyGABP", + "outcome_root": "5XPfzKsF1HLQK6chq9nzPta5fZgEDQZdWCi3ZaaiKji3" + } + }, + { + "outcome_proof_block_hash": "3eYK12fPy78iMoGyFDvFyHWxSb1WzBu5mnqF1vn18Sjo", + "outcome_hash": "DiWFJgJoEPcRzNhj1Z3bYYrW6HXkZXwZQCDksWDGWYCK", + "outcome_proof": [ + { + "hash": "625NncKyggf6JWLXUbJme9eWEBXud75DhK95oyGT8nkz", + "direction": "Right" + } + ], + "outcome_root_proof": [ + { + "hash": "CTUfe8iEXuMrTrv2c1ew2W71Z7qCcNnM3qSnmzcNWJYh", + "direction": "Right" + }, + { + "hash": "CmSXrkzVu4ma3xKQHR7GZUco5j2AXevS3ec5gnVvZTJx", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "CfjMS2dFmuHBsTdfJ5R4umUWobEETQzCZf2rmFYchz1P", + "direction": "Right" + }, + { + "hash": "HEtJH4EmGCpmmYsbUF1sHquzDXUHBvDpVYVE2FN6pojY", + "direction": "Left" + }, + { + "hash": "Fpg9yr4DgCVvNaK8XzUMcFVTTqB5bcbFFme9n4b9erfE", + "direction": "Left" + }, + { + "hash": "8BsiXcWESKyfpUiuV5xwcgH6sznT2wpZ3toHAAntZXgs", + "direction": "Left" + }, + 14, + 11, + 12, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "7dCpZqv62dgXu8FTQhw5LmtR4aGshT54u2kfUA4b7hKQ", + "inner_rest_hash": "BtT5hFjgU5NtddngkK4qMLBtSHw7tM6RawJzhhKmSUjA", + "prev_block_hash": "2SZhhYzYQ9fVAaSuS9Tp9DaP5T31q1Ko6HDbpiTYvSYG", + "outcome_root": "7NrkJm4KqHFWpYTPQkAQizhLpoe7iCNTGQ1oLRY6Nu2y" + } + }, + { + "outcome_proof_block_hash": "E9yLTjsHKyAM2NPmpvsJEo6Ceo87L8tZJeqVJZDFK6fn", + "outcome_hash": "6qiVBWG1CxCtLgwA7tVTEFv3fLkkLBdERv8uicEFmSgn", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "Ay16LXw5XBpLUytXRXCFFxHwXReeT7AM37Ln2TCf9dtp", + "direction": "Right" + }, + 10 + ], + "block_proof": [ + { + "hash": "2pVbpHb3dtgUyN5RsyN9Sksa778XaCYZNNoYwTrjLA6T", + "direction": "Left" + }, + { + "hash": "BGfGaHGspRkV7JNg3oVWj6e8XCNwJrbt1sjfCw6aQhEq", + "direction": "Left" + }, + { + "hash": "GJvQ6Hwa2a82x8Tbu5M5kumcnuuGZCXrfKbSwBTdsRoH", + "direction": "Right" + }, + { + "hash": "333wXQzgiDKiwhBbNo4KUPKYa4cTb1CpmfZ94zgCAeDh", + "direction": "Right" + }, + 14, + 11, + 12, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "67D4C7JBk8eWXwWh4bR1bN4SH4MFLfKvWbqtPrDVixDC", + "inner_rest_hash": "BHknTrn2WxLr7chMpgs9mTQTZxXLsHUNAZAsjSk4mYmj", + "prev_block_hash": "2pVbpHb3dtgUyN5RsyN9Sksa778XaCYZNNoYwTrjLA6T", + "outcome_root": "AXhEXdR689UNzPxDFszniyU4HAPXXoD9hXgyGdXvj3Y" + } + }, + { + "outcome_proof_block_hash": "7N25WHCdNq6f42K2Lu3Jk3KVqa7UyRvqaWGvYTqUrnb5", + "outcome_hash": "FibgVKNrRhQgVHnjhVBB51QB1iZiC4RWv6tHLkRGp3ne", + "outcome_proof": [ + { + "hash": "5o9VMR4xWxaioK6v151TcuGkGzGKwHHnAd64aazuTNkZ", + "direction": "Left" + } + ], + "outcome_root_proof": [ + { + "hash": "Xz8zNimhG5Gsw6tn937kaweMRZcKDQAer5VHtdT7TU4", + "direction": "Right" + }, + { + "hash": "A6xZhTpRjYt6djdxsb7tSNSD8U1b8mapTJFcp2UXSvZ3", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "FKhGQhMfewNK5ZtNkpY9dCx16cmfmmfPBLoBvU8Hn5YW", + "direction": "Right" + }, + { + "hash": "5yTpYT6tu27ty5j32vKYqQYxSERe1qvdTpgsfJxbzhkf", + "direction": "Right" + }, + { + "hash": "39tJXB5oBEKejmjznf6e3WcsQNPwvSPVP5tEPdAhFTW1", + "direction": "Right" + }, + { + "hash": "Dx56BK27qFMPbKrxMNXKRcaDH7LxfyH3NgPCDSUQyTGt", + "direction": "Left" + }, + 15, + 16, + 17, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "3xh1sTbGbuJK7sfzCYJh2yEYWCgmD4voyx9zHhEuiFNw", + "inner_rest_hash": "HFbmky2Kxvdf587biCf1eNM3Vt41mnRhzcM1Zifo2fmh", + "prev_block_hash": "66SDNjto6jXo7PSh2adrKAhKWmg5AAToiVA3RJ6zHWD3", + "outcome_root": "FCx7vEffmUfRkPDgvDDfYTmcHiovDthCzGQh42FkwptX" + } + }, + { + "outcome_proof_block_hash": "4U8vy3NzowZhRkwEiHhzVQkMEFknNiZWSfsnW7DX5xwD", + "outcome_hash": "FArbBCt3GV9c9ttwoVd2vRKLiyHGa1BJgGyaWYKSiPme", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "8kcvRwpMWrJtvhMJVA1732triphkVUAXFJbprfCiUoyU", + "direction": "Right" + }, + 10 + ], + "block_proof": [ + { + "hash": "D6nKsiEojQYvu5kwzZ96CFTt7JGqz53XH4hKHGir2GBc", + "direction": "Left" + }, + { + "hash": "E5guokgex3h4AogwBWSK4Dn8ygA6eXNfnTj4JyeKL1xz", + "direction": "Right" + }, + { + "hash": "B97BmtsGNDLSnfW8BRyG9TDUVPmFtJwjawA2eThGLG9Y", + "direction": "Left" + }, + { + "hash": "7f6qR2dmAyMNqY1Rf7h6iyQFvLuYu7kMoBSSkkwYRcAU", + "direction": "Right" + }, + 15, + 16, + 17, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "9edAZZgpABc3kymJzjZ392jNKbfam7HsUBAhXdhqC6zW", + "inner_rest_hash": "8fQRLSKq3LJyg5dHAZ4FG2FoPUZNxmSDYfJQt7nNQVw9", + "prev_block_hash": "D6nKsiEojQYvu5kwzZ96CFTt7JGqz53XH4hKHGir2GBc", + "outcome_root": "APxigxWvxang1nid7fyBziUA4YLiK4LT1Th3WEzyYwEJ" + } + }, + { + "outcome_proof_block_hash": "HWMyCPRVrAcQKXcXk7aLBuQnTVebTqHEhkg8fMxj3UMj", + "outcome_hash": "DiTUNUxDwN53g2yg9kpNepsiocm2pY9sDqzwNpbg4pD8", + "outcome_proof": [ + { + "hash": "2fhoktUfzE4dStL93byFuEqRrbcMhpBdwxYaLuyrC9SF", + "direction": "Left" + }, + 18, + 19 + ], + "outcome_root_proof": [ + 20, + 21 + ], + "block_proof": [ + 22, + 23, + 24, + 25, + 26, + 27, + 17, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "BfVXSJW3ikZ1DCikaA22LvY6t9Y1UksZNBduRVN2qEJs", + "inner_rest_hash": "77XJVCUNgqURcZctRGf5tHQWx5SMjdVVZgKBjDauD5xc", + "prev_block_hash": "DiSh6mnHWmRJgoFdxGX4Te2sLXCfyvhe2x2uzDHaHc9", + "outcome_root": "DqJaknTVohdThkzXYb6F8rwuzktErtFonBzM4mtDpc3R" + } + }, + { + "outcome_proof_block_hash": "HWMyCPRVrAcQKXcXk7aLBuQnTVebTqHEhkg8fMxj3UMj", + "outcome_hash": "2fhoktUfzE4dStL93byFuEqRrbcMhpBdwxYaLuyrC9SF", + "outcome_proof": [ + { + "hash": "DiTUNUxDwN53g2yg9kpNepsiocm2pY9sDqzwNpbg4pD8", + "direction": "Right" + }, + 18, + 19 + ], + "outcome_root_proof": [ + 20, + 21 + ], + "block_proof": [ + 22, + 23, + 24, + 25, + 26, + 27, + 17, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "BfVXSJW3ikZ1DCikaA22LvY6t9Y1UksZNBduRVN2qEJs", + "inner_rest_hash": "77XJVCUNgqURcZctRGf5tHQWx5SMjdVVZgKBjDauD5xc", + "prev_block_hash": "DiSh6mnHWmRJgoFdxGX4Te2sLXCfyvhe2x2uzDHaHc9", + "outcome_root": "DqJaknTVohdThkzXYb6F8rwuzktErtFonBzM4mtDpc3R" + } + }, + { + "outcome_proof_block_hash": "HWMyCPRVrAcQKXcXk7aLBuQnTVebTqHEhkg8fMxj3UMj", + "outcome_hash": "29DTPvy6CadYYU2nMA8hzbUZK7gyyfNpB9QsPq6DkUa6", + "outcome_proof": [ + { + "hash": "BVML5FcvYMF37914srysXHMwqNR86RwJDqmSREDLHdgZ", + "direction": "Left" + }, + { + "hash": "3bEnHTFiEcyYN7N6RcLGRbMj6PmpsY58NEzxfkNPU3K7", + "direction": "Right" + }, + 19 + ], + "outcome_root_proof": [ + 20, + 21 + ], + "block_proof": [ + 22, + 23, + 24, + 25, + 26, + 27, + 17, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "BfVXSJW3ikZ1DCikaA22LvY6t9Y1UksZNBduRVN2qEJs", + "inner_rest_hash": "77XJVCUNgqURcZctRGf5tHQWx5SMjdVVZgKBjDauD5xc", + "prev_block_hash": "DiSh6mnHWmRJgoFdxGX4Te2sLXCfyvhe2x2uzDHaHc9", + "outcome_root": "DqJaknTVohdThkzXYb6F8rwuzktErtFonBzM4mtDpc3R" + } + }, + { + "outcome_proof_block_hash": "B9JWY5eiYgPS4YEdCVYpjc2Hfi2YiUDcFYfjoVd8Cgaw", + "outcome_hash": "EUa5ZPU3gryV79JsKWYqP1Hr4D6cdLMUyDcWdB4wR5et", + "outcome_proof": [ + { + "hash": "2Ef3q3JqnvQ7ohNbbBnEEXHvY751a31pdT4fMguRjU7f", + "direction": "Left" + } + ], + "outcome_root_proof": [ + { + "hash": "C5enxhie3SY1bT9DbYNujKu7Pj5KVoRemeSHn2Ze47tR", + "direction": "Right" + }, + { + "hash": "6u9qsVCJ2duWmJzCKByosBxLz1vRrcMBCxwQJRiVFZp", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "Hc9gU89D54q6xPkTiotqQQ3uZ4ATpLxyjA3eksfxXamx", + "direction": "Right" + }, + { + "hash": "7gYgpZzSaDBYFHVtzYoU5GarEZLK59d2BRQ3Cy9o2kmN", + "direction": "Right" + }, + { + "hash": "Bd7SGMfcoYoamoDq8NSyySid8HQp9k5myBTGUPVW6A6a", + "direction": "Right" + }, + { + "hash": "EYEnEmjNM9CjxFTtPu8dNyiMdr6WtjpG6eZGucBk175V", + "direction": "Right" + }, + 26, + 27, + 17, + 13, + 6, + 7 + ], + "transaction_header": { + "inner_lite_hash": "BKMXE1PfYHSEoQnG7eXtJdD34CeTypLyX1CJzWNZiwm7", + "inner_rest_hash": "8BHDS1KsJ55UMy2ndpcYYKUDACQYmCrVbh42dDqWiQDp", + "prev_block_hash": "5vxW4UeYrqouAFCs63LwQWjMhKpEM6aabZmbsTwpbqg8", + "outcome_root": "GmBUBAtuqnVwXmGP5KTBdkDCkwo3pCfCNSYYfZ5joRy2" + } + }, + { + "outcome_proof_block_hash": "3Q1GtpF9RPah9cG3HgS2Rc4Nq1ws9NmswQ7WkpYMrC55", + "outcome_hash": "4dgzT7sWDwppkeixFwcNCqMkaNXc5wHdh5uiSx4CaS9r", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "3fCasAkkmwJ7A7GtcKoEEabmRZ7vQibvDWYq7Fyfb5Vu", + "direction": "Right" + }, + 10 + ], + "block_proof": [ + { + "hash": "8AqJ5aEEuSNb1QTvExCdsmEaUEhtZ44aKed5eADGDyYA", + "direction": "Left" + }, + { + "hash": "9SpBDpKY6ufh2hAxbhS7zXUT4spxnx7yLU92xXofAskJ", + "direction": "Right" + }, + { + "hash": "94R941GmSBUpCEp16ori3AWYPcxQPLoixVgAWAZ4Xp7o", + "direction": "Left" + }, + { + "hash": "2qkNJsPjcU3ojZJv8Xngoo4CDwXfHAzU75VExJG4NmSw", + "direction": "Left" + }, + 28, + 29, + 30, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "6aw5LHvcoWfcjmZA9rfc3UQX16MQZELWY1aTddoMWhYP", + "inner_rest_hash": "CQtCaHorfe43v9KumQovuNzMaQh1MTe8oWeKdNBWfYDP", + "prev_block_hash": "8AqJ5aEEuSNb1QTvExCdsmEaUEhtZ44aKed5eADGDyYA", + "outcome_root": "Cm1B6vXdPKoctAPFC9G5Zm8PKyfCZfSFsgjAcSPP5nms" + } + }, + { + "outcome_proof_block_hash": "BgPyMt5WDkXi2No6HCx2W8CYMzqdEABVAisTVKrc397Y", + "outcome_hash": "AFoY4NoL3ARVz585wwV7zn5uWEHNZjFcaRBg2GNsCa2k", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "5u2rogcQMAvBMgPV7JJTVcVU3FF3QH33tZdHqCLHaBFK", + "direction": "Right" + }, + { + "hash": "8fqxCdoCborMxwpGym3kma3ZxFiABYLdPHtuMtXcS7zh", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "7Jcyi4Lc7GfuKYE326CYyM9npVpnLs8CXyAm4sVQ9vDw", + "direction": "Right" + }, + { + "hash": "Hq8HZA8UAuWX8qd4DPHkXm9whzyzxJhZ5YXxDYaVgNYk", + "direction": "Left" + }, + { + "hash": "SBhAbxofw771hJVnZq9w5u5aYPqgHURAKGdmgVkH1DJ", + "direction": "Right" + }, + { + "hash": "3aZ4b4UcPdNLQoMJm2DqH9Vq8GSxuyZcP8Mbfk7Ws7Sc", + "direction": "Right" + }, + 28, + 29, + 30, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "HfhNmHv5NnjbQyUz9TRtTjJ7MGFmvyQb9youeUraznRe", + "inner_rest_hash": "4vfTLdDbNp48ThVsDYcaadu9jb1vyi34E2xqXzP5v19t", + "prev_block_hash": "9kNH12h8DiVQkir4TD7CJETsVu69ftx9vTmiEJJjZvYd", + "outcome_root": "6N1VaEjxmdicjG29CvCkNgKJPy4somoXuCMw3YH6nRhR" + } + }, + { + "outcome_proof_block_hash": "4tk1YUH9P4iPoCxmzfBHeuHPQYXwRTsLAbNkPzqoNg9q", + "outcome_hash": "2cmq7sPMdRwuH2oc2pU2rNUnoEp9bc4qRvPk6TcuuMfV", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "ECJGeNrQ3dHfSZYRNhg4jxTGS5bNgtKX6vBN9XdfAqz7", + "direction": "Right" + }, + { + "hash": "CVxBTkZxBEPnLk5QN5Sj2BrvQzCQdHYjXrFN8rzfY4RP", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "Fe2riqWzXNQH4J55TT1p2kd6oddy3aVHbPtK45HbrBRu", + "direction": "Right" + }, + { + "hash": "C4CExCE3AH61DCGWKwKRKRYLhmpuuRSDiAgmTCxwhZQd", + "direction": "Left" + }, + 34, + 35, + 36, + 29, + 30, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "F6r1aHyovKST91xtCL4QQG1Y6M7JLySMndiZGB8pMLuQ", + "inner_rest_hash": "6RRDMRFPZ5dMNVYzxxHfMNb2MmhaAjaYw1CExKAaKqHX", + "prev_block_hash": "8yd1cSebZzL8UWtqoHrezFoCdQQB7wLrw4MdGjGjsFr7", + "outcome_root": "626b9ru2Ftauwsv61WkLLhfQkkyMVvuUvCeto651kgeG" + } + }, + { + "outcome_proof_block_hash": "8yd1cSebZzL8UWtqoHrezFoCdQQB7wLrw4MdGjGjsFr7", + "outcome_hash": "27WoaRrpCGAGA4WqHzPMkmroDU3SebJfDWQUE4heYLkr", + "outcome_proof": [ + { + "hash": "3j5GY7qi22mk4ETYYFBaRi6WQJAQ9pCirRjcsAs212XY", + "direction": "Left" + }, + { + "hash": "DnnTKLYB2AShsajo3izzS45CeWGi2mG2Yj9ghdSyojwb", + "direction": "Left" + } + ], + "outcome_root_proof": [ + { + "hash": "5A8qY6o7EGe18pciqzJp5nySvioct44XPwFjAqywZArZ", + "direction": "Right" + }, + { + "hash": "gGteAiWEwp2dPNd6knE6Cj8mVza1yr9vdWfBrHJeyyg", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "Dqwd6XnJJE69WGsCwaG6LPL13AjRdmXMmbySzsofeioE", + "direction": "Left" + }, + { + "hash": "E2NP63A6jys6w7k4ubF1172K7udS2MmjVvQD7PYrQA1A", + "direction": "Right" + }, + 34, + 35, + 36, + 29, + 30, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "3V3azAPeKh2psNx3JqpTs7roWVwDeavrMutDmhP3vmfT", + "inner_rest_hash": "6aHDq5gzvFfTPeqtnBAgjnFJVaTS8Z8hFmZiWv9jNESH", + "prev_block_hash": "Dqwd6XnJJE69WGsCwaG6LPL13AjRdmXMmbySzsofeioE", + "outcome_root": "DimBHpdaeyuS3g8p2ibfuURDLn35Cv8NwKT3KyYnbgVG" + } + }, + { + "outcome_proof_block_hash": "CSxoAQRjsbR71AvpNKxynsfnAcpev97M5XrpArmmGUcU", + "outcome_hash": "2DckF8zc4TSb9rokFxSrRenB8k2JB42qqz3BBcmqpCMb", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "CbZ7LgnQvEuwxanwx9TCvsBMH4X9GVvZLh3AcBJ7ynK7", + "direction": "Right" + }, + { + "hash": "33fMYKGtXdg5vb8oP4tXMVnBuPrRgFqRwsR1B6WVLXTd", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "8Xe5YHJbC3g34KkBK6L3DY4duwSRTKt8afBeUA5k8LX6", + "direction": "Right" + }, + { + "hash": "ABYRg4MV2tdXEmSm9oPjVakJsC3N8FPwJkaTHMLoaa5M", + "direction": "Right" + }, + { + "hash": "3MmGjp5YfvkqakpPWNkHjcx8PMAKKf8bcJebA1oa1sjb", + "direction": "Left" + }, + { + "hash": "9jcftm3j58j6khA47ruH7UJPMtPWdUSyuBcQMMWnnruT", + "direction": "Right" + }, + { + "hash": "HGFfeS3jRHU8BuV9m1ZTHoJA22TETKyhyBRyADFFjcXE", + "direction": "Left" + }, + 37, + 30, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "2yc1Q4beLQc4VeK7j4Jg8JQ6dHQ4Q9G2BjeijASbwvku", + "inner_rest_hash": "5Fv3gZhWHTdFBbiuKvPf1PXYvtpCDqrppuZ83AoaqztP", + "prev_block_hash": "8U9aaVFs5oDHn2cNfPVBtJBe3jfTZLhJhE2AWnYz8WBw", + "outcome_root": "BrobZkMDQjmchcPNUCX4PjMzWNyFEuN6QCkakd4vgFTA" + } + }, + { + "outcome_proof_block_hash": "Gtyr4my2Gj1QDY8ia8LghwrtqjjNP14Lyuy5GgctEnZs", + "outcome_hash": "CejXRLCP9P4Trm92WrHHePMxoGc8MowN1ZApFtNuHePy", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "MBhimoKq3P8TLzVRiUoLRZ36Wbr5H6U4ZKX6Ad64zPr", + "direction": "Right" + }, + { + "hash": "6xtt8PPZyxjL3i1zWPQvGQ9mwQeiQfsJCJ5jgzzCiV8N", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "CosDUzMEFjvFEPAsprD36821Gqh5ay9hUuXYYANmD8xP", + "direction": "Left" + }, + { + "hash": "3G8sQy8S4ygZhzbEyJyXxhpRWc64L2ePPZU5Ed9iKjFq", + "direction": "Left" + }, + { + "hash": "3MsGSESQ26LVyjYPFA8q1BTQRcRiN9SA5zau2iANMTPS", + "direction": "Left" + }, + { + "hash": "68T9sLqD7VDJEmWfC8EVvFQFTip5Q6nQdcaCkH9SKvF2", + "direction": "Left" + }, + { + "hash": "7Z6NA3s1k69Cbne9fR9nZtJcxWHSgaPX3c3T7VvUfqhe", + "direction": "Right" + }, + 37, + 30, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "GrLZrWNuuxdDsxdduSy9ouXGfx1Hu5gWeRw71iBztBrW", + "inner_rest_hash": "611qvrf6jZqUDFRayQjHVLehCPJEC3Vvkh4jPFzYgxfq", + "prev_block_hash": "CosDUzMEFjvFEPAsprD36821Gqh5ay9hUuXYYANmD8xP", + "outcome_root": "5kXTUr73JTRJek9BYntDUz5Evigs8B6oZMuTtxSWBG8Q" + } + }, + { + "outcome_proof_block_hash": "CzbnmsRedvnDEM3MUBSExjty8kH9y3b2WdEyXrP46eD7", + "outcome_hash": "qGR914SE71B6bBesYLQg4Y48KydcQQaa3GTuwRCfrXG", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "J3bsa5XwZiusWhHHpGatJjMzgcpTFP67vNfaPf4rtFWy", + "direction": "Right" + }, + { + "hash": "6szQSCDBVVP1GEt6XS6ug4fP1UjkNAxRHGpJt3yPpX7p", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "619pNnyUqtHKWZZ6mqXM5T6joCDAmmWh4Rts2wyViqoA", + "direction": "Right" + }, + { + "hash": "6tLHKinnYPGvNU3veoVF2SYmg9z991pHftdsMj6WyTx4", + "direction": "Right" + }, + { + "hash": "Bf7EuVhcL3kdZcxXaSQmDsEd4taDd4EQ1iQ1Vgk9BYex", + "direction": "Left" + }, + { + "hash": "8t6DvSBRm5GhcshCHMG9oYS1t26UpsXRMfGmMaYMFQ2x", + "direction": "Left" + }, + { + "hash": "B2uwaErow91RfyaA7EgeXnUPXzKQxyLXjiyEk3EBJfap", + "direction": "Left" + }, + 38, + 39, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "6C21hiRuACNnPU4q5WDgM1AJTDYxKkYAfLzRfrxhQcgL", + "inner_rest_hash": "7SXM8unGiwzY29NdBqH83a3jKTYuer3Hpe2jZFFG5oxU", + "prev_block_hash": "7FbfGi8W8afiEDCm3FX8hdKEJp1xweRzVkXYwNmWYWRy", + "outcome_root": "3KAdT3drXT6UUYph9TjDv8ng1L21F1LDubHjhuSCxAxS" + } + }, + { + "outcome_proof_block_hash": "HWSCKxHUYsYabFtJg8VKc9vGPpyU2Z5bc73RYfpyy4mP", + "outcome_hash": "Bk7Ljf5pnjHCxHfwkZEKhK3wdB5PdR8hQ5pQRidkLrii", + "outcome_proof": [ + { + "hash": "JDMZdWvPqj4genGdAEFhzR4VMnGwEwSr8vp16QaKsSif", + "direction": "Right" + } + ], + "outcome_root_proof": [ + { + "hash": "DZCZtz8qDJrDmFPVsS5EfRbFDy4Wsm7ngXaUUo1QCfd4", + "direction": "Right" + }, + { + "hash": "CN8uqmkpwkfzvzZ8gvB5x3nyZfxt8iBUKdUmkYJ1mKox", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "2GuatXrngUbrscCRriUtoAnXovYDK9Pg5TnqBZCKE79d", + "direction": "Right" + }, + { + "hash": "8mYZwhEdGe1256QmRWkc9iCQLsSVbzi8oKgo6NCp785", + "direction": "Right" + }, + { + "hash": "2s1iBFy7YwDvJoUxsmbZ8PpPZCD8awjtevaUA9375n3e", + "direction": "Right" + }, + { + "hash": "F4NN2g77Z2UdW5vjku1s84yZjk4rp4Yi7jiiKZPuYzY", + "direction": "Right" + }, + { + "hash": "EqcTwQLFyUqRNixLtKiXZ42iZGX3FW9Tiz3JP3fw3acv", + "direction": "Right" + }, + 38, + 39, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "ABvfWf3tEaF9vcHM43uqPkucGWb3ynMuoJffHQidF4M7", + "inner_rest_hash": "49RKYDm3uhX7SvLan2epaQQ9tydGWgSHhmVYvczwv8B6", + "prev_block_hash": "GoS88YTweaXQjgeG6N6Km8ZavZoJyB4T9qVwS4UgQe3a", + "outcome_root": "FjWuyieBPuQ8Qijf9uEYnACaKF297T7MZzshbwjxWUW7" + } + }, + { + "outcome_proof_block_hash": "7hUAdcGm5mMuWTLdjsXgdpUsSLRAyd6HccfJDdMSB6Mu", + "outcome_hash": "7tZ7beEAgQNJh1jZrxP2vNwQoqC2aodgqeurbx5kRZDP", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "Bx7SiQY43sSDnwvuP6hVGBRzSmXKabLGv19c4W9e55TR", + "direction": "Right" + }, + 10 + ], + "block_proof": [ + { + "hash": "A6ZCeowAt3xmhgoN1ti7QZf193CtBjUbmfhAz6u5L9uk", + "direction": "Left" + }, + { + "hash": "7Luy9EQ26cz312pG88p5hvx2yjDjt5e462rew887sPx2", + "direction": "Right" + }, + { + "hash": "2efME1zuuRrTE8TAZEafUVi7XSchdCdLU7nz8QRVtC5Q", + "direction": "Right" + }, + { + "hash": "HMo2McnGnstSLsBdSoiDS3qUMZjjsoU6x1qTdj7DrK5R", + "direction": "Left" + }, + { + "hash": "HWiyNgQXdcB2Nz1Vp9s5Z1UX7chtj7nqar32wiZQcyxq", + "direction": "Left" + }, + 40, + 39, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "DGHWpP4P3vkFaRvDdQLnZ61M6ePCnRiCrBet15MHf1iy", + "inner_rest_hash": "HA5ATMhVGgyemdq5WkoJ8o1XSeTEyd37x1xsFjEY5aQC", + "prev_block_hash": "A6ZCeowAt3xmhgoN1ti7QZf193CtBjUbmfhAz6u5L9uk", + "outcome_root": "FGHJ29Q1DPTEA3o1LDbVuPwF87qBfXnMAWHnPbAaRmXY" + } + }, + { + "outcome_proof_block_hash": "5EKaBuzxNa8sPtne3WSt5srWXXrgGvKajagjLdHM2Kkx", + "outcome_hash": "FTGpEtfdFFbazQbfoDdkzU3ZJetYYg2iL2e8NwkvGif6", + "outcome_proof": [], + "outcome_root_proof": [ + { + "hash": "Zm2va2iEbsxrZtF9PKZBcYdQ6pkqvbpiNdSguFuqVq3", + "direction": "Right" + }, + { + "hash": "FrPHNs6LUPfkskFKQEmae3qiTxhGzXVeJSUQ8mKwo5Lg", + "direction": "Left" + } + ], + "block_proof": [ + { + "hash": "CBvz1vjPT1d2KXM12Y4d4dQs1epo8tdFghVm6CdX2ZBW", + "direction": "Right" + }, + { + "hash": "21pATaPjLWmiunMQTEpDbbBnW4woqYUGxaiCouHbe1Kk", + "direction": "Right" + }, + { + "hash": "H4owaaSh1yQ5YYLVzogwH9wxfoqLAV6W4mVHKijjpQAc", + "direction": "Left" + }, + { + "hash": "GzXNkNeKWo6JoLiuejcfCgXCoHcUcot6MQ1ixj8VWuk4", + "direction": "Right" + }, + { + "hash": "A47STR2avAd1GoJUUXqTgmCapwKTnAHHmntvfbJqZgTj", + "direction": "Right" + }, + 40, + 39, + 31, + 32, + 33 + ], + "transaction_header": { + "inner_lite_hash": "FyM9oxg1xG5LG69QYVhkRsW4xzBYdq2XUYGLxVab1XPK", + "inner_rest_hash": "6a7x22aDseHD3YTuN9uvmT3duxcfAAvfWSH6zzapu8uv", + "prev_block_hash": "9JJFsJN5AVrVfgbCqdYEcet34aYLBjPhe9fo3bKgma1S", + "outcome_root": "CRd8sJwWaHUFaXxis8MdoRRiTku94vARshaXoDYhfDuF" + } + } + ], + "ancestry": [ + { + "hash": "6DP4mp6K49q4kQQSAYBo4QRRU3jXa8C5DrNaW1ge8Vrx", + "direction": "Left" + }, + { + "hash": "DgigRuAvBTvEmyX5nJLFCSuRYWSEmWQmrrAGSs9DLQak", + "direction": "Left" + }, + { + "hash": "57utWiKTjxWyZ1tUd4W9nyu7P4jzYhs3teP4m4kpSaDu", + "direction": "Left" + }, + { + "hash": "HisstLDNrDLJy4ahW7chsZGwERcgJiySTXjsHmMRUdfK", + "direction": "Right" + }, + { + "hash": "EB6aGQdtXnLwUGjDRGV7ah8Z2d6T22knUriQRYNg75JY", + "direction": "Left" + }, + { + "hash": "HaaV8tAvLP9P6aj5CZGD9aBf9RSJYy7ohGpTmnu42hoZ", + "direction": "Left" + }, + { + "hash": "27ziHAhqiCo8btQzn82vDJXSkJZJNn7y7hYD9WP2H1vu", + "direction": "Left" + }, + { + "hash": "HwNidihiNH1SgaDhgGaVQbVgkejSQvukLhzGf7TEPoyW", + "direction": "Right" + }, + { + "hash": "4RZmC5ZS7rJ3oQ9LkdbfpiYiVtYZWMxtcdexgJeusBD5", + "direction": "Right" + }, + { + "hash": "F8B161Nx13ohvHNXks3i5A94917Mc3j9dHRMX5Bn4ikb", + "direction": "Right" + }, + { + "hash": "8rLNm99SrgbyYrG9MV9NMiDvCXeU1jFXaJWrUcX4uSce", + "direction": "Right" + }, + { + "hash": "FcDoohS3gUDvd8hBpkjdCQhhKkpCr4w7BFfNXWNsJwNR", + "direction": "Right" + }, + { + "hash": "CHEiVMresScVo6BYKs4zM1uQQxAwa2CkeHWZqBhxxRbL", + "direction": "Left" + }, + { + "hash": "BDTvH8vD78hi8arZ2QpiSZyVV8ybv7yZDaBh8G4UZqqU", + "direction": "Right" + }, + { + "hash": "83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD", + "direction": "Left" + }, + { + "hash": "AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL", + "direction": "Left" + } + ], + "cache": [ + { + "hash": "GkcC5qgUiUrxJtRPbxfpADaybg85VKvDwyvb3JyNVjuA", + "direction": "Left" + }, + { + "hash": "9XuydMRVVEK5nYfwd7pHwKh6KyLWXjCsUJuFEuqxgUa6", + "direction": "Left" + }, + { + "hash": "9dsJ5tU1WJWspTeQXdXXCbefNkqmV1wLtxhwLvvXqucD", + "direction": "Right" + }, + { + "hash": "EYRjWXQ7jadU91wgMtzDciAYQr8sFeEhBDFUb5SJRc6", + "direction": "Left" + }, + { + "hash": "BdBGkEfvUGMwCy1DWyK2noqvRJk8CwdeBVCBwfiN9JFL", + "direction": "Right" + }, + { + "hash": "2sF2yNc5EQj47sJQw6vCo6R6MWukYpDvgRQqRPzP4CJo", + "direction": "Left" + }, + { + "hash": "E6HrEYk7NEWUxrjDVkjZFy8rNx9Zc32y5YnyLyCcU3eC", + "direction": "Left" + }, + { + "hash": "E3fs93nVb9JQvQJ6QMonhGJwgBAQiRa8YHgJwXhGFyak", + "direction": "Left" + }, + { + "hash": "A8F7PAvGs5j2ukYEt8YUycJsQX5nvao8HaiV4ASQcQBP", + "direction": "Left" + }, + { + "hash": "Fwgz3VUsZgR8JYVBCVApitkNRQaqiHs3SMuZ7xazqLd8", + "direction": "Right" + }, + { + "hash": "4A9zZ1umpi36rXiuaKYJZgAjhUH9WoTrnSBXtA3wMdV2", + "direction": "Left" + }, + { + "hash": "61JTPuMXGD99t8vve8b6ERPUYP6VYi2bNUJzgDF1czdX", + "direction": "Left" + }, + { + "hash": "3faT5zQzwuzts2Jri1koSTsF3nT3WyZh3XE3P6miU2NW", + "direction": "Left" + }, + { + "hash": "CJ95SeiV3LX8PaMDopjQD68721wQMZDyNLsQFsn29HMw", + "direction": "Right" + }, + { + "hash": "EKyxaEdwti1tGofq4GUMYamjZ79KuJfti531tPe6bdZK", + "direction": "Right" + }, + { + "hash": "DdVQ7QFgHmZRNmr2RXuzkNc4D4hxUM6dHpgPe865P68b", + "direction": "Left" + }, + { + "hash": "EWzi7eZJTqN1KvGe6M7nH3GksH89jCb6htX3cyBnnAfy", + "direction": "Left" + }, + { + "hash": "Ab4wfYWSHKjKvhpkH2cXnUDN64GzvLfbGDVZNervL1a", + "direction": "Right" + }, + { + "hash": "GFpwD8twK9HBgSBJgNi53j7Q3VNeK1TAKZenMYf2EtoV", + "direction": "Left" + }, + { + "hash": "648a1eGVB7psazSqFUX3Xe1qehQgQjAd5wBWz9x3c5Jq", + "direction": "Right" + }, + { + "hash": "5eJZ5Wnu2m6btsRNfnkJ1subZrEHLGH31RJBBy3tN7SZ", + "direction": "Right" + }, + { + "hash": "gwydeiGUo2j5yjZnkUxsFBaTgxH3Z4GyoNUYqAeGFVN", + "direction": "Left" + }, + { + "hash": "DiSh6mnHWmRJgoFdxGX4Te2sLXCfyvhe2x2uzDHaHc9", + "direction": "Left" + }, + { + "hash": "7Ko1DwMqGwdNRJRgetKSbfghKMRmg8KSAdri6rF1rZBN", + "direction": "Left" + }, + { + "hash": "9nngootZbThL8EJwx7FBet2Cwg3v3aQarbJZPbBuyH6C", + "direction": "Left" + }, + { + "hash": "4YiUJoSwER2LfEw7JNhB129xcn7v7u5jCZ83AD6tjrqR", + "direction": "Left" + }, + { + "hash": "96awqjzUXUJkR3v1qutrALuhKYjtP6K6BBa8LS68BZtm", + "direction": "Left" + }, + { + "hash": "9KeiNWYMvY8L8b4kPk46eL2hSJ2QGp2YzhTgEhRu5q8z", + "direction": "Right" + }, + { + "hash": "J5nLwtmSTV74e65cBmecpcLMSHsx4asGyxsoVSvFQC8h", + "direction": "Left" + }, + { + "hash": "FH9kBGUEeD5Lc7M3y74L9NEJjWVvR9AbAdaZfNhreRQB", + "direction": "Left" + }, + { + "hash": "DDwyotnDJM6XvCPAbdVihrKBLhF3CKJDVCt689ZM52e8", + "direction": "Left" + }, + { + "hash": "6Xk9qWvWRwm3MDWGM8er8geARovhHQHrzjnC8wnLGq3N", + "direction": "Right" + }, + { + "hash": "B3uUcVdrShf5E5zQQVQz1Sns6mcnnj935nnCaCmpev6o", + "direction": "Left" + }, + { + "hash": "1VSeeRdxGpybV7qFqxdALRZUnEeSinDm8uhJrwppquc", + "direction": "Right" + }, + { + "hash": "DQe2UUq9xfeSvoQNX4HnYUBwtaBHqAUP74gAS9KidafU", + "direction": "Left" + }, + { + "hash": "G2RFLaesZ48KDzWTy9NR7ePYGRPaY9t7e5ed4AZpzzX", + "direction": "Right" + }, + { + "hash": "7XXpKJ6aKVhNhC352ZZ2Md4xATig3LM97XnQTuvdRd62", + "direction": "Right" + }, + { + "hash": "BZnUwS1y8Px63EqcjGybtPK5xtpDeZMFJa6GDHYUbf25", + "direction": "Right" + }, + { + "hash": "35K1ntZFk6GkajHVhXG2PZm4v2jREiz7hiq69DPLf3z9", + "direction": "Left" + }, + { + "hash": "FWpT77u5c6KyZHrnGN1J3GRXdhCuQdot1Xq31YxDhagC", + "direction": "Right" + }, + { + "hash": "Aioj5bCB2k3mmiAZ86N4kyUMz7ptmkgQML7Vs8Z824ct", + "direction": "Right" + } + ] +} \ No newline at end of file diff --git a/fixtures/new.json b/fixtures/new.json new file mode 100644 index 0000000..c1c2f98 --- /dev/null +++ b/fixtures/new.json @@ -0,0 +1,277 @@ +{ + "block_header_lite": { + "inner_lite": { + "block_merkle_root": "EZAE5EtGtZ9gHvym5ik8udvz36v6QoaTXaa57ThqZZcG", + "epoch_id": "kfsRFy3JizrUbdbebBbF6qsbmcs6B1c7q1WmR2bAUNu", + "height": 141429462, + "next_bp_hash": "13TSrQRaeiG58P6HFNFgucGqUtw1Uzk5YtuRp6qCSvan", + "next_epoch_id": "363skYdkBLcq7mQqV8D7y9SnKymVUr2uvkUL81Ls2qcU", + "outcome_root": "GPcsQU6MbfuiF6ytJF366PqoeRVNa6c4xcwxmu77JKrA", + "prev_state_root": "BjCJq95Q5V8uTV72Wx9xS3wf7VwknL7e8c95q6TwXWH", + "timestamp": 1697116252972606168, + "timestamp_nanosec": "1697116252972606168" + }, + "inner_rest_hash": "DUcqmmQqLGvDaR9xNcNpCmAWbWgLgAeqfsBEZrqhdcri", + "prev_block_hash": "13Uxh7cLcijnPjA7Y2wxhAHVCetQuibogzrmSB5n8QJc" + }, + "block_proof": [ + { + "direction": "Left", + "hash": "13Uxh7cLcijnPjA7Y2wxhAHVCetQuibogzrmSB5n8QJc" + }, + { + "direction": "Left", + "hash": "8B5D5CfGYTxgLDGkCVDn9hLYDLsiT19unDX5JxZCsiTr" + }, + { + "direction": "Left", + "hash": "TCRtLaDGXdSBysPd4PScjZtrU3d2L3rW4M9ymYY3gWy" + }, + { + "direction": "Left", + "hash": "2R6sC9L5aAbVayxs9sfXCHRuHbuBfyWaZ3JYS5J49W2d" + }, + { + "direction": "Left", + "hash": "QtdhmXvsKEvTX174KCW2meAhSRZ92uWtDE68L5633MB" + }, + { + "direction": "Right", + "hash": "9duwyjTd8bTnDBfNG1QzcpVe4xEdDz345ZKG4jDaaqnA" + }, + { + "direction": "Left", + "hash": "21Nx67mvAPq2CAKkC5oGLNkNfxD5HbjhieXvw4Y2YLTj" + }, + { + "direction": "Left", + "hash": "GUqSFcGbYrDSNs1iSmtdTue729vfPmejfpD5UhC9R6GR" + }, + { + "direction": "Right", + "hash": "5r5uRqdgYX1cC6XunGQn14ehVoQdqYZpFt7Wzez8ecnu" + }, + { + "direction": "Left", + "hash": "9bHgUynfCmXDGmHrC62Xao45giLaubFVumhrdvM4fxvH" + }, + { + "direction": "Right", + "hash": "8enjc461ghBLKwPsGKUvmfnw8w23p594K8KJU8Xd6KML" + }, + { + "direction": "Right", + "hash": "AUZR837iTHpKPhWukNHBC2yxyqgyudW5VtBSGDEYAyT1" + }, + { + "direction": "Right", + "hash": "CU3VpwNsGY7E536b1ZGpaJXBSLjuVNp2yKguXQiMWwis" + }, + { + "direction": "Left", + "hash": "3W6Yj1j2VtJ4CWbem92zc8kkCRPwjVTSdFsZfejPhgiH" + }, + { + "direction": "Right", + "hash": "3PkDGYu1U2Jg6MTTsVtPVJMBEwYHHXxfxbRKxZ4h1THr" + }, + { + "direction": "Left", + "hash": "4MMnD19pGH3Lra8MQASZBu9pd35bPPVyzjgWPRHePmH5" + }, + { + "direction": "Right", + "hash": "Eg537w1JwemL7SYHqfXdJtSLBPBCbfPZt5jj7PZ41RYk" + }, + { + "direction": "Left", + "hash": "HkoV39svoP9UGTHtKK2UBAXrV3Lk2TgHXRB7WCw18knQ" + }, + { + "direction": "Right", + "hash": "8vQXtFKwgJwyGzVdmV9V9QSwFJGAmg9FWhSdaZh7JRo6" + }, + { + "direction": "Left", + "hash": "EWBH9QMZJJyXv4dpadBPih5tU3xGKvkKsFPeKg82xHNv" + }, + { + "direction": "Left", + "hash": "3ZSrvjnjc9pXhq2npc3sZTEhMh6eot8Xgwebk3Q6C6Kb" + }, + { + "direction": "Right", + "hash": "CiUV5kcdjZpqw6iS2nYhiJZd5Prhh2ZHM7AXVoiCzr1R" + }, + { + "direction": "Right", + "hash": "E71mnUNN1jWnyvrfKLsTAAaQ1rJtVwuRLiSjUqR8QKaT" + }, + { + "direction": "Left", + "hash": "7x3QgcmcQu1Di5uYwP29nL4uRzLFv6tDpacZJrYTCDrY" + }, + { + "direction": "Left", + "hash": "83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD" + }, + { + "direction": "Left", + "hash": "AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL" + } + ], + "outcome_proof": { + "block_hash": "2qTqYFHTNsbWHppZQ51UbA42ogKpSY8qiLDQcnvrRLmG", + "id": "j9XRxXGNhwh8GYAdEqs5BAGPaBpPFCgEDb7bUuTauu6", + "outcome": { + "executor_id": "cdk.topgunbakugo.testnet", + "gas_burnt": 3039186723463, + "logs": [ + "submitting 1 blobs" + ], + "metadata": { + "gas_profile": [ + { + "cost": "BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "4501057887" + }, + { + "cost": "CONTRACT_LOADING_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "35445963" + }, + { + "cost": "CONTRACT_LOADING_BYTES", + "cost_category": "WASM_HOST_COST", + "gas_used": "23966697750" + }, + { + "cost": "LOG_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "3543313050" + }, + { + "cost": "LOG_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "237578238" + }, + { + "cost": "READ_CACHED_TRIE_NODE", + "cost_category": "WASM_HOST_COST", + "gas_used": "2280000000" + }, + { + "cost": "READ_MEMORY_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "15659179200" + }, + { + "cost": "READ_MEMORY_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "152053320" + }, + { + "cost": "READ_REGISTER_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "10068660744" + }, + { + "cost": "READ_REGISTER_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "111966432" + }, + { + "cost": "STORAGE_READ_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "112713691500" + }, + { + "cost": "STORAGE_READ_KEY_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "247620264" + }, + { + "cost": "STORAGE_READ_VALUE_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "157108140" + }, + { + "cost": "STORAGE_WRITE_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "64196736000" + }, + { + "cost": "STORAGE_WRITE_KEY_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "352414335" + }, + { + "cost": "TOUCHING_TRIE_NODE", + "cost_category": "WASM_HOST_COST", + "gas_used": "177121515186" + }, + { + "cost": "UTF8_DECODING_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "3111779061" + }, + { + "cost": "UTF8_DECODING_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "5248448622" + }, + { + "cost": "WASM_INSTRUCTION", + "cost_category": "WASM_HOST_COST", + "gas_used": "149320340928" + }, + { + "cost": "WRITE_MEMORY_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "14018974305" + }, + { + "cost": "WRITE_MEMORY_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "3137785344" + }, + { + "cost": "WRITE_REGISTER_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "14327612430" + }, + { + "cost": "WRITE_REGISTER_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "4318576704" + } + ], + "version": 3 + }, + "receipt_ids": [ + "vTtqwAXNFtPzjvE56DGXx7cscGaMwxnedZJDXFf9Lfw" + ], + "status": { + "SuccessValue": "MTQxNDI5NDYx" + }, + "tokens_burnt": "303918672346300000000" + }, + "proof": [ + { + "direction": "Left", + "hash": "FDurSHTQEHFfTtTx7eroLF8HPCkqoTrNuAPToKM3rxVM" + } + ] + }, + "outcome_root_proof": [ + { + "direction": "Right", + "hash": "4v4RMRsH6gMk5hBGdkYDvZ3F4svupZUDaddYE7ipLWEB" + }, + { + "direction": "Left", + "hash": "y41CyyzpXZZd1Wv9DVkdaD6TdmxwSixWTUQtdFUhxEu" + } + ] +} diff --git a/fixtures/old.json b/fixtures/old.json new file mode 100644 index 0000000..03a2987 --- /dev/null +++ b/fixtures/old.json @@ -0,0 +1,281 @@ +{ + "block_header_lite": { + "inner_lite": { + "block_merkle_root": "CRicirgx88qEgqTTXBrP3e8kj2JLoDkNmjtCCLsfgUAi", + "epoch_id": "kfsRFy3JizrUbdbebBbF6qsbmcs6B1c7q1WmR2bAUNu", + "height": 141429413, + "next_bp_hash": "13TSrQRaeiG58P6HFNFgucGqUtw1Uzk5YtuRp6qCSvan", + "next_epoch_id": "363skYdkBLcq7mQqV8D7y9SnKymVUr2uvkUL81Ls2qcU", + "outcome_root": "DvxxqEhshdgePU3gnjrW9iBTEgyb1uiBTLkJsM9FhDzg", + "prev_state_root": "5GSNfFfmYtgUm16LRBgayaZi89aYQ4rKCMuiCyGWuvSS", + "timestamp": 1697116223277387356, + "timestamp_nanosec": "1697116223277387356" + }, + "inner_rest_hash": "HZNv3ipD8QpUCo7NrowN2TwxdUecWGjuArKY7GSkv1qb", + "prev_block_hash": "2SBcD81z1MUc5ws7TicyShVFHuvrDZeu1Xo4s7xYxbFM" + }, + "block_proof": [ + { + "direction": "Right", + "hash": "2JY1XGjtX4sMe5tSLaBLdsCum8Y3vm8mncxoHCPpNwDu" + }, + { + "direction": "Left", + "hash": "6CPqptR9SEffA6zw8eJoUjd8nkAGNSu7cSY67YsA57Z5" + }, + { + "direction": "Left", + "hash": "4UTbjzjUZyUfzQVeFWLa2rYfFAxQAGTd9SzZpW8KcHaU" + }, + { + "direction": "Left", + "hash": "3GkRJMa4YCwxjdcmhRT9s4i7t2WY9M5abrarA7KiGUob" + }, + { + "direction": "Right", + "hash": "7Yp6cw7FFfMqoSfBZD7o1eT1bCU69miZu7P8QHD8m4GC" + }, + { + "direction": "Left", + "hash": "71xCGevBBqwHRgaD3LiqtqEV45kjJsxVQn8QVJcfaoPf" + }, + { + "direction": "Right", + "hash": "2r3BxC8YWKWPxSCq3fg4Vzq7b5JehxKVKWhNEdHFoLYK" + }, + { + "direction": "Left", + "hash": "GUqSFcGbYrDSNs1iSmtdTue729vfPmejfpD5UhC9R6GR" + }, + { + "direction": "Right", + "hash": "5r5uRqdgYX1cC6XunGQn14ehVoQdqYZpFt7Wzez8ecnu" + }, + { + "direction": "Left", + "hash": "9bHgUynfCmXDGmHrC62Xao45giLaubFVumhrdvM4fxvH" + }, + { + "direction": "Right", + "hash": "8enjc461ghBLKwPsGKUvmfnw8w23p594K8KJU8Xd6KML" + }, + { + "direction": "Right", + "hash": "AUZR837iTHpKPhWukNHBC2yxyqgyudW5VtBSGDEYAyT1" + }, + { + "direction": "Right", + "hash": "CU3VpwNsGY7E536b1ZGpaJXBSLjuVNp2yKguXQiMWwis" + }, + { + "direction": "Left", + "hash": "3W6Yj1j2VtJ4CWbem92zc8kkCRPwjVTSdFsZfejPhgiH" + }, + { + "direction": "Right", + "hash": "3PkDGYu1U2Jg6MTTsVtPVJMBEwYHHXxfxbRKxZ4h1THr" + }, + { + "direction": "Left", + "hash": "4MMnD19pGH3Lra8MQASZBu9pd35bPPVyzjgWPRHePmH5" + }, + { + "direction": "Right", + "hash": "Eg537w1JwemL7SYHqfXdJtSLBPBCbfPZt5jj7PZ41RYk" + }, + { + "direction": "Left", + "hash": "HkoV39svoP9UGTHtKK2UBAXrV3Lk2TgHXRB7WCw18knQ" + }, + { + "direction": "Right", + "hash": "8vQXtFKwgJwyGzVdmV9V9QSwFJGAmg9FWhSdaZh7JRo6" + }, + { + "direction": "Left", + "hash": "EWBH9QMZJJyXv4dpadBPih5tU3xGKvkKsFPeKg82xHNv" + }, + { + "direction": "Left", + "hash": "3ZSrvjnjc9pXhq2npc3sZTEhMh6eot8Xgwebk3Q6C6Kb" + }, + { + "direction": "Right", + "hash": "CiUV5kcdjZpqw6iS2nYhiJZd5Prhh2ZHM7AXVoiCzr1R" + }, + { + "direction": "Right", + "hash": "E71mnUNN1jWnyvrfKLsTAAaQ1rJtVwuRLiSjUqR8QKaT" + }, + { + "direction": "Left", + "hash": "7x3QgcmcQu1Di5uYwP29nL4uRzLFv6tDpacZJrYTCDrY" + }, + { + "direction": "Left", + "hash": "83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD" + }, + { + "direction": "Left", + "hash": "AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL" + } + ], + "outcome_proof": { + "block_hash": "CKymwud3BMkyDTnENanADfk9UiBeyHVWvkJWh5XXphZz", + "id": "7LAgw7ETir15YXcKdbSuxLUZ1YNUpcDmKHjqxxAhpZ6U", + "outcome": { + "executor_id": "cdk.topgunbakugo.testnet", + "gas_burnt": 12182960979079, + "logs": [ + "submitting 1 blobs" + ], + "metadata": { + "gas_profile": [ + { + "cost": "BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "4501057887" + }, + { + "cost": "CONTRACT_LOADING_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "35445963" + }, + { + "cost": "CONTRACT_LOADING_BYTES", + "cost_category": "WASM_HOST_COST", + "gas_used": "23966697750" + }, + { + "cost": "LOG_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "3543313050" + }, + { + "cost": "LOG_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "237578238" + }, + { + "cost": "READ_CACHED_TRIE_NODE", + "cost_category": "WASM_HOST_COST", + "gas_used": "2280000000" + }, + { + "cost": "READ_MEMORY_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "15659179200" + }, + { + "cost": "READ_MEMORY_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "152053320" + }, + { + "cost": "READ_REGISTER_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "10068660744" + }, + { + "cost": "READ_REGISTER_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "6942313032" + }, + { + "cost": "STORAGE_READ_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "112713691500" + }, + { + "cost": "STORAGE_READ_KEY_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "247620264" + }, + { + "cost": "STORAGE_READ_VALUE_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "157108140" + }, + { + "cost": "STORAGE_WRITE_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "64196736000" + }, + { + "cost": "STORAGE_WRITE_KEY_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "352414335" + }, + { + "cost": "TOUCHING_TRIE_NODE", + "cost_category": "WASM_HOST_COST", + "gas_used": "177121515186" + }, + { + "cost": "UTF8_DECODING_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "3111779061" + }, + { + "cost": "UTF8_DECODING_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "5248448622" + }, + { + "cost": "WASM_INSTRUCTION", + "cost_category": "WASM_HOST_COST", + "gas_used": "8679108238944" + }, + { + "cost": "WRITE_MEMORY_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "14018974305" + }, + { + "cost": "WRITE_MEMORY_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "191895184944" + }, + { + "cost": "WRITE_REGISTER_BASE", + "cost_category": "WASM_HOST_COST", + "gas_used": "14327612430" + }, + { + "cost": "WRITE_REGISTER_BYTE", + "cost_category": "WASM_HOST_COST", + "gas_used": "267766961904" + } + ], + "version": 3 + }, + "receipt_ids": [ + "3YCUYAFa4ps5QijQdQUHZt4CDfMPnacUAEvsTvaEKtJj" + ], + "status": { + "SuccessValue": "MTQxNDI5NDEy" + }, + "tokens_burnt": "1218296097907900000000" + }, + "proof": [ + { + "direction": "Left", + "hash": "HuHcBZgtbvujxZC2gA6roFTpnzvVy4QwYS55h7CY64bG" + }, + { + "direction": "Right", + "hash": "CTsXUdmEbAMLcnVLw4XoJVaYYeLZoca1zLeFkSkjJPLf" + } + ] + }, + "outcome_root_proof": [ + { + "direction": "Right", + "hash": "AAFFM7AYBVmBXkYVwC8dNa82GjDfmQavW7okHpgyf2uX" + }, + { + "direction": "Left", + "hash": "CYv4pGiLJPiUDaQsqEyTgWeg2S2jbLhxtKymFh2xgaXR" + } + ] + } diff --git a/flake.lock b/flake.lock index 1bea00c..a6622fe 100644 --- a/flake.lock +++ b/flake.lock @@ -5,11 +5,11 @@ "systems": "systems" }, "locked": { - "lastModified": 1689068808, - "narHash": "sha256-6ixXo3wt24N/melDWjq70UuHQLxGV8jZvooRanIHXw0=", + "lastModified": 1694529238, + "narHash": "sha256-zsNZZGTGnMOf9YpHKJqMSsa0dXbfmxeoJ7xHlrt+xmY=", "owner": "numtide", "repo": "flake-utils", - "rev": "919d646de7be200f3bf08cb76ae1f09402b6f9b4", + "rev": "ff7b65b44d01cf9ba6a71320833626af21126384", "type": "github" }, "original": { @@ -38,11 +38,11 @@ }, "nixpkgs": { "locked": { - "lastModified": 1691006197, - "narHash": "sha256-DbtxVWPt+ZP5W0Usg7jAyTomIM//c3Jtfa59Ht7AV8s=", + "lastModified": 1700390070, + "narHash": "sha256-de9KYi8rSJpqvBfNwscWdalIJXPo8NjdIZcEJum1mH0=", "owner": "nixos", "repo": "nixpkgs", - "rev": "66aedfd010204949cb225cf749be08cb13ce1813", + "rev": "e4ad989506ec7d71f7302cc3067abd82730a4beb", "type": "github" }, "original": { @@ -81,11 +81,11 @@ "nixpkgs": "nixpkgs_2" }, "locked": { - "lastModified": 1691115502, - "narHash": "sha256-M4QWfCRrwBZ6HvYpJxB2bZCIs7ZCBEsF4a7mabRy3Zs=", + "lastModified": 1700446608, + "narHash": "sha256-q/87GqBvQoUNBYiI3hwhsDqfyfk972RuZK+EwKab5s0=", "owner": "oxalica", "repo": "rust-overlay", - "rev": "c6f763fe0ecbbcdbec404ff62f546485a03946e7", + "rev": "e17bfe3baa0487f0671c9ed0e9057d10987ba7f7", "type": "github" }, "original": { diff --git a/src/client/mod.rs b/src/client/mod.rs index 4f5f717..c18cb66 100644 --- a/src/client/mod.rs +++ b/src/client/mod.rs @@ -20,6 +20,7 @@ use std::{path::PathBuf, str::FromStr}; use tokio::time; pub mod error; +pub mod proof; pub mod rpc; pub type Header = LightClientBlockLiteView; @@ -129,23 +130,23 @@ impl LightClient { } } Message::ValidateProof { tx, proof } => { - if let Err(e) = tx.send_async(self.validate_proof(*proof).await).await { + if let Err(e) = tx.send_async(self.verify_proof(*proof).await).await { log::error!("Failed to send validation result: {:?}", e); } } }, - r = self.sync() => { - tokio::time::sleep(duration).await; - match r { - Err(e) => log::error!("Error syncing: {:?}", e), - Ok(false) if is_fast => { - log::debug!("Slowing down sync interval to {:?}", default_duration); - duration = default_duration; - is_fast = false; - } - _ => (), - } - } + // r = self.sync() => { + // tokio::time::sleep(duration).await; + // match r { + // Err(e) => log::error!("Error syncing: {:?}", e), + // Ok(false) if is_fast => { + // log::debug!("Slowing down sync interval to {:?}", default_duration); + // duration = default_duration; + // is_fast = false; + // } + // _ => (), + // } + // } } } }); @@ -330,22 +331,20 @@ impl LightClient { } // TODO: no panic - pub async fn validate_proof(&mut self, body: Proof) -> bool { + pub async fn verify_proof(&mut self, body: Proof) -> bool { let block_hash = body.block_header_lite.hash(); assert_eq!(block_hash, body.outcome_proof.block_hash); - let shard_outcome_root = merkle::compute_root_from_path( + let outcome_root = merkle::compute_root_from_path( &body.outcome_proof.proof, CryptoHash::hash_borsh(body.outcome_proof.to_hashes()), ); - log::debug!("shard_outcome_root: {:?}", shard_outcome_root); - let block_outcome = - merkle::compute_root_from_path_and_item(&body.outcome_root_proof, shard_outcome_root); - log::debug!("block_outcome_root: {:?}", block_outcome); + let outcome_root = + merkle::compute_root_from_path_and_item(&body.outcome_root_proof, outcome_root); + log::debug!("outcome_root: {:?}", outcome_root); - let shard_execution_outcome = - block_outcome == body.block_header_lite.inner_lite.outcome_root; + let outcome_root_matches = outcome_root == body.block_header_lite.inner_lite.outcome_root; let mut block_verified = merkle::verify_hash( self.head.inner_lite.block_merkle_root, @@ -388,10 +387,10 @@ impl LightClient { log::debug!( "shard outcome included: {:?}, block included: {:?}", - shard_execution_outcome, + outcome_root_matches, block_verified ); - shard_execution_outcome && block_verified + outcome_root_matches && block_verified } pub async fn shutdown(&self) -> anyhow::Result<()> { @@ -447,6 +446,7 @@ impl LightClientState { current_block_hash.0 )) } + fn reconstruct_light_client_block_view_fields( &mut self, block_view: &LightClientBlockView, @@ -619,7 +619,7 @@ mod tests { use super::*; use core::str::FromStr; use near_primitives::merkle::compute_root_from_path; - use serde_json; + use serde_json::{self, json}; fn fixture(file: &str) -> LightClientBlockView { serde_json::from_reader(std::fs::File::open(file).unwrap()).unwrap() @@ -1024,4 +1024,123 @@ mod tests { CryptoHash::from_str("AC7P5WhmY1kFaL7ZWpCToPcQpH4kuxBnzCx53BSWDzY7").unwrap(); assert_eq!(shard_outcome_root, expected_shard_outcome_root); } + + #[test] + fn test_root() { + let root = "WWrLWbWHwSmjtTn5oBZPYgRCuCYn6fkYVa4yhPWNK4L"; + + let old_block_hash = "7xTJs2wYhLcWKsT2Vcf4HPcXVdKKxTAPFTphhEhf3AxW"; + let old_proof = json!([ + { + "direction": "Left", + "hash": "GCvcb8M3qHWtvmSdEZFUWcTd6TwifECQZbV932S3AJzj" + }, + { + "direction": "Left", + "hash": "RYJXmEszufH799Rzsu669aUo96YCPqN8K93DWVYoj7A" + }, + { + "direction": "Left", + "hash": "C1epDr3Zy7sBLJcrwN1q8JNqCRbL9ZwGXdUdudMkKgZK" + }, + { + "direction": "Right", + "hash": "4S5gtNEiMPPLzSZ2HKQr62PEFa2sFKyGWZ3Hzm3wmmZv" + }, + { + "direction": "Left", + "hash": "3xo4CBANUZFc8bpX1X4cZHd9uEMJtxagxYjq5N9XGkF6" + }, + { + "direction": "Right", + "hash": "2ZecgrRDh4i7CcxHsS6NMRTMvd5NCxo3Mjx8prho7Gd8" + }, + { + "direction": "Left", + "hash": "51BaXGWzouQPb1dme26UmDZQ4PAJqUmcU3utoyKwQs3Z" + }, + { + "direction": "Left", + "hash": "3R2JTnndX74J1zYCnWPM4JdKvTrMwW8Tj3N7tcCmMPr5" + }, + { + "direction": "Right", + "hash": "74pTjaUkKY42tqDSy3wjRmH8X7bkjoiQiQ4AibnvShF2" + }, + { + "direction": "Right", + "hash": "FVKmSZx11uW5itjGZf6B3aCs7JRcL2i51Jb9VVeeY6QP" + }, + { + "direction": "Right", + "hash": "CQoiwaCYqcJxZTT9bqceKTGUffZiKQNKPf6YUvXihEQJ" + }, + { + "direction": "Right", + "hash": "DntvNk8Ugyu6ZdAUTV39MrPUrXQUFJKVaBUput6GMZiq" + }, + { + "direction": "Left", + "hash": "4ZXzvj47NPZXfwV6N2Z2TCs52AX143bKGTUNQR4KuL5F" + }, + { + "direction": "Left", + "hash": "6Czm39v4AvX1Xc5NZdsB94BNhawYhPAXRqyP3BephTyV" + }, + { + "direction": "Right", + "hash": "tojBg1nCgpCEpD3ssM4rTibXuCNQ6Bdfq7iByhPvukD" + }, + { + "direction": "Right", + "hash": "FHTGPaTNPPfcQ1yGuVCz1LgRxkQDjuSX4bk4L8kgDgQq" + }, + { + "direction": "Right", + "hash": "837fc2U3xTc7fMPPPvaXd2GzTkJhVWRxznk4xnzqg7kz" + }, + { + "direction": "Left", + "hash": "CAyu4UBn9JTTbq13pUGjY6nXCvjDaRzKwepwpJcpRLpn" + }, + { + "direction": "Left", + "hash": "6ejd5xCdeRefBX34FoXBv4V2hVcRa2S89dnymWbCdfpj" + }, + { + "direction": "Right", + "hash": "DYzHCZNv5VR8hR6dvYsA2CK8GaruXYA3GJmABpqB4rp" + }, + { + "direction": "Right", + "hash": "D9BmTFSNCoU9ZzKLANz8J11UBhNpWAGnn6LH8YD5Sw1D" + }, + { + "direction": "Right", + "hash": "CiUV5kcdjZpqw6iS2nYhiJZd5Prhh2ZHM7AXVoiCzr1R" + }, + { + "direction": "Right", + "hash": "E71mnUNN1jWnyvrfKLsTAAaQ1rJtVwuRLiSjUqR8QKaT" + }, + { + "direction": "Left", + "hash": "7x3QgcmcQu1Di5uYwP29nL4uRzLFv6tDpacZJrYTCDrY" + }, + { + "direction": "Left", + "hash": "83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD" + }, + { + "direction": "Left", + "hash": "AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL" + }]); + + let old = merkle::verify_hash( + CryptoHash::from_str(root).unwrap(), + &serde_json::from_value(old_proof).unwrap(), + CryptoHash::from_str(old_block_hash).unwrap(), + ); + assert!(old); + } } diff --git a/src/client/proof.rs b/src/client/proof.rs new file mode 100644 index 0000000..dae288d --- /dev/null +++ b/src/client/proof.rs @@ -0,0 +1,605 @@ +use std::{collections::HashMap, sync::Arc}; + +use super::{LightClientState, Proof as FullProof}; +use borsh::{BorshDeserialize, BorshSerialize}; +use either::Either; +use itertools::Itertools; +use near_primitives::{ + merkle::{self, combine_hash, compute_root_from_path, MerklePathItem}, + views::LightClientBlockLiteView, +}; +use near_primitives_core::hash::CryptoHash; +use serde::{Deserialize, Serialize}; + +#[derive(Debug, Clone, BorshSerialize, BorshDeserialize, Serialize, Deserialize)] +pub struct LightweightHeader { + inner_lite_hash: CryptoHash, + inner_rest_hash: CryptoHash, + prev_block_hash: CryptoHash, + outcome_root: CryptoHash, +} + +impl LightweightHeader { + fn hash(&self) -> CryptoHash { + combine_hash( + &combine_hash(&self.inner_lite_hash, &self.inner_rest_hash), + &self.prev_block_hash, + ) + } +} + +impl From for LightweightHeader { + fn from(header: LightClientBlockLiteView) -> Self { + Self { + inner_lite_hash: LightClientState::inner_lite_hash(&header.inner_lite), + inner_rest_hash: header.inner_rest_hash, + prev_block_hash: header.prev_block_hash, + outcome_root: header.inner_lite.outcome_root, + } + } +} + +#[derive(Debug, Clone, BorshSerialize, BorshDeserialize)] +pub struct LightestHeader { + block_hash: CryptoHash, + outcome_root: CryptoHash, +} + +impl From for LightestHeader { + fn from(header: LightweightHeader) -> Self { + Self { + block_hash: header.hash(), + outcome_root: header.outcome_root, + } + } +} + +#[derive(Debug, Clone, BorshSerialize, BorshDeserialize)] +pub enum Header { + Lightweight(LightweightHeader), + Lightest(LightestHeader), +} + +impl Header { + fn hash(&self) -> CryptoHash { + match self { + Header::Lightweight(header) => header.hash(), + Header::Lightest(header) => header.block_hash, + } + } + fn outcome_root(&self) -> CryptoHash { + match self { + Header::Lightweight(header) => header.outcome_root, + Header::Lightest(header) => header.outcome_root, + } + } +} + +#[derive(Debug, Clone, BorshSerialize, BorshDeserialize, Serialize, Deserialize)] +pub struct BlindedProof { + outcome_proof_block_hash: CryptoHash, + outcome_hash: CryptoHash, + outcome_proof: Vec, + outcome_root_proof: Vec, + block_proof: Vec, + // The header for the block that this proof was created from + transaction_header: LightweightHeader, +} + +#[derive(Debug, Clone, Default, BorshSerialize, BorshDeserialize)] +struct MerkleCache { + items: Vec, +} + +impl MerkleCache { + fn cache(&mut self, batch: &mut Vec) { + // TODO: this is super inneficient, we had a nice one without duplicates but + // it unfortunately copies the iterator + let duplicates = batch + .iter() + .cloned() + .map(|fp| [fp.outcome_proof, fp.outcome_root_proof, fp.block_proof].concat()) + .flatten() + // Unfortunately duplicates clones the array, so we can't do this zero + // copy + .duplicates() + .collect_vec(); + + batch + .iter_mut() + .map(|fp| { + fp.outcome_proof + .iter_mut() + .chain(fp.outcome_root_proof.iter_mut()) + .chain(fp.block_proof.iter_mut()) + }) + .flatten() + .for_each(|item| { + duplicates + .iter() + .position(|dup| dup == item) + .map(|i| item.0 = Either::Left(i as u32)); + }); + self.items = duplicates.into_iter().map(|x| x.0.unwrap_right()).collect(); + } +} + +impl From for BlindedProof { + fn from(x: FullProof) -> Self { + Self { + outcome_proof_block_hash: x.outcome_proof.block_hash, + outcome_hash: CryptoHash::hash_borsh(x.outcome_proof.to_hashes()), + outcome_proof: x + .outcome_proof + .proof + .into_iter() + .map(LookupMerklePathItem::from) + .collect(), + outcome_root_proof: x + .outcome_root_proof + .into_iter() + .map(LookupMerklePathItem::from) + .collect(), + block_proof: x + .block_proof + .into_iter() + .map(LookupMerklePathItem::from) + .collect(), + transaction_header: LightweightHeader::from(x.block_header_lite), + } + } +} +impl BlindedProof { + fn collect_path<'a>( + &self, + cache: &'a [MerklePathItem], + path: &[LookupMerklePathItem], + ) -> Vec { + Proof::unpack_cached_path(cache, &path) + .into_iter() + .cloned() + .collect() + } +} + +// TODO: add refcount here +#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] +#[serde(transparent)] +pub struct LookupMerklePathItem( + #[serde(with = "either::serde_untagged")] Either, +); + +impl From for LookupMerklePathItem { + fn from(x: MerklePathItem) -> Self { + Self(Either::Right(x)) + } +} + +impl From for LookupMerklePathItem { + fn from(x: u32) -> Self { + Self(Either::Left(x)) + } +} + +impl LookupMerklePathItem { + fn into<'a>(&'a self, cache: &'a [MerklePathItem]) -> &MerklePathItem { + match &self.0 { + // TODO: no panic + Either::Left(i) => cache.get(*i as usize).unwrap(), + Either::Right(x) => x, + } + } +} + +impl From> for LookupMerklePathItem { + fn from(x: Either) -> Self { + Self(x) + } +} + +impl std::hash::Hash for LookupMerklePathItem { + fn hash(&self, state: &mut H) { + state.write( + match &self.0 { + Either::Left(t) => CryptoHash::hash_borsh(t), + Either::Right(u) => CryptoHash::hash_borsh(u), + } + .as_bytes(), + ) + } +} + +impl BorshSerialize for LookupMerklePathItem { + fn serialize(&self, writer: &mut W) -> std::io::Result<()> { + match &self.0 { + Either::Left(_) => BorshSerialize::serialize(&0u8, writer), + Either::Right(_) => BorshSerialize::serialize(&1u8, writer), + }?; + match &self.0 { + Either::Left(i) => BorshSerialize::serialize(i, writer), + Either::Right(path) => BorshSerialize::serialize(path, writer), + } + } +} + +impl BorshDeserialize for LookupMerklePathItem { + fn deserialize_reader(reader: &mut R) -> std::io::Result { + let kind = u8::deserialize_reader(reader)?; + if kind == 0 { + let i = u32::deserialize_reader(reader)?; + Ok(LookupMerklePathItem(Either::Left(i))) + } else if kind == 1 { + let path = MerklePathItem::deserialize_reader(reader)?; + Ok(LookupMerklePathItem(Either::Right(path))) + } else { + Err(std::io::Error::new( + std::io::ErrorKind::InvalidData, + anyhow::anyhow!("Invalid LookupMerklePathItem kind: {}", kind), + )) + } + } +} + +#[derive(Debug, Clone, BorshSerialize, BorshDeserialize, Serialize, Deserialize)] +pub struct Proof { + /// The block_merkle_root of the header used to create the proof batch + /// It should be at least the header for the latest transactions proven + 1 + client_block_merkle_root: CryptoHash, + batch: Vec, + // common ancestry if there is a common ancestry line in the batch + // we can save g as costs by only passing this once + ancestry: Vec, + // Cache of common paths, indexed by u8 + cache: Vec, +} + +impl Proof { + fn unpack_cached_path<'a>( + cache: &'a [MerklePathItem], + path: &'a [LookupMerklePathItem], + ) -> Vec<&'a MerklePathItem> { + path.iter().map(|x| x.into(cache)).collect() + } + + /// This checks the entire path of all batches and updates any reusable nodes + /// with indices + // fn cache_nodes(cache: &[MerklePathItem], batch: Vec) -> Vec { + // batch + // .into_iter() + // .map(|x| BlindedProof::from(&cache, x)) + // .collect() + // } + + fn common_ancestry( + proof1: &[MerklePathItem], + proof2: &[MerklePathItem], + ) -> Vec { + if proof1.is_empty() { + return proof2.to_vec(); + } + if proof2.is_empty() { + return proof1.to_vec(); + } + let mut common_ancestry = proof1 + .iter() + .rev() + .zip(proof2.iter().rev()) + .peekable() + .peeking_take_while(|(x, y)| x.hash == y.hash && x.direction == y.direction) + .map(|(x, _y)| x.clone()) + .collect_vec(); + common_ancestry.reverse(); + common_ancestry + } + + fn new(created_from: CryptoHash, mut batch: Vec) -> Self { + // First decide common ancestry among all batches + let ancestry = batch + .iter() + .fold(vec![], |acc, x| Self::common_ancestry(&acc, &x.block_proof)); + + // Filter out ancestors + batch.iter_mut().for_each(|x| { + x.block_proof = x + .block_proof + .iter() + .filter(|y| !ancestry.contains(y)) + .cloned() + .collect_vec() + }); + + let mut batch = batch.into_iter().map(BlindedProof::from).collect_vec(); + let mut cache = MerkleCache::default(); + cache.cache(&mut batch); + + Proof { + client_block_merkle_root: created_from, + batch, + ancestry, + cache: cache.items, + } + } +} + +pub fn verify_proof(proof: Proof) -> bool { + // We should know about the known_sync_block_merkle_root + //assert!(blinded_headers.contains_key(&body.created_from)); + //let cache = Proof::build_flat_cache(&body.batch); + + proof.batch.into_iter().all(|blinded| { + let block_hash = blinded.transaction_header.hash(); + log::debug!( + "Verifying blinded proof: {:?}", + blinded.transaction_header.outcome_root + ); + let block_hash_matches = block_hash == blinded.outcome_proof_block_hash; + + let outcome_proof_root = compute_root_from_path( + &blinded.collect_path(&proof.cache, &blinded.outcome_proof), + blinded.outcome_hash, + ); + let outcome_root_root = merkle::compute_root_from_path_and_item( + &blinded.collect_path(&proof.cache, &blinded.outcome_root_proof), + outcome_proof_root, + ); + let outcome_verified = outcome_root_root == blinded.transaction_header.outcome_root; + + let mut block_proof: Vec = + blinded.collect_path(&proof.cache, &blinded.block_proof); + block_proof.extend_from_slice(&proof.ancestry); + + let block_verified = + merkle::verify_hash(proof.client_block_merkle_root, &block_proof, block_hash); + // TODO: here we probably also want to make sure we know about this root + + log::debug!( + "block_hash_matches: {:?}, outcome_verified: {:?}, block_verified: {:?}", + block_hash_matches, + outcome_verified, + block_verified + ); + block_hash_matches && outcome_verified && block_verified + }) +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::client::rpc::{NearRpcClient, Network}; + use futures::FutureExt; + use near_primitives_core::types::AccountId; + use std::{path::Path, str::FromStr}; + + const BLOCK_MERKLE_ROOT: &str = "WWrLWbWHwSmjtTn5oBZPYgRCuCYn6fkYVa4yhPWNK4L"; + const LIGHT_CLIENT_HEAD: &str = "4bM5eXMDGxpFZXbWNT6TqX1HdZsWoHZ11KerCHJ8RKmU"; + + fn get_constants() -> (CryptoHash, CryptoHash, NearRpcClient) { + let client = NearRpcClient::new(Network::Testnet); + let block_root = CryptoHash::from_str(BLOCK_MERKLE_ROOT).unwrap(); + let light_client_head = CryptoHash::from_str(LIGHT_CLIENT_HEAD).unwrap(); + (light_client_head, block_root, client) + } + + fn read_proof Deserialize<'a>>(path: &str) -> T { + println!("Reading {}", path); + serde_json::from_reader(std::fs::File::open(path).unwrap()).unwrap() + } + + fn write_proof(path: &str, proof: &Proof) { + std::fs::write( + path.to_string().replace(".json", ".hex"), + hex::encode(proof.try_to_vec().unwrap()), + ) + .unwrap(); + let json = serde_json::to_string_pretty(&proof).unwrap(); + println!("Writing path: {}\n {}", path, json); + std::fs::write(path, json).unwrap(); + } + + fn proof_fixture(is_new: bool) -> FullProof { + let path = if is_new { + "fixtures/new.json" + } else { + "fixtures/old.json" + }; + read_proof(path) + } + + #[test] + fn test_blinded_same_block_hash() { + let proof = proof_fixture(true); + let full_block_hash = proof.block_header_lite.hash(); + let blinded = LightweightHeader::from(proof.block_header_lite); + assert_eq!(full_block_hash, blinded.hash()); + } + + #[test] + fn test_blinded_outcome() { + let proof = proof_fixture(true); + let outcome_proof_old = proof.outcome_proof.clone(); + let outcome_proof_leaf = CryptoHash::hash_borsh(proof.outcome_proof.to_hashes()); + let blinded = BlindedProof::from(proof); + + let cache = vec![]; + let outcome_proof_root = merkle::compute_root_from_path( + &blinded.collect_path(&cache, &blinded.outcome_proof), + blinded.outcome_hash, + ); + let outcome_root = merkle::compute_root_from_path_and_item( + &blinded.collect_path(&cache, &blinded.outcome_root_proof), + outcome_proof_root, + ); + assert_eq!(outcome_root, blinded.transaction_header.outcome_root); + assert_eq!( + outcome_proof_root, + merkle::compute_root_from_path(&outcome_proof_old.proof, outcome_proof_leaf) + ) + } + + #[test] + fn test_cache() { + let proof1 = proof_fixture(false); + let proof2 = proof_fixture(false); + + // Sort the proof items so the order will be the same in the cache + let paths = proof2 + .outcome_proof + .proof + .iter() + .chain(&proof2.outcome_root_proof) + .chain(&proof2.block_proof) + .cloned() + .collect_vec(); + + let original = vec![proof1, proof2]; + let mut blinded = original.into_iter().map(BlindedProof::from).collect_vec(); + let mut cache = MerkleCache::default(); + cache.cache(&mut blinded); + // Since the proofs were the same, the cache should contain all of the paths for all of the + // proofs + itertools::assert_equal(paths, cache.items); + } + + #[test] + fn test_common_ancestry() { + let proof = proof_fixture(true); + let cached_nodes = Proof::common_ancestry(&proof.block_proof, &proof.block_proof); + assert_eq!(cached_nodes, proof.block_proof); + println!("{:#?}", cached_nodes); + } + + #[test] + fn test_create_e2e() { + let root = CryptoHash::from_str(BLOCK_MERKLE_ROOT).unwrap(); + + fn get_proof_len(proof: &FullProof) -> usize { + let mut bytes = vec![]; + BorshSerialize::serialize(&proof.block_proof, &mut bytes).unwrap(); + BorshSerialize::serialize(&proof.outcome_proof, &mut bytes).unwrap(); + BorshSerialize::serialize(&proof.block_header_lite, &mut bytes).unwrap(); + BorshSerialize::serialize(&proof.outcome_root_proof, &mut bytes).unwrap(); + bytes.len() + } + let proof1 = proof_fixture(true); + let proof2 = proof_fixture(false); + let proof_size = get_proof_len(&proof1) + get_proof_len(&proof2); + println!("{:#?}", proof_size); + let batch = vec![proof1, proof2]; + let proof = Proof::new(root, batch); + let new_proof_size = proof.try_to_vec().unwrap().len(); + assert!(new_proof_size < proof_size / 2); + println!("{:#?}", new_proof_size); + + println!("{}", hex::encode(root)); + assert!(verify_proof(proof)); + } + + // Util for rewriting the original bridge proofs + fn _rewrite_bridge_proofs(rainbow_prover_fixture_path: &str) { + let rewritten = vec![ + ( + "22f00dd154366d758cd3e4fe81c1caed8e0db6227fe4b2b52a8e5a468aa0a723", + "proof2.json", + ), + ( + "0d0776820a9a81481a559c36fd5d69c33718fb7d7fd3be7564a446e043e2cb35", + "proof3.json", + ), + ( + "1f7129496c461c058fb3daf258d89bf7dacb4efad5742351f66098a00bb6fa53", + "proof4.json", + ), + ( + "a9cd8eb4dd92ba5f2fef47d68e1d73ac8c57047959f6f8a2dcc664419e74e4b8", + "proof5.json", + ), + ( + "cc3954a51b7c1a86861df8809f79c2bf839741e3e380e28360b8b3970a5d90bd", + "proof6.json", + ), + ( + "8298c9cd1048df03e9ccefac4b022636a30a2f7e6a8c33cc4104901b92e08dfd", + "proof7.json", + ), + ] + .iter() + .map(|(root, path)| { + ( + CryptoHash(hex::decode(root).unwrap().try_into().unwrap()), + format!("{}{}", rainbow_prover_fixture_path, path), + ) + }) + .all(|(block_root, path)| { + let proof = if path.contains("old") { + let proof: FullProof = read_proof(&path); + Proof::new(block_root, vec![proof]) + } else { + read_proof(&path) + }; + + write_proof(&path, &proof); + verify_proof(proof) + }); + assert!(rewritten); + } + + // This test populates a reasonably large batch of proofs for verifying the + // slimmer onchain light client + #[tokio::test] + async fn batch_proofs() { + let _ = pretty_env_logger::try_init(); + let (head, common_root, client) = get_constants(); + let receiver_id = AccountId::from_str("da.topgunbakugo.testnet").unwrap(); + + let path = "fixtures/batch.json"; + + let p = if Path::new(path).exists() { + read_proof(path) + } else { + // Get a bunch of receipts to test + let receipts = vec![ + "2QUBErafE2zU3dV9HBYCUyqpcP74erBm7k1j7tXduRa2", + "BixAd4vjnYKzxNtSLGD5w7jk7vD3ZedesHizutwWvDDK", + "6W5LXCxFwg21XMxVLqZMzq24g531kFu2hxBr2N5yN67f", + "Dwwv7KeNP8H9Pr6suUkAqDwex8mp1vK3KykE7asjaYHk", + "D75Ej8NJ8kYE62CnaVX3p8SCHiNLBQCA7pLpudeRbnM6", + "ERDrqb65Sa8mKr4XrXCw1uBae3a8mrByLPcEN92zewfG", + "FuNDcy6u3sS5B8j1szxZGTh6BCjsDTWfUbfFWsFwUyPA", + "2EVVdVsw1nM3hRFV4LTkBYYRWPCR77TVmYxZwipLHmYn", + "9Exwa5z51bTGWPzMMkRcVdGZ6a8rpu4CuWfAvrMuqXHn", + "J65AycXr48nCqrRc8eErLp5AiQb6xUAWF1sgG2fRbn9t", + "5axBiFL3GkvizK4KobsJjzychu7UckoKACnX23RefhoS", + "GEG9nhhwbgMHw61fTrFYvtpeRidAB6NKmNFJmkMc8AuE", + "BFdir5jFtyWQWoLPbxWEGtRedbjgt4nSPcDrdNfWLXcd", + "48tF8CXXLLaUgEbvayqhvX1UZLzLrqSjVJBq5bWXFX7T", + "3L2vmfXBZzMGoXx8bLDaSRxtp2dV65SZr3rYKM46wqe1", + "4g3zeZEt4UrXqFh8Vxcdz8UCGaBBdT7K7MELrnNbu3PH", + "2r652tQLr8kpGtY6KQNLhVC4j6S5jwC1Q6VcuLxQM1UQ", + "DNAQsgrEqJ6kYhqaXbhCNP5W4K891jLUhBtY3nsU17dv", + "EJCfAgRT3x8hPmTvLJgqexuwRqhgujUKbby4b81ypXX4", + "3FFWvbPQXkpQNvGwJ6NkkJtya6XXo8x88Wmhoyv6ABQU", + "BVgDgRBcJJoXeFYarSTrsH9NYe6kgfC14Des88hSJp1j", + "EcnvcNZDgwyiBP7NpV2afkDDvPCmw5rwKT2MLiG1jUFD", + "Hbjut1evHnfm1Ee9eeGJe5ydorVE3W66RacprLuciFpY", + "AKknXHv6Y7iyD44yyTjdTLmgC8AM9CmUHtLXvgEh4qdS", + ] + .into_iter() + .map(CryptoHash::from_str) + .map(Result::unwrap) + .map(|receipt_id| { + client + .fetch_light_client_receipt_proof(receipt_id.clone(), receiver_id.clone(), head) + .map(Option::unwrap) + }) + .collect_vec(); + + let proofs: Vec<_> = futures::future::join_all(receipts).await; + + let p = Proof::new(common_root, proofs); + write_proof(path, &p); + p + }; + + assert!(verify_proof(p)); + } +} diff --git a/src/main.rs b/src/main.rs index 6a7edbb..dc4ab87 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,3 +1,5 @@ +#![feature(slice_partition_dedup)] + use crate::client::LightClient; use client::Message; From 362ba10a5ce10a21036a9051b1136a4c2b18650f Mon Sep 17 00:00:00 2001 From: dndll Date: Thu, 21 Dec 2023 12:03:47 +0000 Subject: [PATCH 2/4] feat: use a zero-copy version of near-merkle --- fixtures/batch.json | 50 ++++++------- src/client/proof.rs | 124 ++++++++++++-------------------- src/client/proof/merkle_util.rs | 31 ++++++++ src/main.rs | 1 + 4 files changed, 101 insertions(+), 105 deletions(-) create mode 100644 src/client/proof/merkle_util.rs diff --git a/fixtures/batch.json b/fixtures/batch.json index 1c9e4c1..d8a3bfa 100644 --- a/fixtures/batch.json +++ b/fixtures/batch.json @@ -38,7 +38,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "2hEeSce4tTwiaCQ8AHprkndeg6jXv6rvg3rn7xw8m66V", "inner_rest_hash": "2zxnxeLSfGTFyYJjASAY9gVPVx3VXdL9424FpHCw1UHg", "prev_block_hash": "E7ugr7jwqLMBGHZf3Dszcmf4y9B96W4isaJraZJSrxUj", @@ -90,7 +90,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "5sRv1mRJMEtuG4T7jdQyrP15wWNEJfX6miskUS42pLHS", "inner_rest_hash": "J4RPuwENpZmdkiMfqhqyYMbQ4MwR7idcaDBpP8431gMe", "prev_block_hash": "HbWHEEwcNZrChQxGJL3STkWNiB1UcRtLgAEzUtgkYp2S", @@ -132,7 +132,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "GZp6n6hEkDtfsuEfYFrXD1ZCUKuGwkjGKBGFjmMsS6f4", "inner_rest_hash": "4nXZUEBPSyFk3EM9ovfnTVX8vEh4TiqRpp2p55WwrErJ", "prev_block_hash": "Gkh1v29dTQeZHUBsSmp37KFR9ZnV86nX4TX3fTsKuUbk", @@ -182,7 +182,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "2zNaCSmvYcxriCjGxNnLFPsrqG6nxuterVC6y46kabCf", "inner_rest_hash": "DzGYc3CDLZXLn8EfADzZud6RPiuaaDj4E5o5Snmt2hQp", "prev_block_hash": "HqnvBjc9vvVNiNkGXtpuCEccbSR34J3uaMtX3Kao82dv", @@ -224,7 +224,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "7tJ1xJLMcEoD1RqmVfbBt7d9wVmdC4rzzqXd3AaYAoaf", "inner_rest_hash": "9SNkoN3PaoQ3RzHnVoSetTTRA1yH4TovHZrPCeLZuKQn", "prev_block_hash": "FP9jWLcURrjeZq9Nn4GPRuw1BrS3B4viE3v7RP2MzV4T", @@ -269,7 +269,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "8YJX9Z1N1u4KJfxeSZoxLuNgwYUKmZoTafCaaYxxfTx8", "inner_rest_hash": "3yso176TyBauYDM3xQZF8yDAc3wYJMaATSaLvjud17wo", "prev_block_hash": "37PEb6siLYmuKtv1S1msEj3VmsVgrJhvwKbSN1ZyGABP", @@ -319,7 +319,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "7dCpZqv62dgXu8FTQhw5LmtR4aGshT54u2kfUA4b7hKQ", "inner_rest_hash": "BtT5hFjgU5NtddngkK4qMLBtSHw7tM6RawJzhhKmSUjA", "prev_block_hash": "2SZhhYzYQ9fVAaSuS9Tp9DaP5T31q1Ko6HDbpiTYvSYG", @@ -361,7 +361,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "67D4C7JBk8eWXwWh4bR1bN4SH4MFLfKvWbqtPrDVixDC", "inner_rest_hash": "BHknTrn2WxLr7chMpgs9mTQTZxXLsHUNAZAsjSk4mYmj", "prev_block_hash": "2pVbpHb3dtgUyN5RsyN9Sksa778XaCYZNNoYwTrjLA6T", @@ -411,7 +411,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "3xh1sTbGbuJK7sfzCYJh2yEYWCgmD4voyx9zHhEuiFNw", "inner_rest_hash": "HFbmky2Kxvdf587biCf1eNM3Vt41mnRhzcM1Zifo2fmh", "prev_block_hash": "66SDNjto6jXo7PSh2adrKAhKWmg5AAToiVA3RJ6zHWD3", @@ -453,7 +453,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "9edAZZgpABc3kymJzjZ392jNKbfam7HsUBAhXdhqC6zW", "inner_rest_hash": "8fQRLSKq3LJyg5dHAZ4FG2FoPUZNxmSDYfJQt7nNQVw9", "prev_block_hash": "D6nKsiEojQYvu5kwzZ96CFTt7JGqz53XH4hKHGir2GBc", @@ -487,7 +487,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "BfVXSJW3ikZ1DCikaA22LvY6t9Y1UksZNBduRVN2qEJs", "inner_rest_hash": "77XJVCUNgqURcZctRGf5tHQWx5SMjdVVZgKBjDauD5xc", "prev_block_hash": "DiSh6mnHWmRJgoFdxGX4Te2sLXCfyvhe2x2uzDHaHc9", @@ -521,7 +521,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "BfVXSJW3ikZ1DCikaA22LvY6t9Y1UksZNBduRVN2qEJs", "inner_rest_hash": "77XJVCUNgqURcZctRGf5tHQWx5SMjdVVZgKBjDauD5xc", "prev_block_hash": "DiSh6mnHWmRJgoFdxGX4Te2sLXCfyvhe2x2uzDHaHc9", @@ -558,7 +558,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "BfVXSJW3ikZ1DCikaA22LvY6t9Y1UksZNBduRVN2qEJs", "inner_rest_hash": "77XJVCUNgqURcZctRGf5tHQWx5SMjdVVZgKBjDauD5xc", "prev_block_hash": "DiSh6mnHWmRJgoFdxGX4Te2sLXCfyvhe2x2uzDHaHc9", @@ -608,7 +608,7 @@ 6, 7 ], - "transaction_header": { + "header": { "inner_lite_hash": "BKMXE1PfYHSEoQnG7eXtJdD34CeTypLyX1CJzWNZiwm7", "inner_rest_hash": "8BHDS1KsJ55UMy2ndpcYYKUDACQYmCrVbh42dDqWiQDp", "prev_block_hash": "5vxW4UeYrqouAFCs63LwQWjMhKpEM6aabZmbsTwpbqg8", @@ -650,7 +650,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "6aw5LHvcoWfcjmZA9rfc3UQX16MQZELWY1aTddoMWhYP", "inner_rest_hash": "CQtCaHorfe43v9KumQovuNzMaQh1MTe8oWeKdNBWfYDP", "prev_block_hash": "8AqJ5aEEuSNb1QTvExCdsmEaUEhtZ44aKed5eADGDyYA", @@ -695,7 +695,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "HfhNmHv5NnjbQyUz9TRtTjJ7MGFmvyQb9youeUraznRe", "inner_rest_hash": "4vfTLdDbNp48ThVsDYcaadu9jb1vyi34E2xqXzP5v19t", "prev_block_hash": "9kNH12h8DiVQkir4TD7CJETsVu69ftx9vTmiEJJjZvYd", @@ -734,7 +734,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "F6r1aHyovKST91xtCL4QQG1Y6M7JLySMndiZGB8pMLuQ", "inner_rest_hash": "6RRDMRFPZ5dMNVYzxxHfMNb2MmhaAjaYw1CExKAaKqHX", "prev_block_hash": "8yd1cSebZzL8UWtqoHrezFoCdQQB7wLrw4MdGjGjsFr7", @@ -782,7 +782,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "3V3azAPeKh2psNx3JqpTs7roWVwDeavrMutDmhP3vmfT", "inner_rest_hash": "6aHDq5gzvFfTPeqtnBAgjnFJVaTS8Z8hFmZiWv9jNESH", "prev_block_hash": "Dqwd6XnJJE69WGsCwaG6LPL13AjRdmXMmbySzsofeioE", @@ -830,7 +830,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "2yc1Q4beLQc4VeK7j4Jg8JQ6dHQ4Q9G2BjeijASbwvku", "inner_rest_hash": "5Fv3gZhWHTdFBbiuKvPf1PXYvtpCDqrppuZ83AoaqztP", "prev_block_hash": "8U9aaVFs5oDHn2cNfPVBtJBe3jfTZLhJhE2AWnYz8WBw", @@ -878,7 +878,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "GrLZrWNuuxdDsxdduSy9ouXGfx1Hu5gWeRw71iBztBrW", "inner_rest_hash": "611qvrf6jZqUDFRayQjHVLehCPJEC3Vvkh4jPFzYgxfq", "prev_block_hash": "CosDUzMEFjvFEPAsprD36821Gqh5ay9hUuXYYANmD8xP", @@ -926,7 +926,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "6C21hiRuACNnPU4q5WDgM1AJTDYxKkYAfLzRfrxhQcgL", "inner_rest_hash": "7SXM8unGiwzY29NdBqH83a3jKTYuer3Hpe2jZFFG5oxU", "prev_block_hash": "7FbfGi8W8afiEDCm3FX8hdKEJp1xweRzVkXYwNmWYWRy", @@ -979,7 +979,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "ABvfWf3tEaF9vcHM43uqPkucGWb3ynMuoJffHQidF4M7", "inner_rest_hash": "49RKYDm3uhX7SvLan2epaQQ9tydGWgSHhmVYvczwv8B6", "prev_block_hash": "GoS88YTweaXQjgeG6N6Km8ZavZoJyB4T9qVwS4UgQe3a", @@ -1024,7 +1024,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "DGHWpP4P3vkFaRvDdQLnZ61M6ePCnRiCrBet15MHf1iy", "inner_rest_hash": "HA5ATMhVGgyemdq5WkoJ8o1XSeTEyd37x1xsFjEY5aQC", "prev_block_hash": "A6ZCeowAt3xmhgoN1ti7QZf193CtBjUbmfhAz6u5L9uk", @@ -1072,7 +1072,7 @@ 32, 33 ], - "transaction_header": { + "header": { "inner_lite_hash": "FyM9oxg1xG5LG69QYVhkRsW4xzBYdq2XUYGLxVab1XPK", "inner_rest_hash": "6a7x22aDseHD3YTuN9uvmT3duxcfAAvfWSH6zzapu8uv", "prev_block_hash": "9JJFsJN5AVrVfgbCqdYEcet34aYLBjPhe9fo3bKgma1S", @@ -1312,4 +1312,4 @@ "direction": "Right" } ] -} \ No newline at end of file +} diff --git a/src/client/proof.rs b/src/client/proof.rs index dae288d..1a34851 100644 --- a/src/client/proof.rs +++ b/src/client/proof.rs @@ -1,16 +1,17 @@ -use std::{collections::HashMap, sync::Arc}; - use super::{LightClientState, Proof as FullProof}; use borsh::{BorshDeserialize, BorshSerialize}; use either::Either; use itertools::Itertools; +use merkle_util::compute_root_from_path; use near_primitives::{ - merkle::{self, combine_hash, compute_root_from_path, MerklePathItem}, + merkle::{combine_hash, MerklePathItem}, views::LightClientBlockLiteView, }; use near_primitives_core::hash::CryptoHash; use serde::{Deserialize, Serialize}; +pub(crate) mod merkle_util; + #[derive(Debug, Clone, BorshSerialize, BorshDeserialize, Serialize, Deserialize)] pub struct LightweightHeader { inner_lite_hash: CryptoHash, @@ -83,23 +84,23 @@ pub struct BlindedProof { outcome_root_proof: Vec, block_proof: Vec, // The header for the block that this proof was created from - transaction_header: LightweightHeader, + header: LightweightHeader, } -#[derive(Debug, Clone, Default, BorshSerialize, BorshDeserialize)] +#[derive(Debug, Clone, Default, BorshSerialize, BorshDeserialize, Serialize, Deserialize)] +#[serde(transparent)] struct MerkleCache { items: Vec, } impl MerkleCache { fn cache(&mut self, batch: &mut Vec) { - // TODO: this is super inneficient, we had a nice one without duplicates but - // it unfortunately copies the iterator let duplicates = batch .iter() .cloned() .map(|fp| [fp.outcome_proof, fp.outcome_root_proof, fp.block_proof].concat()) .flatten() + // TODO: this is super inefficient, we had a nice one without duplicates but // Unfortunately duplicates clones the array, so we can't do this zero // copy .duplicates() @@ -122,6 +123,17 @@ impl MerkleCache { }); self.items = duplicates.into_iter().map(|x| x.0.unwrap_right()).collect(); } + + fn collect<'a>( + &'a self, + path: &'a [LookupMerklePathItem], + ) -> impl Iterator { + path.iter().map(|x| match &x.0 { + // TODO: no panic + Either::Left(i) => self.items.get(*i as usize).unwrap(), + Either::Right(x) => x, + }) + } } impl From for BlindedProof { @@ -145,24 +157,11 @@ impl From for BlindedProof { .into_iter() .map(LookupMerklePathItem::from) .collect(), - transaction_header: LightweightHeader::from(x.block_header_lite), + header: LightweightHeader::from(x.block_header_lite), } } } -impl BlindedProof { - fn collect_path<'a>( - &self, - cache: &'a [MerklePathItem], - path: &[LookupMerklePathItem], - ) -> Vec { - Proof::unpack_cached_path(cache, &path) - .into_iter() - .cloned() - .collect() - } -} -// TODO: add refcount here #[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)] #[serde(transparent)] pub struct LookupMerklePathItem( @@ -181,22 +180,6 @@ impl From for LookupMerklePathItem { } } -impl LookupMerklePathItem { - fn into<'a>(&'a self, cache: &'a [MerklePathItem]) -> &MerklePathItem { - match &self.0 { - // TODO: no panic - Either::Left(i) => cache.get(*i as usize).unwrap(), - Either::Right(x) => x, - } - } -} - -impl From> for LookupMerklePathItem { - fn from(x: Either) -> Self { - Self(x) - } -} - impl std::hash::Hash for LookupMerklePathItem { fn hash(&self, state: &mut H) { state.write( @@ -227,10 +210,10 @@ impl BorshDeserialize for LookupMerklePathItem { let kind = u8::deserialize_reader(reader)?; if kind == 0 { let i = u32::deserialize_reader(reader)?; - Ok(LookupMerklePathItem(Either::Left(i))) + Ok(LookupMerklePathItem::from(i)) } else if kind == 1 { let path = MerklePathItem::deserialize_reader(reader)?; - Ok(LookupMerklePathItem(Either::Right(path))) + Ok(LookupMerklePathItem::from(path)) } else { Err(std::io::Error::new( std::io::ErrorKind::InvalidData, @@ -250,26 +233,10 @@ pub struct Proof { // we can save g as costs by only passing this once ancestry: Vec, // Cache of common paths, indexed by u8 - cache: Vec, + cache: MerkleCache, } impl Proof { - fn unpack_cached_path<'a>( - cache: &'a [MerklePathItem], - path: &'a [LookupMerklePathItem], - ) -> Vec<&'a MerklePathItem> { - path.iter().map(|x| x.into(cache)).collect() - } - - /// This checks the entire path of all batches and updates any reusable nodes - /// with indices - // fn cache_nodes(cache: &[MerklePathItem], batch: Vec) -> Vec { - // batch - // .into_iter() - // .map(|x| BlindedProof::from(&cache, x)) - // .collect() - // } - fn common_ancestry( proof1: &[MerklePathItem], proof2: &[MerklePathItem], @@ -316,7 +283,7 @@ impl Proof { client_block_merkle_root: created_from, batch, ancestry, - cache: cache.items, + cache, } } } @@ -327,30 +294,29 @@ pub fn verify_proof(proof: Proof) -> bool { //let cache = Proof::build_flat_cache(&body.batch); proof.batch.into_iter().all(|blinded| { - let block_hash = blinded.transaction_header.hash(); - log::debug!( - "Verifying blinded proof: {:?}", - blinded.transaction_header.outcome_root - ); + let block_hash = blinded.header.hash(); + // TODO: here we probably also want to make sure we know about this root in the light + // client + log::debug!("Verifying blinded proof: {:?}", blinded.header.outcome_root); let block_hash_matches = block_hash == blinded.outcome_proof_block_hash; let outcome_proof_root = compute_root_from_path( - &blinded.collect_path(&proof.cache, &blinded.outcome_proof), + proof.cache.collect(&blinded.outcome_proof), blinded.outcome_hash, ); - let outcome_root_root = merkle::compute_root_from_path_and_item( - &blinded.collect_path(&proof.cache, &blinded.outcome_root_proof), + let outcome_root_root = merkle_util::compute_root_from_path_and_item( + proof.cache.collect(&blinded.outcome_root_proof), outcome_proof_root, ); - let outcome_verified = outcome_root_root == blinded.transaction_header.outcome_root; + let outcome_verified = outcome_root_root == blinded.header.outcome_root; - let mut block_proof: Vec = - blinded.collect_path(&proof.cache, &blinded.block_proof); - block_proof.extend_from_slice(&proof.ancestry); + let block_proof = proof + .cache + .collect(&blinded.block_proof) + .chain(&proof.ancestry); let block_verified = - merkle::verify_hash(proof.client_block_merkle_root, &block_proof, block_hash); - // TODO: here we probably also want to make sure we know about this root + merkle_util::verify_hash(proof.client_block_merkle_root, block_proof, block_hash); log::debug!( "block_hash_matches: {:?}, outcome_verified: {:?}, block_verified: {:?}", @@ -420,19 +386,17 @@ mod tests { let outcome_proof_leaf = CryptoHash::hash_borsh(proof.outcome_proof.to_hashes()); let blinded = BlindedProof::from(proof); - let cache = vec![]; - let outcome_proof_root = merkle::compute_root_from_path( - &blinded.collect_path(&cache, &blinded.outcome_proof), - blinded.outcome_hash, - ); - let outcome_root = merkle::compute_root_from_path_and_item( - &blinded.collect_path(&cache, &blinded.outcome_root_proof), + let cache = MerkleCache::default(); + let outcome_proof_root = + compute_root_from_path(cache.collect(&blinded.outcome_proof), blinded.outcome_hash); + let outcome_root = merkle_util::compute_root_from_path_and_item( + cache.collect(&blinded.outcome_root_proof), outcome_proof_root, ); - assert_eq!(outcome_root, blinded.transaction_header.outcome_root); + assert_eq!(outcome_root, blinded.header.outcome_root); assert_eq!( outcome_proof_root, - merkle::compute_root_from_path(&outcome_proof_old.proof, outcome_proof_leaf) + compute_root_from_path(outcome_proof_old.proof.iter(), outcome_proof_leaf) ) } diff --git a/src/client/proof/merkle_util.rs b/src/client/proof/merkle_util.rs new file mode 100644 index 0000000..2b8a9f4 --- /dev/null +++ b/src/client/proof/merkle_util.rs @@ -0,0 +1,31 @@ +use borsh::BorshSerialize; +use near_primitives::merkle::{combine_hash, Direction, MerklePathItem}; +use near_primitives_core::{hash::CryptoHash, types::MerkleHash}; + +pub trait PathIterator<'a> = Iterator; + +pub fn verify_hash<'a>(root: MerkleHash, path: impl PathIterator<'a>, item_hash: MerkleHash) -> bool { + compute_root_from_path(path, item_hash) == root +} + +pub fn compute_root_from_path<'a>(path: impl PathIterator<'a>, item_hash: MerkleHash) -> MerkleHash { + let mut res = item_hash; + for item in path { + match item.direction { + Direction::Left => { + res = combine_hash(&item.hash, &res); + } + Direction::Right => { + res = combine_hash(&res, &item.hash); + } + } + } + res +} + +pub fn compute_root_from_path_and_item<'a, T: BorshSerialize>( + path: impl PathIterator<'a>, + item: T, +) -> MerkleHash { + compute_root_from_path(path, CryptoHash::hash_borsh(item)) +} diff --git a/src/main.rs b/src/main.rs index dc4ab87..6505b3f 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,4 +1,5 @@ #![feature(slice_partition_dedup)] +#![feature(trait_alias)] use crate::client::LightClient; use client::Message; From 1e796bdcb0a1d6e72c7e4685c757ca2cd1684de7 Mon Sep 17 00:00:00 2001 From: dndll Date: Thu, 21 Dec 2023 17:30:42 +0000 Subject: [PATCH 3/4] feat!: use cleaner protocol --- fixtures/{3_previous_epoch.json => 1.json} | 0 fixtures/{2_previous_epoch.json => 2.json} | 0 ..._current_epoch.json => current_epoch.json} | 0 src/client/error.rs | 16 +- src/client/mod.rs | 978 ++---------------- .../{proof.rs => protocol/experimental.rs} | 69 +- src/client/{proof => protocol}/merkle_util.rs | 14 +- src/client/protocol/mod.rs | 631 +++++++++++ src/controller.rs | 4 +- src/erasure.rs | 5 +- 10 files changed, 781 insertions(+), 936 deletions(-) rename fixtures/{3_previous_epoch.json => 1.json} (100%) rename fixtures/{2_previous_epoch.json => 2.json} (100%) rename fixtures/{1_current_epoch.json => current_epoch.json} (100%) rename src/client/{proof.rs => protocol/experimental.rs} (91%) rename src/client/{proof => protocol}/merkle_util.rs (68%) create mode 100644 src/client/protocol/mod.rs diff --git a/fixtures/3_previous_epoch.json b/fixtures/1.json similarity index 100% rename from fixtures/3_previous_epoch.json rename to fixtures/1.json diff --git a/fixtures/2_previous_epoch.json b/fixtures/2.json similarity index 100% rename from fixtures/2_previous_epoch.json rename to fixtures/2.json diff --git a/fixtures/1_current_epoch.json b/fixtures/current_epoch.json similarity index 100% rename from fixtures/1_current_epoch.json rename to fixtures/current_epoch.json diff --git a/src/client/error.rs b/src/client/error.rs index 2c6d677..79432f9 100644 --- a/src/client/error.rs +++ b/src/client/error.rs @@ -1,9 +1,21 @@ -#[derive(Debug, Eq, PartialEq)] +use thiserror::Error; + +#[derive(Error, Debug, PartialEq, Eq)] pub enum Error { + #[error("Block already verified")] BlockAlreadyVerified, + #[error("Block not current or next epoch")] BlockNotCurrentOrNextEpoch, - BlockMissingNextBps, + #[error("Signature invalid")] SignatureInvalid, + #[error("Not enough approved stake")] NotEnoughApprovedStake, + #[error("Block is in the next epoch but no new set")] NextBpsInvalid, + #[error("Signature len mismatch")] + SignatureLenMismatch, + #[error("Invalid proof")] + InvalidProof, + #[error("Validator not signed")] + ValidatorNotSigned, } diff --git a/src/client/mod.rs b/src/client/mod.rs index c18cb66..758533d 100644 --- a/src/client/mod.rs +++ b/src/client/mod.rs @@ -1,30 +1,39 @@ -use crate::{client::error::Error, config::Config}; +use crate::{client::protocol::Protocol, config::Config}; +use anyhow::{anyhow, Result}; use borsh::{BorshDeserialize, BorshSerialize}; use flume::{Receiver, Sender}; -use near_crypto::Signature; -use near_jsonrpc_client::methods::light_client_proof::RpcLightClientExecutionProofResponse; -use near_primitives::{ - block_header::{ApprovalInner, BlockHeaderInnerLite}, - merkle::{self}, - views::{ - validator_stake_view::ValidatorStakeView, BlockHeaderInnerLiteView, - LightClientBlockLiteView, LightClientBlockView, - }, -}; -use near_primitives_core::{ - hash::CryptoHash, - types::{AccountId, BlockHeight}, -}; +use near_jsonrpc_client::methods::light_client_proof::RpcLightClientExecutionProofResponse as BasicProof; +use near_primitives::{types::validator_stake::ValidatorStake, views::LightClientBlockLiteView}; +use near_primitives_core::{hash::CryptoHash, types::AccountId}; +use serde::{Deserialize, Serialize}; use sled::Tree; use std::{path::PathBuf, str::FromStr}; use tokio::time; pub mod error; -pub mod proof; +pub mod protocol; pub mod rpc; +pub const STATE_PATH: &str = "state.json"; pub type Header = LightClientBlockLiteView; -pub type Proof = RpcLightClientExecutionProofResponse; + +#[derive(Debug, Serialize, Deserialize)] +pub enum Proof { + Basic(BasicProof), + Experimental(protocol::experimental::Proof), +} + +impl From for Proof { + fn from(proof: BasicProof) -> Self { + Self::Basic(proof) + } +} + +impl From for Proof { + fn from(proof: protocol::experimental::Proof) -> Self { + Self::Experimental(proof) + } +} pub enum ProofType { Transaction { @@ -56,20 +65,6 @@ pub enum Message { }, } -pub const STATE_PATH: &str = "state.json"; - -macro_rules! cvec { - ($($x:expr),*) => { - { - let mut temp_vec = Vec::new(); - $( - temp_vec.extend_from_slice(&$x); - )* - temp_vec - } - }; -} - #[derive(Debug, Clone)] pub struct LightClient { client: rpc::NearRpcClient, @@ -118,14 +113,14 @@ impl LightClient { tx, proof: ProofType::Transaction { transaction_id, sender_id } } => { - if let Err(e) = tx.send_async(self.get_transaction_proof(transaction_id, sender_id).await).await { + if let Err(e) = tx.send_async(self.get_transaction_proof(transaction_id, sender_id).await.map(Into::into)).await { log::error!("Failed to send proof: {:?}", e); } }, Message::GetProof { tx, proof: ProofType::Receipt { receipt_id, receiver_id } } => { - if let Err(e) = tx.send_async(self.get_receipt_proof(receipt_id, receiver_id).await).await { + if let Err(e) = tx.send_async(self.get_receipt_proof(receipt_id, receiver_id).await.map(Into::into)).await { log::error!("Failed to send proof: {:?}", e); } } @@ -135,22 +130,24 @@ impl LightClient { } } }, - // r = self.sync() => { - // tokio::time::sleep(duration).await; - // match r { - // Err(e) => log::error!("Error syncing: {:?}", e), - // Ok(false) if is_fast => { - // log::debug!("Slowing down sync interval to {:?}", default_duration); - // duration = default_duration; - // is_fast = false; - // } - // _ => (), - // } - // } + r = self.sync() => { + tokio::time::sleep(duration).await; + match r { + Err(e) => log::error!("Error syncing: {:?}", e), + Ok(false) if is_fast => { + log::debug!("Slowing down sync interval to {:?}", default_duration); + duration = default_duration; + is_fast = false; + } + _ => (), + } + } } } }); } + + // TODO: move to store module pub fn get, T: BorshDeserialize>(&self, key: K, kind: TreeKind) -> Option { let tree = match kind { TreeKind::BlockProducers => &self.block_producers, @@ -167,7 +164,7 @@ impl LightClient { key: K, value: &T, kind: TreeKind, - ) -> anyhow::Result<()> { + ) -> Result<()> { let tree = match kind { TreeKind::BlockProducers => &self.block_producers, TreeKind::ArchivalHeaders => &self.archival_headers, @@ -177,7 +174,7 @@ impl LightClient { .map(|_| ())?) } - pub async fn init(config: &Config) -> anyhow::Result { + pub async fn init(config: &Config) -> Result { let sled = sled::open(config.state_path.clone().unwrap_or(STATE_PATH.into())) .expect("Failed to open store"); let client = rpc::NearRpcClient::new(config.network.clone()); @@ -223,57 +220,39 @@ impl LightClient { } } - // TODO: Try to optimise - // TODO: Remove mutability return - pub async fn sync(&mut self) -> anyhow::Result { - let mut new_state: LightClientState = self.head.clone().into(); - log::trace!("Current head: {:#?}", new_state.head.inner_lite); + pub async fn sync(&mut self) -> Result { + log::trace!("Current head: {:#?}", self.head.inner_lite); - let new_header = self + let next_header = self .client .fetch_latest_header( - &CryptoHash::from_str(&format!("{}", new_state.head.hash())) - .map_err(|e| anyhow::anyhow!("Failed to parse hash: {:?}", e))?, + &CryptoHash::from_str(&format!("{}", self.head.hash())) + .map_err(|e| anyhow!("Failed to parse hash: {:?}", e))?, ) - .await; + .await + .ok_or_else(|| anyhow!("Failed to fetch latest header"))?; + log::trace!("Got new header: {:#?}", next_header.inner_lite); - if let Some(new_header) = new_header { - log::trace!("Got new header: {:#?}", new_header.inner_lite); - let validated = self - .get::<_, Vec>( - &new_state.head.inner_lite.epoch_id, - TreeKind::BlockProducers, - ) - .and_then(|bps| { - new_state - .validate_and_update_head(&new_header, bps.clone()) - .map_err(|x| log::trace!("Failed to validate header: {:?}", x)) - .ok() - }); + let bps = self + .get::<_, Vec>(&self.head.inner_lite.epoch_id, TreeKind::BlockProducers) + .ok_or_else(|| anyhow!("Failed to get block producers"))?; - if validated.is_some() { - log::debug!("Validated new header: {:?}", new_header.inner_lite.height); - if let Some((epoch, next_bps)) = new_state.next_bps { - log::debug!("storing next bps[{:?}]", epoch); - self.insert(epoch, &next_bps, TreeKind::BlockProducers)?; - } + let synced = Protocol::sync(&self.head, &bps, next_header)?; - self.insert( - self.head.inner_lite.epoch_id, - &self.head.clone(), - TreeKind::ArchivalHeaders, - )?; - self.head = new_state.head; - self.store.insert("head", self.head.try_to_vec()?)?; - Ok(true) - } else { - log::debug!("New header is invalid"); - Ok(false) - } - } else { - log::trace!("No new header found"); - Ok(false) + if let Some((epoch, next_bps)) = synced.next_bps { + log::debug!("storing next bps[{:?}]", epoch); + self.insert(epoch, &next_bps, TreeKind::BlockProducers)?; } + + self.insert( + self.head.inner_lite.epoch_id, + &synced.new_head, + TreeKind::ArchivalHeaders, + )?; + self.head = synced.new_head; + self.store.insert("head", self.head.try_to_vec()?)?; + + Ok(true) } fn head(&self) -> &Header { @@ -289,7 +268,7 @@ impl LightClient { &mut self, transaction_id: CryptoHash, sender_id: AccountId, - ) -> Option { + ) -> Option { // TODO: we need much more logic here for headers in different epochs & the head // since there are race conditions when we come to validate the proof header, we'd need to short-cache the proof validation let last_verified_hash = self.head.hash(); @@ -313,7 +292,7 @@ impl LightClient { &mut self, receipt_id: CryptoHash, receiver_id: AccountId, - ) -> Option { + ) -> Option { let last_verified_hash = self.head.hash(); self.client .fetch_light_client_receipt_proof(receipt_id, receiver_id, last_verified_hash) @@ -330,70 +309,21 @@ impl LightClient { }) } - // TODO: no panic - pub async fn verify_proof(&mut self, body: Proof) -> bool { - let block_hash = body.block_header_lite.hash(); - assert_eq!(block_hash, body.outcome_proof.block_hash); - - let outcome_root = merkle::compute_root_from_path( - &body.outcome_proof.proof, - CryptoHash::hash_borsh(body.outcome_proof.to_hashes()), - ); - - let outcome_root = - merkle::compute_root_from_path_and_item(&body.outcome_root_proof, outcome_root); - log::debug!("outcome_root: {:?}", outcome_root); - - let outcome_root_matches = outcome_root == body.block_header_lite.inner_lite.outcome_root; - - let mut block_verified = merkle::verify_hash( - self.head.inner_lite.block_merkle_root, - &body.block_proof, - block_hash, - ); - - // Functionality to cover race conditions between previously synced heads since the current head may of changed - // since the proof was generated - // FIXME: bug here if the same proof is requested - if !block_verified { - if let Some(r) = self - .get::<_, LightClientBlockLiteView>( - &body.outcome_proof.block_hash, - TreeKind::ArchivalHeaders, - ) - .and_then(|archive| { - log::debug!( - "Trying to verify against archival header: {:?}", - archive.inner_lite.height - ); - if merkle::verify_hash( - archive.inner_lite.block_merkle_root, - &body.block_proof, - block_hash, - ) { - log::debug!("Verified against archival header"); - block_verified = true; - Some(archive.inner_lite.block_merkle_root) - } else { - None - } - }) - { - if let Err(e) = self.archival_headers.remove(r) { - log::warn!("Failed to remove archival header: {:?}", e) - } - } - } - - log::debug!( - "shard outcome included: {:?}, block included: {:?}", - outcome_root_matches, - block_verified - ); - outcome_root_matches && block_verified + pub async fn verify_proof(&mut self, p: Proof) -> bool { + // TODO: this is a hack + let archive = match &p { + Proof::Basic(p) => self.get::<_, LightClientBlockLiteView>( + p.outcome_proof.block_hash, + TreeKind::ArchivalHeaders, + ), + Proof::Experimental(_) => None, + }; + Protocol::inclusion_proof_verify(&self.head, archive.as_ref(), p) + .inspect_err(|e| log::warn!("Failed to verify proof: {:?}", e)) + .unwrap_or_default() } - pub async fn shutdown(&self) -> anyhow::Result<()> { + pub async fn shutdown(&self) -> Result<()> { log::info!("Shutting down light client"); if let Err(e) = self.store.insert("head", self.head.try_to_vec()?) { log::warn!("Failed to insert head: {:?}", e) @@ -404,743 +334,3 @@ impl LightClient { todo!("Deliberate panic here, need to make this graceful"); } } - -#[derive(Debug, Clone)] -pub struct LightClientState { - pub head: LightClientBlockLiteView, - pub next_bps: Option<(CryptoHash, Vec)>, -} - -impl From for LightClientState { - fn from(head: LightClientBlockLiteView) -> Self { - Self { - head, - next_bps: None, - } - } -} - -impl LightClientState { - fn inner_lite_hash(ilv: &BlockHeaderInnerLiteView) -> CryptoHash { - let full_inner: BlockHeaderInnerLite = ilv.clone().into(); - CryptoHash::hash_borsh(full_inner) - } - - fn calculate_current_block_hash(block_view: &LightClientBlockView) -> CryptoHash { - let inner_light_hash = Self::inner_lite_hash(&block_view.inner_lite); - CryptoHash::hash_bytes(&cvec!( - CryptoHash::hash_bytes(&cvec!(inner_light_hash.0, block_view.inner_rest_hash.0)) - .as_bytes() - .as_ref(), - block_view.prev_block_hash.as_bytes().as_ref() - )) - } - - fn next_block_hash( - &self, - current_block_hash: &CryptoHash, - block_view: &LightClientBlockView, - ) -> CryptoHash { - CryptoHash::hash_bytes(&cvec!( - block_view.next_block_inner_hash.0, - current_block_hash.0 - )) - } - - fn reconstruct_light_client_block_view_fields( - &mut self, - block_view: &LightClientBlockView, - ) -> ([u8; 32], [u8; 32], Vec) { - let current_block_hash = Self::calculate_current_block_hash(block_view); - let next_block_hash: CryptoHash = self.next_block_hash(¤t_block_hash, block_view); - - let endorsement = ApprovalInner::Endorsement(next_block_hash); - - let approval_message = cvec!( - endorsement.try_to_vec().unwrap()[..], - (block_view.inner_lite.height + 2).to_le_bytes()[..] - ); - - log::debug!("Current block hash: {}", current_block_hash); - log::debug!("Next block hash: {}", next_block_hash); - log::debug!("Approval message: {:?}", approval_message); - ( - *current_block_hash.as_bytes(), - next_block_hash.into(), - approval_message, - ) - } - - pub fn ensure_not_already_verified( - head: &LightClientBlockLiteView, - block_height: &BlockHeight, - ) -> Result<(), Error> { - if block_height <= &head.inner_lite.height { - Err(Error::BlockAlreadyVerified) - } else { - Ok(()) - } - } - - pub fn ensure_epoch_is_current_or_next( - head: &LightClientBlockLiteView, - epoch_id: &CryptoHash, - ) -> Result<(), Error> { - if ![head.inner_lite.epoch_id, head.inner_lite.next_epoch_id].contains(epoch_id) { - log::debug!("Block is not in the current or next epoch"); - Err(Error::BlockNotCurrentOrNextEpoch) - } else { - Ok(()) - } - } - - pub fn ensure_if_next_epoch_contains_next_bps( - head: &LightClientBlockLiteView, - epoch_id: &CryptoHash, - next_bps: &Option>, - ) -> Result<(), Error> { - if &head.inner_lite.next_epoch_id == epoch_id && next_bps.is_none() { - log::debug!("Block is in the next epoch but no new set"); - Err(Error::BlockMissingNextBps) - } else { - Ok(()) - } - } - - pub fn validate_signatures( - signatures: &[Option], - epoch_block_producers: Vec, - approval_message: &[u8], - ) -> Result<(u128, u128), Error> { - let mut total_stake = 0; - let mut approved_stake = 0; - - for (maybe_signature, block_producer) in signatures.iter().zip(epoch_block_producers.iter()) - { - let vs = block_producer.clone().into_validator_stake(); - let bps_stake = vs.stake(); - let bps_pk = vs.public_key(); - - total_stake += bps_stake; - - if let Some(signature) = maybe_signature { - log::trace!( - "Checking if signature {} and message {:?} was signed by {}", - signature, - approval_message, - bps_pk - ); - if !signature.verify(approval_message, bps_pk) { - log::debug!("Signature is invalid"); - return Err(Error::SignatureInvalid); - } - approved_stake += bps_stake; - } - } - Ok((total_stake, approved_stake)) - } - - pub fn ensure_stake_is_sufficient( - total_stake: &u128, - approved_stake: &u128, - ) -> Result<(), Error> { - let threshold = total_stake * 2 / 3; - - if approved_stake <= &threshold { - log::debug!("Not enough stake approved"); - Err(Error::NotEnoughApprovedStake) - } else { - Ok(()) - } - } - - pub fn ensure_next_bps_is_valid( - expected_hash: &CryptoHash, - next_bps: &Option>, - ) -> Result, Error> { - if let Some(next_bps) = next_bps { - let next_bps_hash = CryptoHash::hash_borsh(next_bps); - if &next_bps_hash != expected_hash { - log::warn!("Next block producers hash is invalid"); - return Err(Error::NextBpsInvalid); - } - - Ok(next_bps.clone()) - } else { - // No new bps - Ok(Default::default()) - } - } - - pub fn validate_and_update_head( - &mut self, - block_view: &LightClientBlockView, - epoch_block_producers: Vec, - ) -> Result<(), Error> { - let (_, _, approval_message) = self.reconstruct_light_client_block_view_fields(block_view); - - Self::ensure_not_already_verified(&self.head, &block_view.inner_lite.height)?; - Self::ensure_epoch_is_current_or_next(&self.head, &block_view.inner_lite.epoch_id)?; - Self::ensure_if_next_epoch_contains_next_bps( - &self.head, - &block_view.inner_lite.epoch_id, - &block_view.next_bps, - )?; - let (total_stake, approved_stake) = Self::validate_signatures( - &block_view.approvals_after_next, - epoch_block_producers, - &approval_message, - )?; - Self::ensure_stake_is_sufficient(&total_stake, &approved_stake)?; - - if let Ok(next_bps) = Self::ensure_next_bps_is_valid( - &block_view.inner_lite.next_bp_hash, - &block_view.next_bps, - ) { - log::trace!("Setting next bps[{:?}]", self.head.inner_lite.next_epoch_id); - self.next_bps = Some((self.head.inner_lite.next_epoch_id, next_bps)); - } - - let prev_head = self.head.inner_lite.height; - self.head = LightClientBlockLiteView { - prev_block_hash: block_view.prev_block_hash, - inner_rest_hash: block_view.inner_rest_hash, - inner_lite: block_view.inner_lite.clone(), - }; - let new_head = self.head.inner_lite.height; - log::debug!("prev/current head: {}/{}", prev_head, new_head); - - Ok(()) - } -} - -#[cfg(test)] -mod tests { - use super::*; - use core::str::FromStr; - use near_primitives::merkle::compute_root_from_path; - use serde_json::{self, json}; - - fn fixture(file: &str) -> LightClientBlockView { - serde_json::from_reader(std::fs::File::open(file).unwrap()).unwrap() - } - - fn get_previous() -> LightClientBlockView { - fixture("fixtures/2_previous_epoch.json") - } - fn get_previous_previous() -> LightClientBlockView { - fixture("fixtures/3_previous_epoch.json") - } - - fn get_current() -> LightClientBlockView { - fixture("fixtures/1_current_epoch.json") - } - - fn get_epochs() -> Vec<(u32, LightClientBlockView, CryptoHash)> { - vec![ - ( - 1, - get_previous_previous(), - CryptoHash::from_str("B35Jn6mLXACRcsf6PATMixqgzqJZd71JaNh1LScJjFuJ").unwrap(), - ), - ( - 2, - get_previous(), - CryptoHash::from_str("Doy7Y7aVMgN8YhdAseGBMHNmYoqzWsXszqJ7MFLNMcQ7").unwrap(), - ), // Doy7Y7aVMgN8YhdAseGBMHNmYoqzWsXszqJ7MFLNMcQ7 - ( - 3, - get_current(), - CryptoHash::from_str("3tyxRRBgbYTo5DYd1LpX3EZtEiRYbDAAji6kcsf9QRge").unwrap(), - ), // 3tyxRRBgbYTo5DYd1LpX3EZtEiRYbDAAji6kcsf9QRge - ] - } - fn view_to_lite_view(h: LightClientBlockView) -> LightClientBlockLiteView { - LightClientBlockLiteView { - prev_block_hash: h.prev_block_hash, - inner_rest_hash: h.inner_rest_hash, - inner_lite: h.inner_lite, - } - } - - #[test] - fn test_headers() { - let headers_by_epoch = get_epochs(); - let next_epoch_id = headers_by_epoch[1].1.inner_lite.epoch_id.clone(); - - let mut state = LightClientState { - head: view_to_lite_view(headers_by_epoch[0].1.clone()), - next_bps: Some(( - next_epoch_id, - headers_by_epoch[0] - .1 - .next_bps - .clone() - .unwrap() - .into_iter() - .map(Into::into) - .collect(), - )), - }; - - let (current, _, approval_message) = - state.reconstruct_light_client_block_view_fields(&headers_by_epoch[1].1); - assert_eq!(current, headers_by_epoch[1].2 .0); - - let signature = headers_by_epoch[1].1.approvals_after_next[0] - .clone() - .unwrap(); - assert!(signature.verify( - &approval_message, - get_previous_previous().next_bps.unwrap()[0] - .clone() - .into_validator_stake() - .public_key() - )); - } - - fn bootstrap_state() -> (LightClientState, LightClientBlockView) { - let headers_by_epoch = get_epochs(); - let next_epoch_id = headers_by_epoch[1].1.inner_lite.epoch_id.clone(); - let new_header = headers_by_epoch[1].1.clone(); - - ( - LightClientState { - head: view_to_lite_view(headers_by_epoch[0].1.clone()), - next_bps: Some(( - next_epoch_id, - headers_by_epoch[0] - .1 - .next_bps - .clone() - .unwrap() - .into_iter() - .map(Into::into) - .collect(), - )), - }, - new_header, - ) - } - #[test] - fn test_blackbox_validate() { - pretty_env_logger::try_init().ok(); - let headers_by_epoch = get_epochs(); - - let (mut state, next_header) = bootstrap_state(); - - state - .validate_and_update_head( - &next_header, - headers_by_epoch[0].1.next_bps.clone().unwrap(), - ) - .unwrap(); - } - - #[test] - fn test_validate_already_verified() { - pretty_env_logger::try_init().ok(); - assert_eq!( - LightClientState::ensure_not_already_verified( - &bootstrap_state().0.head, - &BlockHeight::MIN - ), - Err(Error::BlockAlreadyVerified) - ); - } - #[test] - fn test_validate_bad_epoch() { - pretty_env_logger::try_init().ok(); - assert_eq!( - LightClientState::ensure_epoch_is_current_or_next( - &bootstrap_state().0.head, - &CryptoHash::hash_bytes(b"bogus hash") - ), - Err(Error::BlockNotCurrentOrNextEpoch) - ); - } - - #[test] - fn test_next_epoch_bps() { - pretty_env_logger::try_init().ok(); - - let (state, mut next_block) = bootstrap_state(); - next_block.next_bps = None; - - assert_eq!( - LightClientState::ensure_if_next_epoch_contains_next_bps( - &state.head, - &next_block.inner_lite.epoch_id, - &next_block.next_bps - ), - Err(Error::BlockMissingNextBps) - ); - } - - #[test] - fn test_next_invalid_signature() { - pretty_env_logger::try_init().ok(); - let (_, next_block) = bootstrap_state(); - let headers_by_epoch = get_epochs(); - - assert_eq!( - LightClientState::validate_signatures( - &next_block.approvals_after_next, - headers_by_epoch[0].1.next_bps.clone().unwrap(), - &b"bogus approval message"[..] - ), - Err(Error::SignatureInvalid) - ); - } - - #[test] - fn test_next_invalid_signatures_no_approved_stake() { - pretty_env_logger::try_init().ok(); - - let (mut state, mut next_block) = bootstrap_state(); - let headers_by_epoch = get_epochs(); - - let (_, _, approval_message) = - state.reconstruct_light_client_block_view_fields(&next_block); - - next_block.approvals_after_next = next_block - .approvals_after_next - .iter() - .cloned() - .map(|_| None) - .collect(); - - let (total_stake, approved_stake) = LightClientState::validate_signatures( - &next_block.approvals_after_next, - headers_by_epoch[0].1.next_bps.clone().unwrap(), - &approval_message, - ) - .unwrap(); - - assert_eq!( - (total_stake, approved_stake), - (512915271547861520119028536348929, 0) - ); - } - - #[test] - fn test_next_invalid_signatures_stake_isnt_sufficient() { - pretty_env_logger::try_init().ok(); - - let (mut state, next_block) = bootstrap_state(); - let headers_by_epoch = get_epochs(); - - let (_, _, approval_message) = - state.reconstruct_light_client_block_view_fields(&next_block); - - let (total_stake, approved_stake) = LightClientState::validate_signatures( - &next_block.approvals_after_next, - headers_by_epoch[0].1.next_bps.clone().unwrap(), - &approval_message, - ) - .unwrap(); - - assert_eq!( - (total_stake, approved_stake), - ( - 512915271547861520119028536348929, - 345140782903867823005444871054881 - ) - ); - - assert!( - LightClientState::ensure_stake_is_sufficient(&total_stake, &approved_stake).is_ok() - ); - - assert_eq!( - LightClientState::ensure_stake_is_sufficient( - &total_stake, - &((total_stake * 2) / 3 - 1) - ), - Err(Error::NotEnoughApprovedStake) - ); - } - - #[test] - fn test_next_invalid_bps_stake() { - let (state, next_block) = bootstrap_state(); - - assert_eq!( - LightClientState::ensure_if_next_epoch_contains_next_bps( - &state.head, - &next_block.inner_lite.epoch_id, - &None - ), - Err(Error::BlockMissingNextBps) - ); - } - - #[test] - fn test_next_bps_invalid_hash() { - pretty_env_logger::try_init().ok(); - - let (_, next_header) = bootstrap_state(); - - assert_eq!( - LightClientState::ensure_next_bps_is_valid( - &CryptoHash::hash_borsh(b"invalid"), - &next_header.next_bps - ), - Err(Error::NextBpsInvalid) - ); - } - - #[test] - fn test_next_bps() { - pretty_env_logger::try_init().ok(); - - let (_, next_block) = bootstrap_state(); - - assert_eq!( - LightClientState::ensure_next_bps_is_valid( - &next_block.inner_lite.next_bp_hash, - &next_block.next_bps - ) - .unwrap(), - next_block.next_bps.unwrap() - ); - } - - #[test] - fn test_next_bps_noop() { - pretty_env_logger::try_init().ok(); - - let (_, next_block) = bootstrap_state(); - - assert_eq!( - LightClientState::ensure_next_bps_is_valid(&next_block.inner_lite.next_bp_hash, &None) - .unwrap(), - vec![] - ); - } - - #[test] - fn test_can_create_current_hash() { - let file = "fixtures/well_known_header.json"; - let prev_hash = - CryptoHash::from_str("BUcVEkMq3DcZzDGgeh1sb7FFuD86XYcXpEt25Cf34LuP").unwrap(); - let well_known_header_view: BlockHeaderInnerLiteView = - serde_json::from_reader(std::fs::File::open(file).unwrap()).unwrap(); - - let inner_light_hash = LightClientState::inner_lite_hash(&well_known_header_view); - - // Inner rest hashing - for (inner_rest, expected) in vec![ - ( - "FaU2VzTNqxfouDtkQWcmrmU2UdvtSES3rQuccnZMtWAC", - "3ckGjcedZiN3RnvfiuEN83BtudDTVa9Pub4yZ8R737qt", - ), - ( - "BEqJyfEYNNrKmhHToZTvbeYqVjoqSe2w2uTMMT9KDaqb", - "Hezx56VTH815G6JTzWqJ7iuWxdR9X4ZqGwteaDF8q2z", - ), - ( - "7Kg3Uqeg7XSoQ7qXbed5JUR48YE49EgLiuqBTRuiDq6j", - "Finjr87adnUqpFHVXbmAWiVAY12EA9G4DfUw27XYHox", - ), - ] { - let inner_rest_hash = CryptoHash::from_str(inner_rest).unwrap(); - let expected = CryptoHash::from_str(expected).unwrap(); - - let inner_hash = CryptoHash::hash_bytes(&cvec!(inner_light_hash.0, inner_rest_hash.0)) - .0 - .to_vec(); - let current_hash = - CryptoHash::hash_bytes(&cvec!(inner_hash, prev_hash.as_bytes().to_vec())); - assert_eq!(current_hash, expected); - } - } - - #[test] - fn test_can_recreate_current_block_hash() { - let file = "fixtures/well_known_header.json"; - let prev_hash = - CryptoHash::from_str("BUcVEkMq3DcZzDGgeh1sb7FFuD86XYcXpEt25Cf34LuP").unwrap(); - let well_known_header_view: BlockHeaderInnerLiteView = - serde_json::from_reader(std::fs::File::open(file).unwrap()).unwrap(); - - // Inner rest hashing - for (inner_rest, expected) in vec![ - ( - "FaU2VzTNqxfouDtkQWcmrmU2UdvtSES3rQuccnZMtWAC", - "3ckGjcedZiN3RnvfiuEN83BtudDTVa9Pub4yZ8R737qt", - ), - ( - "BEqJyfEYNNrKmhHToZTvbeYqVjoqSe2w2uTMMT9KDaqb", - "Hezx56VTH815G6JTzWqJ7iuWxdR9X4ZqGwteaDF8q2z", - ), - ( - "7Kg3Uqeg7XSoQ7qXbed5JUR48YE49EgLiuqBTRuiDq6j", - "Finjr87adnUqpFHVXbmAWiVAY12EA9G4DfUw27XYHox", - ), - ] { - let inner_rest_hash = CryptoHash::from_str(inner_rest).unwrap(); - let expected = CryptoHash::from_str(expected).unwrap(); - - let view = LightClientBlockView { - inner_lite: well_known_header_view.clone(), - inner_rest_hash, - prev_block_hash: prev_hash, - // Not needed for this test - approvals_after_next: vec![], - next_bps: None, - next_block_inner_hash: CryptoHash::default(), - }; - - let current_hash = LightClientState::calculate_current_block_hash(&view); - assert_eq!(current_hash, expected); - } - } - - #[test] - fn test_shard_outcome_root() { - pretty_env_logger::try_init().ok(); - let req = r#"{"outcome_proof":{"proof":[],"block_hash":"5CY72FinjVV2Hd5zRikYYMaKh67pftXJsw8vwRXAUAQF","id":"9UhBumQ3eEmPH5ALc3NwiDCQfDrFakteRD7rHE9CfZ32","outcome":{"logs":[],"receipt_ids":["2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"],"gas_burnt":2434069818500,"tokens_burnt":"243406981850000000000","executor_id":"datayalla.testnet","status":{"SuccessReceiptId":"2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"},"metadata":{"version":1,"gas_profile":null}}},"outcome_root_proof":[{"hash":"9f7YjLvzvSspJMMJ3DDTrFaEyPQ5qFqQDNoWzAbSTjTy","direction":"Right"},{"hash":"67ZxFmzWXbWJSyi7Wp9FTSbbJx2nMr7wSuW3EP1cJm4K","direction":"Left"}],"block_header_lite":{"prev_block_hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","inner_rest_hash":"G25j8jSWRyrXV317cPC3qYA4SyJWXsBfErjhBYQkxw5A","inner_lite":{"height":134481525,"epoch_id":"4tBzDozzGED3QiCRURfViVuyJy5ikaN9dVH7m2MYkTyw","next_epoch_id":"9gYJSiT3TQbKbwui5bdbzBA9PCMSSfiffWhBdMtcasm2","prev_state_root":"EwkRecSP8GRvaxL7ynCEoHhsL1ksU6FsHVLCevcccF5q","outcome_root":"8Eu5qpDUMpW5nbmTrTKmDH2VYqFEHTKPETSTpPoyGoGc","timestamp":1691615068679535000,"timestamp_nanosec":"1691615068679535094","next_bp_hash":"8LCFsP6LeueT4X3PEni9CMvH7maDYpBtfApWZdXmagss","block_merkle_root":"583vb6csYnczHyt5z6Msm4LzzGkceTZHdvXjC8vcWeGK"}},"block_proof":[{"hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","direction":"Left"},{"hash":"HgZaHXpb5zs4rxUQTeW69XBNLBJoo4sz2YEDh7aFnMpC","direction":"Left"},{"hash":"EYNXYsnESQkXo7B27a9xu6YgbDSyynNcByW5Q2SqAaKH","direction":"Right"},{"hash":"AbKbsD7snoSnmzAtwNqXLBT5sm7bZr48GCCLSdksFuzi","direction":"Left"},{"hash":"7KKmS7n3MtCfv7UqciidJ24Abqsk8m85jVQTh94KTjYS","direction":"Left"},{"hash":"5nKA1HCZMJbdCccZ16abZGEng4sMoZhKez74rcCFjnhL","direction":"Left"},{"hash":"BupagAycSLD7v42ksgMKJFiuCzCdZ6ksrGLwukw7Vfe3","direction":"Right"},{"hash":"D6v37P4kcVJh8N9bV417eqJoyMeQbuZ743oNsbKxsU7z","direction":"Right"},{"hash":"8sWxxbe1rdquP5VdYfQbw1UvtcXDRansJYJV5ySzyow4","direction":"Right"},{"hash":"CmKVKWRqEqi4UaeKKYXpPSesYqdQYwHQM3E4xLKEUAj8","direction":"Left"},{"hash":"3TvjFzVyPBvPpph5zL6VCASLCxdNeiKV6foPwUpAGqRv","direction":"Left"},{"hash":"AnzSG9f91ePS6L6ii3eAkocp4iKjp6wjzSwWsDYWLnMX","direction":"Right"},{"hash":"FYVJDL4T6c87An3pdeBvntB68NzpcPtpvLP6ifjxxNkr","direction":"Left"},{"hash":"2YMF6KE8XTz7Axj3uyAoFbZisWej9Xo8mxgVtauWCZaV","direction":"Left"},{"hash":"4BHtLcxqNfWSneBdW76qsd8om8Gjg58Qw5BX8PHz93hf","direction":"Left"},{"hash":"7G3QUT7NQSHyXNQyzm8dsaYrFk5LGhYaG7aVafKAekyG","direction":"Left"},{"hash":"3XaMNnvnX69gGqBJX43Na1bSTJ4VUe7z6h5ZYJsaSZZR","direction":"Left"},{"hash":"FKu7GtfviPioyAGXGZLBVTJeG7KY5BxGwuL447oAZxiL","direction":"Right"},{"hash":"BePd7DPKUQnGtnSds5fMJGBUwHGxSNBpaNLwceJGUcJX","direction":"Left"},{"hash":"2BVKWMd9pXZTEyE9D3KL52hAWAyMrXj1NqutamyurrY1","direction":"Left"},{"hash":"EWavHKhwQiT8ApnXvybvc9bFY6aJYJWqBhcrZpubKXtA","direction":"Left"},{"hash":"83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD","direction":"Left"},{"hash":"AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL","direction":"Left"}]}"#; - let body: RpcLightClientExecutionProofResponse = serde_json::from_str(req).unwrap(); - // shard_outcome_root = compute_root(sha256(borsh(execution_outcome)), outcome_proof.proof) - - println!( - "Verifying light client proof for txn id: {:?}", - body.outcome_proof.id - ); - - let outcome_hash = CryptoHash::hash_borsh(&body.outcome_proof.to_hashes()); - println!("Hash of the outcome is: {:?}", outcome_hash); - - let shard_outcome_root = compute_root_from_path(&body.outcome_proof.proof, outcome_hash); - println!("Shard outcome root is: {:?}", shard_outcome_root); - let block_outcome_root = compute_root_from_path( - &body.outcome_root_proof, - CryptoHash::hash_borsh(shard_outcome_root), - ); - println!("Block outcome root is: {:?}", block_outcome_root); - - let expected_shard_outcome_root = - CryptoHash::from_str("AC7P5WhmY1kFaL7ZWpCToPcQpH4kuxBnzCx53BSWDzY7").unwrap(); - assert_eq!(shard_outcome_root, expected_shard_outcome_root); - } - - #[test] - fn test_root() { - let root = "WWrLWbWHwSmjtTn5oBZPYgRCuCYn6fkYVa4yhPWNK4L"; - - let old_block_hash = "7xTJs2wYhLcWKsT2Vcf4HPcXVdKKxTAPFTphhEhf3AxW"; - let old_proof = json!([ - { - "direction": "Left", - "hash": "GCvcb8M3qHWtvmSdEZFUWcTd6TwifECQZbV932S3AJzj" - }, - { - "direction": "Left", - "hash": "RYJXmEszufH799Rzsu669aUo96YCPqN8K93DWVYoj7A" - }, - { - "direction": "Left", - "hash": "C1epDr3Zy7sBLJcrwN1q8JNqCRbL9ZwGXdUdudMkKgZK" - }, - { - "direction": "Right", - "hash": "4S5gtNEiMPPLzSZ2HKQr62PEFa2sFKyGWZ3Hzm3wmmZv" - }, - { - "direction": "Left", - "hash": "3xo4CBANUZFc8bpX1X4cZHd9uEMJtxagxYjq5N9XGkF6" - }, - { - "direction": "Right", - "hash": "2ZecgrRDh4i7CcxHsS6NMRTMvd5NCxo3Mjx8prho7Gd8" - }, - { - "direction": "Left", - "hash": "51BaXGWzouQPb1dme26UmDZQ4PAJqUmcU3utoyKwQs3Z" - }, - { - "direction": "Left", - "hash": "3R2JTnndX74J1zYCnWPM4JdKvTrMwW8Tj3N7tcCmMPr5" - }, - { - "direction": "Right", - "hash": "74pTjaUkKY42tqDSy3wjRmH8X7bkjoiQiQ4AibnvShF2" - }, - { - "direction": "Right", - "hash": "FVKmSZx11uW5itjGZf6B3aCs7JRcL2i51Jb9VVeeY6QP" - }, - { - "direction": "Right", - "hash": "CQoiwaCYqcJxZTT9bqceKTGUffZiKQNKPf6YUvXihEQJ" - }, - { - "direction": "Right", - "hash": "DntvNk8Ugyu6ZdAUTV39MrPUrXQUFJKVaBUput6GMZiq" - }, - { - "direction": "Left", - "hash": "4ZXzvj47NPZXfwV6N2Z2TCs52AX143bKGTUNQR4KuL5F" - }, - { - "direction": "Left", - "hash": "6Czm39v4AvX1Xc5NZdsB94BNhawYhPAXRqyP3BephTyV" - }, - { - "direction": "Right", - "hash": "tojBg1nCgpCEpD3ssM4rTibXuCNQ6Bdfq7iByhPvukD" - }, - { - "direction": "Right", - "hash": "FHTGPaTNPPfcQ1yGuVCz1LgRxkQDjuSX4bk4L8kgDgQq" - }, - { - "direction": "Right", - "hash": "837fc2U3xTc7fMPPPvaXd2GzTkJhVWRxznk4xnzqg7kz" - }, - { - "direction": "Left", - "hash": "CAyu4UBn9JTTbq13pUGjY6nXCvjDaRzKwepwpJcpRLpn" - }, - { - "direction": "Left", - "hash": "6ejd5xCdeRefBX34FoXBv4V2hVcRa2S89dnymWbCdfpj" - }, - { - "direction": "Right", - "hash": "DYzHCZNv5VR8hR6dvYsA2CK8GaruXYA3GJmABpqB4rp" - }, - { - "direction": "Right", - "hash": "D9BmTFSNCoU9ZzKLANz8J11UBhNpWAGnn6LH8YD5Sw1D" - }, - { - "direction": "Right", - "hash": "CiUV5kcdjZpqw6iS2nYhiJZd5Prhh2ZHM7AXVoiCzr1R" - }, - { - "direction": "Right", - "hash": "E71mnUNN1jWnyvrfKLsTAAaQ1rJtVwuRLiSjUqR8QKaT" - }, - { - "direction": "Left", - "hash": "7x3QgcmcQu1Di5uYwP29nL4uRzLFv6tDpacZJrYTCDrY" - }, - { - "direction": "Left", - "hash": "83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD" - }, - { - "direction": "Left", - "hash": "AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL" - }]); - - let old = merkle::verify_hash( - CryptoHash::from_str(root).unwrap(), - &serde_json::from_value(old_proof).unwrap(), - CryptoHash::from_str(old_block_hash).unwrap(), - ); - assert!(old); - } -} diff --git a/src/client/proof.rs b/src/client/protocol/experimental.rs similarity index 91% rename from src/client/proof.rs rename to src/client/protocol/experimental.rs index 1a34851..89a17ba 100644 --- a/src/client/proof.rs +++ b/src/client/protocol/experimental.rs @@ -1,17 +1,18 @@ -use super::{LightClientState, Proof as FullProof}; +use super::BasicProof as FullProof; +use crate::client::protocol::Protocol; use borsh::{BorshDeserialize, BorshSerialize}; use either::Either; use itertools::Itertools; -use merkle_util::compute_root_from_path; use near_primitives::{ + block_header::BlockHeaderInnerLite, merkle::{combine_hash, MerklePathItem}, views::LightClientBlockLiteView, }; use near_primitives_core::hash::CryptoHash; use serde::{Deserialize, Serialize}; -pub(crate) mod merkle_util; - +/// Requires only needed parts of the LightClientBlockLiteView, and pre hashes +/// the inner_lite. #[derive(Debug, Clone, BorshSerialize, BorshDeserialize, Serialize, Deserialize)] pub struct LightweightHeader { inner_lite_hash: CryptoHash, @@ -31,8 +32,9 @@ impl LightweightHeader { impl From for LightweightHeader { fn from(header: LightClientBlockLiteView) -> Self { + let full_inner: BlockHeaderInnerLite = header.inner_lite.clone().into(); Self { - inner_lite_hash: LightClientState::inner_lite_hash(&header.inner_lite), + inner_lite_hash: CryptoHash::hash_borsh(&full_inner), inner_rest_hash: header.inner_rest_hash, prev_block_hash: header.prev_block_hash, outcome_root: header.inner_lite.outcome_root, @@ -40,13 +42,16 @@ impl From for LightweightHeader { } } +/// Reduces the need to hash inner_lite, inner_rest and prev_block_hash +/// +/// Useful for extremely expensive environments. #[derive(Debug, Clone, BorshSerialize, BorshDeserialize)] -pub struct LightestHeader { +pub struct BlindHeader { block_hash: CryptoHash, outcome_root: CryptoHash, } -impl From for LightestHeader { +impl From for BlindHeader { fn from(header: LightweightHeader) -> Self { Self { block_hash: header.hash(), @@ -58,7 +63,7 @@ impl From for LightestHeader { #[derive(Debug, Clone, BorshSerialize, BorshDeserialize)] pub enum Header { Lightweight(LightweightHeader), - Lightest(LightestHeader), + Lightest(BlindHeader), } impl Header { @@ -124,13 +129,13 @@ impl MerkleCache { self.items = duplicates.into_iter().map(|x| x.0.unwrap_right()).collect(); } + // TODO: no panic fn collect<'a>( &'a self, path: &'a [LookupMerklePathItem], ) -> impl Iterator { path.iter().map(|x| match &x.0 { - // TODO: no panic - Either::Left(i) => self.items.get(*i as usize).unwrap(), + Either::Left(i) => &self.items[*i as usize], Either::Right(x) => x, }) } @@ -289,34 +294,29 @@ impl Proof { } pub fn verify_proof(proof: Proof) -> bool { - // We should know about the known_sync_block_merkle_root + // TODO: We should know about the known_sync_block_merkle_root //assert!(blinded_headers.contains_key(&body.created_from)); - //let cache = Proof::build_flat_cache(&body.batch); proof.batch.into_iter().all(|blinded| { let block_hash = blinded.header.hash(); - // TODO: here we probably also want to make sure we know about this root in the light - // client log::debug!("Verifying blinded proof: {:?}", blinded.header.outcome_root); let block_hash_matches = block_hash == blinded.outcome_proof_block_hash; - let outcome_proof_root = compute_root_from_path( + let outcome_verified = Protocol::verify_outcome( + &blinded.outcome_hash, proof.cache.collect(&blinded.outcome_proof), - blinded.outcome_hash, - ); - let outcome_root_root = merkle_util::compute_root_from_path_and_item( proof.cache.collect(&blinded.outcome_root_proof), - outcome_proof_root, + &blinded.header.outcome_root, ); - let outcome_verified = outcome_root_root == blinded.header.outcome_root; - let block_proof = proof - .cache - .collect(&blinded.block_proof) - .chain(&proof.ancestry); - - let block_verified = - merkle_util::verify_hash(proof.client_block_merkle_root, block_proof, block_hash); + let block_verified = Protocol::verify_block( + &proof.client_block_merkle_root, + proof + .cache + .collect(&blinded.block_proof) + .chain(&proof.ancestry), + &block_hash, + ); log::debug!( "block_hash_matches: {:?}, outcome_verified: {:?}, block_verified: {:?}", @@ -329,17 +329,20 @@ pub fn verify_proof(proof: Proof) -> bool { } #[cfg(test)] -mod tests { +pub(crate) mod tests { use super::*; - use crate::client::rpc::{NearRpcClient, Network}; + use crate::client::{ + protocol::merkle_util::{compute_root_from_path, compute_root_from_path_and_item}, + rpc::{NearRpcClient, Network}, + }; use futures::FutureExt; use near_primitives_core::types::AccountId; use std::{path::Path, str::FromStr}; - const BLOCK_MERKLE_ROOT: &str = "WWrLWbWHwSmjtTn5oBZPYgRCuCYn6fkYVa4yhPWNK4L"; - const LIGHT_CLIENT_HEAD: &str = "4bM5eXMDGxpFZXbWNT6TqX1HdZsWoHZ11KerCHJ8RKmU"; + pub const BLOCK_MERKLE_ROOT: &str = "WWrLWbWHwSmjtTn5oBZPYgRCuCYn6fkYVa4yhPWNK4L"; + pub const LIGHT_CLIENT_HEAD: &str = "4bM5eXMDGxpFZXbWNT6TqX1HdZsWoHZ11KerCHJ8RKmU"; - fn get_constants() -> (CryptoHash, CryptoHash, NearRpcClient) { + pub fn get_constants() -> (CryptoHash, CryptoHash, NearRpcClient) { let client = NearRpcClient::new(Network::Testnet); let block_root = CryptoHash::from_str(BLOCK_MERKLE_ROOT).unwrap(); let light_client_head = CryptoHash::from_str(LIGHT_CLIENT_HEAD).unwrap(); @@ -389,7 +392,7 @@ mod tests { let cache = MerkleCache::default(); let outcome_proof_root = compute_root_from_path(cache.collect(&blinded.outcome_proof), blinded.outcome_hash); - let outcome_root = merkle_util::compute_root_from_path_and_item( + let outcome_root = compute_root_from_path_and_item( cache.collect(&blinded.outcome_root_proof), outcome_proof_root, ); diff --git a/src/client/proof/merkle_util.rs b/src/client/protocol/merkle_util.rs similarity index 68% rename from src/client/proof/merkle_util.rs rename to src/client/protocol/merkle_util.rs index 2b8a9f4..cbae81c 100644 --- a/src/client/proof/merkle_util.rs +++ b/src/client/protocol/merkle_util.rs @@ -1,14 +1,22 @@ use borsh::BorshSerialize; -use near_primitives::merkle::{combine_hash, Direction, MerklePathItem}; +pub use near_primitives::merkle::combine_hash; +use near_primitives::merkle::{Direction, MerklePathItem}; use near_primitives_core::{hash::CryptoHash, types::MerkleHash}; pub trait PathIterator<'a> = Iterator; -pub fn verify_hash<'a>(root: MerkleHash, path: impl PathIterator<'a>, item_hash: MerkleHash) -> bool { +pub fn verify_hash<'a>( + root: MerkleHash, + path: impl PathIterator<'a>, + item_hash: MerkleHash, +) -> bool { compute_root_from_path(path, item_hash) == root } -pub fn compute_root_from_path<'a>(path: impl PathIterator<'a>, item_hash: MerkleHash) -> MerkleHash { +pub fn compute_root_from_path<'a>( + path: impl PathIterator<'a>, + item_hash: MerkleHash, +) -> MerkleHash { let mut res = item_hash; for item in path { match item.direction { diff --git a/src/client/protocol/mod.rs b/src/client/protocol/mod.rs new file mode 100644 index 0000000..3014083 --- /dev/null +++ b/src/client/protocol/mod.rs @@ -0,0 +1,631 @@ +use super::{error::Error, BasicProof, Proof}; +use anyhow::Result; +use borsh::BorshSerialize; +use merkle_util::*; +use near_crypto::{PublicKey, Signature}; +use near_primitives::{ + block_header::ApprovalInner, + types::{validator_stake::ValidatorStake, EpochId}, + views::{ + validator_stake_view::ValidatorStakeView, LightClientBlockLiteView, LightClientBlockView, + }, +}; +use near_primitives_core::{hash::CryptoHash, types::BlockHeight}; + +pub mod merkle_util; + +// Lightweight batch protocol with lookups for proofs +pub mod experimental; + +#[derive(Debug)] +pub struct Synced { + pub new_head: LightClientBlockLiteView, + pub next_bps: Option<(EpochId, Vec)>, +} + +pub struct Protocol; + +impl Protocol { + pub fn sync( + head: &LightClientBlockLiteView, + epoch_bps: &[ValidatorStake], + next_block: LightClientBlockView, + ) -> Result { + println!("current_head: {:?}", head.inner_lite.height); + println!("next_block: {:?}", next_block.inner_lite.height); + Self::ensure_not_already_verified(&head, &next_block.inner_lite.height)?; + Self::ensure_epoch_is_current_or_next(&head, &next_block.inner_lite.epoch_id)?; + Self::ensure_if_next_epoch_contains_next_bps( + &head, + &next_block.inner_lite.epoch_id, + &next_block.next_bps, + )?; + + let new_head = LightClientBlockLiteView { + prev_block_hash: next_block.prev_block_hash, + inner_rest_hash: next_block.inner_rest_hash, + inner_lite: next_block.inner_lite.clone(), + }; + + let approval_message = Self::reconstruct_approval_message(&next_block).unwrap(); + + let StakeInfo { total, approved } = Self::validate_signatures( + &next_block.approvals_after_next, + epoch_bps, + &approval_message, + )?; + println!("total_stake: {:?}", total); + println!("approved_stake: {:?}", approved); + Self::ensure_stake_is_sufficient(&total, &approved)?; + + log::debug!( + "prev/current head: {}/{}", + head.inner_lite.height, + new_head.inner_lite.height + ); + + Ok(Synced { + new_head, + next_bps: Self::ensure_next_bps_is_valid( + &next_block.inner_lite.next_bp_hash, + next_block.next_bps, + )? + .map(|next_bps| next_bps.into_iter().map(Into::into).collect()) + .map(|next_bps| (EpochId(head.inner_lite.next_epoch_id), next_bps)), + }) + .inspect(|synced| log::debug!("Synced new head: {:?}", synced)) + } + + pub fn inclusion_proof_verify( + head: &LightClientBlockLiteView, + archive_header: Option<&LightClientBlockLiteView>, + proof: Proof, + ) -> Result { + match proof { + Proof::Experimental(proof) => Ok(experimental::verify_proof(proof)), + Proof::Basic(proof) => { + let block_hash = proof.block_header_lite.hash(); + let block_hash_matches = block_hash == proof.outcome_proof.block_hash; + + let outcome_hash = CryptoHash::hash_borsh(proof.outcome_proof.to_hashes()); + + let outcome_verified = Self::verify_outcome( + &outcome_hash, + proof.outcome_proof.proof.iter(), + proof.outcome_root_proof.iter(), + &proof.block_header_lite.inner_lite.outcome_root, + ); + + let mut block_verified = Self::verify_block( + &head.inner_lite.block_merkle_root, + proof.block_proof.iter(), + &block_hash, + ); + + // TODO: refactor me + // Functionality to cover race conditions between previously synced heads since the current head may of changed + // since the proof was generated + match archive_header { + Some(archive_header) if !block_verified => { + log::debug!( + "Trying to verify against archival header: {:?}", + archive_header.inner_lite.height + ); + if verify_hash( + archive_header.inner_lite.block_merkle_root, + proof.block_proof.iter(), + block_hash, + ) { + log::debug!("Verified against archival header"); + block_verified = true; + } else { + return Err(Error::InvalidProof.into()); + } + } + Some(_) => {} + None => return Err(Error::InvalidProof.into()), + } + + log::debug!( + "shard outcome included: {:?}, block included: {:?}", + outcome_verified, + block_verified + ); + Ok(block_hash_matches && outcome_verified && block_verified) + .inspect(|verified| log::debug!("Verified proof {:?}", verified)) + } + } + } + + pub(crate) fn verify_outcome<'a>( + outcome_hash: &CryptoHash, + outcome_proof: impl PathIterator<'a>, + outcome_root_proof: impl PathIterator<'a>, + expected_outcome_root: &CryptoHash, + ) -> bool { + let outcome_root = compute_root_from_path(outcome_proof, *outcome_hash); + + let outcome_root = compute_root_from_path_and_item(outcome_root_proof, outcome_root); + log::debug!("outcome_root: {:?}", outcome_root); + + &outcome_root == expected_outcome_root + } + + pub(crate) fn verify_block<'a>( + block_merkle_root: &CryptoHash, + block_proof: impl PathIterator<'a>, + block_hash: &CryptoHash, + ) -> bool { + verify_hash(*block_merkle_root, block_proof, *block_hash) + } + + fn next_block_hash( + current_block_hash: &CryptoHash, + block_view: &LightClientBlockView, + ) -> CryptoHash { + // TODO: remove the check once tests pass + let old = CryptoHash::hash_bytes(&{ + let mut temp_vec = Vec::new(); + temp_vec.extend_from_slice(&(block_view.next_block_inner_hash.0)); + temp_vec.extend_from_slice(&(current_block_hash.0)); + temp_vec + }); + let new = combine_hash(&block_view.next_block_inner_hash, ¤t_block_hash); + assert_eq!(old, new); + log::debug!("Current block hash: {}", current_block_hash); + new + } + + fn reconstruct_approval_message(block_view: &LightClientBlockView) -> Option> { + let new_head = LightClientBlockLiteView { + prev_block_hash: block_view.prev_block_hash, + inner_rest_hash: block_view.inner_rest_hash, + inner_lite: block_view.inner_lite.clone(), + }; + + let next_block_hash = Self::next_block_hash(&new_head.hash(), &block_view); + let endorsement = ApprovalInner::Endorsement(next_block_hash); + + let approval_message = { + let mut temp_vec = Vec::new(); + temp_vec.extend_from_slice(&(endorsement.try_to_vec().ok()?[..])); + temp_vec.extend_from_slice(&((block_view.inner_lite.height + 2).to_le_bytes()[..])); + temp_vec + }; + + log::debug!("Next block hash: {}", next_block_hash); + log::debug!("Approval message: {:?}", approval_message); + Option::Some(approval_message) + } + + pub fn ensure_not_already_verified( + head: &LightClientBlockLiteView, + block_height: &BlockHeight, + ) -> Result<(), Error> { + if block_height <= &head.inner_lite.height { + Err(Error::BlockAlreadyVerified) + } else { + Ok(()) + } + } + + pub fn ensure_epoch_is_current_or_next( + head: &LightClientBlockLiteView, + epoch_id: &CryptoHash, + ) -> Result<(), Error> { + if ![head.inner_lite.epoch_id, head.inner_lite.next_epoch_id].contains(epoch_id) { + log::debug!("Block is not in the current or next epoch"); + Err(Error::BlockNotCurrentOrNextEpoch.into()) + } else { + Ok(()) + } + } + + pub fn ensure_if_next_epoch_contains_next_bps( + head: &LightClientBlockLiteView, + epoch_id: &CryptoHash, + next_bps: &Option>, + ) -> Result<(), Error> { + if &head.inner_lite.next_epoch_id == epoch_id && next_bps.is_none() { + log::debug!("Block is in the next epoch but no new set"); + Err(Error::NextBpsInvalid) + } else { + Ok(()) + } + } + + pub fn validate_signatures( + signatures: &[Option], + epoch_bps: &[ValidatorStake], + approval_message: &[u8], + ) -> Result { + // Signatures can contain more than the number of bps due to epoch switch + if signatures.len() < epoch_bps.len() { + log::debug!( + "Signature len {} does not match BPS len {}", + signatures.len(), + epoch_bps.len() + ); + return Err(Error::SignatureLenMismatch); + } + + Ok(signatures + .iter() + .zip(epoch_bps.iter()) + .fold((0, 0), |(total_stake, approved_stake), (sig, vs)| { + let pk = vs.public_key(); + let stake = vs.stake(); + let total_stake = total_stake + stake; + + let approved_stake = match Self::validate_signature(approval_message, sig, pk) { + Ok(_) => approved_stake + stake, + Err(Error::SignatureInvalid) | Err(Error::ValidatorNotSigned) => approved_stake, + Err(_) => approved_stake, + }; + + (total_stake, approved_stake) + }) + .into()) + } + + fn validate_signature( + msg: &[u8], + sig: &Option, + pk: &PublicKey, + ) -> Result<(), Error> { + match sig { + Some(signature) if signature.verify(msg, pk) => { + log::trace!("{} {:?} pk {} approved", signature, msg, pk); + Ok(()) + } + Some(signature) => { + log::debug!("{} invalid sig {}", pk, signature); + Err(Error::SignatureInvalid) + } + _ => { + log::trace!("{} signature not approved", pk); + Err(Error::ValidatorNotSigned) + } + } + } + + pub fn ensure_stake_is_sufficient( + total_stake: &u128, + approved_stake: &u128, + ) -> Result<(), Error> { + let threshold = total_stake / 3 * 2; + + if approved_stake <= &threshold { + log::debug!("Not enough stake approved"); + Err(Error::NotEnoughApprovedStake.into()) + } else { + Ok(()) + } + } + + pub fn ensure_next_bps_is_valid( + expected_hash: &CryptoHash, + next_bps: Option>, + ) -> Result>, Error> { + if let Some(next_bps) = next_bps { + let next_bps_hash = CryptoHash::hash_borsh(next_bps.clone()); + + if &next_bps_hash == expected_hash { + Ok(Some(next_bps)) + } else { + log::warn!("Next block producers hash is invalid"); + return Err(Error::NextBpsInvalid); + } + } else { + Ok(None) + } + } +} + +#[derive(Debug, Clone, PartialEq, Eq)] +pub(crate) struct StakeInfo { + total: u128, + approved: u128, +} + +impl From<(u128, u128)> for StakeInfo { + fn from((total, approved): (u128, u128)) -> Self { + Self { total, approved } + } +} + +#[macro_export] +macro_rules! cvec { + ($($x:expr),*) => { + { + let mut temp_vec = Vec::new(); + $( + temp_vec.extend_from_slice(&$x); + )* + temp_vec + } + }; +} + +#[cfg(test)] +mod tests { + use super::*; + use crate::client::protocol::experimental::tests::get_constants; + use core::str::FromStr; + use itertools::Itertools; + use near_jsonrpc_client::methods::light_client_proof::RpcLightClientExecutionProofResponse; + use serde_json::{self}; + + fn fixture(file: &str) -> LightClientBlockView { + serde_json::from_reader(std::fs::File::open(format!("fixtures/{}", file)).unwrap()).unwrap() + } + + fn get_next_epoch() -> LightClientBlockView { + fixture("2.json") + } + + fn get_first_epoch() -> LightClientBlockView { + fixture("1.json") + } + + fn get_current_epoch() -> LightClientBlockView { + fixture("current_epoch.json") + } + + fn view_to_lite_view(h: LightClientBlockView) -> LightClientBlockLiteView { + LightClientBlockLiteView { + prev_block_hash: h.prev_block_hash, + inner_rest_hash: h.inner_rest_hash, + inner_lite: h.inner_lite, + } + } + + fn test_state() -> ( + LightClientBlockLiteView, + Vec, + LightClientBlockView, + ) { + let first = get_first_epoch(); + let next = get_next_epoch(); + + ( + view_to_lite_view(first.clone()), + first + .next_bps + .clone() + .unwrap() + .into_iter() + .map(Into::into) + .collect(), + next, + ) + } + + #[test] + fn test_sync_across_epoch_boundaries() { + let (mut head, mut next_bps, next_block) = test_state(); + let mut next_epoch_id = EpochId(head.inner_lite.next_epoch_id); + + let mut sync_and_update = |next_block: LightClientBlockView| { + let sync_next = Protocol::sync(&head, &next_bps, next_block.clone()).unwrap(); + // Assert we matched the epoch id for the new BPS + assert_eq!( + head.inner_lite.next_epoch_id, + sync_next.next_bps.as_ref().unwrap().0 .0 + ); + + head = sync_next.new_head; + next_bps = sync_next.next_bps.unwrap().1; + + // Assert new head is the new block + assert_eq!(head.inner_lite, next_block.inner_lite); + // Assert new BPS is from the next block producers because we're + // in an epoch boundary + assert_eq!( + &next_bps, + &next_block + .next_bps + .unwrap() + .into_iter() + .map(Into::into) + .collect_vec() + ); + next_epoch_id.0 = head.inner_lite.next_epoch_id; + }; + + // Do first sync + sync_and_update(next_block.clone()); + + // Get next header, do next sync + let next_block = get_current_epoch(); + sync_and_update(next_block.clone()); + } + + #[test] + fn test_validate_already_verified() { + pretty_env_logger::try_init().ok(); + + let (head, _, _) = test_state(); + assert_eq!( + Protocol::ensure_not_already_verified(&head, &BlockHeight::MIN), + Err(Error::BlockAlreadyVerified) + ); + } + + #[test] + fn test_validate_bad_epoch() { + pretty_env_logger::try_init().ok(); + + let (head, _, _) = test_state(); + assert_eq!( + Protocol::ensure_epoch_is_current_or_next( + &head, + &CryptoHash::hash_bytes(b"bogus hash") + ), + Err(Error::BlockNotCurrentOrNextEpoch) + ); + } + + #[test] + fn test_next_epoch_bps_invalid() { + pretty_env_logger::try_init().ok(); + + let (head, _, mut next_block) = test_state(); + next_block.next_bps = None; + + assert_eq!( + Protocol::ensure_if_next_epoch_contains_next_bps( + &head, + &next_block.inner_lite.epoch_id, + &next_block.next_bps + ), + Err(Error::NextBpsInvalid) + ); + } + + #[test] + fn test_next_invalid_signature() { + pretty_env_logger::try_init().ok(); + + let (_, next_bps, next_block) = test_state(); + assert_eq!( + Protocol::validate_signature( + &b"bogus approval message"[..], + &next_block.approvals_after_next[0], + &next_bps[0].public_key(), + ), + Err(Error::SignatureInvalid) + ); + } + + #[test] + fn test_next_invalid_signatures_no_approved_stake() { + pretty_env_logger::try_init().ok(); + + let (_, next_bps, mut next_block) = test_state(); + + let approval_message = Protocol::reconstruct_approval_message(&next_block); + // Nobody signed anything + next_block.approvals_after_next = next_block + .approvals_after_next + .iter() + .cloned() + .map(|_| None) + .collect(); + + let StakeInfo { total, approved } = Protocol::validate_signatures( + &next_block.approvals_after_next, + &next_bps.clone(), + &approval_message.unwrap(), + ) + .unwrap(); + + assert_eq!((total, approved), (512915271547861520119028536348929, 0)); + } + + #[test] + fn test_next_invalid_signatures_stake_isnt_sufficient() { + pretty_env_logger::try_init().ok(); + + let (_, next_bps, next_block) = test_state(); + + let approval_message = Protocol::reconstruct_approval_message(&next_block); + + let StakeInfo { total, approved } = Protocol::validate_signatures( + &next_block.approvals_after_next, + &next_bps, + &approval_message.unwrap(), + ) + .unwrap(); + + assert_eq!( + (total, approved), + ( + 512915271547861520119028536348929, + 345140782903867823005444871054881 + ) + ); + + assert!(Protocol::ensure_stake_is_sufficient(&total, &approved).is_ok()); + + let min_approval_amount = (total / 3) * 2; + + assert_eq!( + Protocol::ensure_stake_is_sufficient(&total, &(min_approval_amount - 1)), + Err(Error::NotEnoughApprovedStake) + ); + } + + #[test] + fn test_next_bps_invalid_hash() { + pretty_env_logger::try_init().ok(); + + let (_, _, next_block) = test_state(); + + assert_eq!( + Protocol::ensure_next_bps_is_valid( + &CryptoHash::hash_borsh(b"invalid"), + next_block.next_bps + ), + Err(Error::NextBpsInvalid) + ); + } + + #[test] + fn test_next_bps() { + pretty_env_logger::try_init().ok(); + + let (_, _, next_block) = test_state(); + + assert_eq!( + Protocol::ensure_next_bps_is_valid( + &next_block.inner_lite.next_bp_hash, + next_block.next_bps.clone() + ) + .unwrap(), + next_block.next_bps + ); + } + + #[test] + fn test_next_bps_noop_on_empty() { + pretty_env_logger::try_init().ok(); + + let (_, _, next_block) = test_state(); + assert_eq!( + Protocol::ensure_next_bps_is_valid(&next_block.inner_lite.next_bp_hash, None).unwrap(), + None + ); + } + + #[test] + fn test_outcome_root() { + pretty_env_logger::try_init().ok(); + let req = r#"{"outcome_proof":{"proof":[],"block_hash":"5CY72FinjVV2Hd5zRikYYMaKh67pftXJsw8vwRXAUAQF","id":"9UhBumQ3eEmPH5ALc3NwiDCQfDrFakteRD7rHE9CfZ32","outcome":{"logs":[],"receipt_ids":["2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"],"gas_burnt":2434069818500,"tokens_burnt":"243406981850000000000","executor_id":"datayalla.testnet","status":{"SuccessReceiptId":"2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"},"metadata":{"version":1,"gas_profile":null}}},"outcome_root_proof":[{"hash":"9f7YjLvzvSspJMMJ3DDTrFaEyPQ5qFqQDNoWzAbSTjTy","direction":"Right"},{"hash":"67ZxFmzWXbWJSyi7Wp9FTSbbJx2nMr7wSuW3EP1cJm4K","direction":"Left"}],"block_header_lite":{"prev_block_hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","inner_rest_hash":"G25j8jSWRyrXV317cPC3qYA4SyJWXsBfErjhBYQkxw5A","inner_lite":{"height":134481525,"epoch_id":"4tBzDozzGED3QiCRURfViVuyJy5ikaN9dVH7m2MYkTyw","next_epoch_id":"9gYJSiT3TQbKbwui5bdbzBA9PCMSSfiffWhBdMtcasm2","prev_state_root":"EwkRecSP8GRvaxL7ynCEoHhsL1ksU6FsHVLCevcccF5q","outcome_root":"8Eu5qpDUMpW5nbmTrTKmDH2VYqFEHTKPETSTpPoyGoGc","timestamp":1691615068679535000,"timestamp_nanosec":"1691615068679535094","next_bp_hash":"8LCFsP6LeueT4X3PEni9CMvH7maDYpBtfApWZdXmagss","block_merkle_root":"583vb6csYnczHyt5z6Msm4LzzGkceTZHdvXjC8vcWeGK"}},"block_proof":[{"hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","direction":"Left"},{"hash":"HgZaHXpb5zs4rxUQTeW69XBNLBJoo4sz2YEDh7aFnMpC","direction":"Left"},{"hash":"EYNXYsnESQkXo7B27a9xu6YgbDSyynNcByW5Q2SqAaKH","direction":"Right"},{"hash":"AbKbsD7snoSnmzAtwNqXLBT5sm7bZr48GCCLSdksFuzi","direction":"Left"},{"hash":"7KKmS7n3MtCfv7UqciidJ24Abqsk8m85jVQTh94KTjYS","direction":"Left"},{"hash":"5nKA1HCZMJbdCccZ16abZGEng4sMoZhKez74rcCFjnhL","direction":"Left"},{"hash":"BupagAycSLD7v42ksgMKJFiuCzCdZ6ksrGLwukw7Vfe3","direction":"Right"},{"hash":"D6v37P4kcVJh8N9bV417eqJoyMeQbuZ743oNsbKxsU7z","direction":"Right"},{"hash":"8sWxxbe1rdquP5VdYfQbw1UvtcXDRansJYJV5ySzyow4","direction":"Right"},{"hash":"CmKVKWRqEqi4UaeKKYXpPSesYqdQYwHQM3E4xLKEUAj8","direction":"Left"},{"hash":"3TvjFzVyPBvPpph5zL6VCASLCxdNeiKV6foPwUpAGqRv","direction":"Left"},{"hash":"AnzSG9f91ePS6L6ii3eAkocp4iKjp6wjzSwWsDYWLnMX","direction":"Right"},{"hash":"FYVJDL4T6c87An3pdeBvntB68NzpcPtpvLP6ifjxxNkr","direction":"Left"},{"hash":"2YMF6KE8XTz7Axj3uyAoFbZisWej9Xo8mxgVtauWCZaV","direction":"Left"},{"hash":"4BHtLcxqNfWSneBdW76qsd8om8Gjg58Qw5BX8PHz93hf","direction":"Left"},{"hash":"7G3QUT7NQSHyXNQyzm8dsaYrFk5LGhYaG7aVafKAekyG","direction":"Left"},{"hash":"3XaMNnvnX69gGqBJX43Na1bSTJ4VUe7z6h5ZYJsaSZZR","direction":"Left"},{"hash":"FKu7GtfviPioyAGXGZLBVTJeG7KY5BxGwuL447oAZxiL","direction":"Right"},{"hash":"BePd7DPKUQnGtnSds5fMJGBUwHGxSNBpaNLwceJGUcJX","direction":"Left"},{"hash":"2BVKWMd9pXZTEyE9D3KL52hAWAyMrXj1NqutamyurrY1","direction":"Left"},{"hash":"EWavHKhwQiT8ApnXvybvc9bFY6aJYJWqBhcrZpubKXtA","direction":"Left"},{"hash":"83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD","direction":"Left"},{"hash":"AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL","direction":"Left"}]}"#; + let p: RpcLightClientExecutionProofResponse = serde_json::from_str(req).unwrap(); + + let outcome_hash = CryptoHash::hash_borsh(&p.outcome_proof.to_hashes()); + + let root_matches = Protocol::verify_outcome( + &outcome_hash, + p.outcome_proof.proof.iter(), + p.outcome_root_proof.iter(), + &p.block_header_lite.inner_lite.outcome_root, + ); + assert!(root_matches); + } + + #[test] + fn test_root() { + pretty_env_logger::try_init().ok(); + let (_head, root, _) = get_constants(); + + let block_hash = + CryptoHash::from_str("7xTJs2wYhLcWKsT2Vcf4HPcXVdKKxTAPFTphhEhf3AxW").unwrap(); + let req = r#"{"outcome_proof":{"proof":[],"block_hash":"5CY72FinjVV2Hd5zRikYYMaKh67pftXJsw8vwRXAUAQF","id":"9UhBumQ3eEmPH5ALc3NwiDCQfDrFakteRD7rHE9CfZ32","outcome":{"logs":[],"receipt_ids":["2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"],"gas_burnt":2434069818500,"tokens_burnt":"243406981850000000000","executor_id":"datayalla.testnet","status":{"SuccessReceiptId":"2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"},"metadata":{"version":1,"gas_profile":null}}},"outcome_root_proof":[{"hash":"9f7YjLvzvSspJMMJ3DDTrFaEyPQ5qFqQDNoWzAbSTjTy","direction":"Right"},{"hash":"67ZxFmzWXbWJSyi7Wp9FTSbbJx2nMr7wSuW3EP1cJm4K","direction":"Left"}],"block_header_lite":{"prev_block_hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","inner_rest_hash":"G25j8jSWRyrXV317cPC3qYA4SyJWXsBfErjhBYQkxw5A","inner_lite":{"height":134481525,"epoch_id":"4tBzDozzGED3QiCRURfViVuyJy5ikaN9dVH7m2MYkTyw","next_epoch_id":"9gYJSiT3TQbKbwui5bdbzBA9PCMSSfiffWhBdMtcasm2","prev_state_root":"EwkRecSP8GRvaxL7ynCEoHhsL1ksU6FsHVLCevcccF5q","outcome_root":"8Eu5qpDUMpW5nbmTrTKmDH2VYqFEHTKPETSTpPoyGoGc","timestamp":1691615068679535000,"timestamp_nanosec":"1691615068679535094","next_bp_hash":"8LCFsP6LeueT4X3PEni9CMvH7maDYpBtfApWZdXmagss","block_merkle_root":"583vb6csYnczHyt5z6Msm4LzzGkceTZHdvXjC8vcWeGK"}},"block_proof":[{"hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","direction":"Left"},{"hash":"HgZaHXpb5zs4rxUQTeW69XBNLBJoo4sz2YEDh7aFnMpC","direction":"Left"},{"hash":"EYNXYsnESQkXo7B27a9xu6YgbDSyynNcByW5Q2SqAaKH","direction":"Right"},{"hash":"AbKbsD7snoSnmzAtwNqXLBT5sm7bZr48GCCLSdksFuzi","direction":"Left"},{"hash":"7KKmS7n3MtCfv7UqciidJ24Abqsk8m85jVQTh94KTjYS","direction":"Left"},{"hash":"5nKA1HCZMJbdCccZ16abZGEng4sMoZhKez74rcCFjnhL","direction":"Left"},{"hash":"BupagAycSLD7v42ksgMKJFiuCzCdZ6ksrGLwukw7Vfe3","direction":"Right"},{"hash":"D6v37P4kcVJh8N9bV417eqJoyMeQbuZ743oNsbKxsU7z","direction":"Right"},{"hash":"8sWxxbe1rdquP5VdYfQbw1UvtcXDRansJYJV5ySzyow4","direction":"Right"},{"hash":"CmKVKWRqEqi4UaeKKYXpPSesYqdQYwHQM3E4xLKEUAj8","direction":"Left"},{"hash":"3TvjFzVyPBvPpph5zL6VCASLCxdNeiKV6foPwUpAGqRv","direction":"Left"},{"hash":"AnzSG9f91ePS6L6ii3eAkocp4iKjp6wjzSwWsDYWLnMX","direction":"Right"},{"hash":"FYVJDL4T6c87An3pdeBvntB68NzpcPtpvLP6ifjxxNkr","direction":"Left"},{"hash":"2YMF6KE8XTz7Axj3uyAoFbZisWej9Xo8mxgVtauWCZaV","direction":"Left"},{"hash":"4BHtLcxqNfWSneBdW76qsd8om8Gjg58Qw5BX8PHz93hf","direction":"Left"},{"hash":"7G3QUT7NQSHyXNQyzm8dsaYrFk5LGhYaG7aVafKAekyG","direction":"Left"},{"hash":"3XaMNnvnX69gGqBJX43Na1bSTJ4VUe7z6h5ZYJsaSZZR","direction":"Left"},{"hash":"FKu7GtfviPioyAGXGZLBVTJeG7KY5BxGwuL447oAZxiL","direction":"Right"},{"hash":"BePd7DPKUQnGtnSds5fMJGBUwHGxSNBpaNLwceJGUcJX","direction":"Left"},{"hash":"2BVKWMd9pXZTEyE9D3KL52hAWAyMrXj1NqutamyurrY1","direction":"Left"},{"hash":"EWavHKhwQiT8ApnXvybvc9bFY6aJYJWqBhcrZpubKXtA","direction":"Left"},{"hash":"83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD","direction":"Left"},{"hash":"AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL","direction":"Left"}]}"#; + let p: RpcLightClientExecutionProofResponse = serde_json::from_str(req).unwrap(); + + let block_verified = Protocol::verify_block(&root, p.block_proof.iter(), &block_hash); + assert!(block_verified); + } +} diff --git a/src/controller.rs b/src/controller.rs index c53abea..880bb0b 100644 --- a/src/controller.rs +++ b/src/controller.rs @@ -14,6 +14,7 @@ use crate::client::Message; type ClientState = flume::Sender; +// TODO: replace with jsonrpc pub(crate) fn init(ctx: Sender) -> JoinHandle> { let proof_channel = flume::bounded(64); @@ -107,7 +108,6 @@ mod proof { use super::*; use crate::client::{Proof, ProofType}; use axum::Json; - use near_jsonrpc_client::methods::light_client_proof::RpcLightClientExecutionProofResponse; use near_primitives_core::types::AccountId; pub type ProofChannel = (flume::Sender>, flume::Receiver>); @@ -173,7 +173,7 @@ mod proof { pub(super) async fn post_proof( State((client, (tx, rx))): State<(ClientState, ValidateProofChannel)>, - Json(proof): Json, + Json(proof): Json, ) -> impl IntoResponse { log::debug!("post_proof: {:?}", proof); if let Err(e) = client diff --git a/src/erasure.rs b/src/erasure.rs index b7229f5..f1df845 100644 --- a/src/erasure.rs +++ b/src/erasure.rs @@ -1,3 +1,4 @@ +use anyhow::Result; use near_primitives::merkle::{self, MerklePath}; use near_primitives_core::hash::CryptoHash; use reed_solomon_novelpoly::WrappedShard; @@ -15,7 +16,7 @@ pub struct Erasure { #[allow(unused)] impl Erasure { - pub fn encodify(data: &[u8]) -> anyhow::Result { + pub fn encodify(data: &[u8]) -> Result { Ok(Self { shards: reed_solomon_novelpoly::encode(data, VALIDATORS)? .into_iter() @@ -24,7 +25,7 @@ impl Erasure { }) } - pub fn recover(&self) -> anyhow::Result> { + pub fn recover(&self) -> Result> { Ok(reed_solomon_novelpoly::reconstruct( self.shards.clone(), VALIDATORS, From 47d6d31aa16ae499e92f7b2dea2433139fa422fc Mon Sep 17 00:00:00 2001 From: dndll Date: Wed, 3 Jan 2024 17:39:06 +0000 Subject: [PATCH 4/4] feat: expose experimental batch proofs BREAKING CHANGE: store has been refactored dramatically to introduce a plugin api. BREAKING CHANGE: controller has been refactored to expose experimental api. --- .gitignore | 2 +- Cargo.lock | 1867 ++++++++++++++++++--------- Cargo.toml | 27 +- default.toml | 4 +- fixtures/batch.json | 4 +- flake.lock | 18 +- flake.nix | 7 +- rust-toolchain.toml | 5 +- src/client/error.rs | 6 +- src/client/message.rs | 58 + src/client/mod.rs | 532 ++++---- src/client/protocol/experimental.rs | 41 +- src/client/protocol/merkle_util.rs | 8 +- src/client/protocol/mod.rs | 127 +- src/client/rpc.rs | 50 +- src/client/store.rs | 291 +++++ src/config.rs | 28 +- src/controller.rs | 221 ++-- src/erasure.rs | 116 -- src/main.rs | 23 +- testnet.toml | 7 +- 21 files changed, 2107 insertions(+), 1335 deletions(-) create mode 100644 src/client/message.rs create mode 100644 src/client/store.rs delete mode 100644 src/erasure.rs diff --git a/.gitignore b/.gitignore index ca47641..4ea0f70 100644 --- a/.gitignore +++ b/.gitignore @@ -4,4 +4,4 @@ local.toml -db.sled/ +state.db diff --git a/Cargo.lock b/Cargo.lock index 68af862..869ba91 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -4,13 +4,14 @@ version = 3 [[package]] name = "actix" -version = "0.13.0" +version = "0.13.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f728064aca1c318585bf4bb04ffcfac9e75e508ab4e8b1bd9ba5dfe04e2cbed5" +checksum = "cba56612922b907719d4a01cf11c8d5b458e7d3dba946d0435f20f58d6795ed2" dependencies = [ + "actix-macros", "actix-rt", "actix_derive", - "bitflags 1.3.2", + "bitflags 2.4.1", "bytes", "crossbeam-channel", "futures-core", @@ -23,14 +24,24 @@ dependencies = [ "pin-project-lite", "smallvec", "tokio", - "tokio-util 0.7.8", + "tokio-util 0.7.10", +] + +[[package]] +name = "actix-macros" +version = "0.2.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e01ed3140b2f8d422c68afa1ed2e85d996ea619c988ac834d255db32138655cb" +dependencies = [ + "quote", + "syn 2.0.46", ] [[package]] name = "actix-rt" -version = "2.8.0" +version = "2.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "15265b6b8e2347670eb363c47fc8c75208b4a4994b27192f345fcbe707804f3e" +checksum = "28f32d40287d3f402ae0028a9d54bef51af15c8769492826a69d28f81893151d" dependencies = [ "futures-core", "tokio", @@ -38,13 +49,13 @@ dependencies = [ [[package]] name = "actix_derive" -version = "0.6.0" +version = "0.6.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6d44b8fee1ced9671ba043476deddef739dd0959bf77030b26b738cc591737a7" +checksum = "7c7db3d5a9718568e4cf4a537cfd7070e6e6ff7481510d0237fb529ac850f6d3" dependencies = [ "proc-macro2", "quote", - "syn 1.0.109", + "syn 2.0.46", ] [[package]] @@ -55,31 +66,41 @@ checksum = "f26201604c87b1e01bd3d98f8d5d9a8fcbb815e8cedb41ffccbeb4bf593a35fe" [[package]] name = "ahash" -version = "0.7.6" +version = "0.7.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fcb51a0695d8f838b1ee009b3fbf66bda078cd64590202a864a8f3e8c4315c47" +checksum = "5a824f2aa7e75a0c98c5a504fceb80649e9c35265d44525b5f94de4771a395cd" dependencies = [ - "getrandom 0.2.10", + "getrandom 0.2.11", "once_cell", "version_check", ] [[package]] name = "ahash" -version = "0.8.3" +version = "0.8.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2c99f64d1e06488f620f932677e24bc6e2897582980441ae90a671415bd7ec2f" +checksum = "77c3a9648d43b9cd48db467b3f87fdd6e146bcc88ab0180006cef2179fe11d01" dependencies = [ "cfg-if", "once_cell", "version_check", + "zerocopy", ] [[package]] name = "aho-corasick" -version = "1.0.2" +version = "0.7.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cc936419f96fa211c1b9166887b38e5e40b19958e5b895be7c1f93adec7071ac" +dependencies = [ + "memchr", +] + +[[package]] +name = "aho-corasick" +version = "1.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "43f6cb1bf222025340178f382c426f13757b2960e89779dfcb319c32542a5a41" +checksum = "b2969dcb958b36655471fc61f7e416fa76033bdd4bfed0678d8fee1e2d07a1f0" dependencies = [ "memchr", ] @@ -99,6 +120,12 @@ dependencies = [ "alloc-no-stdlib", ] +[[package]] +name = "allocator-api2" +version = "0.2.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0942ffc6dcaadf03badf6e6a2d0228460359d5e34b57ccdc720b7382dfbd5ec5" + [[package]] name = "android-tzdata" version = "0.1.1" @@ -116,73 +143,72 @@ dependencies = [ [[package]] name = "anstream" -version = "0.3.2" +version = "0.6.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0ca84f3628370c59db74ee214b3263d58f9aadd9b4fe7e711fd87dc452b7f163" +checksum = "d664a92ecae85fd0a7392615844904654d1d5f5514837f471ddef4a057aba1b6" dependencies = [ "anstyle", "anstyle-parse", "anstyle-query", "anstyle-wincon", "colorchoice", - "is-terminal", "utf8parse", ] [[package]] name = "anstyle" -version = "1.0.1" +version = "1.0.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3a30da5c5f2d5e72842e00bcb57657162cdabef0931f40e2deb9b4140440cecd" +checksum = "7079075b41f533b8c61d2a4d073c4676e1f8b249ff94a393b0595db304e0dd87" [[package]] name = "anstyle-parse" -version = "0.2.1" +version = "0.2.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "938874ff5980b03a87c5524b3ae5b59cf99b1d6bc836848df7bc5ada9643c333" +checksum = "c75ac65da39e5fe5ab759307499ddad880d724eed2f6ce5b5e8a26f4f387928c" dependencies = [ "utf8parse", ] [[package]] name = "anstyle-query" -version = "1.0.0" +version = "1.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5ca11d4be1bab0c8bc8734a9aa7bf4ee8316d462a08c6ac5052f888fef5b494b" +checksum = "e28923312444cdd728e4738b3f9c9cac739500909bb3d3c94b43551b16517648" dependencies = [ - "windows-sys", + "windows-sys 0.52.0", ] [[package]] name = "anstyle-wincon" -version = "1.0.2" +version = "3.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c677ab05e09154296dd37acecd46420c17b9713e8366facafa8fc0885167cf4c" +checksum = "1cd54b81ec8d6180e24654d0b371ad22fc3dd083b6ff8ba325b72e00c87660a7" dependencies = [ "anstyle", - "windows-sys", + "windows-sys 0.52.0", ] [[package]] name = "anyhow" -version = "1.0.72" +version = "1.0.79" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3b13c32d80ecc7ab747b80c3784bce54ee8a7a0cc4fbda9bf4cda2cf6fe90854" +checksum = "080e9890a082662b09c1ad45f567faeeb47f22b5fb23895fbe1e651e718e25ca" [[package]] name = "arbitrary" -version = "1.3.0" +version = "1.3.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e2d098ff73c1ca148721f37baad5ea6a465a13f9573aba8641fbbbae8164a54e" +checksum = "7d5a26814d8dcb93b0e5a0ff3c6d80a8843bafb21b39e8e18a6f05471870e110" dependencies = [ "derive_arbitrary", ] [[package]] name = "async-compression" -version = "0.4.1" +version = "0.4.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "62b74f44609f0f91493e3082d3734d98497e094777144380ea4db9f9905dd5b6" +checksum = "bc2d0cfb2a7388d34f590e76686704c494ed7aaceed62ee1ba35cbf363abc2a5" dependencies = [ "brotli", "flate2", @@ -211,18 +237,18 @@ checksum = "16e62a023e7c117e27523144c5d2459f4397fcc3cab0085af8e2224f643a0193" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "async-trait" -version = "0.1.72" +version = "0.1.77" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cc6dde6e4ed435a4c1ee4e73592f5ba9da2151af10076cc04858746af9352d09" +checksum = "c980ee35e870bd1a4d2c8294d4c04d0499e67bca1e4b5cefcc693c2fa00caea9" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -249,13 +275,46 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3b829e4e32b91e643de6eafe82b1d90675f5874230191a4ffbc1b336dec4d6bf" dependencies = [ "async-trait", - "axum-core", + "axum-core 0.3.4", "bitflags 1.3.2", "bytes", "futures-util", - "http", - "http-body", - "hyper", + "http 0.2.11", + "http-body 0.4.6", + "hyper 0.14.28", + "itoa", + "matchit", + "memchr", + "mime", + "percent-encoding", + "pin-project-lite", + "rustversion", + "serde", + "serde_json", + "serde_path_to_error", + "serde_urlencoded", + "sync_wrapper", + "tokio", + "tower", + "tower-layer", + "tower-service", +] + +[[package]] +name = "axum" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d09dbe0e490df5da9d69b36dca48a76635288a82f92eca90024883a56202026d" +dependencies = [ + "async-trait", + "axum-core 0.4.2", + "bytes", + "futures-util", + "http 1.0.0", + "http-body 1.0.0", + "http-body-util", + "hyper 1.1.0", + "hyper-util", "itoa", "matchit", "memchr", @@ -272,6 +331,7 @@ dependencies = [ "tower", "tower-layer", "tower-service", + "tracing", ] [[package]] @@ -283,12 +343,33 @@ dependencies = [ "async-trait", "bytes", "futures-util", - "http", - "http-body", + "http 0.2.11", + "http-body 0.4.6", + "mime", + "rustversion", + "tower-layer", + "tower-service", +] + +[[package]] +name = "axum-core" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e87c8503f93e6d144ee5690907ba22db7ba79ab001a932ab99034f0fe836b3df" +dependencies = [ + "async-trait", + "bytes", + "futures-util", + "http 1.0.0", + "http-body 1.0.0", + "http-body-util", "mime", + "pin-project-lite", "rustversion", + "sync_wrapper", "tower-layer", "tower-service", + "tracing", ] [[package]] @@ -299,9 +380,9 @@ checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8" [[package]] name = "base64" -version = "0.21.2" +version = "0.21.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "604178f6c5c21f02dc555784810edfb88d34ac2c73b2eae109655649ee73ce3d" +checksum = "35636a1494ede3b646cc98f74f8e62c773a38a659ebc777a2cf26b9b74171df9" [[package]] name = "bitflags" @@ -311,9 +392,9 @@ checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" [[package]] name = "bitflags" -version = "2.3.3" +version = "2.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "630be753d4e58660abd17930c71b647fe46c27ea6b63cc59e1e3851406972e42" +checksum = "327762f6e5a765692301e5bb513e0d9fef63be86bbc14528052b1cd3e6f03e07" [[package]] name = "blake2" @@ -391,9 +472,9 @@ dependencies = [ [[package]] name = "brotli" -version = "3.3.4" +version = "3.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a1a0b1dbcc8ae29329621f8d4f0d835787c1c38bb1401979b49d13b0b305ff68" +checksum = "516074a47ef4bce09577a3b379392300159ce5b1ba2e501ff1c819950066100f" dependencies = [ "alloc-no-stdlib", "alloc-stdlib", @@ -402,9 +483,9 @@ dependencies = [ [[package]] name = "brotli-decompressor" -version = "2.3.4" +version = "2.5.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4b6561fd3f895a11e8f72af2cb7d22e08366bebc2b6b57f7744c4bda27034744" +checksum = "4e2e4afe60d7dd600fdd3de8d0f08c2b7ec039712e3b6137ff98b7004e82de4f" dependencies = [ "alloc-no-stdlib", "alloc-stdlib", @@ -418,27 +499,27 @@ checksum = "771fe0050b883fcc3ea2359b1a96bcfbc090b7116eae7c3c512c7a083fdf23d3" [[package]] name = "bumpalo" -version = "3.13.0" +version = "3.14.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a3e2c3daef883ecc1b5d58c15adae93470a91d425f3532ba1695849656af3fc1" +checksum = "7f30e7476521f6f8af1a1c4c0b8cc94f0bee37d91763d0ca2665f299b6cd8aec" [[package]] name = "byteorder" -version = "1.4.3" +version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "14c189c53d098945499cdfa7ecc63567cf3886b3332b312a5b4585d8d3a6a610" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" [[package]] name = "bytes" -version = "1.4.0" +version = "1.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "89b2fd2a0dcf38d7971e2194b6b6eebab45ae01067456a7fd93d5547a61b70be" +checksum = "a2bd12c1caf447e69cd4528f47f94d203fd2582878ecb9e9465484c4148a8223" [[package]] name = "bytesize" -version = "1.2.0" +version = "1.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "38fcc2979eff34a4b84e1cf9a1e3da42a7d44b3b690a40cdcb23e3d556cfb2e5" +checksum = "a3e368af43e418a04d52505cf3dbc23dda4e3407ae2fa99fd0e4f308ce546acc" dependencies = [ "serde", ] @@ -455,9 +536,9 @@ dependencies = [ [[package]] name = "cc" -version = "1.0.82" +version = "1.0.83" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "305fe645edc1442a0fa8b6726ba61d422798d37a52e12eaecf4b022ebbb88f01" +checksum = "f1174fb0b6ec23863f8b971027804a42614e347eafb0a95bf0b12cdae21fc4d0" dependencies = [ "libc", ] @@ -468,26 +549,19 @@ 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 = "chrono" -version = "0.4.26" +version = "0.4.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ec837a71355b28f6556dbd569b37b3f363091c0bd4b2e735674521b4c5fd9bc5" +checksum = "7f2c685bad3eb3d45a01354cedb7d5faa66194d1d58ba6e267a8de788f79db38" dependencies = [ "android-tzdata", "iana-time-zone", "js-sys", "num-traits", "serde", - "time 0.1.45", "wasm-bindgen", - "winapi", + "windows-targets 0.48.5", ] [[package]] @@ -501,20 +575,19 @@ dependencies = [ [[package]] name = "clap" -version = "4.3.21" +version = "4.4.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c27cdf28c0f604ba3f512b0c9a409f8de8513e4816705deb0498b627e7c3a3fd" +checksum = "dcfab8ba68f3668e89f6ff60f5b205cea56aa7b769451a59f34b8682f51c056d" dependencies = [ "clap_builder", "clap_derive", - "once_cell", ] [[package]] name = "clap_builder" -version = "4.3.21" +version = "4.4.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "08a9f1ab5e9f01a9b81f202e8562eb9a10de70abf9eaeac1be465c28b75aa4aa" +checksum = "fb7fb5e4e979aec3be7791562fcba452f94ad85e954da024396433e0e25a79e9" dependencies = [ "anstream", "anstyle", @@ -524,21 +597,57 @@ dependencies = [ [[package]] name = "clap_derive" -version = "4.3.12" +version = "4.4.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "54a9bb5758fc5dfe728d1019941681eccaf0cf8a4189b692a0ee2f2ecf90a050" +checksum = "cf9804afaaf59a91e75b022a30fb7229a7901f60c755489cc61c9b423b836442" dependencies = [ "heck 0.4.1", "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "clap_lex" -version = "0.5.0" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "702fc72eb24e5a1e48ce58027a675bc24edd52096d5397d4aea7c6dd9eca0bd1" + +[[package]] +name = "coerce" +version = "0.8.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2da6da31387c7e4ef160ffab6d5e7f00c42626fe39aea70a7b0f1773f7dd6c1b" +checksum = "ba30a4bf4bebf9112b9f42124de78b0e5dccad05c86181072663c7d8567aedec" +dependencies = [ + "anyhow", + "async-trait", + "axum 0.6.20", + "byteorder", + "bytes", + "chrono", + "futures", + "hashring", + "hmac", + "jwt", + "lazy_static", + "metrics", + "metrics-exporter-prometheus", + "metrics-util", + "parking_lot 0.12.1", + "protobuf 3.2.0", + "rand 0.8.5", + "serde", + "serde_json", + "sha2 0.10.8", + "tokio", + "tokio-stream", + "tokio-util 0.7.10", + "tracing", + "utoipa", + "utoipa-swagger-ui", + "uuid", + "valuable", +] [[package]] name = "colorchoice" @@ -548,9 +657,9 @@ checksum = "acbf1af155f9b9ef647e42cdc158db4b64a1b61f743629225fde6f3e0be2a7c7" [[package]] name = "config" -version = "0.13.3" +version = "0.13.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d379af7f68bfc21714c6c7dea883544201741d2ce8274bb12fa54f89507f52a7" +checksum = "23738e11972c7643e4ec947840fc463b6a571afcd3e735bdfce7d03c7a784aca" dependencies = [ "async-trait", "json5", @@ -573,9 +682,9 @@ checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e" [[package]] name = "core-foundation" -version = "0.9.3" +version = "0.9.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "194a7a9e6de53fa55116934067c844d9d749312f75c6f6d0980e8c252f8c2146" +checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f" dependencies = [ "core-foundation-sys", "libc", @@ -583,15 +692,15 @@ dependencies = [ [[package]] name = "core-foundation-sys" -version = "0.8.4" +version = "0.8.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e496a50fda8aacccc86d7529e2c1e0892dbd0f898a6b5645b5561b89c3210efa" +checksum = "06ea2b9bc92be3c2baa9334a323ebca2d6f074ff852cd1d7b11064035cd3868f" [[package]] name = "cpufeatures" -version = "0.2.9" +version = "0.2.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a17b76ff3a4162b0b27f354a0c87015ddad39d35f9c0c36607a3bdd175dde1f1" +checksum = "ce420fe07aecd3e67c5f910618fe65e94158f6dcc0adf44e00d69ce2bdfe0fd0" dependencies = [ "libc", ] @@ -607,9 +716,9 @@ dependencies = [ [[package]] name = "crossbeam-channel" -version = "0.5.8" +version = "0.5.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a33c2bf77f2df06183c3aa30d1e96c0695a313d4f9c453cc3762a6db39f99200" +checksum = "82a9b73a36529d9c47029b9fb3a6f0ea3cc916a261195352ba19e770fc1748b2" dependencies = [ "cfg-if", "crossbeam-utils", @@ -617,22 +726,20 @@ dependencies = [ [[package]] name = "crossbeam-epoch" -version = "0.9.15" +version = "0.9.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ae211234986c545741a7dc064309f67ee1e5ad243d0e48335adc0484d960bcc7" +checksum = "0e3681d554572a651dda4186cd47240627c3d0114d45a95f6ad27f2f22e7548d" dependencies = [ "autocfg", "cfg-if", "crossbeam-utils", - "memoffset", - "scopeguard", ] [[package]] name = "crossbeam-utils" -version = "0.8.16" +version = "0.8.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5a22b2d63d4d1dc0b7f1b6b2747dd0088008a9be28b6ddf0b1e7d335e3037294" +checksum = "c3a430a770ebd84726f584a90ee7f020d28db52c6d02138900f22341f866d39c" dependencies = [ "cfg-if", ] @@ -697,7 +804,7 @@ dependencies = [ "proc-macro2", "quote", "strsim", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -708,27 +815,28 @@ checksum = "836a9bbc7ad63342d6d6e7b815ccab164bc77a2d95d84bc3117a8c0d5c98e2d5" dependencies = [ "darling_core", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "deranged" -version = "0.3.7" +version = "0.3.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7684a49fb1af197853ef7b2ee694bc1f5b4179556f1e5710e1760c5db6f5e929" +checksum = "b42b6fa04a440b495c8b04d0e71b707c585f83cb9cb28cf8cd0d976c315e31b4" dependencies = [ + "powerfmt", "serde", ] [[package]] name = "derive_arbitrary" -version = "1.3.1" +version = "1.3.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "53e0efad4403bfc52dc201159c4b842a246a14b98c64b55dfd0f2d89729dfeb8" +checksum = "67e77553c4162a157adbf834ebae5b415acbecbeafc7a74b0e886657506a7611" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -761,6 +869,27 @@ checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" dependencies = [ "block-buffer 0.10.4", "crypto-common", + "subtle", +] + +[[package]] +name = "dirs" +version = "4.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ca3aa72a6f96ea37bbc5aa912f6788242832f75369bdfdadcb0e38423f100059" +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]] @@ -809,41 +938,47 @@ dependencies = [ [[package]] name = "encoding_rs" -version = "0.8.32" +version = "0.8.33" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "071a31f4ee85403370b58aca746f01041ede6f0da2730960ad001edc2b71b394" +checksum = "7268b386296a025e474d5140678f75d6de9493ae55a5d709eeb9dd08149945e1" dependencies = [ "cfg-if", ] +[[package]] +name = "endian-type" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c34f04666d835ff5d62e058c3995147c06f42fe86ff053337632bca83e42702d" + [[package]] name = "enum-map" -version = "2.6.1" +version = "2.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9705d8de4776df900a4a0b2384f8b0ab42f775e93b083b42f8ce71bdc32a47e3" +checksum = "6866f3bfdf8207509a033af1a75a7b08abda06bbaaeae6669323fd5a097df2e9" dependencies = [ "enum-map-derive", ] [[package]] name = "enum-map-derive" -version = "0.13.0" +version = "0.17.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ccb14d927583dd5c2eac0f2cf264fc4762aefe1ae14c47a8a20fc1939d3a5fc0" +checksum = "f282cfdfe92516eb26c2af8589c274c7c17681f5ecc03c18255fe741c6aa64eb" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "env_logger" -version = "0.7.1" +version = "0.10.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "44533bbbb3bb3c1fa17d9f2e4e38bbbaf8396ba82193c4cb1b6445d711445d36" +checksum = "95b3f3e67048839cb0d0781f445682a35113da7121f7c949db0e2be96a4fbece" dependencies = [ - "atty", "humantime", + "is-terminal", "log", "regex", "termcolor", @@ -857,30 +992,19 @@ checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5" [[package]] name = "errno" -version = "0.3.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6b30f669a7961ef1631673d2766cc92f52d64f7ef354d4fe0ddfd30ed52f0f4f" -dependencies = [ - "errno-dragonfly", - "libc", - "windows-sys", -] - -[[package]] -name = "errno-dragonfly" -version = "0.1.2" +version = "0.3.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "aa68f1b12764fab894d2755d2518754e71b4fd80ecfb822714a1206c2aab39bf" +checksum = "a258e46cdc063eb8519c00b9fc845fc47bcfca4130e2f08e88665ceda8474245" dependencies = [ - "cc", "libc", + "windows-sys 0.52.0", ] [[package]] name = "fastrand" -version = "2.0.0" +version = "2.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6999dc1837253364c2ebb0704ba97994bd874e8f195d665c50b7548f6ea92764" +checksum = "25cbce373ec4653f1a01a31e8a5e5ec0c622dc27ff9c4e6606eefef5cbbed4a5" [[package]] name = "fixed-hash" @@ -899,9 +1023,9 @@ checksum = "0ce7134b9999ecaf8bcd65542e436736ef32ddca1b3e06094cb6ec5755203b80" [[package]] name = "flate2" -version = "1.0.26" +version = "1.0.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3b9429470923de8e8cbd4d2dc513535400b4b3fef0319fb5c4e1f520a7bef743" +checksum = "46303f565772937ffe1d394a4fac6f411c6013172fadde9dcdb1e147a086940e" dependencies = [ "crc32fast", "miniz_oxide", @@ -943,22 +1067,13 @@ checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b" [[package]] name = "form_urlencoded" -version = "1.2.0" +version = "1.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a62bc1cf6f830c2ec14a513a9fb124d0a213a629668a4186f329db21fe045652" +checksum = "e13624c2627564efccf4934284bdd98cbaa14e79b0b5a141218e507b3a823456" dependencies = [ "percent-encoding", ] -[[package]] -name = "fs-err" -version = "2.10.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fb5fd9bcbe8b1087cbd395b51498c01bc997cef73e778a80b77a811af5e2d29f" -dependencies = [ - "autocfg", -] - [[package]] name = "fs2" version = "0.4.3" @@ -971,9 +1086,9 @@ dependencies = [ [[package]] name = "futures" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "23342abe12aba583913b2e62f22225ff9c950774065e4bfb61a19cd9770fec40" +checksum = "645c6916888f6cb6350d2550b80fb63e734897a8498abe35cfb732b6487804b0" dependencies = [ "futures-channel", "futures-core", @@ -986,9 +1101,9 @@ dependencies = [ [[package]] name = "futures-channel" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "955518d47e09b25bbebc7a18df10b81f0c766eaf4c4f1cccef2fca5f2a4fb5f2" +checksum = "eac8f7d7865dcb88bd4373ab671c8cf4508703796caa2b1985a9ca867b3fcb78" dependencies = [ "futures-core", "futures-sink", @@ -996,15 +1111,15 @@ dependencies = [ [[package]] name = "futures-core" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4bca583b7e26f571124fe5b7561d49cb2868d79116cfa0eefce955557c6fee8c" +checksum = "dfc6580bb841c5a68e9ef15c77ccc837b40a7504914d52e47b8b0e9bbda25a1d" [[package]] name = "futures-executor" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ccecee823288125bd88b4d7f565c9e58e41858e47ab72e8ea2d64e93624386e0" +checksum = "a576fc72ae164fca6b9db127eaa9a9dda0d61316034f33a0a0d4eda41f02b01d" dependencies = [ "futures-core", "futures-task", @@ -1013,38 +1128,38 @@ dependencies = [ [[package]] name = "futures-io" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4fff74096e71ed47f8e023204cfd0aa1289cd54ae5430a9523be060cdb849964" +checksum = "a44623e20b9681a318efdd71c299b6b222ed6f231972bfe2f224ebad6311f0c1" [[package]] name = "futures-macro" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "89ca545a94061b6365f2c7355b4b32bd20df3ff95f02da9329b34ccc3bd6ee72" +checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "futures-sink" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f43be4fe21a13b9781a69afa4985b0f6ee0e1afab2c6f454a8cf30e2b2237b6e" +checksum = "9fb8e00e87438d937621c1c6269e53f536c14d3fbd6a042bb24879e57d474fb5" [[package]] name = "futures-task" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "76d3d132be6c0e6aa1534069c705a74a5997a356c0dc2f86a47765e5617c5b65" +checksum = "38d84fa142264698cdce1a9f9172cf383a0c82de1bddcf3092901442c4097004" [[package]] name = "futures-util" -version = "0.3.28" +version = "0.3.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "26b01e40b772d54cf6c6d721c1d1abd0647a0106a12ecaa1c186273392a69533" +checksum = "3d6401deb83407ab3da39eba7e33987a73c3df0c82b4bb5813ee871c19c41d48" dependencies = [ "futures-channel", "futures-core", @@ -1090,9 +1205,9 @@ dependencies = [ [[package]] name = "getrandom" -version = "0.2.10" +version = "0.2.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "be4136b2a15dd319360be1c07d9933517ccf0be8f16bf62a3bee4f0d618df427" +checksum = "fe9006bed769170c11f845cf00c7c1e9092aeb3f268e007c3e760ac68008070f" dependencies = [ "cfg-if", "js-sys", @@ -1103,20 +1218,39 @@ dependencies = [ [[package]] name = "h2" -version = "0.3.20" +version = "0.3.22" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "97ec8491ebaf99c8eaa73058b045fe58073cd6be7f596ac993ced0b0a0c01049" +checksum = "4d6250322ef6e60f93f9a2162799302cd6f68f79f6e5d85c8c16f14d1d958178" dependencies = [ "bytes", "fnv", "futures-core", "futures-sink", "futures-util", - "http", - "indexmap 1.9.3", + "http 0.2.11", + "indexmap 2.1.0", + "slab", + "tokio", + "tokio-util 0.7.10", + "tracing", +] + +[[package]] +name = "h2" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e1d308f63daf4181410c242d34c11f928dcb3aa105852019e043c9d1f4e4368a" +dependencies = [ + "bytes", + "fnv", + "futures-core", + "futures-sink", + "futures-util", + "http 1.0.0", + "indexmap 2.1.0", "slab", "tokio", - "tokio-util 0.7.8", + "tokio-util 0.7.10", "tracing", ] @@ -1126,7 +1260,7 @@ version = "0.12.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8a9ee70c43aaf417c914396645a0fa852624801b24ebb7ae78fe8272889ac888" dependencies = [ - "ahash 0.7.6", + "ahash 0.7.7", ] [[package]] @@ -1135,14 +1269,27 @@ version = "0.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "43a3c133739dddd0d2990f9a4bdf8eb4b21ef50e4851ca85ab661199821d510e" dependencies = [ - "ahash 0.8.3", + "ahash 0.8.7", ] [[package]] name = "hashbrown" -version = "0.14.0" +version = "0.14.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2c6201b9ff9fd90a5a3bac2e56a830d0caa509576f0e503818ee82c181b3437a" +checksum = "290f1a1d9242c78d09ce40a5e87e7554ee637af1351968159f4952f028f75604" +dependencies = [ + "ahash 0.8.7", + "allocator-api2", +] + +[[package]] +name = "hashring" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aa283406d74fcfeb4778f4e300beaae30db96793371da168d003cbc833e149e0" +dependencies = [ + "siphasher", +] [[package]] name = "heck" @@ -1170,9 +1317,9 @@ dependencies = [ [[package]] name = "hermit-abi" -version = "0.3.2" +version = "0.3.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "443144c8cdadd93ebf52ddb4056d257f5b52c04d3c804e657d19eb73fc33668b" +checksum = "d77f7ec81a6d05a3abb01ab6eb7590f6083d08449fe5a1c8b1e620283546ccb7" [[package]] name = "hex" @@ -1183,11 +1330,40 @@ dependencies = [ "serde", ] +[[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 = "home" +version = "0.5.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3d1354bf6b7235cb4a0576c2619fd4ed18183f689b12b006a0ee7329eeff9a5" +dependencies = [ + "windows-sys 0.52.0", +] + [[package]] name = "http" -version = "0.2.9" +version = "0.2.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bd6effc99afb63425aff9b05836f029929e345a6148a14b7ecd5ab67af944482" +checksum = "8947b1a6fad4393052c7ba1f4cd97bed3e953a95c79c92ad9b051a04611d9fbb" +dependencies = [ + "bytes", + "fnv", + "itoa", +] + +[[package]] +name = "http" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b32afd38673a8016f7c9ae69e5af41a58f81b1d31689040f2f1959594ce194ea" dependencies = [ "bytes", "fnv", @@ -1196,12 +1372,35 @@ dependencies = [ [[package]] name = "http-body" -version = "0.4.5" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ceab25649e9960c0311ea418d17bee82c0dcec1bd053b5f9a66e265a693bed2" +dependencies = [ + "bytes", + "http 0.2.11", + "pin-project-lite", +] + +[[package]] +name = "http-body" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1cac85db508abc24a2e48553ba12a996e87244a0395ce011e62b37158745d643" +dependencies = [ + "bytes", + "http 1.0.0", +] + +[[package]] +name = "http-body-util" +version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d5f38f16d184e36f2408a55281cd658ecbd3ca05cce6d6510a176eca393e26d1" +checksum = "41cb79eb393015dadd30fc252023adb0b2400a0caee0fa2a077e6e21a551e840" dependencies = [ "bytes", - "http", + "futures-util", + "http 1.0.0", + "http-body 1.0.0", "pin-project-lite", ] @@ -1213,37 +1412,34 @@ checksum = "d897f394bad6a705d5f4104762e116a75639e470d80901eed05a860a95cb1904" [[package]] name = "httpdate" -version = "1.0.2" +version = "1.0.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c4a1e36c821dbe04574f602848a19f742f4fb3c98d40449f11bcad18d6b17421" +checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9" [[package]] name = "humantime" -version = "1.3.0" +version = "2.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "df004cfca50ef23c36850aaaa59ad52cc70d0e90243c3c7737a4dd32dc7a3c4f" -dependencies = [ - "quick-error", -] +checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4" [[package]] name = "hyper" -version = "0.14.27" +version = "0.14.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ffb1cfd654a8219eaef89881fdb3bb3b1cdc5fa75ded05d6933b2b382e395468" +checksum = "bf96e135eb83a2a8ddf766e426a841d8ddd7449d5f00d34ea02b41d2f19eef80" dependencies = [ "bytes", "futures-channel", "futures-core", "futures-util", - "h2", - "http", - "http-body", + "h2 0.3.22", + "http 0.2.11", + "http-body 0.4.6", "httparse", "httpdate", "itoa", "pin-project-lite", - "socket2", + "socket2 0.5.5", "tokio", "tower-service", "tracing", @@ -1251,42 +1447,79 @@ dependencies = [ ] [[package]] -name = "hyper-timeout" -version = "0.4.1" +name = "hyper" +version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bbb958482e8c7be4bc3cf272a766a2b0bf1a6755e7a6ae777f017a31d11b13b1" +checksum = "fb5aa53871fc917b1a9ed87b683a5d86db645e23acb32c2e0785a353e522fb75" dependencies = [ - "hyper", + "bytes", + "futures-channel", + "futures-util", + "h2 0.4.0", + "http 1.0.0", + "http-body 1.0.0", + "httparse", + "httpdate", + "itoa", "pin-project-lite", "tokio", - "tokio-io-timeout", ] [[package]] -name = "hyper-tls" -version = "0.5.0" +name = "hyper-timeout" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bbb958482e8c7be4bc3cf272a766a2b0bf1a6755e7a6ae777f017a31d11b13b1" +dependencies = [ + "hyper 0.14.28", + "pin-project-lite", + "tokio", + "tokio-io-timeout", +] + +[[package]] +name = "hyper-tls" +version = "0.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d6183ddfa99b85da61a140bea0efc93fdf56ceaa041b37d553518030827f9905" dependencies = [ "bytes", - "hyper", + "hyper 0.14.28", "native-tls", "tokio", "tokio-native-tls", ] +[[package]] +name = "hyper-util" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bdea9aac0dbe5a9240d68cfd9501e2db94222c6dc06843e06640b9e07f0fdc67" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "http 1.0.0", + "http-body 1.0.0", + "hyper 1.1.0", + "pin-project-lite", + "socket2 0.5.5", + "tokio", + "tracing", +] + [[package]] name = "iana-time-zone" -version = "0.1.57" +version = "0.1.59" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2fad5b825842d2b38bd206f3e81d6957625fd7f0a361e345c30e01a0ae2dd613" +checksum = "b6a67363e2aa4443928ce15e57ebae94fd8949958fd1223c4cfc0cd473ad7539" dependencies = [ "android_system_properties", "core-foundation-sys", "iana-time-zone-haiku", "js-sys", "wasm-bindgen", - "windows", + "windows-core", ] [[package]] @@ -1306,9 +1539,9 @@ checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" [[package]] name = "idna" -version = "0.4.0" +version = "0.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7d20d6b07bfbc108882d88ed8e37d39636dcc260e15e30c45e6ba089610b917c" +checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" dependencies = [ "unicode-bidi", "unicode-normalization", @@ -1327,12 +1560,12 @@ dependencies = [ [[package]] name = "indexmap" -version = "2.0.0" +version = "2.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d5477fe2230a79769d8dc68e0eabf5437907c0457a5614a9e8dddb67f65eb65d" +checksum = "d530e1a18b1cb4c484e6e34556a0d948706958449fca0cab753d649f2bce3d1f" dependencies = [ "equivalent", - "hashbrown 0.14.0", + "hashbrown 0.14.3", "serde", ] @@ -1347,19 +1580,19 @@ dependencies = [ [[package]] name = "ipnet" -version = "2.8.0" +version = "2.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "28b29a3cd74f0f4598934efe3aeba42bae0eb4680554128851ebbecb02af14e6" +checksum = "8f518f335dce6725a761382244631d86cf0ccb2863413590b31338feb467f9c3" [[package]] name = "is-terminal" -version = "0.4.9" +version = "0.4.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cb0889898416213fab133e1d33a0e5858a48177452750691bde3666d0fdbaf8b" +checksum = "0bad00257d07be169d870ab665980b06cdb366d792ad690bf2e76876dc503455" dependencies = [ - "hermit-abi 0.3.2", + "hermit-abi 0.3.3", "rustix", - "windows-sys", + "windows-sys 0.52.0", ] [[package]] @@ -1371,15 +1604,6 @@ dependencies = [ "either", ] -[[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.0" @@ -1391,15 +1615,15 @@ dependencies = [ [[package]] name = "itoa" -version = "1.0.9" +version = "1.0.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "af150ab688ff2122fcef229be89cb50dd66af9e01a4ff320cc137eecc9bacc38" +checksum = "b1a46d1a171d865aa5f83f92695765caa047a9b4cbae2cbf37dbd613a793fd4c" [[package]] name = "js-sys" -version = "0.3.64" +version = "0.3.66" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c5f195fe497f702db0f318b07fdd68edb16955aed830df8363d837542f8f935a" +checksum = "cee9c64da59eae3b50095c18d3e74f8b73c0b86d2792824ff01bbce68ba229ca" dependencies = [ "wasm-bindgen", ] @@ -1417,9 +1641,24 @@ dependencies = [ [[package]] name = "json_comments" -version = "0.2.1" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9dbbfed4e59ba9750e15ba154fdfd9329cee16ff3df539c2666b70f58cc32105" + +[[package]] +name = "jwt" +version = "0.16.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "41ee439ee368ba4a77ac70d04f14015415af8600d6c894dc1f11bd79758c57d5" +checksum = "6204285f77fe7d9784db3fdc449ecce1a0114927a51d5a41c4c7a292011c015f" +dependencies = [ + "base64 0.13.1", + "crypto-common", + "digest 0.10.7", + "hmac", + "serde", + "serde_json", + "sha2 0.10.8", +] [[package]] name = "lazy_static" @@ -1429,9 +1668,20 @@ checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" [[package]] name = "libc" -version = "0.2.147" +version = "0.2.151" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "302d7ab3130588088d277783b1e2d2e10c9e9e4a16dd9050e6ec93fb3e7048f4" + +[[package]] +name = "libredox" +version = "0.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b4668fb0ea861c1df094127ac5f1da3409a82116a4ba74fca2e58ef927159bb3" +checksum = "85c833ca1e66078851dba29046874e38f08b2c883700aa29a03ddd3b23814ee8" +dependencies = [ + "bitflags 2.4.1", + "libc", + "redox_syscall 0.4.1", +] [[package]] name = "linked-hash-map" @@ -1441,15 +1691,15 @@ checksum = "0717cef1bc8b636c6e1c1bbdefc09e6322da8a9321966e8928ef80d20f7f770f" [[package]] name = "linux-raw-sys" -version = "0.4.5" +version = "0.4.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "57bcfdad1b858c2db7c38303a6d2ad4dfaf5eb53dfeb0910128b2c26d6158503" +checksum = "c4cd1a83af159aa67994778be9070f0ae1bd732942279cabb14f86f986a21456" [[package]] name = "lock_api" -version = "0.4.10" +version = "0.4.11" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c1cc9717a20b1bb222f333e6a92fd32f7d8a18ddc5a3191a11af45dcbf4dcd16" +checksum = "3c168f8615b12bc01f9c17e2eb0cc07dcae1940121185446edc3744920e8ef45" dependencies = [ "autocfg", "scopeguard", @@ -1457,9 +1707,18 @@ dependencies = [ [[package]] name = "log" -version = "0.4.19" +version = "0.4.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b5e6163cb8c49088c2c36f57875e58ccd8c87c7427f7fbd50ea6710b2f3f2e8f" + +[[package]] +name = "mach2" +version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b06a4cde4c0f271a446782e3eff8de789548ce57dbc8eca9292c27f4a42004b4" +checksum = "19b955cdeb2a02b9117f121ce63aa52d08ade45de53e48fe6a38b39c10f6f709" +dependencies = [ + "libc", +] [[package]] name = "matchers" @@ -1472,23 +1731,73 @@ dependencies = [ [[package]] name = "matchit" -version = "0.7.2" +version = "0.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ed1202b2a6f884ae56f04cff409ab315c5ce26b5e58d7412e484f01fd52f52ef" +checksum = "0e7465ac9959cc2b1404e8e2367b43684a6d13790fe23056cc8c6c5a6b7bcb94" [[package]] name = "memchr" -version = "2.5.0" +version = "2.7.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2dffe52ecf27772e601905b7522cb4ef790d2cc203488bbd0e2fe85fcb74566d" +checksum = "523dc4f511e55ab87b694dc30d0f820d60906ef06413f93d4d7a1385599cc149" [[package]] -name = "memoffset" -version = "0.9.0" +name = "metrics" +version = "0.21.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5a634b1c61a95585bd15607c6ab0c4e5b226e695ff2800ba0cdccddf208c406c" +checksum = "fde3af1a009ed76a778cb84fdef9e7dbbdf5775ae3e4cc1f434a6a307f6f76c5" dependencies = [ - "autocfg", + "ahash 0.8.7", + "metrics-macros", + "portable-atomic", +] + +[[package]] +name = "metrics-exporter-prometheus" +version = "0.12.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d4fa7ce7c4862db464a37b0b31d89bca874562f034bd7993895572783d02950" +dependencies = [ + "base64 0.21.5", + "hyper 0.14.28", + "indexmap 1.9.3", + "ipnet", + "metrics", + "metrics-util", + "quanta", + "thiserror", + "tokio", + "tracing", +] + +[[package]] +name = "metrics-macros" +version = "0.7.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38b4faf00617defe497754acde3024865bc143d44a86799b24e191ecff91354f" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.46", +] + +[[package]] +name = "metrics-util" +version = "0.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "111cb375987443c3de8d503580b536f77dc8416d32db62d9456db5d93bd7ac47" +dependencies = [ + "aho-corasick 0.7.20", + "crossbeam-epoch", + "crossbeam-utils", + "hashbrown 0.13.2", + "indexmap 1.9.3", + "metrics", + "num_cpus", + "ordered-float", + "quanta", + "radix_trie", + "sketches-ddsketch", ] [[package]] @@ -1497,6 +1806,16 @@ version = "0.3.17" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a" +[[package]] +name = "mime_guess" +version = "2.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4192263c238a5f0d0c6bfd21f336a313a4ce1c450542449ca191bb657b4642ef" +dependencies = [ + "mime", + "unicase", +] + [[package]] name = "minimal-lexical" version = "0.2.1" @@ -1514,13 +1833,13 @@ dependencies = [ [[package]] name = "mio" -version = "0.8.8" +version = "0.8.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "927a765cd3fc26206e66b296465fa9d3e5ab003e651c1b3c060e7956d96b19d2" +checksum = "8f3d0b296e374a4e6f3c7b0a1f5a51d748a0d34c85e7dc48fc3fa9a87657fe09" dependencies = [ "libc", "wasi 0.11.0+wasi-snapshot-preview1", - "windows-sys", + "windows-sys 0.48.0", ] [[package]] @@ -1535,7 +1854,7 @@ version = "0.7.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6a51313c5820b0b02bd422f4b44776fbf47961755c74ce64afc73bfad10226c3" dependencies = [ - "getrandom 0.2.10", + "getrandom 0.2.11", ] [[package]] @@ -1583,7 +1902,7 @@ dependencies = [ "once_cell", "serde", "serde_json", - "sha2 0.10.7", + "sha2 0.10.8", "smart-default", "tracing", ] @@ -1672,39 +1991,14 @@ dependencies = [ ] [[package]] -name = "near-o11y" -version = "0.17.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "af7d35397b02b131c188c72f3885e97daeccab134ec2fc8cc0073a94cf1cfe19" -dependencies = [ - "actix", - "atty", - "clap", - "near-crypto", - "near-primitives-core", - "once_cell", - "opentelemetry", - "opentelemetry-otlp", - "opentelemetry-semantic-conventions", - "prometheus", - "serde", - "strum", - "thiserror", - "tokio", - "tracing", - "tracing-appender", - "tracing-opentelemetry", - "tracing-subscriber", -] - -[[package]] -name = "near-offchain-lightclient" -version = "0.1.0" +name = "near-light-client" +version = "0.2.0" dependencies = [ "anyhow", "async-trait", - "axum", + "axum 0.7.3", "borsh", + "coerce", "config", "either", "flume", @@ -1717,19 +2011,42 @@ dependencies = [ "near-primitives", "near-primitives-core", "pretty_env_logger", - "prost 0.11.9", + "protobuf 3.2.0", "rand 0.8.5", - "reed-solomon-novelpoly", "reqwest", - "rustc-hex", "serde", "serde_json", - "sha256", "sled", "thiserror", "tokio", ] +[[package]] +name = "near-o11y" +version = "0.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af7d35397b02b131c188c72f3885e97daeccab134ec2fc8cc0073a94cf1cfe19" +dependencies = [ + "actix", + "atty", + "clap", + "near-crypto", + "near-primitives-core", + "once_cell", + "opentelemetry", + "opentelemetry-otlp", + "opentelemetry-semantic-conventions", + "prometheus", + "serde", + "strum", + "thiserror", + "tokio", + "tracing", + "tracing-appender", + "tracing-opentelemetry", + "tracing-subscriber", +] + [[package]] name = "near-primitives" version = "0.17.0" @@ -1763,7 +2080,7 @@ dependencies = [ "smart-default", "strum", "thiserror", - "time 0.3.25", + "time", "tracing", ] @@ -1774,7 +2091,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "775fec19ef51a341abdbf792a9dda5b4cb89f488f681b2fd689b9321d24db47b" dependencies = [ "arbitrary", - "base64 0.21.2", + "base64 0.21.5", "borsh", "bs58", "derive_more", @@ -1784,7 +2101,7 @@ dependencies = [ "serde", "serde_repr", "serde_with", - "sha2 0.10.7", + "sha2 0.10.8", "strum", "thiserror", ] @@ -1797,7 +2114,7 @@ checksum = "84c1eda300e2e78f4f945ae58117d49e806899f4a51ee2faa09eda5ebc2e6571" dependencies = [ "quote", "serde", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -1809,7 +2126,7 @@ dependencies = [ "fs2", "near-rpc-error-core", "serde", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -1832,6 +2149,15 @@ dependencies = [ "thiserror", ] +[[package]] +name = "nibble_vec" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "77a5d83df9f36fe23f0c3648c6bbb8b0298bb5f1939c8f2704431371f4b84d43" +dependencies = [ + "smallvec", +] + [[package]] name = "nom" version = "7.1.3" @@ -1888,9 +2214,9 @@ dependencies = [ [[package]] name = "num-traits" -version = "0.2.16" +version = "0.2.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f30b0abd723be7e2ffca1272140fac1a2f084c77ec3e123c192b66af1ee9e6c2" +checksum = "39e3200413f237f41ab11ad6d161bc7239c84dcb631773ccd7de3dfe4b5c267c" dependencies = [ "autocfg", ] @@ -1901,15 +2227,15 @@ version = "1.16.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" dependencies = [ - "hermit-abi 0.3.2", + "hermit-abi 0.3.3", "libc", ] [[package]] name = "once_cell" -version = "1.18.0" +version = "1.19.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dd8b5dd2ae5ed71462c540258bedcb51965123ad7e7ccf4b9a8cafaa4a63576d" +checksum = "3fdb12b2476b595f9358c5161aa467c2438859caa136dec86c26fdd2efe17b92" [[package]] name = "opaque-debug" @@ -1919,11 +2245,11 @@ checksum = "624a8340c38c1b80fd549087862da4ba43e08858af025b236e509b6649fc13d5" [[package]] name = "openssl" -version = "0.10.56" +version = "0.10.62" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "729b745ad4a5575dd06a3e1af1414bd330ee561c01b3899eb584baeaa8def17e" +checksum = "8cde4d2d9200ad5909f8dac647e29482e07c3a35de8a13fce7c9c7747ad9f671" dependencies = [ - "bitflags 1.3.2", + "bitflags 2.4.1", "cfg-if", "foreign-types", "libc", @@ -1940,7 +2266,7 @@ checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -1951,9 +2277,9 @@ checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" [[package]] name = "openssl-sys" -version = "0.9.91" +version = "0.9.98" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "866b5f16f90776b9bb8dc1e1802ac6f0513de3a7a7465867bfbc563dc737faac" +checksum = "c1665caf8ab2dc9aef43d1c0023bd904633a6a05cb30b0ad59bec2ae986e57a7" dependencies = [ "cc", "libc", @@ -1991,9 +2317,9 @@ dependencies = [ "async-trait", "futures", "futures-util", - "http", + "http 0.2.11", "opentelemetry", - "prost 0.9.0", + "prost", "thiserror", "tokio", "tonic", @@ -2009,6 +2335,15 @@ dependencies = [ "opentelemetry", ] +[[package]] +name = "ordered-float" +version = "3.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f1e1c390732d15f1d48471625cd92d154e66db2c56645e29a9cd26f4699f72dc" +dependencies = [ + "num-traits", +] + [[package]] name = "ordered-multimap" version = "0.4.3" @@ -2043,7 +2378,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3742b2c103b9f06bc9fff0a37ff4912935851bee6d36f3c02bcc755bcfec228f" dependencies = [ "lock_api", - "parking_lot_core 0.9.8", + "parking_lot_core 0.9.9", ] [[package]] @@ -2062,15 +2397,15 @@ dependencies = [ [[package]] name = "parking_lot_core" -version = "0.9.8" +version = "0.9.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "93f00c865fe7cabf650081affecd3871070f26767e7b2070a3ffae14c654b447" +checksum = "4c42a9226546d68acdd9c0a280d17ce19bfe27a46bf68784e4066115788d008e" dependencies = [ "cfg-if", "libc", - "redox_syscall 0.3.5", + "redox_syscall 0.4.1", "smallvec", - "windows-targets", + "windows-targets 0.48.5", ] [[package]] @@ -2081,25 +2416,26 @@ checksum = "8835116a5c179084a830efb3adc117ab007512b535bc1a21c991d3b32a6b44dd" [[package]] name = "percent-encoding" -version = "2.3.0" +version = "2.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9b2a4787296e9989611394c33f193f676704af1686e70b8f8033ab5ba9a35a94" +checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e" [[package]] name = "pest" -version = "2.7.2" +version = "2.7.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1acb4a4365a13f749a93f1a094a7805e5cfa0955373a9de860d962eaa3a5fe5a" +checksum = "ae9cee2a55a544be8b89dc6848072af97a20f2422603c10865be2a42b580fff5" dependencies = [ + "memchr", "thiserror", "ucd-trie", ] [[package]] name = "pest_derive" -version = "2.7.2" +version = "2.7.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "666d00490d4ac815001da55838c500eafb0320019bbaa44444137c48b443a853" +checksum = "81d78524685f5ef2a3b3bd1cafbc9fcabb036253d9b1463e726a91cd16e2dfc2" dependencies = [ "pest", "pest_generator", @@ -2107,36 +2443,36 @@ dependencies = [ [[package]] name = "pest_generator" -version = "2.7.2" +version = "2.7.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "68ca01446f50dbda87c1786af8770d535423fa8a53aec03b8f4e3d7eb10e0929" +checksum = "68bd1206e71118b5356dae5ddc61c8b11e28b09ef6a31acbd15ea48a28e0c227" dependencies = [ "pest", "pest_meta", "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "pest_meta" -version = "2.7.2" +version = "2.7.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "56af0a30af74d0445c0bf6d9d051c979b516a1a5af790d251daee76005420a48" +checksum = "7c747191d4ad9e4a4ab9c8798f1e82a39affe7ef9648390b7e5548d18e099de6" dependencies = [ "once_cell", "pest", - "sha2 0.10.7", + "sha2 0.10.8", ] [[package]] name = "petgraph" -version = "0.6.3" +version = "0.6.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4dd7d28ee937e54fe3080c91faa1c3a46c06de6252988a7f4592ba2310ef22a4" +checksum = "e1d3afd2628e69da2be385eb6f2fd57c8ac7977ceeff6dc166ff1657b0e386a9" dependencies = [ "fixedbitset", - "indexmap 1.9.3", + "indexmap 2.1.0", ] [[package]] @@ -2156,14 +2492,14 @@ checksum = "4359fd9c9171ec6e8c62926d6faaf553a8dc3f64e1507e76da7911b4f6a04405" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "pin-project-lite" -version = "0.2.11" +version = "0.2.13" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2c516611246607d0c04186886dbb3a754368ef82c79e9827a802c6d836dd111c" +checksum = "8afb450f006bf6385ca15ef45d71d2288452bc3683ce2e2cacc0d18e4be60b58" [[package]] name = "pin-utils" @@ -2173,9 +2509,21 @@ checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" [[package]] name = "pkg-config" -version = "0.3.27" +version = "0.3.28" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "69d3587f8a9e599cc7ec2c00e331f71c4e69a5f9a4b8a6efd5b07466b9736f9a" + +[[package]] +name = "portable-atomic" +version = "1.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "26072860ba924cbfa98ea39c8c19b4dd6a4a25423dbdf219c1eca91aa0cf6964" +checksum = "7170ef9988bc169ba16dd36a7fa041e5c4cbeb6a35b76d4c03daded371eae7c0" + +[[package]] +name = "powerfmt" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391" [[package]] name = "ppv-lite86" @@ -2185,9 +2533,9 @@ checksum = "5b40af805b3121feab8a3c29f04d8ad262fa8e0561883e7653e024ae4479e6de" [[package]] name = "pretty_env_logger" -version = "0.4.0" +version = "0.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "926d36b9553851b8b0005f1275891b392ee4d2d833852c417ed025477350fb9d" +checksum = "865724d4dbe39d9f3dd3b52b88d859d66bcb2d6a0acfd5ea68a65fb66d4bdc1c" dependencies = [ "env_logger", "log", @@ -2212,11 +2560,35 @@ dependencies = [ "toml", ] +[[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", + "syn 1.0.109", + "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.66" +version = "1.0.74" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "18fb31db3f9bddb2ea821cde30a9f70117e3f119938b5ee630b7403aa6e2ead9" +checksum = "2de98502f212cfcea8d0bb305bd0f49d7ebdd75b64ba0a68f937d888f4e0d6db" dependencies = [ "unicode-ident", ] @@ -2232,7 +2604,7 @@ dependencies = [ "lazy_static", "memchr", "parking_lot 0.12.1", - "protobuf", + "protobuf 2.28.0", "thiserror", ] @@ -2243,17 +2615,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "444879275cb4fd84958b1a1d5420d15e6fcf7c235fe47f053c9c2a80aceb6001" dependencies = [ "bytes", - "prost-derive 0.9.0", -] - -[[package]] -name = "prost" -version = "0.11.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0b82eaa1d779e9a4bc1c3217db8ffbeabaae1dca241bf70183242128d48681cd" -dependencies = [ - "bytes", - "prost-derive 0.11.9", + "prost-derive", ] [[package]] @@ -2269,7 +2631,7 @@ dependencies = [ "log", "multimap", "petgraph", - "prost 0.9.0", + "prost", "prost-types", "regex", "tempfile", @@ -2289,19 +2651,6 @@ dependencies = [ "syn 1.0.109", ] -[[package]] -name = "prost-derive" -version = "0.11.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e5d2d8d10f3c6ded6da8b05b5fb3b8a5082514344d56c9f871412d29b4e075b4" -dependencies = [ - "anyhow", - "itertools 0.10.5", - "proc-macro2", - "quote", - "syn 1.0.109", -] - [[package]] name = "prost-types" version = "0.9.0" @@ -2309,7 +2658,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "534b7a0e836e3c482d2693070f982e39e7611da9695d4d1f5a4b186b51faef0a" dependencies = [ "bytes", - "prost 0.9.0", + "prost", ] [[package]] @@ -2319,20 +2668,60 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "106dd99e98437432fed6519dedecfade6a06a73bb7b2a1e019fdd2bee5778d94" [[package]] -name = "quick-error" -version = "1.2.3" +name = "protobuf" +version = "3.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b55bad9126f378a853655831eb7363b7b01b81d19f8cb1218861086ca4a1a61e" +dependencies = [ + "once_cell", + "protobuf-support", + "thiserror", +] + +[[package]] +name = "protobuf-support" +version = "3.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a5d4d7b8601c814cfb36bcebb79f0e61e45e1e93640cf778837833bbed05c372" +dependencies = [ + "thiserror", +] + +[[package]] +name = "quanta" +version = "0.11.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a1d01941d82fa2ab50be1e79e6714289dd7cde78eba4c074bc5a4374f650dfe0" +checksum = "a17e662a7a8291a865152364c20c7abc5e60486ab2001e8ec10b24862de0b9ab" +dependencies = [ + "crossbeam-utils", + "libc", + "mach2", + "once_cell", + "raw-cpuid", + "wasi 0.11.0+wasi-snapshot-preview1", + "web-sys", + "winapi", +] [[package]] name = "quote" -version = "1.0.32" +version = "1.0.35" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "50f3b39ccfb720540debaa0164757101c08ecb8d326b15358ce76a62c7e85965" +checksum = "291ec9ab5efd934aaf503a6466c5d5251535d108ee747472c3977cc5acc868ef" dependencies = [ "proc-macro2", ] +[[package]] +name = "radix_trie" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c069c179fcdc6a2fe24d8d18305cf085fdbd4f922c041943e203685d6a1c58fd" +dependencies = [ + "endian-type", + "nibble_vec", +] + [[package]] name = "rand" version = "0.7.3" @@ -2392,7 +2781,7 @@ version = "0.6.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" dependencies = [ - "getrandom 0.2.10", + "getrandom 0.2.11", ] [[package]] @@ -2405,7 +2794,16 @@ dependencies = [ ] [[package]] -name = "redox_syscall" +name = "raw-cpuid" +version = "10.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c297679cb867470fa8c9f67dbba74a78d78e3e98d7cf2b08d6d71540f797332" +dependencies = [ + "bitflags 1.3.2", +] + +[[package]] +name = "redox_syscall" version = "0.2.16" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a" @@ -2415,45 +2813,43 @@ dependencies = [ [[package]] name = "redox_syscall" -version = "0.3.5" +version = "0.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "567664f262709473930a4bf9e51bf2ebf3348f2e748ccc50dea20646858f8f29" +checksum = "4722d768eff46b75989dd134e5c353f0d6296e5aaa3132e776cbdb56be7731aa" dependencies = [ "bitflags 1.3.2", ] [[package]] -name = "reed-solomon-erasure" -version = "4.0.2" +name = "redox_users" +version = "0.4.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a415a013dd7c5d4221382329a5a3482566da675737494935cbbbcdec04662f9d" +checksum = "a18479200779601e498ada4e8c1e1f50e3ee19deb0259c25825a98b5603b2cb4" dependencies = [ - "smallvec", + "getrandom 0.2.11", + "libredox", + "thiserror", ] [[package]] -name = "reed-solomon-novelpoly" -version = "1.0.2" +name = "reed-solomon-erasure" +version = "4.0.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "58130877ca403ab42c864fbac74bb319a0746c07a634a92a5cfc7f54af272582" +checksum = "a415a013dd7c5d4221382329a5a3482566da675737494935cbbbcdec04662f9d" dependencies = [ - "derive_more", - "fs-err", - "itertools 0.11.0", - "static_init", - "thiserror", + "smallvec", ] [[package]] name = "regex" -version = "1.9.3" +version = "1.10.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "81bc1d4caf89fac26a70747fe603c130093b53c773888797a6329091246d651a" +checksum = "380b951a9c5e80ddfd6136919eef32310721aa4aacd4889a8d39124b026ab343" dependencies = [ - "aho-corasick", + "aho-corasick 1.1.2", "memchr", - "regex-automata 0.3.6", - "regex-syntax 0.7.4", + "regex-automata 0.4.3", + "regex-syntax 0.8.2", ] [[package]] @@ -2467,13 +2863,13 @@ dependencies = [ [[package]] name = "regex-automata" -version = "0.3.6" +version = "0.4.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fed1ceff11a1dddaee50c9dc8e4938bd106e9d89ae372f192311e7da498e3b69" +checksum = "5f804c7828047e88b2d32e2d7fe5a105da8ee3264f01902f796c8e067dc2483f" dependencies = [ - "aho-corasick", + "aho-corasick 1.1.2", "memchr", - "regex-syntax 0.7.4", + "regex-syntax 0.8.2", ] [[package]] @@ -2484,26 +2880,26 @@ checksum = "f162c6dd7b008981e4d40210aca20b4bd0f9b60ca9271061b07f78537722f2e1" [[package]] name = "regex-syntax" -version = "0.7.4" +version = "0.8.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e5ea92a5b6195c6ef2a0295ea818b312502c6fc94dde986c5553242e18fd4ce2" +checksum = "c08c74e62047bb2de4ff487b251e4a92e24f48745648451635cec7d591162d9f" [[package]] name = "reqwest" -version = "0.11.18" +version = "0.11.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cde824a14b7c14f85caff81225f411faacc04a2013f41670f41443742b1c1c55" +checksum = "37b1ae8d9ac08420c66222fb9096fc5de435c3c48542bc5336c51892cffafb41" dependencies = [ "async-compression", - "base64 0.21.2", + "base64 0.21.5", "bytes", "encoding_rs", "futures-core", "futures-util", - "h2", - "http", - "http-body", - "hyper", + "h2 0.3.22", + "http 0.2.11", + "http-body 0.4.6", + "hyper 0.14.28", "hyper-tls", "ipnet", "js-sys", @@ -2516,9 +2912,10 @@ dependencies = [ "serde", "serde_json", "serde_urlencoded", + "system-configuration", "tokio", "tokio-native-tls", - "tokio-util 0.7.8", + "tokio-util 0.7.10", "tower-service", "url", "wasm-bindgen", @@ -2538,6 +2935,41 @@ dependencies = [ "serde", ] +[[package]] +name = "rust-embed" +version = "6.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a36224c3276f8c4ebc8c20f158eca7ca4359c8db89991c4925132aaaf6702661" +dependencies = [ + "rust-embed-impl", + "rust-embed-utils", + "walkdir", +] + +[[package]] +name = "rust-embed-impl" +version = "6.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "49b94b81e5b2c284684141a2fb9e2a31be90638caf040bf9afbc5a0416afe1ac" +dependencies = [ + "proc-macro2", + "quote", + "rust-embed-utils", + "shellexpand", + "syn 2.0.46", + "walkdir", +] + +[[package]] +name = "rust-embed-utils" +version = "7.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d38ff6bf570dc3bb7100fce9f7b60c33fa71d80e88da3f2580df4ff2bdded74" +dependencies = [ + "sha2 0.10.8", + "walkdir", +] + [[package]] name = "rust-ini" version = "0.18.0" @@ -2548,12 +2980,6 @@ dependencies = [ "ordered-multimap", ] -[[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.4.0" @@ -2565,15 +2991,15 @@ dependencies = [ [[package]] name = "rustix" -version = "0.38.7" +version = "0.38.28" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "172891ebdceb05aa0005f533a6cbfca599ddd7d966f6f5d4d9b2e70478e70399" +checksum = "72e572a5e8ca657d7366229cdde4bd14c4eb5499a9573d4d366fe1b599daa316" dependencies = [ - "bitflags 2.3.3", + "bitflags 2.4.1", "errno", "libc", "linux-raw-sys", - "windows-sys", + "windows-sys 0.52.0", ] [[package]] @@ -2584,17 +3010,26 @@ checksum = "7ffc183a10b4478d04cbbbfc96d0873219d962dd5accaff2ffbd4ceb7df837f4" [[package]] name = "ryu" -version = "1.0.15" +version = "1.0.16" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1ad4cc8da4ef723ed60bced201181d83791ad433213d8c24efffda1eec85d741" +checksum = "f98d2aa92eebf49b69786be48e4477826b256916e84a57ff2a4f21923b48eb4c" + +[[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 = "schannel" -version = "0.1.22" +version = "0.1.23" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0c3733bf4cf7ea0880754e19cb5a462007c4a8c1914bff372ccc95b464f1df88" +checksum = "fbc91545643bcf3a0bbb6569265615222618bdf33ce4ffbbd13c4bbd4c093534" dependencies = [ - "windows-sys", + "windows-sys 0.52.0", ] [[package]] @@ -2647,35 +3082,35 @@ dependencies = [ [[package]] name = "semver" -version = "1.0.18" +version = "1.0.21" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b0293b4b29daaf487284529cc2f5675b8e57c61f70167ba415a463651fd6a918" +checksum = "b97ed7a9823b74f99c7742f5336af7be5ecd3eeafcb1507d1fa93347b1d589b0" [[package]] name = "serde" -version = "1.0.183" +version = "1.0.194" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "32ac8da02677876d532745a130fc9d8e6edfa81a269b107c5b00829b91d8eb3c" +checksum = "0b114498256798c94a0689e1a15fec6005dee8ac1f41de56404b67afc2a4b773" dependencies = [ "serde_derive", ] [[package]] name = "serde_derive" -version = "1.0.183" +version = "1.0.194" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "aafe972d60b0b9bee71a91b92fee2d4fb3c9d7e8f6b179aa99f27203d99a4816" +checksum = "a3385e45322e8f9931410f01b3031ec534c3947d0e94c18049af4d9f9907d4e0" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "serde_json" -version = "1.0.104" +version = "1.0.110" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "076066c5f1078eac5b722a31827a8832fe108bed65dfa75e233c89f8206e976c" +checksum = "6fbd975230bada99c8bb618e0c365c2eefa219158d5c6c29610fd09ff1833257" dependencies = [ "itoa", "ryu", @@ -2684,9 +3119,9 @@ dependencies = [ [[package]] name = "serde_path_to_error" -version = "0.1.14" +version = "0.1.15" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4beec8bce849d58d06238cb50db2e1c417cfeafa4c63f692b15c82b7c80f8335" +checksum = "ebd154a240de39fdebcf5775d2675c204d7c13cf39a4c697be6493c8e734337c" dependencies = [ "itoa", "serde", @@ -2694,13 +3129,13 @@ dependencies = [ [[package]] name = "serde_repr" -version = "0.1.16" +version = "0.1.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8725e1dfadb3a50f7e5ce0b1a540466f6ed3fe7a0fca2ac2b8b831d31316bd00" +checksum = "0b2e6b945e9d3df726b65d6ee24060aff8e3533d431f677a9695db04eff9dfdb" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -2717,40 +3152,40 @@ dependencies = [ [[package]] name = "serde_with" -version = "3.2.0" +version = "3.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1402f54f9a3b9e2efe71c1cea24e648acce55887983553eeb858cf3115acfd49" +checksum = "64cd236ccc1b7a29e7e2739f27c0b2dd199804abc4290e32f59f3b68d6405c23" dependencies = [ - "base64 0.21.2", + "base64 0.21.5", "chrono", "hex", "indexmap 1.9.3", - "indexmap 2.0.0", + "indexmap 2.1.0", "serde", "serde_json", "serde_with_macros", - "time 0.3.25", + "time", ] [[package]] name = "serde_with_macros" -version = "3.2.0" +version = "3.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9197f1ad0e3c173a0222d3c4404fb04c3afe87e962bcb327af73e8301fa203c7" +checksum = "93634eb5f75a2323b16de4748022ac4297f9e76b6dced2be287a099f41b5e788" dependencies = [ "darling", "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "serde_yaml" -version = "0.9.25" +version = "0.9.30" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1a49e178e4452f45cb61d0cd8cebc1b0fafd3e41929e996cef79aa3aca91f574" +checksum = "b1bf28c79a99f70ee1f1d83d10c875d2e70618417fda01ad1785e027579d9d38" dependencies = [ - "indexmap 2.0.0", + "indexmap 2.1.0", "itoa", "ryu", "serde", @@ -2772,9 +3207,9 @@ dependencies = [ [[package]] name = "sha2" -version = "0.10.7" +version = "0.10.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "479fb9d862239e610720565ca91403019f2f00410f1864c5aa7479b950a76ed8" +checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8" dependencies = [ "cfg-if", "cpufeatures", @@ -2782,25 +3217,21 @@ dependencies = [ ] [[package]] -name = "sha256" -version = "1.3.0" +name = "sharded-slab" +version = "0.1.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7f5ed5ebbe2d9fb5c5e67be64aa462053d707941e02ffb5e65b6200c00b6161c" +checksum = "f40ca3c46823713e0d4209592e8d6e826aa57e928f09752619fc696c499637f6" dependencies = [ - "async-trait", - "bytes", - "hex", - "sha2 0.10.7", - "tokio", + "lazy_static", ] [[package]] -name = "sharded-slab" -version = "0.1.4" +name = "shellexpand" +version = "2.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "900fba806f70c630b0a382d0d825e17a0f19fcd059a2ade1ff237bcddf446b31" +checksum = "7ccc8076840c4da029af4f87e4e8daeb0fca6b87bbb02e10cb60b791450e11e4" dependencies = [ - "lazy_static", + "dirs", ] [[package]] @@ -2818,11 +3249,23 @@ version = "1.6.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "74233d3b3b2f6d4b006dc19dee745e73e2a6bfb6f93607cd3b02bd5b00797d7c" +[[package]] +name = "siphasher" +version = "0.3.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "38b58827f4464d87d377d175e90bf58eb00fd8716ff0a62f80356b5e61555d0d" + +[[package]] +name = "sketches-ddsketch" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "68a406c1882ed7f29cd5e248c9848a80e7cb6ae0fea82346d2746f2f941c07e1" + [[package]] name = "slab" -version = "0.4.8" +version = "0.4.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6528351c9bc8ab22353f9d776db39a20288e8d6c37ef8cfe3317cf875eecfc2d" +checksum = "8f92a496fb766b417c996b9c5e57daf2f7ad3b0bebe1ccfca4856390e3d3bb67" dependencies = [ "autocfg", ] @@ -2845,9 +3288,9 @@ dependencies = [ [[package]] name = "smallvec" -version = "1.11.0" +version = "1.11.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "62bb4feee49fdd9f707ef802e22365a35de4b7b299de4763d44bfea899442ff9" +checksum = "4dccd0940a2dcdf68d092b8cbab7dc0ad8fa938bf95787e1b916b0e3d0e8e970" [[package]] name = "smart-default" @@ -2862,14 +3305,24 @@ dependencies = [ [[package]] name = "socket2" -version = "0.4.9" +version = "0.4.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "64a4a911eed85daf18834cfaa86a79b7d266ff93ff5ba14005426219480ed662" +checksum = "9f7916fc008ca5542385b89a3d3ce689953c143e9304a9bf8beec1de48994c0d" dependencies = [ "libc", "winapi", ] +[[package]] +name = "socket2" +version = "0.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7b5fac59a5cb5dd637972e5fca70daf0523c9067fcdc4842f053dae04a18f8e9" +dependencies = [ + "libc", + "windows-sys 0.48.0", +] + [[package]] name = "spin" version = "0.9.8" @@ -2885,34 +3338,6 @@ version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f" -[[package]] -name = "static_init" -version = "1.0.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8a2a1c578e98c1c16fc3b8ec1328f7659a500737d7a0c6d625e73e830ff9c1f6" -dependencies = [ - "bitflags 1.3.2", - "cfg_aliases", - "libc", - "parking_lot 0.11.2", - "parking_lot_core 0.8.6", - "static_init_macro", - "winapi", -] - -[[package]] -name = "static_init_macro" -version = "1.0.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "70a2595fc3aa78f2d0e45dd425b22282dd863273761cc77780914b2cf3003acf" -dependencies = [ - "cfg_aliases", - "memchr", - "proc-macro2", - "quote", - "syn 1.0.109", -] - [[package]] name = "strsim" version = "0.10.0" @@ -2960,9 +3385,9 @@ dependencies = [ [[package]] name = "syn" -version = "2.0.28" +version = "2.0.46" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "04361975b3f5e348b2189d8dc55bc942f278b2d482a6a0365de5bdd62d351567" +checksum = "89456b690ff72fddcecf231caedbe615c59480c93358a93dfae7fc29e3ebbf0e" dependencies = [ "proc-macro2", "quote", @@ -2975,46 +3400,67 @@ 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 = "tempfile" -version = "3.7.1" +version = "3.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dc02fddf48964c42031a0b3fe0428320ecf3a73c401040fc0096f97794310651" +checksum = "01ce4141aa927a6d1bd34a041795abd0db1cccba5d5f24b009f694bdf3a1f3fa" dependencies = [ "cfg-if", "fastrand", - "redox_syscall 0.3.5", + "redox_syscall 0.4.1", "rustix", - "windows-sys", + "windows-sys 0.52.0", ] [[package]] name = "termcolor" -version = "1.2.0" +version = "1.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "be55cf8942feac5c765c2c993422806843c9a9a45d4d5c407ad6dd2ea95eb9b6" +checksum = "ff1bc3d3f05aff0403e8ac0d92ced918ec05b666a43f83297ccef5bea8a3d449" dependencies = [ "winapi-util", ] [[package]] name = "thiserror" -version = "1.0.44" +version = "1.0.56" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "611040a08a0439f8248d1990b111c95baa9c704c805fa1f62104b39655fd7f90" +checksum = "d54378c645627613241d077a3a79db965db602882668f9136ac42af9ecb730ad" dependencies = [ "thiserror-impl", ] [[package]] name = "thiserror-impl" -version = "1.0.44" +version = "1.0.56" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "090198534930841fab3a5d1bb637cde49e339654e606195f8d9c76eeb081dc96" +checksum = "fa0faa943b50f3db30a20aa7e265dbc66076993efed8463e8de414e5d06d3471" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -3029,23 +3475,13 @@ dependencies = [ [[package]] name = "time" -version = "0.1.45" +version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1b797afad3f312d1c66a56d11d0316f916356d11bd158fbc6ca6389ff6bf805a" -dependencies = [ - "libc", - "wasi 0.10.0+wasi-snapshot-preview1", - "winapi", -] - -[[package]] -name = "time" -version = "0.3.25" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b0fdd63d58b18d663fbdf70e049f00a22c8e42be082203be7f26589213cd75ea" +checksum = "f657ba42c3f86e7680e53c8cd3af8abbe56b5491790b46e22e19c0d57463583e" dependencies = [ "deranged", "itoa", + "powerfmt", "serde", "time-core", "time-macros", @@ -3053,15 +3489,15 @@ dependencies = [ [[package]] name = "time-core" -version = "0.1.1" +version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7300fbefb4dadc1af235a9cef3737cea692a9d97e1b9cbcd4ebdae6f8868e6fb" +checksum = "ef927ca75afb808a4d64dd374f00a2adf8d0fcff8e7b184af886c3c87ec4a3f3" [[package]] name = "time-macros" -version = "0.2.11" +version = "0.2.16" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eb71511c991639bb078fd5bf97757e03914361c48100d52878b8e52b46fb92cd" +checksum = "26197e33420244aeb70c3e8c78376ca46571bc4e701e4791c2cd9f57dcb3a43f" dependencies = [ "time-core", ] @@ -3095,9 +3531,9 @@ dependencies = [ "parking_lot 0.12.1", "pin-project-lite", "signal-hook-registry", - "socket2", + "socket2 0.4.10", "tokio-macros", - "windows-sys", + "windows-sys 0.48.0", ] [[package]] @@ -3118,7 +3554,7 @@ checksum = "630bdcf245f78637c13ec01ffae6187cca34625e8c63150d424b59e55af2675e" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] @@ -3158,14 +3594,18 @@ dependencies = [ [[package]] name = "tokio-util" -version = "0.7.8" +version = "0.7.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "806fe8c2c87eccc8b3267cbae29ed3ab2d0bd37fca70ab622e46aaa9375ddb7d" +checksum = "5419f34732d9eb6ee4c3578b7989078579b7f039cbbb9ca2c4da015749371e15" dependencies = [ "bytes", "futures-core", + "futures-io", "futures-sink", + "futures-util", + "hashbrown 0.14.3", "pin-project-lite", + "slab", "tokio", "tracing", ] @@ -3191,15 +3631,15 @@ dependencies = [ "bytes", "futures-core", "futures-util", - "h2", - "http", - "http-body", - "hyper", + "h2 0.3.22", + "http 0.2.11", + "http-body 0.4.6", + "hyper 0.14.28", "hyper-timeout", "percent-encoding", "pin-project", - "prost 0.9.0", - "prost-derive 0.9.0", + "prost", + "prost-derive", "tokio", "tokio-stream", "tokio-util 0.6.10", @@ -3236,7 +3676,7 @@ dependencies = [ "rand 0.8.5", "slab", "tokio", - "tokio-util 0.7.8", + "tokio-util 0.7.10", "tower-layer", "tower-service", "tracing", @@ -3256,11 +3696,10 @@ checksum = "b6bc1c9ce2b5135ac7f93c72918fc37feb872bdc6a5533a8b85eb4b86bfdae52" [[package]] name = "tracing" -version = "0.1.37" +version = "0.1.40" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8ce8c33a8d48bd45d624a6e523445fd21ec13d3653cd51f681abf67418f54eb8" +checksum = "c3523ab5a71916ccf420eebdf5521fcef02141234bbc0b8a49f2fdc4544364ef" dependencies = [ - "cfg-if", "log", "pin-project-lite", "tracing-attributes", @@ -3269,31 +3708,32 @@ dependencies = [ [[package]] name = "tracing-appender" -version = "0.2.2" +version = "0.2.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "09d48f71a791638519505cefafe162606f706c25592e4bde4d97600c0195312e" +checksum = "3566e8ce28cc0a3fe42519fc80e6b4c943cc4c8cef275620eb8dac2d3d4e06cf" dependencies = [ "crossbeam-channel", - "time 0.3.25", + "thiserror", + "time", "tracing-subscriber", ] [[package]] name = "tracing-attributes" -version = "0.1.26" +version = "0.1.27" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f4f31f56159e98206da9efd823404b79b6ef3143b4a7ab76e67b1751b25a4ab" +checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", ] [[package]] name = "tracing-core" -version = "0.1.31" +version = "0.1.32" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0955b8137a1df6f1a2e9a37d8a6656291ff0297c1a97c24e0d8425fe2312f79a" +checksum = "c06d3da6113f116aaee68e4d601191614c9053067f9ab7f6edbcb161237daa54" dependencies = [ "once_cell", "valuable", @@ -3311,12 +3751,23 @@ dependencies = [ [[package]] name = "tracing-log" -version = "0.1.3" +version = "0.1.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "78ddad33d2d10b1ed7eb9d1f518a5674713876e97e5bb9b7345a7984fbb4f922" +checksum = "f751112709b4e791d8ce53e32c4ed2d353565a795ce84da2285393f41557bdf2" dependencies = [ - "lazy_static", "log", + "once_cell", + "tracing-core", +] + +[[package]] +name = "tracing-log" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ee855f1f400bd0e5c02d150ae5de3840039a3f54b025156404e34c23c03f47c3" +dependencies = [ + "log", + "once_cell", "tracing-core", ] @@ -3330,15 +3781,15 @@ dependencies = [ "opentelemetry", "tracing", "tracing-core", - "tracing-log", + "tracing-log 0.1.4", "tracing-subscriber", ] [[package]] name = "tracing-subscriber" -version = "0.3.17" +version = "0.3.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "30a651bc37f915e81f087d86e62a18eec5f79550c7faff886f7090b4ea757c77" +checksum = "ad0f048c97dbd9faa9b7df56362b8ebcaa52adb06b498c050d2f4e32f90a7a8b" dependencies = [ "matchers", "nu-ansi-term", @@ -3349,20 +3800,20 @@ dependencies = [ "thread_local", "tracing", "tracing-core", - "tracing-log", + "tracing-log 0.2.0", ] [[package]] name = "try-lock" -version = "0.2.4" +version = "0.2.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3528ecfd12c466c6f163363caf2d02a71161dd5e1cc6ae7b34207ea2d42d81ed" +checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b" [[package]] name = "typenum" -version = "1.16.0" +version = "1.17.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "497961ef93d974e23eb6f433eb5fe1b7930b659f06d12dec6fc44a8f554c0bba" +checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825" [[package]] name = "ucd-trie" @@ -3382,17 +3833,26 @@ dependencies = [ "static_assertions", ] +[[package]] +name = "unicase" +version = "2.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f7d2d4dafb69621809a81864c9c1b864479e1235c0dd4e199924b9742439ed89" +dependencies = [ + "version_check", +] + [[package]] name = "unicode-bidi" -version = "0.3.13" +version = "0.3.14" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "92888ba5573ff080736b3648696b70cafad7d250551175acbaa4e0385b3e1460" +checksum = "6f2528f27a9eb2b21e69c95319b30bd0efd85d09c379741b0f78ea1d86be2416" [[package]] name = "unicode-ident" -version = "1.0.11" +version = "1.0.12" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "301abaae475aa91687eb82514b328ab47a211a533026cb25fc3e519b86adfc3c" +checksum = "3354b9ac3fae1ff6755cb6db53683adb661634f67557942dea4facebec0fee4b" [[package]] name = "unicode-normalization" @@ -3411,15 +3871,15 @@ checksum = "1dd624098567895118886609431a7c3b8f516e41d30e0643f03d94592a147e36" [[package]] name = "unsafe-libyaml" -version = "0.2.9" +version = "0.2.10" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f28467d3e1d3c6586d8f25fa243f544f5800fec42d97032474e17222c2b75cfa" +checksum = "ab4c90930b95a82d00dc9e9ac071b4991924390d46cbd0dfe566148667605e4b" [[package]] name = "url" -version = "2.4.0" +version = "2.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "50bff7831e19200a85b17131d085c25d7811bc4e186efdaf54bbd132994a88cb" +checksum = "31e6302e3bb753d46e83516cae55ae196fc0c309407cf11ab35cc51a4c2a4633" dependencies = [ "form_urlencoded", "idna", @@ -3432,11 +3892,76 @@ version = "0.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "711b9620af191e0cdc7468a8d14e709c3dcdb115b36f838e601583af800a370a" +[[package]] +name = "utoipa" +version = "3.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d82b1bc5417102a73e8464c686eef947bdfb99fcdfc0a4f228e81afa9526470a" +dependencies = [ + "indexmap 2.1.0", + "serde", + "serde_json", + "utoipa-gen", +] + +[[package]] +name = "utoipa-gen" +version = "3.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05d96dcd6fc96f3df9b3280ef480770af1b7c5d14bc55192baa9b067976d920c" +dependencies = [ + "proc-macro-error", + "proc-macro2", + "quote", + "regex", + "syn 2.0.46", +] + +[[package]] +name = "utoipa-swagger-ui" +version = "3.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "84614caa239fb25b2bb373a52859ffd94605ceb256eeb1d63436325cf81e3653" +dependencies = [ + "axum 0.6.20", + "mime_guess", + "regex", + "rust-embed", + "serde", + "serde_json", + "utoipa", + "zip", +] + +[[package]] +name = "uuid" +version = "1.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5e395fcf16a7a3d8127ec99782007af141946b4795001f876d54fb0d55978560" +dependencies = [ + "getrandom 0.2.11", + "serde", +] + [[package]] name = "valuable" version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "830b7e5d4d90034032940e4ace0d9a9a057e7a45cd94e6c007832e39edb82f6d" +dependencies = [ + "valuable-derive", +] + +[[package]] +name = "valuable-derive" +version = "0.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d44690c645190cfce32f91a1582281654b2338c6073fa250b0949fd25c55b32" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] [[package]] name = "vcpkg" @@ -3450,6 +3975,16 @@ version = "0.9.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "49874b5167b65d7193b8aba1567f5c7d93d001cafc34600cee003eda787e483f" +[[package]] +name = "walkdir" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d71d857dc86794ca4c280d616f7da00d2dbfd8cd788846559a6813e6aa4b54ee" +dependencies = [ + "same-file", + "winapi-util", +] + [[package]] name = "want" version = "0.3.1" @@ -3465,12 +4000,6 @@ version = "0.9.0+wasi-snapshot-preview1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519" -[[package]] -name = "wasi" -version = "0.10.0+wasi-snapshot-preview1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1a143597ca7c7793eff794def352d41792a93c481eb1042423ff7ff72ba2c31f" - [[package]] name = "wasi" version = "0.11.0+wasi-snapshot-preview1" @@ -3479,9 +4008,9 @@ checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" [[package]] name = "wasm-bindgen" -version = "0.2.87" +version = "0.2.89" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7706a72ab36d8cb1f80ffbf0e071533974a60d0a308d01a5d0375bf60499a342" +checksum = "0ed0d4f68a3015cc185aff4db9506a015f4b96f95303897bfa23f846db54064e" dependencies = [ "cfg-if", "wasm-bindgen-macro", @@ -3489,24 +4018,24 @@ dependencies = [ [[package]] name = "wasm-bindgen-backend" -version = "0.2.87" +version = "0.2.89" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5ef2b6d3c510e9625e5fe6f509ab07d66a760f0885d858736483c32ed7809abd" +checksum = "1b56f625e64f3a1084ded111c4d5f477df9f8c92df113852fa5a374dbda78826" dependencies = [ "bumpalo", "log", "once_cell", "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", "wasm-bindgen-shared", ] [[package]] name = "wasm-bindgen-futures" -version = "0.4.37" +version = "0.4.39" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c02dbc21516f9f1f04f187958890d7e6026df8d16540b7ad9492bc34a67cea03" +checksum = "ac36a15a220124ac510204aec1c3e5db8a22ab06fd6706d881dc6149f8ed9a12" dependencies = [ "cfg-if", "js-sys", @@ -3516,9 +4045,9 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro" -version = "0.2.87" +version = "0.2.89" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dee495e55982a3bd48105a7b947fd2a9b4a8ae3010041b9e0faab3f9cd028f1d" +checksum = "0162dbf37223cd2afce98f3d0785506dcb8d266223983e4b5b525859e6e182b2" dependencies = [ "quote", "wasm-bindgen-macro-support", @@ -3526,28 +4055,28 @@ dependencies = [ [[package]] name = "wasm-bindgen-macro-support" -version = "0.2.87" +version = "0.2.89" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "54681b18a46765f095758388f2d0cf16eb8d4169b639ab575a8f5693af210c7b" +checksum = "f0eb82fcb7930ae6219a7ecfd55b217f5f0893484b7a13022ebb2b2bf20b5283" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", "wasm-bindgen-backend", "wasm-bindgen-shared", ] [[package]] name = "wasm-bindgen-shared" -version = "0.2.87" +version = "0.2.89" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ca6ad05a4870b2bf5fe995117d3728437bd27d7cd5f06f13c17443ef369775a1" +checksum = "7ab9b36309365056cd639da3134bf87fa8f3d86008abf99e612384a6eecd459f" [[package]] name = "web-sys" -version = "0.3.64" +version = "0.3.66" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9b85cbef8c220a6abc02aefd892dfc0fc23afb1c6a426316ec33253a3877249b" +checksum = "50c24a44ec86bb68fbecd1b3efed7e85ea5621b39b35ef2766b66cd984f8010f" dependencies = [ "js-sys", "wasm-bindgen", @@ -3555,13 +4084,14 @@ dependencies = [ [[package]] name = "which" -version = "4.4.0" +version = "4.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2441c784c52b289a054b7201fc93253e288f094e2f4be9058343127c4226a269" +checksum = "87ba24419a2078cd2b0f2ede2691b6c66d8e47836da3b6db8265ebad47afbfc7" dependencies = [ "either", - "libc", + "home", "once_cell", + "rustix", ] [[package]] @@ -3582,9 +4112,9 @@ checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" [[package]] name = "winapi-util" -version = "0.1.5" +version = "0.1.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "70ec6ce85bb158151cae5e5c87f95a8e97d2c0c4b001223f33a334e3ce5de178" +checksum = "f29e6f9198ba0d26b4c9f07dbe6f9ed633e1f3d5b8b414090084349e46a52596" dependencies = [ "winapi", ] @@ -3596,12 +4126,12 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" [[package]] -name = "windows" -version = "0.48.0" +name = "windows-core" +version = "0.52.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e686886bc078bc1b0b600cac0147aadb815089b6e4da64016cbd754b6342700f" +checksum = "33ab640c8d7e35bf8ba19b884ba838ceb4fba93a4e8c65a9059d08afcfc683d9" dependencies = [ - "windows-targets", + "windows-targets 0.52.0", ] [[package]] @@ -3610,73 +4140,140 @@ version = "0.48.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9" dependencies = [ - "windows-targets", + "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.0", ] [[package]] name = "windows-targets" -version = "0.48.1" +version = "0.48.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "05d4b17490f70499f20b9e791dcf6a299785ce8af4d709018206dc5b4953e95f" +checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c" dependencies = [ - "windows_aarch64_gnullvm", - "windows_aarch64_msvc", - "windows_i686_gnu", - "windows_i686_msvc", - "windows_x86_64_gnu", - "windows_x86_64_gnullvm", - "windows_x86_64_msvc", + "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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8a18201040b24831fbb9e4eb208f8892e1f50a37feb53cc7ff887feb8f50e7cd" +dependencies = [ + "windows_aarch64_gnullvm 0.52.0", + "windows_aarch64_msvc 0.52.0", + "windows_i686_gnu 0.52.0", + "windows_i686_msvc 0.52.0", + "windows_x86_64_gnu 0.52.0", + "windows_x86_64_gnullvm 0.52.0", + "windows_x86_64_msvc 0.52.0", ] [[package]] name = "windows_aarch64_gnullvm" -version = "0.48.0" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8" + +[[package]] +name = "windows_aarch64_gnullvm" +version = "0.52.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "91ae572e1b79dba883e0d315474df7305d12f569b400fcf90581b06062f7e1bc" +checksum = "cb7764e35d4db8a7921e09562a0304bf2f93e0a51bfccee0bd0bb0b666b015ea" [[package]] name = "windows_aarch64_msvc" -version = "0.48.0" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc" + +[[package]] +name = "windows_aarch64_msvc" +version = "0.52.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b2ef27e0d7bdfcfc7b868b317c1d32c641a6fe4629c171b8928c7b08d98d7cf3" +checksum = "bbaa0368d4f1d2aaefc55b6fcfee13f41544ddf36801e793edbbfd7d7df075ef" [[package]] name = "windows_i686_gnu" -version = "0.48.0" +version = "0.48.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e" + +[[package]] +name = "windows_i686_gnu" +version = "0.52.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "622a1962a7db830d6fd0a69683c80a18fda201879f0f447f065a3b7467daa241" +checksum = "a28637cb1fa3560a16915793afb20081aba2c92ee8af57b4d5f28e4b3e7df313" [[package]] name = "windows_i686_msvc" -version = "0.48.0" +version = "0.48.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4542c6e364ce21bf45d69fdd2a8e455fa38d316158cfd43b3ac1c5b1b19f8e00" +checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406" + +[[package]] +name = "windows_i686_msvc" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ffe5e8e31046ce6230cc7215707b816e339ff4d4d67c65dffa206fd0f7aa7b9a" [[package]] name = "windows_x86_64_gnu" -version = "0.48.0" +version = "0.48.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ca2b8a661f7628cbd23440e50b05d705db3686f894fc9580820623656af974b1" +checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e" + +[[package]] +name = "windows_x86_64_gnu" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3d6fa32db2bc4a2f5abeacf2b69f7992cd09dca97498da74a151a3132c26befd" [[package]] name = "windows_x86_64_gnullvm" -version = "0.48.0" +version = "0.48.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7896dbc1f41e08872e9d5e8f8baa8fdd2677f29468c4e156210174edc7f7b953" +checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc" + +[[package]] +name = "windows_x86_64_gnullvm" +version = "0.52.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1a657e1e9d3f514745a572a6846d3c7aa7dbe1658c056ed9c3344c4109a6949e" [[package]] name = "windows_x86_64_msvc" -version = "0.48.0" +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.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1a515f5799fe4961cb532f983ce2b23082366b898e52ffbce459c86f67c8378a" +checksum = "dff9641d1cd4be8d1a070daf9e3773c5f67e78b4d9d42263020c057706765c04" [[package]] name = "winreg" -version = "0.10.1" +version = "0.50.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "80d0f4e272c85def139476380b12f9ac60926689dd2e01d4923222f40580869d" +checksum = "524e57b2c537c0f9b1e69f1965311ec12182b4122e45035b1508cd24d2adadb1" dependencies = [ - "winapi", + "cfg-if", + "windows-sys 0.48.0", ] [[package]] @@ -3688,6 +4285,26 @@ dependencies = [ "linked-hash-map", ] +[[package]] +name = "zerocopy" +version = "0.7.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "74d4d3961e53fa4c9a25a8637fc2bfaf2595b3d3ae34875568a5cf64787716be" +dependencies = [ + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.7.32" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ce1b18ccd8e73a9321186f97e46f9f04b778851177567b1975109d26a08d2a6" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.46", +] + [[package]] name = "zeroize" version = "1.3.0" @@ -3705,5 +4322,17 @@ checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69" dependencies = [ "proc-macro2", "quote", - "syn 2.0.28", + "syn 2.0.46", +] + +[[package]] +name = "zip" +version = "0.6.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "760394e246e4c28189f19d488c058bf16f564016aefac5d32bb1f3b51d5e9261" +dependencies = [ + "byteorder", + "crc32fast", + "crossbeam-utils", + "flate2", ] diff --git a/Cargo.toml b/Cargo.toml index 4d98a1f..8685447 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,9 +1,7 @@ [package] edition = "2021" -name = "near-offchain-lightclient" -version = "0.1.0" - -# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html +name = "near-light-client" +version = "0.2.0" [dependencies] anyhow = "1.0" @@ -12,13 +10,12 @@ config = "0.13" either = { version = "1.9", features = [ "serde" ] } itertools = "0.12" log = "0.4" -pretty_env_logger = "0.4" -reed-solomon-novelpoly = "1.0.1" -sha256 = "*" -sled = "0.34" +pretty_env_logger = "0.5" +sled = "0.34" # TODO: maybe heavy thiserror = "1.0" # Async +coerce = { version = "0.8", features = ["full"] } axum = "*" flume = "0.10" futures = "0.3.28" @@ -26,18 +23,18 @@ reqwest = { version = "0.11.17", features = [ "gzip", "brotli", "deflate", "json tokio = { version = "1", features = [ "full" ] } # Codec -borsh = { version = "0.10.2", features = [ "rc" ] } +borsh = { version = "0.10", features = [ "rc" ] } +protobuf = "=3.2.0" hex = { version = "0.4", features = [ "serde" ] } -prost = "0.11" -rustc-hex = "2.1.0" serde = { version = "1.0", features = [ "derive" ] } serde_json = "1.0" +# TODO: upgrade # Near specific -near-crypto = "0.17.0" -near-jsonrpc-client = "0.6.0" -near-primitives = "0.17.0" -near-primitives-core = "0.17.0" +near-crypto = "0.17" +near-jsonrpc-client = "0.6" +near-primitives = "0.17" +near-primitives-core = "0.17" [dev-dependencies] rand = "*" diff --git a/default.toml b/default.toml index dd47d44..11609af 100644 --- a/default.toml +++ b/default.toml @@ -1,4 +1,4 @@ -debug = false +catchup = false network = "Testnet" starting_head = "4zwZQzjQDpimeLK3tX39nzok6UjDU9edS57EFhkAa4Sk" -state_path = "db.sled" +state_path = "state.db" diff --git a/fixtures/batch.json b/fixtures/batch.json index d8a3bfa..022afa8 100644 --- a/fixtures/batch.json +++ b/fixtures/batch.json @@ -1,5 +1,5 @@ { - "client_block_merkle_root": "WWrLWbWHwSmjtTn5oBZPYgRCuCYn6fkYVa4yhPWNK4L", + "head_block_root": "WWrLWbWHwSmjtTn5oBZPYgRCuCYn6fkYVa4yhPWNK4L", "batch": [ { "outcome_proof_block_hash": "3r5dS7bXYG2w35qLXx78nHjPPs8psdYZUdStUYkvHr8N", @@ -1312,4 +1312,4 @@ "direction": "Right" } ] -} +} \ No newline at end of file diff --git a/flake.lock b/flake.lock index a6622fe..260077f 100644 --- a/flake.lock +++ b/flake.lock @@ -5,11 +5,11 @@ "systems": "systems" }, "locked": { - "lastModified": 1694529238, - "narHash": "sha256-zsNZZGTGnMOf9YpHKJqMSsa0dXbfmxeoJ7xHlrt+xmY=", + "lastModified": 1701680307, + "narHash": "sha256-kAuep2h5ajznlPMD9rnQyffWG8EM/C73lejGofXvdM8=", "owner": "numtide", "repo": "flake-utils", - "rev": "ff7b65b44d01cf9ba6a71320833626af21126384", + "rev": "4022d587cbbfd70fe950c1e2083a02621806a725", "type": "github" }, "original": { @@ -38,11 +38,11 @@ }, "nixpkgs": { "locked": { - "lastModified": 1700390070, - "narHash": "sha256-de9KYi8rSJpqvBfNwscWdalIJXPo8NjdIZcEJum1mH0=", + "lastModified": 1703961334, + "narHash": "sha256-M1mV/Cq+pgjk0rt6VxoyyD+O8cOUiai8t9Q6Yyq4noY=", "owner": "nixos", "repo": "nixpkgs", - "rev": "e4ad989506ec7d71f7302cc3067abd82730a4beb", + "rev": "b0d36bd0a420ecee3bc916c91886caca87c894e9", "type": "github" }, "original": { @@ -81,11 +81,11 @@ "nixpkgs": "nixpkgs_2" }, "locked": { - "lastModified": 1700446608, - "narHash": "sha256-q/87GqBvQoUNBYiI3hwhsDqfyfk972RuZK+EwKab5s0=", + "lastModified": 1704075545, + "narHash": "sha256-L3zgOuVKhPjKsVLc3yTm2YJ6+BATyZBury7wnhyc8QU=", "owner": "oxalica", "repo": "rust-overlay", - "rev": "e17bfe3baa0487f0671c9ed0e9057d10987ba7f7", + "rev": "a0df72e106322b67e9c6e591fe870380bd0da0d5", "type": "github" }, "original": { diff --git a/flake.nix b/flake.nix index a093b6e..b5e880b 100644 --- a/flake.nix +++ b/flake.nix @@ -20,13 +20,10 @@ rustc = rustVersion; }; in { - # stdenv = pkgs.clangStdenv; + stdenv = pkgs.fastStdenv; devShell = pkgs.mkShell { LIBCLANG_PATH = pkgs.libclang.lib + "/lib/"; - NIXPKGS_ALLOW_INSECURE=1; - SEQUENCER_BATCH_INBOX_ADDRESS="0xff00000000000000000000000000000000000000"; - L2OO_ADDRESS="0x70997970C51812dc3A010C7d01b50e0d17dc79C"; - # LD_LIBRARY_PATH = "${pkgs.stdenv.cc.cc.lib}/lib/:$LD_LIBRARY_PATH"; + PROTOC = pkgs.protobuf + "/bin/protoc"; nativeBuildInputs = with pkgs; [ bashInteractive diff --git a/rust-toolchain.toml b/rust-toolchain.toml index 5685cde..037198b 100644 --- a/rust-toolchain.toml +++ b/rust-toolchain.toml @@ -2,6 +2,5 @@ # This specifies the version of Rust we use to build. # Individual crates in the workspace may support a lower version, as indicated by `rust-version` field in each crate's `Cargo.toml`. # The version specified below, should be at least as high as the maximum `rust-version` within the workspace. -channel = "1.72.0" -components = [ "rustfmt", "rust-src" ] -targets = [ "wasm32-unknown-unknown" ] +channel = "1.75.0" +components = [ "rustfmt", "clippy"] diff --git a/src/client/error.rs b/src/client/error.rs index 79432f9..8bf62b5 100644 --- a/src/client/error.rs +++ b/src/client/error.rs @@ -4,7 +4,7 @@ use thiserror::Error; pub enum Error { #[error("Block already verified")] BlockAlreadyVerified, - #[error("Block not current or next epoch")] + #[error("Block not in current or next epoch")] BlockNotCurrentOrNextEpoch, #[error("Signature invalid")] SignatureInvalid, @@ -12,10 +12,6 @@ pub enum Error { NotEnoughApprovedStake, #[error("Block is in the next epoch but no new set")] NextBpsInvalid, - #[error("Signature len mismatch")] - SignatureLenMismatch, - #[error("Invalid proof")] - InvalidProof, #[error("Validator not signed")] ValidatorNotSigned, } diff --git a/src/client/message.rs b/src/client/message.rs new file mode 100644 index 0000000..2ad5895 --- /dev/null +++ b/src/client/message.rs @@ -0,0 +1,58 @@ +use super::{protocol::experimental::Proof as ExperimentalProof, Header, Proof}; +use anyhow::Result; +use coerce::actor::message::Message; +use near_primitives::types::TransactionOrReceiptId; +use near_primitives_core::hash::CryptoHash; +use serde::{Deserialize, Serialize}; + +pub struct Shutdown; + +impl Message for Shutdown { + type Result = Result<()>; +} + +pub struct Head; + +impl Message for Head { + type Result = Option
; +} + +pub struct Archive { + pub epoch: CryptoHash, +} + +impl Message for Archive { + type Result = Option
; +} + +#[derive(Debug, Deserialize, Serialize)] +pub struct GetProof(pub TransactionOrReceiptId); + +impl Message for GetProof { + type Result = Option; +} + +#[derive(Debug, Deserialize, Serialize)] +pub struct BatchGetProof(pub Vec); + +impl Message for BatchGetProof { + type Result = Option<(ExperimentalProof, Vec)>; +} + +pub struct VerifyProof { + pub proof: Proof, +} + +impl Message for VerifyProof { + type Result = Result; +} + +// TODO: batch messages +// +// +#[cfg(test)] +mod tests { + + #[test] + fn test_name() {} +} diff --git a/src/client/mod.rs b/src/client/mod.rs index 758533d..e2034a9 100644 --- a/src/client/mod.rs +++ b/src/client/mod.rs @@ -1,31 +1,50 @@ -use crate::{client::protocol::Protocol, config::Config}; +use self::{ + message::BatchGetProof, protocol::experimental::Proof as ExperimentalProof, store::Store, +}; +use crate::{ + client::{ + protocol::Protocol, + store::{head_key, Collection, Entity}, + }, + config::Config, +}; use anyhow::{anyhow, Result}; -use borsh::{BorshDeserialize, BorshSerialize}; -use flume::{Receiver, Sender}; +use async_trait::async_trait; +use coerce::actor::{context::ActorContext, message::Handler, Actor}; +use itertools::Itertools; +use message::{Archive, GetProof, Head, Shutdown, VerifyProof}; use near_jsonrpc_client::methods::light_client_proof::RpcLightClientExecutionProofResponse as BasicProof; -use near_primitives::{types::validator_stake::ValidatorStake, views::LightClientBlockLiteView}; -use near_primitives_core::{hash::CryptoHash, types::AccountId}; +use near_primitives::views::{validator_stake_view::ValidatorStakeView, LightClientBlockLiteView}; +use near_primitives_core::hash::CryptoHash; use serde::{Deserialize, Serialize}; -use sled::Tree; -use std::{path::PathBuf, str::FromStr}; +use std::{str::FromStr, sync::Arc}; use tokio::time; pub mod error; +pub mod message; pub mod protocol; pub mod rpc; +mod store; -pub const STATE_PATH: &str = "state.json"; pub type Header = LightClientBlockLiteView; +pub type BlockMerkleRoot = CryptoHash; #[derive(Debug, Serialize, Deserialize)] +#[serde(untagged)] pub enum Proof { - Basic(BasicProof), + Basic { + head_block_root: BlockMerkleRoot, + proof: Box, + }, Experimental(protocol::experimental::Proof), } -impl From for Proof { - fn from(proof: BasicProof) -> Self { - Self::Basic(proof) +impl From<(CryptoHash, BasicProof)> for Proof { + fn from((head_block_root, proof): (CryptoHash, BasicProof)) -> Self { + Self::Basic { + head_block_root, + proof: Box::new(proof), + } } } @@ -35,302 +54,303 @@ impl From for Proof { } } -pub enum ProofType { - Transaction { - transaction_id: CryptoHash, - sender_id: AccountId, - }, - Receipt { - receipt_id: CryptoHash, - receiver_id: AccountId, - }, -} - -pub enum Message { - Shutdown(Option), - Head { - tx: Sender
, - }, - Archive { - tx: Sender>, - epoch: CryptoHash, - }, - GetProof { - tx: Sender>, - proof: ProofType, - }, - ValidateProof { - tx: Sender, - proof: Box, - }, +impl Proof { + pub fn block_merkle_root(&self) -> &CryptoHash { + match self { + Self::Basic { + head_block_root, .. + } => head_block_root, + Self::Experimental(proof) => &proof.head_block_root, + } + } } -#[derive(Debug, Clone)] pub struct LightClient { + config: Config, client: rpc::NearRpcClient, - store: sled::Db, - head: LightClientBlockLiteView, - block_producers: Tree, - archival_headers: Tree, + store: Arc>, } -pub enum TreeKind { - BlockProducers, - ArchivalHeaders, +#[async_trait] +impl Actor for LightClient { + async fn started(&mut self, _ctx: &mut ActorContext) { + self.bootstrap_store().await.unwrap(); + // TODO: anonymous ctx.spawn(id, actor) + let catchup = self.config.catchup; + let store = self.store.clone(); + let client = self.client.clone(); + tokio::task::spawn(async move { Self::start_syncing(catchup, store, client).await }); + } } -impl LightClient { - pub fn start(mut self, mut is_fast: bool, rx: Receiver) { - tokio::spawn(async move { - // TODO: make configurable, currently set to ~block time - let default_duration = time::Duration::from_secs(2); +#[async_trait] +impl Handler for LightClient { + async fn handle( + &mut self, + _message: Head, + _ctx: &mut ActorContext, + ) -> ::Result { + self.store.head().await.ok() + } +} - let mut duration = if is_fast { - time::Duration::from_millis(100) - } else { - default_duration - }; +#[async_trait] +impl Handler for LightClient { + async fn handle( + &mut self, + message: Archive, + _ctx: &mut ActorContext, + ) -> ::Result { + self.header(message.epoch).await + } +} - loop { - tokio::select! { - Ok(msg) = rx.recv_async() => match msg { - Message::Shutdown(_path) => { - if let Err(e) = self.shutdown().await { - log::error!("Failed to shutdown: {:?}", e); - } - } - Message::Head { tx } => { - if let Err(e) = tx.send_async(self.head().clone()).await { - log::error!("Failed to send head: {:?}", e); - } - }, - Message::Archive { tx, epoch } => { - if let Err(e) = tx.send_async(self.header(epoch)).await { - log::error!("Failed to send archival header: {:?}", e); - } - }, - Message::GetProof { - tx, - proof: ProofType::Transaction { transaction_id, sender_id } - } => { - if let Err(e) = tx.send_async(self.get_transaction_proof(transaction_id, sender_id).await.map(Into::into)).await { - log::error!("Failed to send proof: {:?}", e); - } - }, - Message::GetProof { - tx, proof: ProofType::Receipt { receipt_id, receiver_id } - } => { - if let Err(e) = tx.send_async(self.get_receipt_proof(receipt_id, receiver_id).await.map(Into::into)).await { - log::error!("Failed to send proof: {:?}", e); - } - } - Message::ValidateProof { tx, proof } => { - if let Err(e) = tx.send_async(self.verify_proof(*proof).await).await { - log::error!("Failed to send validation result: {:?}", e); - } - } - }, - r = self.sync() => { - tokio::time::sleep(duration).await; - match r { - Err(e) => log::error!("Error syncing: {:?}", e), - Ok(false) if is_fast => { - log::debug!("Slowing down sync interval to {:?}", default_duration); - duration = default_duration; - is_fast = false; - } - _ => (), - } - } - } - } - }); +#[async_trait] +impl Handler for LightClient { + async fn handle( + &mut self, + _message: Shutdown, + ctx: &mut ActorContext, + ) -> ::Result { + self.store.shutdown().await; + ctx.stop(None); + Ok(()) } +} - // TODO: move to store module - pub fn get, T: BorshDeserialize>(&self, key: K, kind: TreeKind) -> Option { - let tree = match kind { - TreeKind::BlockProducers => &self.block_producers, - TreeKind::ArchivalHeaders => &self.archival_headers, - }; - tree.get(key) +#[async_trait] +impl Handler for LightClient { + async fn handle( + &mut self, + message: GetProof, + _ctx: &mut ActorContext, + ) -> ::Result { + self.get_proofs(BatchGetProof(vec![message])) + .await .ok() - .unwrap_or_default() - .and_then(|v| BorshDeserialize::try_from_slice(&v).ok()) + .and_then(|proofs| proofs.into_iter().next()) } +} - pub fn insert, T: BorshSerialize>( +#[async_trait] +impl Handler for LightClient { + async fn handle( &mut self, - key: K, - value: &T, - kind: TreeKind, - ) -> Result<()> { - let tree = match kind { - TreeKind::BlockProducers => &self.block_producers, - TreeKind::ArchivalHeaders => &self.archival_headers, - }; - Ok(tree - .insert(key, BorshSerialize::try_to_vec(value)?) - .map(|_| ())?) + message: VerifyProof, + _ctx: &mut ActorContext, + ) -> ::Result { + self.verify_proof(message.proof).await.map_err(|e| { + log::error!("{:?}", e); + e + }) } +} + +#[async_trait] +impl Handler for LightClient { + async fn handle( + &mut self, + message: BatchGetProof, + _ctx: &mut ActorContext, + ) -> ::Result { + self.experimental_get_proofs(message).await.ok() + } +} - pub async fn init(config: &Config) -> Result { - let sled = sled::open(config.state_path.clone().unwrap_or(STATE_PATH.into())) - .expect("Failed to open store"); +impl LightClient { + pub fn new(config: &Config) -> Result { let client = rpc::NearRpcClient::new(config.network.clone()); - let head = sled.get("head")?; - let block_producers = sled.open_tree("bps")?; - let archival_headers = sled.open_tree("archive")?; - - if let Some(head) = head { - let head: LightClientBlockLiteView = - borsh::BorshDeserialize::try_from_slice(&head).expect("Failed to deserialize head"); - Ok(Self { - client, - store: sled, - head, - block_producers, - archival_headers, - }) - } else { - let starting_head = client - .fetch_latest_header(&CryptoHash::from_str(&config.starting_head).unwrap()) + // TODO: store selector in config + let store = store::sled::init(config)?; + + Ok(Self { + client, + config: config.clone(), + store: Store(store.into()).into(), + }) + } + + async fn bootstrap_store(&mut self) -> Result<()> { + let head = self.store.head().await; + if head.is_err() { + let sync_from = + CryptoHash::from_str(&self.config.starting_head).map_err(anyhow::Error::msg)?; + + let starting_head = self + .client + .fetch_latest_header(&sync_from) .await - .expect("We need a starting header"); + .ok_or_else(|| anyhow::anyhow!("We need a starting header"))?; - log::info!("Got starting head: {:?}", starting_head.inner_lite.height); + log::info!("starting head: {:?}", starting_head.inner_lite.height); - block_producers.insert( + let mut inserts: Vec<(CryptoHash, Entity)> = vec![]; + + inserts.push(( starting_head.inner_lite.epoch_id, - borsh::to_vec(&starting_head.next_bps.clone().unwrap())?, - )?; - - Ok(Self { - client, - store: sled, - head: LightClientBlockLiteView { - prev_block_hash: starting_head.prev_block_hash, - inner_rest_hash: starting_head.inner_rest_hash, - inner_lite: starting_head.inner_lite, - }, - block_producers, - archival_headers, - }) + starting_head + .next_bps + .unwrap() + .into_iter() + .map(ValidatorStakeView::into_validator_stake) + .collect_vec() + .into(), + )); + + let boostrapped_head = LightClientBlockLiteView { + prev_block_hash: starting_head.prev_block_hash, + inner_rest_hash: starting_head.inner_rest_hash, + inner_lite: starting_head.inner_lite, + }; + + inserts.push((head_key(), boostrapped_head.into())); + + self.store.insert(&inserts).await?; } - } - pub async fn sync(&mut self) -> Result { - log::trace!("Current head: {:#?}", self.head.inner_lite); + Ok(()) + } - let next_header = self - .client + // TODO: dynamically determine if should catchup + pub async fn start_syncing( + mut catching_up: bool, + store: Arc>, + client: rpc::NearRpcClient, + ) { + // TODO: make configurable, currently set to ~block time + let default_duration = time::Duration::from_secs(2); + + loop { + let duration = if catching_up { + time::Duration::from_millis(100) + } else { + default_duration + }; + tokio::select! { + r = Self::sync(store.clone(), client.clone()) => { + tokio::time::sleep(duration).await; + match r { + Err(e) => { + log::error!("Error syncing: {:?}", e); + catching_up = false; + }, + Ok(false) if catching_up => { + log::debug!("Slowing down sync interval to {:?}", default_duration); + catching_up = false; + } + _ => (), + } + } + } + } + } + pub async fn sync( + store: Arc>, + client: rpc::NearRpcClient, + ) -> Result { + let head = store.head().await?; + log::debug!("Current head: {:#?}", head.inner_lite); + + let next_header = client .fetch_latest_header( - &CryptoHash::from_str(&format!("{}", self.head.hash())) + &CryptoHash::from_str(&format!("{}", head.hash())) .map_err(|e| anyhow!("Failed to parse hash: {:?}", e))?, ) .await .ok_or_else(|| anyhow!("Failed to fetch latest header"))?; log::trace!("Got new header: {:#?}", next_header.inner_lite); - let bps = self - .get::<_, Vec>(&self.head.inner_lite.epoch_id, TreeKind::BlockProducers) - .ok_or_else(|| anyhow!("Failed to get block producers"))?; + let bps = store + .get(&Collection::BlockProducers, &head.inner_lite.epoch_id) + .await + .and_then(|x| x.bps())?; + + let synced = Protocol::sync(&head, &bps, next_header)?; - let synced = Protocol::sync(&self.head, &bps, next_header)?; + let mut inserts: Vec<(CryptoHash, Entity)> = vec![]; if let Some((epoch, next_bps)) = synced.next_bps { log::debug!("storing next bps[{:?}]", epoch); - self.insert(epoch, &next_bps, TreeKind::BlockProducers)?; + inserts.push((epoch.0, next_bps.into())); } - self.insert( - self.head.inner_lite.epoch_id, - &synced.new_head, - TreeKind::ArchivalHeaders, - )?; - self.head = synced.new_head; - self.store.insert("head", self.head.try_to_vec()?)?; + inserts.push((head.inner_lite.epoch_id, synced.new_head.clone().into())); + inserts.push((head_key(), synced.new_head.into())); + store.insert(&inserts).await?; Ok(true) } - fn head(&self) -> &Header { - &self.head + async fn header(&self, epoch: CryptoHash) -> Option
{ + self.store + .get(&Collection::Headers, &epoch) + .await + .and_then(|e| e.header()) + .ok() } - fn header(&self, epoch: CryptoHash) -> Option
{ - self.get::<_, LightClientBlockLiteView>(&epoch, TreeKind::ArchivalHeaders) - } + pub async fn batch_fetch_proofs( + &self, + last_verified_hash: &CryptoHash, + p: BatchGetProof, + collect_errors: bool, + ) -> Result<(Vec, Vec)> { + let mut futs = vec![]; + for req in p.0 { + let proof = self + .client + .fetch_light_client_proof(req, *last_verified_hash); + futs.push(proof); + } + let unpin_futs: Vec<_> = futs.into_iter().map(Box::pin).collect(); - // TODO: memoize these in a real cache - pub async fn get_transaction_proof( - &mut self, - transaction_id: CryptoHash, - sender_id: AccountId, - ) -> Option { - // TODO: we need much more logic here for headers in different epochs & the head - // since there are race conditions when we come to validate the proof header, we'd need to short-cache the proof validation - let last_verified_hash = self.head.hash(); - - self.client - .fetch_light_client_tx_proof(transaction_id, sender_id, last_verified_hash) - .await - .map(|proof| { - if let Err(e) = self.insert( - proof.outcome_proof.block_hash, - &self.head.clone(), - TreeKind::ArchivalHeaders, - ) { - log::warn!("Failed to insert archival header: {:?}", e) - } - proof - }) + let proofs: Vec> = futures::future::join_all(unpin_futs).await; + let (proofs, errors): (Vec<_>, Vec<_>) = if collect_errors { + proofs.into_iter().partition_result() + } else { + (proofs.into_iter().collect::>>()?, vec![]) + }; + Ok((proofs, errors)) } - pub async fn get_receipt_proof( - &mut self, - receipt_id: CryptoHash, - receiver_id: AccountId, - ) -> Option { - let last_verified_hash = self.head.hash(); - self.client - .fetch_light_client_receipt_proof(receipt_id, receiver_id, last_verified_hash) - .await - .map(|proof| { - if let Err(e) = self.insert( - proof.outcome_proof.block_hash, - &self.head.clone(), - TreeKind::ArchivalHeaders, - ) { - log::warn!("Failed to insert archival header: {:?}", e) - } - proof - }) + pub async fn verify_proof(&self, p: Proof) -> Result { + anyhow::ensure!( + self.store + .contains(&Collection::UsedRoots, p.block_merkle_root()) + .await?, + "Root {:?} is not known", + p.block_merkle_root() + ); + Protocol::inclusion_proof_verify(p) } - pub async fn verify_proof(&mut self, p: Proof) -> bool { - // TODO: this is a hack - let archive = match &p { - Proof::Basic(p) => self.get::<_, LightClientBlockLiteView>( - p.outcome_proof.block_hash, - TreeKind::ArchivalHeaders, - ), - Proof::Experimental(_) => None, - }; - Protocol::inclusion_proof_verify(&self.head, archive.as_ref(), p) - .inspect_err(|e| log::warn!("Failed to verify proof: {:?}", e)) - .unwrap_or_default() + // TODO: this and below are the same except from two lines + pub async fn get_proofs(&self, p: BatchGetProof) -> Result> { + let head = self.store.head().await?; + let proofs = self.batch_fetch_proofs(&head.hash(), p, false).await?.0; + + self.store + .insert(&[(head.inner_lite.block_merkle_root, Entity::UsedRoot)]) + .await?; + Ok(proofs + .into_iter() + .map(|x| (head.inner_lite.block_merkle_root, x)) + .map(Into::into) + .collect()) } - pub async fn shutdown(&self) -> Result<()> { - log::info!("Shutting down light client"); - if let Err(e) = self.store.insert("head", self.head.try_to_vec()?) { - log::warn!("Failed to insert head: {:?}", e) - } - log::info!("Head: {:?}", self.head.inner_lite.height); - self.store.flush_async().await?; - log::info!("Flushed store"); - todo!("Deliberate panic here, need to make this graceful"); + pub async fn experimental_get_proofs( + &self, + p: BatchGetProof, + ) -> Result<(ExperimentalProof, Vec)> { + let head = self.store.head().await?; + let (proofs, errors) = self.batch_fetch_proofs(&head.hash(), p, true).await?; + let p = protocol::experimental::Proof::new(head.inner_lite.block_merkle_root, proofs); + self.store + .insert(&[(head.inner_lite.block_merkle_root, Entity::UsedRoot)]) + .await?; + + Ok((p, errors)) } } diff --git a/src/client/protocol/experimental.rs b/src/client/protocol/experimental.rs index 89a17ba..d75fd60 100644 --- a/src/client/protocol/experimental.rs +++ b/src/client/protocol/experimental.rs @@ -34,7 +34,7 @@ impl From for LightweightHeader { fn from(header: LightClientBlockLiteView) -> Self { let full_inner: BlockHeaderInnerLite = header.inner_lite.clone().into(); Self { - inner_lite_hash: CryptoHash::hash_borsh(&full_inner), + inner_lite_hash: CryptoHash::hash_borsh(full_inner), inner_rest_hash: header.inner_rest_hash, prev_block_hash: header.prev_block_hash, outcome_root: header.inner_lite.outcome_root, @@ -67,12 +67,14 @@ pub enum Header { } impl Header { + #[allow(unused)] fn hash(&self) -> CryptoHash { match self { Header::Lightweight(header) => header.hash(), Header::Lightest(header) => header.block_hash, } } + #[allow(unused)] fn outcome_root(&self) -> CryptoHash { match self { Header::Lightweight(header) => header.outcome_root, @@ -99,12 +101,11 @@ struct MerkleCache { } impl MerkleCache { - fn cache(&mut self, batch: &mut Vec) { + fn cache(&mut self, batch: &mut [BlindedProof]) { let duplicates = batch .iter() .cloned() - .map(|fp| [fp.outcome_proof, fp.outcome_root_proof, fp.block_proof].concat()) - .flatten() + .flat_map(|fp| [fp.outcome_proof, fp.outcome_root_proof, fp.block_proof].concat()) // TODO: this is super inefficient, we had a nice one without duplicates but // Unfortunately duplicates clones the array, so we can't do this zero // copy @@ -113,23 +114,20 @@ impl MerkleCache { batch .iter_mut() - .map(|fp| { + .flat_map(|fp| { fp.outcome_proof .iter_mut() .chain(fp.outcome_root_proof.iter_mut()) .chain(fp.block_proof.iter_mut()) }) - .flatten() .for_each(|item| { - duplicates - .iter() - .position(|dup| dup == item) - .map(|i| item.0 = Either::Left(i as u32)); + if let Some(i) = duplicates.iter().position(|dup| dup == item) { + item.0 = Either::Left(i as u32); + } }); self.items = duplicates.into_iter().map(|x| x.0.unwrap_right()).collect(); } - // TODO: no panic fn collect<'a>( &'a self, path: &'a [LookupMerklePathItem], @@ -232,7 +230,7 @@ impl BorshDeserialize for LookupMerklePathItem { pub struct Proof { /// The block_merkle_root of the header used to create the proof batch /// It should be at least the header for the latest transactions proven + 1 - client_block_merkle_root: CryptoHash, + pub head_block_root: CryptoHash, batch: Vec, // common ancestry if there is a common ancestry line in the batch // we can save g as costs by only passing this once @@ -264,7 +262,7 @@ impl Proof { common_ancestry } - fn new(created_from: CryptoHash, mut batch: Vec) -> Self { + pub fn new(head_block_root: CryptoHash, mut batch: Vec) -> Self { // First decide common ancestry among all batches let ancestry = batch .iter() @@ -285,7 +283,7 @@ impl Proof { cache.cache(&mut batch); Proof { - client_block_merkle_root: created_from, + head_block_root, batch, ancestry, cache, @@ -310,7 +308,7 @@ pub fn verify_proof(proof: Proof) -> bool { ); let block_verified = Protocol::verify_block( - &proof.client_block_merkle_root, + &proof.head_block_root, proof .cache .collect(&blinded.block_proof) @@ -336,6 +334,7 @@ pub(crate) mod tests { rpc::{NearRpcClient, Network}, }; use futures::FutureExt; + use near_primitives::types::TransactionOrReceiptId; use near_primitives_core::types::AccountId; use std::{path::Path, str::FromStr}; @@ -463,7 +462,7 @@ pub(crate) mod tests { // Util for rewriting the original bridge proofs fn _rewrite_bridge_proofs(rainbow_prover_fixture_path: &str) { - let rewritten = vec![ + let rewritten = [ ( "22f00dd154366d758cd3e4fe81c1caed8e0db6227fe4b2b52a8e5a468aa0a723", "proof2.json", @@ -555,8 +554,14 @@ pub(crate) mod tests { .map(Result::unwrap) .map(|receipt_id| { client - .fetch_light_client_receipt_proof(receipt_id.clone(), receiver_id.clone(), head) - .map(Option::unwrap) + .fetch_light_client_proof( + crate::client::message::GetProof(TransactionOrReceiptId::Receipt { + receipt_id, + receiver_id: receiver_id.clone(), + }), + head, + ) + .map(Result::unwrap) }) .collect_vec(); diff --git a/src/client/protocol/merkle_util.rs b/src/client/protocol/merkle_util.rs index cbae81c..1ff444c 100644 --- a/src/client/protocol/merkle_util.rs +++ b/src/client/protocol/merkle_util.rs @@ -3,18 +3,16 @@ pub use near_primitives::merkle::combine_hash; use near_primitives::merkle::{Direction, MerklePathItem}; use near_primitives_core::{hash::CryptoHash, types::MerkleHash}; -pub trait PathIterator<'a> = Iterator; - pub fn verify_hash<'a>( root: MerkleHash, - path: impl PathIterator<'a>, + path: impl Iterator, item_hash: MerkleHash, ) -> bool { compute_root_from_path(path, item_hash) == root } pub fn compute_root_from_path<'a>( - path: impl PathIterator<'a>, + path: impl Iterator, item_hash: MerkleHash, ) -> MerkleHash { let mut res = item_hash; @@ -32,7 +30,7 @@ pub fn compute_root_from_path<'a>( } pub fn compute_root_from_path_and_item<'a, T: BorshSerialize>( - path: impl PathIterator<'a>, + path: impl Iterator, item: T, ) -> MerkleHash { compute_root_from_path(path, CryptoHash::hash_borsh(item)) diff --git a/src/client/protocol/mod.rs b/src/client/protocol/mod.rs index 3014083..7129214 100644 --- a/src/client/protocol/mod.rs +++ b/src/client/protocol/mod.rs @@ -5,6 +5,7 @@ use merkle_util::*; use near_crypto::{PublicKey, Signature}; use near_primitives::{ block_header::ApprovalInner, + merkle::MerklePathItem, types::{validator_stake::ValidatorStake, EpochId}, views::{ validator_stake_view::ValidatorStakeView, LightClientBlockLiteView, LightClientBlockView, @@ -31,12 +32,10 @@ impl Protocol { epoch_bps: &[ValidatorStake], next_block: LightClientBlockView, ) -> Result { - println!("current_head: {:?}", head.inner_lite.height); - println!("next_block: {:?}", next_block.inner_lite.height); - Self::ensure_not_already_verified(&head, &next_block.inner_lite.height)?; - Self::ensure_epoch_is_current_or_next(&head, &next_block.inner_lite.epoch_id)?; + Self::ensure_not_already_verified(head, &next_block.inner_lite.height)?; + Self::ensure_epoch_is_current_or_next(head, &next_block.inner_lite.epoch_id)?; Self::ensure_if_next_epoch_contains_next_bps( - &head, + head, &next_block.inner_lite.epoch_id, &next_block.next_bps, )?; @@ -53,12 +52,11 @@ impl Protocol { &next_block.approvals_after_next, epoch_bps, &approval_message, - )?; - println!("total_stake: {:?}", total); - println!("approved_stake: {:?}", approved); + ); + Self::ensure_stake_is_sufficient(&total, &approved)?; - log::debug!( + log::trace!( "prev/current head: {}/{}", head.inner_lite.height, new_head.inner_lite.height @@ -73,17 +71,19 @@ impl Protocol { .map(|next_bps| next_bps.into_iter().map(Into::into).collect()) .map(|next_bps| (EpochId(head.inner_lite.next_epoch_id), next_bps)), }) - .inspect(|synced| log::debug!("Synced new head: {:?}", synced)) + .map(|synced| { + log::debug!("Synced new head: {:?}", synced.new_head); + synced + }) } - pub fn inclusion_proof_verify( - head: &LightClientBlockLiteView, - archive_header: Option<&LightClientBlockLiteView>, - proof: Proof, - ) -> Result { + pub fn inclusion_proof_verify(proof: Proof) -> Result { match proof { Proof::Experimental(proof) => Ok(experimental::verify_proof(proof)), - Proof::Basic(proof) => { + Proof::Basic { + head_block_root, + proof, + } => { let block_hash = proof.block_header_lite.hash(); let block_hash_matches = block_hash == proof.outcome_proof.block_hash; @@ -96,51 +96,26 @@ impl Protocol { &proof.block_header_lite.inner_lite.outcome_root, ); - let mut block_verified = Self::verify_block( - &head.inner_lite.block_merkle_root, - proof.block_proof.iter(), - &block_hash, - ); - - // TODO: refactor me - // Functionality to cover race conditions between previously synced heads since the current head may of changed - // since the proof was generated - match archive_header { - Some(archive_header) if !block_verified => { - log::debug!( - "Trying to verify against archival header: {:?}", - archive_header.inner_lite.height - ); - if verify_hash( - archive_header.inner_lite.block_merkle_root, - proof.block_proof.iter(), - block_hash, - ) { - log::debug!("Verified against archival header"); - block_verified = true; - } else { - return Err(Error::InvalidProof.into()); - } - } - Some(_) => {} - None => return Err(Error::InvalidProof.into()), - } + let block_verified = + Self::verify_block(&head_block_root, proof.block_proof.iter(), &block_hash); log::debug!( "shard outcome included: {:?}, block included: {:?}", outcome_verified, block_verified ); - Ok(block_hash_matches && outcome_verified && block_verified) - .inspect(|verified| log::debug!("Verified proof {:?}", verified)) + Ok(block_hash_matches && outcome_verified && block_verified).map(|verified| { + log::debug!("Verified proof {:?}", verified); + verified + }) } } } pub(crate) fn verify_outcome<'a>( outcome_hash: &CryptoHash, - outcome_proof: impl PathIterator<'a>, - outcome_root_proof: impl PathIterator<'a>, + outcome_proof: impl Iterator, + outcome_root_proof: impl Iterator, expected_outcome_root: &CryptoHash, ) -> bool { let outcome_root = compute_root_from_path(outcome_proof, *outcome_hash); @@ -153,7 +128,7 @@ impl Protocol { pub(crate) fn verify_block<'a>( block_merkle_root: &CryptoHash, - block_proof: impl PathIterator<'a>, + block_proof: impl Iterator, block_hash: &CryptoHash, ) -> bool { verify_hash(*block_merkle_root, block_proof, *block_hash) @@ -170,7 +145,7 @@ impl Protocol { temp_vec.extend_from_slice(&(current_block_hash.0)); temp_vec }); - let new = combine_hash(&block_view.next_block_inner_hash, ¤t_block_hash); + let new = combine_hash(&block_view.next_block_inner_hash, current_block_hash); assert_eq!(old, new); log::debug!("Current block hash: {}", current_block_hash); new @@ -183,7 +158,7 @@ impl Protocol { inner_lite: block_view.inner_lite.clone(), }; - let next_block_hash = Self::next_block_hash(&new_head.hash(), &block_view); + let next_block_hash = Self::next_block_hash(&new_head.hash(), block_view); let endorsement = ApprovalInner::Endorsement(next_block_hash); let approval_message = { @@ -215,7 +190,7 @@ impl Protocol { ) -> Result<(), Error> { if ![head.inner_lite.epoch_id, head.inner_lite.next_epoch_id].contains(epoch_id) { log::debug!("Block is not in the current or next epoch"); - Err(Error::BlockNotCurrentOrNextEpoch.into()) + Err(Error::BlockNotCurrentOrNextEpoch) } else { Ok(()) } @@ -238,18 +213,8 @@ impl Protocol { signatures: &[Option], epoch_bps: &[ValidatorStake], approval_message: &[u8], - ) -> Result { - // Signatures can contain more than the number of bps due to epoch switch - if signatures.len() < epoch_bps.len() { - log::debug!( - "Signature len {} does not match BPS len {}", - signatures.len(), - epoch_bps.len() - ); - return Err(Error::SignatureLenMismatch); - } - - Ok(signatures + ) -> StakeInfo { + signatures .iter() .zip(epoch_bps.iter()) .fold((0, 0), |(total_stake, approved_stake), (sig, vs)| { @@ -265,7 +230,7 @@ impl Protocol { (total_stake, approved_stake) }) - .into()) + .into() } fn validate_signature( @@ -297,7 +262,7 @@ impl Protocol { if approved_stake <= &threshold { log::debug!("Not enough stake approved"); - Err(Error::NotEnoughApprovedStake.into()) + Err(Error::NotEnoughApprovedStake) } else { Ok(()) } @@ -314,7 +279,7 @@ impl Protocol { Ok(Some(next_bps)) } else { log::warn!("Next block producers hash is invalid"); - return Err(Error::NextBpsInvalid); + Err(Error::NextBpsInvalid) } } else { Ok(None) @@ -350,8 +315,6 @@ macro_rules! cvec { #[cfg(test)] mod tests { use super::*; - use crate::client::protocol::experimental::tests::get_constants; - use core::str::FromStr; use itertools::Itertools; use near_jsonrpc_client::methods::light_client_proof::RpcLightClientExecutionProofResponse; use serde_json::{self}; @@ -492,7 +455,7 @@ mod tests { Protocol::validate_signature( &b"bogus approval message"[..], &next_block.approvals_after_next[0], - &next_bps[0].public_key(), + next_bps[0].public_key(), ), Err(Error::SignatureInvalid) ); @@ -517,8 +480,7 @@ mod tests { &next_block.approvals_after_next, &next_bps.clone(), &approval_message.unwrap(), - ) - .unwrap(); + ); assert_eq!((total, approved), (512915271547861520119028536348929, 0)); } @@ -535,8 +497,7 @@ mod tests { &next_block.approvals_after_next, &next_bps, &approval_message.unwrap(), - ) - .unwrap(); + ); assert_eq!( (total, approved), @@ -604,7 +565,7 @@ mod tests { let req = r#"{"outcome_proof":{"proof":[],"block_hash":"5CY72FinjVV2Hd5zRikYYMaKh67pftXJsw8vwRXAUAQF","id":"9UhBumQ3eEmPH5ALc3NwiDCQfDrFakteRD7rHE9CfZ32","outcome":{"logs":[],"receipt_ids":["2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"],"gas_burnt":2434069818500,"tokens_burnt":"243406981850000000000","executor_id":"datayalla.testnet","status":{"SuccessReceiptId":"2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"},"metadata":{"version":1,"gas_profile":null}}},"outcome_root_proof":[{"hash":"9f7YjLvzvSspJMMJ3DDTrFaEyPQ5qFqQDNoWzAbSTjTy","direction":"Right"},{"hash":"67ZxFmzWXbWJSyi7Wp9FTSbbJx2nMr7wSuW3EP1cJm4K","direction":"Left"}],"block_header_lite":{"prev_block_hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","inner_rest_hash":"G25j8jSWRyrXV317cPC3qYA4SyJWXsBfErjhBYQkxw5A","inner_lite":{"height":134481525,"epoch_id":"4tBzDozzGED3QiCRURfViVuyJy5ikaN9dVH7m2MYkTyw","next_epoch_id":"9gYJSiT3TQbKbwui5bdbzBA9PCMSSfiffWhBdMtcasm2","prev_state_root":"EwkRecSP8GRvaxL7ynCEoHhsL1ksU6FsHVLCevcccF5q","outcome_root":"8Eu5qpDUMpW5nbmTrTKmDH2VYqFEHTKPETSTpPoyGoGc","timestamp":1691615068679535000,"timestamp_nanosec":"1691615068679535094","next_bp_hash":"8LCFsP6LeueT4X3PEni9CMvH7maDYpBtfApWZdXmagss","block_merkle_root":"583vb6csYnczHyt5z6Msm4LzzGkceTZHdvXjC8vcWeGK"}},"block_proof":[{"hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","direction":"Left"},{"hash":"HgZaHXpb5zs4rxUQTeW69XBNLBJoo4sz2YEDh7aFnMpC","direction":"Left"},{"hash":"EYNXYsnESQkXo7B27a9xu6YgbDSyynNcByW5Q2SqAaKH","direction":"Right"},{"hash":"AbKbsD7snoSnmzAtwNqXLBT5sm7bZr48GCCLSdksFuzi","direction":"Left"},{"hash":"7KKmS7n3MtCfv7UqciidJ24Abqsk8m85jVQTh94KTjYS","direction":"Left"},{"hash":"5nKA1HCZMJbdCccZ16abZGEng4sMoZhKez74rcCFjnhL","direction":"Left"},{"hash":"BupagAycSLD7v42ksgMKJFiuCzCdZ6ksrGLwukw7Vfe3","direction":"Right"},{"hash":"D6v37P4kcVJh8N9bV417eqJoyMeQbuZ743oNsbKxsU7z","direction":"Right"},{"hash":"8sWxxbe1rdquP5VdYfQbw1UvtcXDRansJYJV5ySzyow4","direction":"Right"},{"hash":"CmKVKWRqEqi4UaeKKYXpPSesYqdQYwHQM3E4xLKEUAj8","direction":"Left"},{"hash":"3TvjFzVyPBvPpph5zL6VCASLCxdNeiKV6foPwUpAGqRv","direction":"Left"},{"hash":"AnzSG9f91ePS6L6ii3eAkocp4iKjp6wjzSwWsDYWLnMX","direction":"Right"},{"hash":"FYVJDL4T6c87An3pdeBvntB68NzpcPtpvLP6ifjxxNkr","direction":"Left"},{"hash":"2YMF6KE8XTz7Axj3uyAoFbZisWej9Xo8mxgVtauWCZaV","direction":"Left"},{"hash":"4BHtLcxqNfWSneBdW76qsd8om8Gjg58Qw5BX8PHz93hf","direction":"Left"},{"hash":"7G3QUT7NQSHyXNQyzm8dsaYrFk5LGhYaG7aVafKAekyG","direction":"Left"},{"hash":"3XaMNnvnX69gGqBJX43Na1bSTJ4VUe7z6h5ZYJsaSZZR","direction":"Left"},{"hash":"FKu7GtfviPioyAGXGZLBVTJeG7KY5BxGwuL447oAZxiL","direction":"Right"},{"hash":"BePd7DPKUQnGtnSds5fMJGBUwHGxSNBpaNLwceJGUcJX","direction":"Left"},{"hash":"2BVKWMd9pXZTEyE9D3KL52hAWAyMrXj1NqutamyurrY1","direction":"Left"},{"hash":"EWavHKhwQiT8ApnXvybvc9bFY6aJYJWqBhcrZpubKXtA","direction":"Left"},{"hash":"83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD","direction":"Left"},{"hash":"AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL","direction":"Left"}]}"#; let p: RpcLightClientExecutionProofResponse = serde_json::from_str(req).unwrap(); - let outcome_hash = CryptoHash::hash_borsh(&p.outcome_proof.to_hashes()); + let outcome_hash = CryptoHash::hash_borsh(p.outcome_proof.to_hashes()); let root_matches = Protocol::verify_outcome( &outcome_hash, @@ -614,18 +575,4 @@ mod tests { ); assert!(root_matches); } - - #[test] - fn test_root() { - pretty_env_logger::try_init().ok(); - let (_head, root, _) = get_constants(); - - let block_hash = - CryptoHash::from_str("7xTJs2wYhLcWKsT2Vcf4HPcXVdKKxTAPFTphhEhf3AxW").unwrap(); - let req = r#"{"outcome_proof":{"proof":[],"block_hash":"5CY72FinjVV2Hd5zRikYYMaKh67pftXJsw8vwRXAUAQF","id":"9UhBumQ3eEmPH5ALc3NwiDCQfDrFakteRD7rHE9CfZ32","outcome":{"logs":[],"receipt_ids":["2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"],"gas_burnt":2434069818500,"tokens_burnt":"243406981850000000000","executor_id":"datayalla.testnet","status":{"SuccessReceiptId":"2mrt6jXKwWzkGrhucAtSc8R3mjrhkwCjnqVckPdCMEDo"},"metadata":{"version":1,"gas_profile":null}}},"outcome_root_proof":[{"hash":"9f7YjLvzvSspJMMJ3DDTrFaEyPQ5qFqQDNoWzAbSTjTy","direction":"Right"},{"hash":"67ZxFmzWXbWJSyi7Wp9FTSbbJx2nMr7wSuW3EP1cJm4K","direction":"Left"}],"block_header_lite":{"prev_block_hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","inner_rest_hash":"G25j8jSWRyrXV317cPC3qYA4SyJWXsBfErjhBYQkxw5A","inner_lite":{"height":134481525,"epoch_id":"4tBzDozzGED3QiCRURfViVuyJy5ikaN9dVH7m2MYkTyw","next_epoch_id":"9gYJSiT3TQbKbwui5bdbzBA9PCMSSfiffWhBdMtcasm2","prev_state_root":"EwkRecSP8GRvaxL7ynCEoHhsL1ksU6FsHVLCevcccF5q","outcome_root":"8Eu5qpDUMpW5nbmTrTKmDH2VYqFEHTKPETSTpPoyGoGc","timestamp":1691615068679535000,"timestamp_nanosec":"1691615068679535094","next_bp_hash":"8LCFsP6LeueT4X3PEni9CMvH7maDYpBtfApWZdXmagss","block_merkle_root":"583vb6csYnczHyt5z6Msm4LzzGkceTZHdvXjC8vcWeGK"}},"block_proof":[{"hash":"AEnTyGRrk2roQkYSWoqYhzkbp5SWWJtCd71ZYyj1P26i","direction":"Left"},{"hash":"HgZaHXpb5zs4rxUQTeW69XBNLBJoo4sz2YEDh7aFnMpC","direction":"Left"},{"hash":"EYNXYsnESQkXo7B27a9xu6YgbDSyynNcByW5Q2SqAaKH","direction":"Right"},{"hash":"AbKbsD7snoSnmzAtwNqXLBT5sm7bZr48GCCLSdksFuzi","direction":"Left"},{"hash":"7KKmS7n3MtCfv7UqciidJ24Abqsk8m85jVQTh94KTjYS","direction":"Left"},{"hash":"5nKA1HCZMJbdCccZ16abZGEng4sMoZhKez74rcCFjnhL","direction":"Left"},{"hash":"BupagAycSLD7v42ksgMKJFiuCzCdZ6ksrGLwukw7Vfe3","direction":"Right"},{"hash":"D6v37P4kcVJh8N9bV417eqJoyMeQbuZ743oNsbKxsU7z","direction":"Right"},{"hash":"8sWxxbe1rdquP5VdYfQbw1UvtcXDRansJYJV5ySzyow4","direction":"Right"},{"hash":"CmKVKWRqEqi4UaeKKYXpPSesYqdQYwHQM3E4xLKEUAj8","direction":"Left"},{"hash":"3TvjFzVyPBvPpph5zL6VCASLCxdNeiKV6foPwUpAGqRv","direction":"Left"},{"hash":"AnzSG9f91ePS6L6ii3eAkocp4iKjp6wjzSwWsDYWLnMX","direction":"Right"},{"hash":"FYVJDL4T6c87An3pdeBvntB68NzpcPtpvLP6ifjxxNkr","direction":"Left"},{"hash":"2YMF6KE8XTz7Axj3uyAoFbZisWej9Xo8mxgVtauWCZaV","direction":"Left"},{"hash":"4BHtLcxqNfWSneBdW76qsd8om8Gjg58Qw5BX8PHz93hf","direction":"Left"},{"hash":"7G3QUT7NQSHyXNQyzm8dsaYrFk5LGhYaG7aVafKAekyG","direction":"Left"},{"hash":"3XaMNnvnX69gGqBJX43Na1bSTJ4VUe7z6h5ZYJsaSZZR","direction":"Left"},{"hash":"FKu7GtfviPioyAGXGZLBVTJeG7KY5BxGwuL447oAZxiL","direction":"Right"},{"hash":"BePd7DPKUQnGtnSds5fMJGBUwHGxSNBpaNLwceJGUcJX","direction":"Left"},{"hash":"2BVKWMd9pXZTEyE9D3KL52hAWAyMrXj1NqutamyurrY1","direction":"Left"},{"hash":"EWavHKhwQiT8ApnXvybvc9bFY6aJYJWqBhcrZpubKXtA","direction":"Left"},{"hash":"83Fsd3sdx5tsJkb6maBE1yViKiqbWCCNfJ4XZRsKnRZD","direction":"Left"},{"hash":"AaT9jQmUvVpgDHdFkLR2XctaUVdTti49enmtbT5hsoyL","direction":"Left"}]}"#; - let p: RpcLightClientExecutionProofResponse = serde_json::from_str(req).unwrap(); - - let block_verified = Protocol::verify_block(&root, p.block_proof.iter(), &block_hash); - assert!(block_verified); - } } diff --git a/src/client/rpc.rs b/src/client/rpc.rs index 030b740..5e61013 100644 --- a/src/client/rpc.rs +++ b/src/client/rpc.rs @@ -1,14 +1,19 @@ +use anyhow::Result; use futures::TryFutureExt; use log::debug; use near_jsonrpc_client::{ methods::{self, light_client_proof::RpcLightClientExecutionProofResponse}, JsonRpcClient, }; -use near_primitives::{types::TransactionOrReceiptId, views::LightClientBlockView}; -use near_primitives_core::{hash::CryptoHash, types::AccountId}; +use near_primitives::views::LightClientBlockView; +use near_primitives_core::hash::CryptoHash; use serde::Deserialize; use std::fmt::{Display, Formatter}; +use super::message::GetProof; + +// TODO: retry, failover rpcs + #[derive(Debug, Clone, Deserialize, Default)] pub enum Network { Mainnet, @@ -75,6 +80,7 @@ impl NearRpcClient { let req = methods::next_light_client_block::RpcLightClientNextBlockRequest { last_block_hash: *latest_verified, }; + log::debug!("requesting next block: {:?}", req); self.client .call(&req) .or_else(|e| { @@ -82,42 +88,20 @@ impl NearRpcClient { self.archive.call(&req) }) .await + .map_err(|e| { + log::error!("Error fetching latest header: {:?}", e); + e + }) .ok()? } - pub async fn fetch_light_client_tx_proof( - &self, - transaction_id: CryptoHash, - sender_id: AccountId, - latest_verified: CryptoHash, - ) -> Option { - let req = methods::light_client_proof::RpcLightClientExecutionProofRequest { - id: TransactionOrReceiptId::Transaction { - transaction_hash: transaction_id, - sender_id, - }, - light_client_head: latest_verified, - }; - self.client - .call(&req) - .or_else(|e| { - debug!("Error hitting main rpc, falling back to archive: {:?}", e); - self.archive.call(&req) - }) - .await - .ok() - } - pub async fn fetch_light_client_receipt_proof( + pub async fn fetch_light_client_proof( &self, - receipt_id: CryptoHash, - receiver_id: AccountId, + req: GetProof, latest_verified: CryptoHash, - ) -> Option { + ) -> Result { let req = methods::light_client_proof::RpcLightClientExecutionProofRequest { - id: TransactionOrReceiptId::Receipt { - receipt_id, - receiver_id, - }, + id: req.0, light_client_head: latest_verified, }; self.client @@ -127,6 +111,6 @@ impl NearRpcClient { self.archive.call(&req) }) .await - .ok() + .map_err(|e| anyhow::format_err!("{:?}:{}", req.id, e)) } } diff --git a/src/client/store.rs b/src/client/store.rs new file mode 100644 index 0000000..f29f869 --- /dev/null +++ b/src/client/store.rs @@ -0,0 +1,291 @@ +use super::Header; +use ::sled::IVec; +use anyhow::Result; +use borsh::{BorshDeserialize, BorshSerialize}; +use near_primitives::types::validator_stake::ValidatorStake; +use near_primitives_core::hash::CryptoHash; +use tokio::sync::RwLock; + +pub struct Store(pub RwLock); + +impl Store { + pub async fn head(&self) -> Result
{ + self.get(&Collection::Headers, &head_key()) + .await + .and_then(|e| e.header()) + } + + pub async fn insert(&self, entries: &[(CryptoHash, Entity)]) -> Result<()> { + self.0.write().await.insert(entries) + } + + pub async fn get(&self, collection: &Collection, k: &CryptoHash) -> Result { + self.0.read().await.get(collection, k) + } + + pub async fn shutdown(&self) { + self.0.write().await.shutdown(); + } + + pub async fn contains(&self, collection: &Collection, k: &CryptoHash) -> Result { + self.0.read().await.contains(collection, k) + } +} + +#[derive(Debug)] +pub enum Collection { + BlockProducers, + Headers, + UsedRoots, +} + +#[derive(Debug, BorshSerialize, BorshDeserialize)] +pub enum Entity { + BlockProducers(Vec), + Header(Box
), + UsedRoot, +} + +// Maybe tryinto +impl Entity { + pub fn bps(self) -> Result> { + match self { + Entity::BlockProducers(stake) => Ok(stake), + _ => Err(anyhow::format_err!("Not a block producer")), + } + } + pub fn header(self) -> Result
{ + match self { + Entity::Header(header) => Ok(*header), + _ => Err(anyhow::format_err!("Not a header")), + } + } +} + +impl From> for Entity { + fn from(stake: Vec) -> Self { + Self::BlockProducers(stake) + } +} + +impl From
for Entity { + fn from(header: Header) -> Self { + Self::Header(Box::new(header)) + } +} + +pub trait LightClientStore { + fn insert(&mut self, entries: &[(CryptoHash, Entity)]) -> Result<()>; + fn get(&self, collection: &Collection, k: &CryptoHash) -> Result; + fn head(&self) -> Result
; + fn contains(&self, collection: &Collection, k: &CryptoHash) -> Result; + fn shutdown(&mut self); +} + +pub trait DatabaseOperations { + fn raw_insert, V: Into>( + &mut self, + inserts: Vec<(Collection, Vec<(K, V)>)>, + ) -> Result<()>; + fn raw_get, T: BorshDeserialize>( + &self, + collection: &Collection, + key: K, + ) -> Result; + fn raw_contains>(&self, collection: &Collection, key: K) -> Result; + fn shutdown(&mut self); +} + +fn encode(x: &T) -> Result> { + x.try_to_vec().map_err(|e| { + let e = anyhow::format_err!("Failed to encode: {:?}", e); + log::error!("{:?}", e); + e + }) +} + +fn decode(x: &[u8]) -> Result { + T::try_from_slice(x).map_err(|e| { + let e = anyhow::format_err!("Failed to decode: {:?}", e); + log::error!("{:?}", e); + e + }) +} + +pub fn head_key() -> CryptoHash { + CryptoHash::default() +} + +pub mod sled { + use super::*; + use ::sled::{open, transaction::TransactionError, Batch, Db, Transactional, Tree}; + use itertools::Itertools; + + pub struct Store { + db: Db, + block_producers: Tree, + headers: Tree, + used_roots: Tree, + } + + pub(crate) fn init(config: &crate::config::Config) -> Result { + log::info!("Opening store at {:?}", config.state_path); + let db = open(&config.state_path)?; + + log::debug!("Initializing block producers tree"); + let block_producers = db.open_tree("bps")?; + + log::debug!("Initializing headers tree"); + let headers = db.open_tree("archive")?; + + log::debug!("Initializing used_roots tree"); + let used_roots = db.open_tree("used_roots")?; + used_roots.set_merge_operator(increment_ref); + + Ok(Store { + db, + block_producers, + headers, + used_roots, + }) + } + + impl DatabaseOperations for Store { + fn raw_get, T: BorshDeserialize>( + &self, + collection: &Collection, + key: K, + ) -> Result { + log::debug!("Get {:?} {:?}", collection, key.as_ref()); + + match collection { + Collection::BlockProducers => self.block_producers.get(key), + Collection::Headers => self.headers.get(key), + Collection::UsedRoots => self.used_roots.get(key), + }? + .ok_or_else(|| anyhow::anyhow!("Key not found")) + .and_then(|value| decode(&value)) + } + + fn shutdown(&mut self) { + self.db.flush().unwrap(); + } + + fn raw_insert(&mut self, inserts: Vec<(Collection, Vec<(K, V)>)>) -> Result<()> + where + K: Into, + V: Into, + { + let mut used_roots_entries = vec![]; + let batches = inserts + .into_iter() + .filter_map(|(collection, entries)| { + if let Collection::UsedRoots = collection { + used_roots_entries = entries; + None + } else { + let mut b = Batch::default(); + for (k, v) in entries { + b.insert(k, v); + } + Some((collection, b)) + } + }) + .collect_vec(); + (&self.block_producers, &self.headers) + .transaction(|(bps, headers)| { + for (collection, b) in &batches { + match collection { + Collection::BlockProducers => bps.apply_batch(b)?, + Collection::Headers => headers.apply_batch(b)?, + Collection::UsedRoots => {} + }; + } + Ok(()) + }) + .map_err(|e: TransactionError| anyhow::anyhow!("{:?}", e))?; + + if !used_roots_entries.is_empty() { + for (k, v) in used_roots_entries { + self.used_roots.merge(k.into(), v.into())?; + } + } + Ok(()) + } + + fn raw_contains>(&self, collection: &Collection, key: K) -> Result { + match collection { + Collection::BlockProducers => self.block_producers.contains_key(key), + Collection::Headers => self.headers.contains_key(key), + Collection::UsedRoots => self.used_roots.contains_key(key), + } + .map_err(|e| anyhow::anyhow!("Contains: {:?}", e)) + } + } + + impl LightClientStore for Store { + fn insert(&mut self, inserts: &[(CryptoHash, Entity)]) -> Result<()> { + let inserts = inserts + .iter() + .map(|(k, v)| { + log::debug!("Insert {:?}", k); + log::trace!("Insert {:?}", v); + encode(k).and_then(|ek| { + encode(v).map(|ev| { + let collection = match v { + Entity::BlockProducers(_) => Collection::BlockProducers, + Entity::Header(_) => Collection::Headers, + Entity::UsedRoot => Collection::UsedRoots, + }; + (collection, ek, ev) + }) + }) + }) + .fold_ok(vec![], |mut acc, (collection, k, v)| { + acc.push((collection, vec![(k, v)])); + acc + })?; + self.raw_insert(inserts) + } + + fn get(&self, collection: &Collection, k: &CryptoHash) -> Result { + self.raw_get(collection, encode(k)?) + } + + fn head(&self) -> Result
{ + let head = self + .headers + .get(encode(&head_key())?)? + .ok_or_else(|| anyhow::anyhow!("Failed to get head, no head in store"))?; + let h: Entity = decode(&head)?; + h.header() + } + + fn shutdown(&mut self) { + ::shutdown(self); + } + + fn contains(&self, collection: &Collection, k: &CryptoHash) -> Result { + self.raw_contains(collection, encode(k)?) + } + } + + fn increment_ref( + key: &[u8], // the key being merged + old_ref: Option<&[u8]>, // the previous value, if one existed + _merged_bytes: &[u8], // the new bytes being merged in + ) -> Option> { + let ref_count = old_ref + .map(|ov| ov.to_vec()) + .and_then(|ov| u32::try_from_slice(&ov).ok()) + .unwrap_or_else(|| 0); + log::debug!("Incrementing ref count for {:?}, {}", key, ref_count); + (ref_count + 1).try_to_vec().ok() + } + #[cfg(test)] + mod tests { + + #[test] + fn test_name() {} + } +} diff --git a/src/config.rs b/src/config.rs index e632c04..3bd48b6 100644 --- a/src/config.rs +++ b/src/config.rs @@ -1,30 +1,40 @@ +use crate::client::rpc::Network; use config::{Config as ConfigTrait, ConfigError, Environment, File}; use serde::Deserialize; use std::{env, path::PathBuf}; -use crate::client::rpc::Network; - #[derive(Debug, Deserialize, Clone)] -#[allow(unused)] pub struct Config { - pub debug: bool, - pub state_path: Option, + #[serde(default = "default_db_path")] + pub state_path: PathBuf, pub starting_head: String, pub network: Network, + #[serde(default = "default_host")] + pub host: String, + pub catchup: bool, +} + +fn default_db_path() -> PathBuf { + "state.db".into() +} + +fn default_host() -> String { + "0.0.0.0:3000".into() } impl Config { pub fn new() -> Result { - let run_mode = env::var("NEAR_LIGHT_CLIENT_NETWORK").unwrap_or_else(|_| "Testnet".into()); + let run_mode = env::var("NEAR_LIGHT_CLIENT_NETWORK") + .unwrap_or_else(|_| "testnet".into()) + .to_lowercase(); let default_path = env::var("NEAR_LIGHT_CLIENT_CONFIG_FILE").unwrap_or_else(|_| "default".to_string()); let s = ConfigTrait::builder() - .add_source(File::with_name(&default_path)) - .add_source(File::with_name(&run_mode.to_string()).required(false)) + .add_source(File::with_name(&default_path).required(false)) + .add_source(File::with_name(&run_mode).required(false)) // This file shouldn't be checked in to git .add_source(File::with_name("local").required(false)) - // Eg.. `RELAYER_DEBUG=1 ./target/app` would set the `debug` key .add_source(Environment::with_prefix("NEAR_LIGHT_CLIENT")) .build()?; diff --git a/src/controller.rs b/src/controller.rs index 880bb0b..94f2a21 100644 --- a/src/controller.rs +++ b/src/controller.rs @@ -1,3 +1,5 @@ +use crate::{client::LightClient, config::Config}; +use anyhow::Result; use axum::{ extract::{Path, State}, http::StatusCode, @@ -5,43 +7,37 @@ use axum::{ routing::{get, post}, Router, }; -use flume::Sender; +use coerce::actor::LocalActorRef; use near_primitives_core::hash::CryptoHash; use serde::{Deserialize, Serialize}; use tokio::task::JoinHandle; -use crate::client::Message; - -type ClientState = flume::Sender; - // TODO: replace with jsonrpc -pub(crate) fn init(ctx: Sender) -> JoinHandle> { - let proof_channel = flume::bounded(64); - +pub(crate) fn init(config: &Config, ctx: LocalActorRef) -> JoinHandle> { let controller = Router::new() .route("/health", get(health_check)) .route("/head", get(header::get_head)) - .with_state((ctx.clone(), flume::bounded(64))) + .with_state(ctx.clone()) .route("/header/:epoch", get(header::get_by_epoch)) - .with_state((ctx.clone(), flume::bounded(64))) - .route( - "/proof/tx/:transaction_id/:sender_id", - get(proof::get_tx_proof), - ) - .with_state((ctx.clone(), proof_channel.clone())) - .route( - "/proof/receipt/:receipt_id/:receiver_id", - get(proof::get_receipt_proof), - ) - .with_state((ctx.clone(), proof_channel)) - .route("/proof", post(proof::post_proof)) - .with_state((ctx.clone(), flume::bounded(64))); - + .with_state(ctx.clone()) + .route("/proof", post(proof::post_get_proof)) + .with_state(ctx.clone()) + .route("/proof/verify", post(proof::post_verify_proof)) + .with_state(ctx.clone()) + .route("/proof/experimental", post(proof::post_get_batch_proof)) + .with_state(ctx.clone()); + + let host = config.host.clone(); tokio::spawn(async { - let r = axum::Server::bind(&"0.0.0.0:3000".parse().unwrap()) - .serve(controller.into_make_service()) - .await; - r.map_err(|e| todo!("{:?}", e)) + let listener = tokio::net::TcpListener::bind(host).await.map_err(|e| { + log::error!("Failed to start server: {:?}", e); + anyhow::anyhow!(e) + })?; + println!("listening on {}", listener.local_addr().unwrap()); + axum::serve(listener, controller).await.map_err(|e| { + log::error!("Failed to start server: {:?}", e); + anyhow::anyhow!(e) + }) }) } @@ -51,17 +47,7 @@ async fn health_check() -> StatusCode { mod header { use super::*; - use near_primitives::views::LightClientBlockLiteView; - - pub type HeadChannel = ( - flume::Sender, - flume::Receiver, - ); - - pub type HeaderChannel = ( - flume::Sender>, - flume::Receiver>, - ); + use crate::client::message::{Archive, Head}; #[derive(Debug, Deserialize, Serialize)] pub struct Params { @@ -69,127 +55,98 @@ mod header { } pub(super) async fn get_by_epoch( - State((client, (tx, rx))): State<(ClientState, HeaderChannel)>, + State(client): State>, Path(params): Path, ) -> impl IntoResponse { - log::debug!("get_by_epoch: {:?}", params); - if let Err(e) = client - .send_async(Message::Archive { - tx, + client + .send(Archive { epoch: params.epoch, }) .await - { - log::error!("Failed to send get_by_epoch: {:?}", e); - } - - rx.recv_async().await.map(axum::Json).map_err(|_| { - log::error!("Failed to receive result, channel closed"); - internal_server_error() - }) + .map(axum::Json) + .map_err(|_| internal_server_error()) } pub(super) async fn get_head( - State((client, (tx, rx))): State<(ClientState, HeadChannel)>, + State(client): State>, ) -> impl IntoResponse { - log::debug!("get_head"); - if let Err(e) = client.send_async(Message::Head { tx }).await { - log::error!("Failed to send get_head: {:?}", e); - } - - rx.recv_async().await.map(axum::Json).map_err(|_| { - log::error!("Failed to receive result, channel closed"); - internal_server_error() - }) + client + .send(Head) + .await + .map(axum::Json) + .map_err(ErrorMapper) + .map_err(IntoResponse::into_response) } } mod proof { use super::*; - use crate::client::{Proof, ProofType}; + use crate::client::{ + message::{BatchGetProof, GetProof, VerifyProof}, + Proof, + }; use axum::Json; - use near_primitives_core::types::AccountId; - - pub type ProofChannel = (flume::Sender>, flume::Receiver>); - pub type ValidateProofChannel = (flume::Sender, flume::Receiver); - - #[derive(Debug, Deserialize, Serialize)] - pub struct TransactionParams { - transaction_id: CryptoHash, - sender_id: AccountId, - } - #[derive(Debug, Deserialize, Serialize)] - pub struct ReceiptParams { - receipt_id: CryptoHash, - receiver_id: AccountId, - } - - pub(super) async fn get_tx_proof( - State((client, (tx, rx))): State<(ClientState, ProofChannel)>, - Path(params): Path, + pub(super) async fn post_get_proof( + State(client): State>, + Json(params): Json, ) -> impl IntoResponse { - log::debug!("get_proof: {:?}", params); - if let Err(e) = client - .send_async(Message::GetProof { - tx, - proof: ProofType::Transaction { - transaction_id: params.transaction_id, - sender_id: params.sender_id, - }, - }) + client + .send(params) .await - { - log::error!("Failed to send get_proof: {:?}", e); - } - rx.recv_async().await.map(axum::Json).map_err(|_| { - log::error!("Failed to receive result, channel closed"); - internal_server_error() - }) + .map(axum::Json) + .map_err(ErrorMapper) + .map_err(IntoResponse::into_response) } - pub(super) async fn get_receipt_proof( - State((client, (tx, rx))): State<(ClientState, ProofChannel)>, - Path(params): Path, + pub(super) async fn post_verify_proof( + State(client): State>, + Json(proof): Json, ) -> impl IntoResponse { - log::debug!("get_proof: {:?}", params); - if let Err(e) = client - .send_async(Message::GetProof { - tx, - proof: ProofType::Receipt { - receipt_id: params.receipt_id, - receiver_id: params.receiver_id, - }, - }) + client + .send(VerifyProof { proof }) .await - { - log::error!("Failed to send get_proof: {:?}", e); - } - rx.recv_async().await.map(axum::Json).map_err(|_| { - log::error!("Failed to receive result, channel closed"); - internal_server_error() - }) + .map_err(|e| anyhow::anyhow!(e)) + .and_then(|x| x) + .map(axum::Json) + .map_err(ErrorMapper) + .map_err(IntoResponse::into_response) } - pub(super) async fn post_proof( - State((client, (tx, rx))): State<(ClientState, ValidateProofChannel)>, - Json(proof): Json, + #[derive(Debug, Serialize)] + pub struct BatchProofWithErrors { + proofs: crate::client::protocol::experimental::Proof, + errors: Vec, + } + + pub(super) async fn post_get_batch_proof( + State(client): State>, + Json(body): Json, ) -> impl IntoResponse { - log::debug!("post_proof: {:?}", proof); - if let Err(e) = client - .send_async(Message::ValidateProof { - tx, - proof: Box::new(proof), - }) + client + .send(body) .await - { - log::error!("Failed to send post_proof: {:?}", e); - } + .map_err(|e| anyhow::anyhow!(e)) + .and_then(|x| x.ok_or_else(|| anyhow::anyhow!("Failed to get batch proof"))) + .map_err(ErrorMapper) + .map_err(IntoResponse::into_response) + .map(|(proofs, errors)| BatchProofWithErrors { + proofs, + errors: errors.into_iter().map(|e| e.to_string()).collect(), + }) + .map(axum::Json) + } +} - rx.recv_async().await.map(axum::Json).map_err(|_| { - log::error!("Failed to receive result, channel closed"); - internal_server_error() - }) +struct ErrorMapper(pub T); +impl IntoResponse for ErrorMapper +where + T: ToString, +{ + fn into_response(self) -> Response { + let mut r = Response::new(self.0.to_string().into()); + *r.status_mut() = StatusCode::INTERNAL_SERVER_ERROR; + r } } diff --git a/src/erasure.rs b/src/erasure.rs deleted file mode 100644 index f1df845..0000000 --- a/src/erasure.rs +++ /dev/null @@ -1,116 +0,0 @@ -use anyhow::Result; -use near_primitives::merkle::{self, MerklePath}; -use near_primitives_core::hash::CryptoHash; -use reed_solomon_novelpoly::WrappedShard; - -/// Here we provide a module which can create optimal erasure codes for a given number of expected code validators. -/// -/// We also provide merkleization functionality to generate a succint commitment to the erasure -/// coding scheme. -/// -/// Note: the current merkleization is not optimal, we can do better. -#[allow(unused)] -pub struct Erasure { - shards: Vec>, -} - -#[allow(unused)] -impl Erasure { - pub fn encodify(data: &[u8]) -> Result { - Ok(Self { - shards: reed_solomon_novelpoly::encode(data, VALIDATORS)? - .into_iter() - .map(Some) - .collect(), - }) - } - - pub fn recover(&self) -> Result> { - Ok(reed_solomon_novelpoly::reconstruct( - self.shards.clone(), - VALIDATORS, - )?) - } - - pub fn merklize(&self) -> (CryptoHash, Vec) { - let flattened_data: Vec> = self - .shards - .clone() - .into_iter() - .filter(|s| s.is_some()) - .map(|x| x.unwrap().into_inner()) - .collect(); - merkle::merklize(&flattened_data) - } -} - -#[cfg(test)] -mod tests { - use super::*; - - #[test] - fn test_encode_recover() { - let e = Erasure::<4>::encodify(b"hello").unwrap(); - - let recover = e.recover().unwrap(); - - assert!(recover.starts_with(b"hello")); - } - - #[test] - fn test_encode_recover_one_third() { - let data = b"he1lohe2lohe3lohe4lohe5lohe6lohe7lohe8lo"; - println!("original {:?}", data); - const N: usize = 8; - - let mut codewords = Erasure::::encodify(data).unwrap(); - - println!( - "codewords {:#?}({})", - codewords.shards, - codewords.shards.len() - ); - - codewords.shards[0] = None; - codewords.shards[1] = None; - codewords.shards[2] = None; - codewords.shards[N - 3] = None; - codewords.shards[N - 2] = None; - codewords.shards[N - 1] = None; - println!( - "codewords {:#?}({})", - codewords.shards, - codewords.shards.len() - ); - - let recover = codewords.recover().unwrap(); - println!("recover {:?}", recover); - } - - #[test] - fn test_root() { - let data = b"he1lohe2lohe3lohe4lohe5lohe6lohe7lohe8lo"; - println!("original {:?}", data); - const N: usize = 8; - - let codewords = Erasure::::encodify(data).unwrap(); - let (root, _path) = codewords.merklize(); - - println!("root {:?}", root); - } - - #[test] - fn test_path_len() { - let data = b"he1lohe2lohe3lohe4lohe5lohe6lohe7lohe8lo"; - println!("original {:?}", data); - const N: usize = 8; - - let codewords = Erasure::::encodify(data).unwrap(); - let (_root, path) = codewords.merklize(); - - println!("path {:#?}", path); - let size = std::mem::size_of_val(&*path); - println!("proof size {}", size); - assert_eq!(size, 192); - } -} diff --git a/src/main.rs b/src/main.rs index 6505b3f..56697f3 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,13 +1,9 @@ -#![feature(slice_partition_dedup)] -#![feature(trait_alias)] - -use crate::client::LightClient; -use client::Message; +use crate::client::{message::Shutdown, LightClient}; +use coerce::actor::{system::ActorSystem, IntoActor}; mod client; mod config; mod controller; -mod erasure; pub struct ShutdownMsg; @@ -16,16 +12,19 @@ async fn main() -> anyhow::Result<()> { pretty_env_logger::init(); let config = config::Config::new()?; + let system = ActorSystem::builder() + .system_name("near-light-client") + .build(); - let (ctx, crx) = flume::bounded::(256); - - LightClient::init(&config).await?.start(true, crx); - let webapi = controller::init(ctx.clone()); + let client_actor = LightClient::new(&config)? + .into_actor(Some("light-client"), &system) + .await?; + let webapi = controller::init(&config, client_actor.clone()); if tokio::signal::ctrl_c().await.is_ok() { - log::info!("Shutting down due to ctrlc"); - let _ = ctx.send(Message::Shutdown(config.state_path)); + log::info!("Shutting down.."); webapi.abort(); + client_actor.notify(Shutdown)?; } Ok(()) diff --git a/testnet.toml b/testnet.toml index 1ce246f..c22ad3a 100644 --- a/testnet.toml +++ b/testnet.toml @@ -1,4 +1,5 @@ -debug = false +catchup = true +host = "0.0.0.0:3030" network = "Testnet" -starting_head = "HaWkNoZujD1t9ftJAuZH1Wy2xnXrht8qzENZSDrka3y5" -state_path = "db.sled" +starting_head = "4bM5eXMDGxpFZXbWNT6TqX1HdZsWoHZ11KerCHJ8RKmU" +state_path = "state.db"