Ver Fonte

add svm/wormhole-core-shims (#4311)

* svm: wormhole-post-message-shim

* svm: compare shim costs

* svm: save 1994 CU by nixing CpiContext

* svm: wormhole-vaa-verification-example existing core post vaa method

* svm: add consume_vaas

* svm: wormhole-verify-vaa-shim

* svm: move guardian_set_index into guardian signatures account

* svm: wormhole-integrator-example

* svm: rename verification-example to comparison

* svm: comments

* svm: readme

* svm: rename

* svm: initial e2e test

* svm: add anchor interfaces

* svm: add wormhole-svm-definitions

svm: add wormhole-svm-shim

svm: add rust tests for post message

svm: update js tests for post message

* svm: remove anchor from post message

* simplify post message program

update post message instruction builder

* svm: birdie

* svm: update post message interface

* svm: clean up

* svm: add post signatures to wormhole-svm-shim

svm: reorganize wormhole-svm-definitions

svm: reorganize wormhole-svm-shim

add rust tests for post signatures

* svm: remove anchor from post signatures

svm: remove errors from verify vaa interface

* svm: add close signatures to wormhole-svm-shim

svm: add rust tests for close signatures

svm: remove anchor from close signatures

svm: clean up post signatures

* svm: add verify vaa to wormhole-svm-shim

svm: add rust tests for verify vaa

svm: add guardian set account reader

svm: fix CUs in tests

* svm: rename verify vaa ix to verify hash

svm: clean up anchor interfaces

* svm: add guardian set bump instruction arg

svm: use seed consts

* svm: remove anchor completely from verify vaa

* svm: birdie

svm: rename left/right to actual/expected

svm: fix Makefile

* svm: solana 2.0

svm: update Cargo.lock

svm: clippy

* svm: move anchor event selector to defs

* svm: rename to DataDiscriminator

* svm: birdie

* svm: clippy

* svm: fix create account logic

* svm: add err msg for guardian signatures

* svm: remove rent sysvar

* svm: fix index comments

update Makefile

* svm: add missing signer checks

* svm: fix anchor interfaces

svm: clean up docs

* svm: fix make build

solana: fix test artifacts

* svm: add missing owner check in verify hash

svm: remove realloc and assign in close signatures

* svm: remove unnecessary enum

* svm: fix create account logic

* svm: safer?

* svm: update READMEs

* svm: fix spelling

---------

Co-authored-by: Evan Gray <battledingo@gmail.com>
Co-authored-by: Paul Noel <panoel007@gmail.com>
Co-authored-by: A5 Pickle <a5-pickle@users.noreply.github.com>
A5 Pickle há 7 meses atrás
pai
commit
9b9c07b4b1
95 ficheiros alterados com 22832 adições e 1 exclusões
  1. 4 1
      Tiltfile
  2. 2 0
      cspell-custom-words.txt
  3. 6 0
      devnet/solana-devnet.yaml
  4. 2 0
      solana/Dockerfile.test-validator
  5. BIN
      solana/tests/artifacts/wormhole_integrator_example.so
  6. BIN
      solana/tests/artifacts/wormhole_post_message_shim.so
  7. 7 0
      svm/wormhole-core-shims/.gitignore
  8. 6959 0
      svm/wormhole-core-shims/Cargo.lock
  9. 34 0
      svm/wormhole-core-shims/Cargo.toml
  10. 63 0
      svm/wormhole-core-shims/Makefile
  11. 56 0
      svm/wormhole-core-shims/README.md
  12. 7 0
      svm/wormhole-core-shims/anchor/.prettierignore
  13. 81 0
      svm/wormhole-core-shims/anchor/Anchor.toml
  14. 2767 0
      svm/wormhole-core-shims/anchor/Cargo.lock
  15. 24 0
      svm/wormhole-core-shims/anchor/Cargo.toml
  16. 50 0
      svm/wormhole-core-shims/anchor/Makefile
  17. 23 0
      svm/wormhole-core-shims/anchor/README.md
  18. 314 0
      svm/wormhole-core-shims/anchor/idls/wormhole_post_message_shim.json
  19. 321 0
      svm/wormhole-core-shims/anchor/idls/wormhole_post_message_shim.ts
  20. 257 0
      svm/wormhole-core-shims/anchor/idls/wormhole_verify_vaa_shim.json
  21. 264 0
      svm/wormhole-core-shims/anchor/idls/wormhole_verify_vaa_shim.ts
  22. 21 0
      svm/wormhole-core-shims/anchor/interfaces/wormhole-post-message-shim/Cargo.toml
  23. 9 0
      svm/wormhole-core-shims/anchor/interfaces/wormhole-post-message-shim/README.md
  24. 133 0
      svm/wormhole-core-shims/anchor/interfaces/wormhole-post-message-shim/src/lib.rs
  25. 22 0
      svm/wormhole-core-shims/anchor/interfaces/wormhole-verify-vaa-shim/Cargo.toml
  26. 9 0
      svm/wormhole-core-shims/anchor/interfaces/wormhole-verify-vaa-shim/README.md
  27. 136 0
      svm/wormhole-core-shims/anchor/interfaces/wormhole-verify-vaa-shim/src/lib.rs
  28. 23 0
      svm/wormhole-core-shims/anchor/package.json
  29. 26 0
      svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/Cargo.toml
  30. 2 0
      svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/Xargo.toml
  31. 45 0
      svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/instructions/consume_vaa.rs
  32. 94 0
      svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/instructions/initialize.rs
  33. 8 0
      svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/instructions/mod.rs
  34. 103 0
      svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/instructions/post_message.rs
  35. 37 0
      svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/lib.rs
  36. 28 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/Cargo.toml
  37. 2 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/Xargo.toml
  38. 28 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/error.rs
  39. 27 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/consume_core_posted_vaa.rs
  40. 141 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/consume_vaa.rs
  41. 42 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/consume_vaa_via_shim.rs
  42. 11 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/mod.rs
  43. 70 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/post_signatures.rs
  44. 46 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/lib.rs
  45. 52 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/state/ext/mod.rs
  46. 24 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/state/guardian_signatures.rs
  47. 5 0
      svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/state/mod.rs
  48. 3 0
      svm/wormhole-core-shims/anchor/rust-toolchain.toml
  49. 13 0
      svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_devnet/bridge_config.json
  50. 13 0
      svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_devnet/fee_collector.json
  51. 13 0
      svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_devnet/guardian_set.json
  52. 11 0
      svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_mainnet/config.json
  53. 14 0
      svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_mainnet/fee_collector.json
  54. 14 0
      svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_mainnet/guardian_set_4.json
  55. BIN
      svm/wormhole-core-shims/anchor/tests/artifacts/devnet_core_bridge.so
  56. BIN
      svm/wormhole-core-shims/anchor/tests/artifacts/mainnet_core_bridge.so
  57. 253 0
      svm/wormhole-core-shims/anchor/tests/e2e.ts
  58. 61 0
      svm/wormhole-core-shims/anchor/tests/helpers.ts
  59. 407 0
      svm/wormhole-core-shims/anchor/tests/idls/devnet/wormhole_integrator_example.json
  60. 414 0
      svm/wormhole-core-shims/anchor/tests/idls/devnet/wormhole_integrator_example.ts
  61. 155 0
      svm/wormhole-core-shims/anchor/tests/wormhole-integrator-example.ts
  62. 168 0
      svm/wormhole-core-shims/anchor/tests/wormhole-post-message-shim.ts
  63. 12 0
      svm/wormhole-core-shims/anchor/tests/wormhole-vaa-verification-comparison.ts
  64. 68 0
      svm/wormhole-core-shims/anchor/tests/wormhole-verify-vaa-shim.ts
  65. 11 0
      svm/wormhole-core-shims/anchor/tsconfig.json
  66. 3951 0
      svm/wormhole-core-shims/anchor/yarn.lock
  67. 32 0
      svm/wormhole-core-shims/crates/definitions/Cargo.toml
  68. 34 0
      svm/wormhole-core-shims/crates/definitions/README.md
  69. 54 0
      svm/wormhole-core-shims/crates/definitions/src/borsh.rs
  70. 170 0
      svm/wormhole-core-shims/crates/definitions/src/lib.rs
  71. 97 0
      svm/wormhole-core-shims/crates/definitions/src/solana.rs
  72. 398 0
      svm/wormhole-core-shims/crates/definitions/src/zero_copy/guardian_set.rs
  73. 106 0
      svm/wormhole-core-shims/crates/definitions/src/zero_copy/guardian_signatures.rs
  74. 5 0
      svm/wormhole-core-shims/crates/definitions/src/zero_copy/mod.rs
  75. 18 0
      svm/wormhole-core-shims/crates/shim/Cargo.toml
  76. 4 0
      svm/wormhole-core-shims/crates/shim/README.md
  77. 5 0
      svm/wormhole-core-shims/crates/shim/src/lib.rs
  78. 281 0
      svm/wormhole-core-shims/crates/shim/src/post_message.rs
  79. 41 0
      svm/wormhole-core-shims/crates/shim/src/verify_vaa/close_signatures.rs
  80. 87 0
      svm/wormhole-core-shims/crates/shim/src/verify_vaa/mod.rs
  81. 166 0
      svm/wormhole-core-shims/crates/shim/src/verify_vaa/post_signatures.rs
  82. 120 0
      svm/wormhole-core-shims/crates/shim/src/verify_vaa/verify_hash.rs
  83. 33 0
      svm/wormhole-core-shims/programs/post-message/Cargo.toml
  84. 274 0
      svm/wormhole-core-shims/programs/post-message/README.md
  85. 272 0
      svm/wormhole-core-shims/programs/post-message/src/lib.rs
  86. 303 0
      svm/wormhole-core-shims/programs/post-message/tests/integration_test.rs
  87. 36 0
      svm/wormhole-core-shims/programs/verify-vaa/Cargo.toml
  88. 261 0
      svm/wormhole-core-shims/programs/verify-vaa/README.md
  89. 497 0
      svm/wormhole-core-shims/programs/verify-vaa/src/lib.rs
  90. 75 0
      svm/wormhole-core-shims/programs/verify-vaa/tests/common/close_signatures.rs
  91. 178 0
      svm/wormhole-core-shims/programs/verify-vaa/tests/common/mod.rs
  92. 85 0
      svm/wormhole-core-shims/programs/verify-vaa/tests/common/post_signatures.rs
  93. 78 0
      svm/wormhole-core-shims/programs/verify-vaa/tests/common/verify_hash.rs
  94. 1153 0
      svm/wormhole-core-shims/programs/verify-vaa/tests/integration_test.rs
  95. 3 0
      svm/wormhole-core-shims/rust-toolchain.toml

+ 4 - 1
Tiltfile

@@ -253,6 +253,8 @@ def build_node_yaml():
                     "http://solana-devnet:8899",
                     "--solanaContract",
                     "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o",
+                    "--solanaShimContract",
+                    "EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX",
                 ]
 
             if pythnet:
@@ -472,7 +474,7 @@ if solana or pythnet:
         only = ["./proto", "./solana", "./clients"],
         dockerfile = "solana/Dockerfile.client",
         # Ignore target folders from local (non-container) development.
-        ignore = ["./solana/*/target"],
+        ignore = ["./solana/*/target", "./solana/tests"],
     )
 
     # solana smart contract
@@ -482,6 +484,7 @@ if solana or pythnet:
         context = "solana",
         dockerfile = "solana/Dockerfile",
         target = "builder",
+        ignore = ["./solana/*/target", "./solana/tests"],
         build_args = {"BRIDGE_ADDRESS": "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o"}
     )
 

+ 2 - 0
cspell-custom-words.txt

@@ -71,6 +71,7 @@ holesky
 Holesky
 horcrux
 ICCO
+idls
 incentivized
 incentivizing
 initialisation
@@ -102,6 +103,7 @@ lamports
 lastrun
 libp
 Linea
+localnet
 localterra
 lockfiles
 merkle

+ 6 - 0
devnet/solana-devnet.yaml

@@ -56,6 +56,12 @@ spec:
             - --bpf-program
             - Ex9bCdVMSfx7EzB3pgSi2R4UHwJAXvTw18rBQm5YQ8gK
             - /opt/solana/deps/wormhole_migration.so
+            - --bpf-program
+            - EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX
+            - /opt/solana/deps/wormhole_post_message_shim.so
+            - --bpf-program
+            - AEwubmehHNvkMXoH2C5MgDSemZgQ3HUSYpeaF3UrNZdQ
+            - /opt/solana/deps/wormhole_integrator_example.so
             # - --log
             - -q
           ports:

+ 2 - 0
solana/Dockerfile.test-validator

@@ -1,2 +1,4 @@
 FROM ghcr.io/wormholelabs-xyz/solana-test-validator:1.17.29@sha256:b1f85eed2d33a2bd0378204ab4d1e16537de35407cdcfeedbd021b31636618bc
+COPY tests/artifacts/wormhole_post_message_shim.so /opt/solana/deps/wormhole_post_message_shim.so
+COPY tests/artifacts/wormhole_integrator_example.so /opt/solana/deps/wormhole_integrator_example.so
 COPY --from=solana-contract /opt/solana/deps/ /opt/solana/deps/

BIN
solana/tests/artifacts/wormhole_integrator_example.so


BIN
solana/tests/artifacts/wormhole_post_message_shim.so


+ 7 - 0
svm/wormhole-core-shims/.gitignore

@@ -0,0 +1,7 @@
+.*
+artifacts-*
+fixtures
+target
+**/*.rs.bk
+node_modules
+test-ledger

+ 6959 - 0
svm/wormhole-core-shims/Cargo.lock

@@ -0,0 +1,6959 @@
+# This file is automatically @generated by Cargo.
+# It is not intended for manual editing.
+version = 3
+
+[[package]]
+name = "Inflector"
+version = "0.11.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fe438c63458706e03479442743baae6c88256498e6431708f6dfc520a26515d3"
+dependencies = [
+ "lazy_static",
+ "regex",
+]
+
+[[package]]
+name = "addr2line"
+version = "0.24.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dfbe277e56a376000877090da837660b4427aad530e3028d44e0bffe4f89a1c1"
+dependencies = [
+ "gimli",
+]
+
+[[package]]
+name = "adler2"
+version = "2.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627"
+
+[[package]]
+name = "aead"
+version = "0.5.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d122413f284cf2d62fb1b7db97e02edb8cda96d769b16e443a4f6195e35662b0"
+dependencies = [
+ "crypto-common",
+ "generic-array",
+]
+
+[[package]]
+name = "aes"
+version = "0.8.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0"
+dependencies = [
+ "cfg-if",
+ "cipher",
+ "cpufeatures",
+]
+
+[[package]]
+name = "aes-gcm-siv"
+version = "0.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ae0784134ba9375416d469ec31e7c5f9fa94405049cf08c5ce5b4698be673e0d"
+dependencies = [
+ "aead",
+ "aes",
+ "cipher",
+ "ctr",
+ "polyval",
+ "subtle",
+ "zeroize",
+]
+
+[[package]]
+name = "agave-transaction-view"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "55e9798cd7db3f5e634edb0e43ad53aa6b39acfcfdfe037a9eed033f82cc131e"
+dependencies = [
+ "solana-sdk",
+ "solana-svm-transaction",
+]
+
+[[package]]
+name = "ahash"
+version = "0.8.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e89da841a80418a9b391ebaea17f5c112ffaaa96f621d2c285b5174da76b9011"
+dependencies = [
+ "cfg-if",
+ "getrandom 0.2.15",
+ "once_cell",
+ "version_check",
+ "zerocopy",
+]
+
+[[package]]
+name = "aho-corasick"
+version = "1.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "alloc-no-stdlib"
+version = "2.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cc7bb162ec39d46ab1ca8c77bf72e890535becd1751bb45f64c597edb4c8c6b3"
+
+[[package]]
+name = "alloc-stdlib"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94fb8275041c72129eb51b7d0322c29b8387a0386127718b096429201a5d6ece"
+dependencies = [
+ "alloc-no-stdlib",
+]
+
+[[package]]
+name = "android-tzdata"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e999941b234f3131b00bc13c22d06e8c5ff726d1b6318ac7eb276997bbb4fef0"
+
+[[package]]
+name = "android_system_properties"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "819e7219dbd41043ac279b19830f2efc897156490d7fd6ea916720117ee66311"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "anyhow"
+version = "1.0.95"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "34ac096ce696dc2fcabef30516bb13c0a68a11d30131d3df6f04711467681b04"
+
+[[package]]
+name = "aquamarine"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d1da02abba9f9063d786eab1509833ebb2fac0f966862ca59439c76b9c566760"
+dependencies = [
+ "include_dir",
+ "itertools 0.10.5",
+ "proc-macro-error",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "ark-bn254"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a22f4561524cd949590d78d7d4c5df8f592430d221f7f3c9497bbafd8972120f"
+dependencies = [
+ "ark-ec",
+ "ark-ff",
+ "ark-std",
+]
+
+[[package]]
+name = "ark-ec"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "defd9a439d56ac24968cca0571f598a61bc8c55f71d50a89cda591cb750670ba"
+dependencies = [
+ "ark-ff",
+ "ark-poly",
+ "ark-serialize",
+ "ark-std",
+ "derivative",
+ "hashbrown 0.13.2",
+ "itertools 0.10.5",
+ "num-traits",
+ "zeroize",
+]
+
+[[package]]
+name = "ark-ff"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ec847af850f44ad29048935519032c33da8aa03340876d351dfab5660d2966ba"
+dependencies = [
+ "ark-ff-asm",
+ "ark-ff-macros",
+ "ark-serialize",
+ "ark-std",
+ "derivative",
+ "digest 0.10.7",
+ "itertools 0.10.5",
+ "num-bigint 0.4.6",
+ "num-traits",
+ "paste",
+ "rustc_version",
+ "zeroize",
+]
+
+[[package]]
+name = "ark-ff-asm"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3ed4aa4fe255d0bc6d79373f7e31d2ea147bcf486cba1be5ba7ea85abdb92348"
+dependencies = [
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "ark-ff-macros"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7abe79b0e4288889c4574159ab790824d0033b9fdcb2a112a3182fac2e514565"
+dependencies = [
+ "num-bigint 0.4.6",
+ "num-traits",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "ark-poly"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d320bfc44ee185d899ccbadfa8bc31aab923ce1558716e1997a1e74057fe86bf"
+dependencies = [
+ "ark-ff",
+ "ark-serialize",
+ "ark-std",
+ "derivative",
+ "hashbrown 0.13.2",
+]
+
+[[package]]
+name = "ark-serialize"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "adb7b85a02b83d2f22f89bd5cac66c9c89474240cb6207cb1efc16d098e822a5"
+dependencies = [
+ "ark-serialize-derive",
+ "ark-std",
+ "digest 0.10.7",
+ "num-bigint 0.4.6",
+]
+
+[[package]]
+name = "ark-serialize-derive"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ae3281bc6d0fd7e549af32b52511e1302185bd688fd3359fa36423346ff682ea"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "ark-std"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94893f1e0c6eeab764ade8dc4c0db24caf4fe7cbbaafc0eba0a9030f447b5185"
+dependencies = [
+ "num-traits",
+ "rand 0.8.5",
+]
+
+[[package]]
+name = "arrayref"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "76a2e8124351fda1ef8aaaa3bbd7ebbcb486bbcd4225aca0aa0d84bb2db8fecb"
+
+[[package]]
+name = "arrayvec"
+version = "0.7.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50"
+
+[[package]]
+name = "ascii"
+version = "0.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "eab1c04a571841102f5345a8fc0f6bb3d31c315dec879b5c6e42e40ce7ffa34e"
+
+[[package]]
+name = "asn1-rs"
+version = "0.5.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7f6fd5ddaf0351dff5b8da21b2fb4ff8e08ddd02857f0bf69c47639106c0fff0"
+dependencies = [
+ "asn1-rs-derive",
+ "asn1-rs-impl",
+ "displaydoc",
+ "nom",
+ "num-traits",
+ "rusticata-macros",
+ "thiserror 1.0.69",
+ "time",
+]
+
+[[package]]
+name = "asn1-rs-derive"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "726535892e8eae7e70657b4c8ea93d26b8553afb1ce617caee529ef96d7dee6c"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+ "synstructure 0.12.6",
+]
+
+[[package]]
+name = "asn1-rs-impl"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2777730b2039ac0f95f093556e61b6d26cebed5393ca6f152717777cec3a42ed"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "assert_matches"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b34d609dfbaf33d6889b2b7106d3ca345eacad44200913df5ba02bfd31d2ba9"
+
+[[package]]
+name = "async-channel"
+version = "1.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "81953c529336010edd6d8e358f886d9581267795c61b19475b71314bffa46d35"
+dependencies = [
+ "concurrent-queue",
+ "event-listener 2.5.3",
+ "futures-core",
+]
+
+[[package]]
+name = "async-compression"
+version = "0.4.18"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "df895a515f70646414f4b45c0b79082783b80552b373a68283012928df56f522"
+dependencies = [
+ "brotli",
+ "flate2",
+ "futures-core",
+ "memchr",
+ "pin-project-lite",
+ "tokio",
+]
+
+[[package]]
+name = "async-lock"
+version = "3.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ff6e472cdea888a4bd64f342f09b3f50e1886d32afe8df3d663c01140b811b18"
+dependencies = [
+ "event-listener 5.4.0",
+ "event-listener-strategy",
+ "pin-project-lite",
+]
+
+[[package]]
+name = "async-trait"
+version = "0.1.86"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "644dd749086bf3771a2fbc5f256fdb982d53f011c7d5d560304eafeecebce79d"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "atty"
+version = "0.2.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8"
+dependencies = [
+ "hermit-abi 0.1.19",
+ "libc",
+ "winapi",
+]
+
+[[package]]
+name = "autocfg"
+version = "1.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ace50bade8e6234aa140d9a2f552bbee1db4d353f69b8217bc503490fc1a9f26"
+
+[[package]]
+name = "backtrace"
+version = "0.3.74"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8d82cb332cdfaed17ae235a638438ac4d4839913cc2af585c3c6746e8f8bee1a"
+dependencies = [
+ "addr2line",
+ "cfg-if",
+ "libc",
+ "miniz_oxide",
+ "object",
+ "rustc-demangle",
+ "windows-targets 0.52.6",
+]
+
+[[package]]
+name = "base64"
+version = "0.12.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3441f0f7b02788e948e47f457ca01f1d7e6d92c693bc132c22b087d3141c03ff"
+
+[[package]]
+name = "base64"
+version = "0.13.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8"
+
+[[package]]
+name = "base64"
+version = "0.21.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567"
+
+[[package]]
+name = "base64"
+version = "0.22.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6"
+
+[[package]]
+name = "bincode"
+version = "1.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b1f45e9417d87227c7a56d22e471c6206462cba514c7590c09aff4cf6d1ddcad"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "bitflags"
+version = "1.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a"
+
+[[package]]
+name = "bitflags"
+version = "2.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8f68f53c83ab957f72c32642f3868eec03eb974d1fb82e453128456482613d36"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "bitmaps"
+version = "2.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "031043d04099746d8db04daf1fa424b2bc8bd69d92b25962dcde24da39ab64a2"
+dependencies = [
+ "typenum",
+]
+
+[[package]]
+name = "blake3"
+version = "1.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b8ee0c1824c4dea5b5f81736aff91bae041d2c07ee1192bec91054e10e3e601e"
+dependencies = [
+ "arrayref",
+ "arrayvec",
+ "cc",
+ "cfg-if",
+ "constant_time_eq",
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "block-buffer"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "block-buffer"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "borsh"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "115e54d64eb62cdebad391c19efc9dce4981c690c85a33a12199d99bb9546fee"
+dependencies = [
+ "borsh-derive 0.10.4",
+ "hashbrown 0.13.2",
+]
+
+[[package]]
+name = "borsh"
+version = "1.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5430e3be710b68d984d1391c854eb431a9d548640711faa54eecb1df93db91cc"
+dependencies = [
+ "borsh-derive 1.5.5",
+ "cfg_aliases",
+]
+
+[[package]]
+name = "borsh-derive"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "831213f80d9423998dd696e2c5345aba6be7a0bd8cd19e31c5243e13df1cef89"
+dependencies = [
+ "borsh-derive-internal",
+ "borsh-schema-derive-internal",
+ "proc-macro-crate 0.1.5",
+ "proc-macro2",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "borsh-derive"
+version = "1.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f8b668d39970baad5356d7c83a86fee3a539e6f93bf6764c97368243e17a0487"
+dependencies = [
+ "once_cell",
+ "proc-macro-crate 3.2.0",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "borsh-derive-internal"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "65d6ba50644c98714aa2a70d13d7df3cd75cd2b523a2b452bf010443800976b3"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "borsh-schema-derive-internal"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "276691d96f063427be83e6692b86148e488ebba9f48f77788724ca027ba3b6d4"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "brotli"
+version = "7.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cc97b8f16f944bba54f0433f07e30be199b6dc2bd25937444bbad560bcea29bd"
+dependencies = [
+ "alloc-no-stdlib",
+ "alloc-stdlib",
+ "brotli-decompressor",
+]
+
+[[package]]
+name = "brotli-decompressor"
+version = "4.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "74fa05ad7d803d413eb8380983b092cbbaf9a85f151b871360e7b00cd7060b37"
+dependencies = [
+ "alloc-no-stdlib",
+ "alloc-stdlib",
+]
+
+[[package]]
+name = "bs58"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bf88ba1141d185c399bee5288d850d63b8369520c1eafc32a0430b5b6c287bf4"
+dependencies = [
+ "tinyvec",
+]
+
+[[package]]
+name = "bumpalo"
+version = "3.17.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1628fb46dfa0b37568d12e5edd512553eccf6a22a78e8bde00bb4aed84d5bdbf"
+
+[[package]]
+name = "bv"
+version = "0.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8834bb1d8ee5dc048ee3124f2c7c1afcc6bc9aed03f11e9dfd8c69470a5db340"
+dependencies = [
+ "feature-probe",
+ "serde",
+]
+
+[[package]]
+name = "bytemuck"
+version = "1.21.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ef657dfab802224e671f5818e9a4935f9b1957ed18e58292690cc39e7a4092a3"
+dependencies = [
+ "bytemuck_derive",
+]
+
+[[package]]
+name = "bytemuck_derive"
+version = "1.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3fa76293b4f7bb636ab88fd78228235b5248b4d05cc589aed610f954af5d7c7a"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "byteorder"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
+
+[[package]]
+name = "bytes"
+version = "1.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f61dac84819c6588b558454b194026eb1f09c293b9036ae9b159e74e73ab6cf9"
+
+[[package]]
+name = "bzip2"
+version = "0.4.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bdb116a6ef3f6c3698828873ad02c3014b3c85cadb88496095628e3ef1e347f8"
+dependencies = [
+ "bzip2-sys",
+ "libc",
+]
+
+[[package]]
+name = "bzip2-sys"
+version = "0.1.11+1.0.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "736a955f3fa7875102d57c82b8cac37ec45224a07fd32d58f9f7a186b6cd4cdc"
+dependencies = [
+ "cc",
+ "libc",
+ "pkg-config",
+]
+
+[[package]]
+name = "caps"
+version = "0.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "190baaad529bcfbde9e1a19022c42781bdb6ff9de25721abdb8fd98c0807730b"
+dependencies = [
+ "libc",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "cc"
+version = "1.2.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e4730490333d58093109dc02c23174c3f4d490998c3fed3cc8e82d57afedb9cf"
+dependencies = [
+ "jobserver",
+ "libc",
+ "shlex",
+]
+
+[[package]]
+name = "cesu8"
+version = "1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6d43a04d8753f35258c91f8ec639f792891f748a1edbd759cf1dcea3382ad83c"
+
+[[package]]
+name = "cfg-if"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
+
+[[package]]
+name = "cfg_aliases"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724"
+
+[[package]]
+name = "cfg_eval"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "45565fc9416b9896014f5732ac776f810ee53a66730c17e4020c3ec064a8f88f"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "chrono"
+version = "0.4.39"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7e36cc9d416881d2e24f9a963be5fb1cd90966419ac844274161d10488b3e825"
+dependencies = [
+ "android-tzdata",
+ "iana-time-zone",
+ "js-sys",
+ "num-traits",
+ "serde",
+ "wasm-bindgen",
+ "windows-targets 0.52.6",
+]
+
+[[package]]
+name = "chrono-humanize"
+version = "0.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "799627e6b4d27827a814e837b9d8a504832086081806d45b1afa34dc982b023b"
+dependencies = [
+ "chrono",
+]
+
+[[package]]
+name = "cipher"
+version = "0.4.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad"
+dependencies = [
+ "crypto-common",
+ "inout",
+]
+
+[[package]]
+name = "combine"
+version = "3.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "da3da6baa321ec19e1cc41d31bf599f00c783d0517095cdaf0332e3fe8d20680"
+dependencies = [
+ "ascii",
+ "byteorder",
+ "either",
+ "memchr",
+ "unreachable",
+]
+
+[[package]]
+name = "combine"
+version = "4.6.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ba5a308b75df32fe02788e748662718f03fde005016435c444eea572398219fd"
+dependencies = [
+ "bytes",
+ "memchr",
+]
+
+[[package]]
+name = "concurrent-queue"
+version = "2.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4ca0197aee26d1ae37445ee532fefce43251d24cc7c166799f4d46817f1d3973"
+dependencies = [
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "console"
+version = "0.15.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ea3c6ecd8059b57859df5c69830340ed3c41d30e3da0c1cbed90a96ac853041b"
+dependencies = [
+ "encode_unicode",
+ "libc",
+ "once_cell",
+ "unicode-width",
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "console_error_panic_hook"
+version = "0.1.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a06aeb73f470f66dcdbf7223caeebb85984942f22f1adb2a088cf9668146bbbc"
+dependencies = [
+ "cfg-if",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "console_log"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e89f72f65e8501878b8a004d5a1afb780987e2ce2b4532c562e367a72c57499f"
+dependencies = [
+ "log",
+ "web-sys",
+]
+
+[[package]]
+name = "constant_time_eq"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7c74b8349d32d297c9134b8c88677813a227df8f779daa29bfc29c183fe3dca6"
+
+[[package]]
+name = "core-foundation"
+version = "0.9.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f"
+dependencies = [
+ "core-foundation-sys",
+ "libc",
+]
+
+[[package]]
+name = "core-foundation-sys"
+version = "0.8.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "773648b94d0e5d620f64f280777445740e61fe701025087ec8b57f45c791888b"
+
+[[package]]
+name = "cpufeatures"
+version = "0.2.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "crc32fast"
+version = "1.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a97769d94ddab943e4510d138150169a2758b5ef3eb191a9ee688de3e23ef7b3"
+dependencies = [
+ "cfg-if",
+]
+
+[[package]]
+name = "crossbeam-channel"
+version = "0.5.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "06ba6d68e24814cb8de6bb986db8222d3a027d15872cabc0d18817bc3c0e4471"
+dependencies = [
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "crossbeam-deque"
+version = "0.8.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9dd111b7b7f7d55b72c0a6ae361660ee5853c9af73f70c3c2ef6858b950e2e51"
+dependencies = [
+ "crossbeam-epoch",
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "crossbeam-epoch"
+version = "0.9.18"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e"
+dependencies = [
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "crossbeam-utils"
+version = "0.8.21"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28"
+
+[[package]]
+name = "crunchy"
+version = "0.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "43da5946c66ffcc7745f48db692ffbb10a83bfe0afd96235c5c2a4fb23994929"
+
+[[package]]
+name = "crypto-common"
+version = "0.1.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3"
+dependencies = [
+ "generic-array",
+ "rand_core 0.6.4",
+ "typenum",
+]
+
+[[package]]
+name = "crypto-mac"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b584a330336237c1eecd3e94266efb216c56ed91225d634cb2991c5f3fd1aeab"
+dependencies = [
+ "generic-array",
+ "subtle",
+]
+
+[[package]]
+name = "ctr"
+version = "0.9.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0369ee1ad671834580515889b80f2ea915f23b8be8d0daa4bbaf2ac5c7590835"
+dependencies = [
+ "cipher",
+]
+
+[[package]]
+name = "curve25519-dalek"
+version = "3.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b9fdf9972b2bd6af2d913799d9ebc165ea4d2e65878e329d9c6b372c4491b61"
+dependencies = [
+ "byteorder",
+ "digest 0.9.0",
+ "rand_core 0.5.1",
+ "subtle",
+ "zeroize",
+]
+
+[[package]]
+name = "curve25519-dalek"
+version = "4.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "97fb8b7c4503de7d6ae7b42ab72a5a59857b4c937ec27a3d4539dba95b5ab2be"
+dependencies = [
+ "cfg-if",
+ "cpufeatures",
+ "curve25519-dalek-derive",
+ "digest 0.10.7",
+ "fiat-crypto",
+ "rand_core 0.6.4",
+ "rustc_version",
+ "serde",
+ "subtle",
+ "zeroize",
+]
+
+[[package]]
+name = "curve25519-dalek-derive"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f46882e17999c6cc590af592290432be3bce0428cb0d5f8b6715e4dc7b383eb3"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "darling"
+version = "0.20.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6f63b86c8a8826a49b8c21f08a2d07338eec8d900540f8630dc76284be802989"
+dependencies = [
+ "darling_core",
+ "darling_macro",
+]
+
+[[package]]
+name = "darling_core"
+version = "0.20.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "95133861a8032aaea082871032f5815eb9e98cef03fa916ab4500513994df9e5"
+dependencies = [
+ "fnv",
+ "ident_case",
+ "proc-macro2",
+ "quote",
+ "strsim",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "darling_macro"
+version = "0.20.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d336a2a514f6ccccaa3e09b02d41d35330c07ddf03a62165fcec10bb561c7806"
+dependencies = [
+ "darling_core",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "dashmap"
+version = "5.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "978747c1d849a7d2ee5e8adc0159961c48fb7e5db2f06af6723b80123bb53856"
+dependencies = [
+ "cfg-if",
+ "hashbrown 0.14.5",
+ "lock_api",
+ "once_cell",
+ "parking_lot_core",
+ "rayon",
+]
+
+[[package]]
+name = "data-encoding"
+version = "2.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0e60eed09d8c01d3cee5b7d30acb059b76614c918fa0f992e0dd6eeb10daad6f"
+
+[[package]]
+name = "der-parser"
+version = "8.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dbd676fbbab537128ef0278adb5576cf363cff6aa22a7b24effe97347cfab61e"
+dependencies = [
+ "asn1-rs",
+ "displaydoc",
+ "nom",
+ "num-bigint 0.4.6",
+ "num-traits",
+ "rusticata-macros",
+]
+
+[[package]]
+name = "deranged"
+version = "0.3.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b42b6fa04a440b495c8b04d0e71b707c585f83cb9cb28cf8cd0d976c315e31b4"
+dependencies = [
+ "powerfmt",
+]
+
+[[package]]
+name = "derivation-path"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6e5c37193a1db1d8ed868c03ec7b152175f26160a5b740e5e484143877e0adf0"
+
+[[package]]
+name = "derivative"
+version = "2.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fcc3dd5e9e9c0b295d6e1e4d811fb6f157d5ffd784b8d202fc62eac8035a770b"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "difflib"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6184e33543162437515c2e2b48714794e37845ec9851711914eec9d308f6ebe8"
+
+[[package]]
+name = "digest"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "digest"
+version = "0.10.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292"
+dependencies = [
+ "block-buffer 0.10.4",
+ "crypto-common",
+ "subtle",
+]
+
+[[package]]
+name = "dir-diff"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a7ad16bf5f84253b50d6557681c58c3ab67c47c77d39fed9aeb56e947290bd10"
+dependencies = [
+ "walkdir",
+]
+
+[[package]]
+name = "displaydoc"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "dlopen2"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "09b4f5f101177ff01b8ec4ecc81eead416a8aa42819a2869311b3420fa114ffa"
+dependencies = [
+ "dlopen2_derive",
+ "libc",
+ "once_cell",
+ "winapi",
+]
+
+[[package]]
+name = "dlopen2_derive"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a6cbae11b3de8fce2a456e8ea3dada226b35fe791f0dc1d360c0941f0bb681f3"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "downcast"
+version = "0.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1435fa1053d8b2fbbe9be7e97eca7f33d37b28409959813daefc1446a14247f1"
+
+[[package]]
+name = "eager"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "abe71d579d1812060163dff96056261deb5bf6729b100fa2e36a68b9649ba3d3"
+
+[[package]]
+name = "ed25519"
+version = "1.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "91cff35c70bba8a626e3185d8cd48cc11b5437e1a5bcd15b9b5fa3c64b6dfee7"
+dependencies = [
+ "signature",
+]
+
+[[package]]
+name = "ed25519-dalek"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c762bae6dcaf24c4c84667b8579785430908723d5c889f469d76a41d59cc7a9d"
+dependencies = [
+ "curve25519-dalek 3.2.0",
+ "ed25519",
+ "rand 0.7.3",
+ "serde",
+ "sha2 0.9.9",
+ "zeroize",
+]
+
+[[package]]
+name = "ed25519-dalek-bip32"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d2be62a4061b872c8c0873ee4fc6f101ce7b889d039f019c5fa2af471a59908"
+dependencies = [
+ "derivation-path",
+ "ed25519-dalek",
+ "hmac 0.12.1",
+ "sha2 0.10.8",
+]
+
+[[package]]
+name = "educe"
+version = "0.4.23"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0f0042ff8246a363dbe77d2ceedb073339e85a804b9a47636c6e016a9a32c05f"
+dependencies = [
+ "enum-ordinalize",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "either"
+version = "1.13.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "60b1af1c220855b6ceac025d3f6ecdd2b7c4894bfe9cd9bda4fbb4bc7c0d4cf0"
+
+[[package]]
+name = "encode_unicode"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "34aa73646ffb006b8f5147f3dc182bd4bcb190227ce861fc4a4844bf8e3cb2c0"
+
+[[package]]
+name = "encoding_rs"
+version = "0.8.35"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "75030f3c4f45dafd7586dd6780965a8c7e8e285a5ecb86713e63a79c5b2766f3"
+dependencies = [
+ "cfg-if",
+]
+
+[[package]]
+name = "enum-iterator"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9fd242f399be1da0a5354aa462d57b4ab2b4ee0683cc552f7c007d2d12d36e94"
+dependencies = [
+ "enum-iterator-derive",
+]
+
+[[package]]
+name = "enum-iterator-derive"
+version = "1.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a1ab991c1362ac86c61ab6f556cff143daa22e5a15e4e189df818b2fd19fe65b"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "enum-ordinalize"
+version = "3.1.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1bf1fa3f06bbff1ea5b1a9c7b14aa992a39657db60a2759457328d7e058f49ee"
+dependencies = [
+ "num-bigint 0.4.6",
+ "num-traits",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "env_logger"
+version = "0.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a12e6657c4c97ebab115a42dcee77225f7f482cdd841cf7088c657a42e9e00e7"
+dependencies = [
+ "atty",
+ "humantime",
+ "log",
+ "regex",
+ "termcolor",
+]
+
+[[package]]
+name = "equivalent"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5"
+
+[[package]]
+name = "errno"
+version = "0.3.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "33d852cb9b869c2a9b3df2f71a3074817f01e1844f839a144f5fcef059a4eb5d"
+dependencies = [
+ "libc",
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "event-listener"
+version = "2.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0206175f82b8d6bf6652ff7d71a1e27fd2e4efde587fd368662814d6ec1d9ce0"
+
+[[package]]
+name = "event-listener"
+version = "5.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3492acde4c3fc54c845eaab3eed8bd00c7a7d881f78bfc801e43a93dec1331ae"
+dependencies = [
+ "concurrent-queue",
+ "parking",
+ "pin-project-lite",
+]
+
+[[package]]
+name = "event-listener-strategy"
+version = "0.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3c3e4e0dd3673c1139bf041f3008816d9cf2946bbfac2945c09e523b8d7b05b2"
+dependencies = [
+ "event-listener 5.4.0",
+ "pin-project-lite",
+]
+
+[[package]]
+name = "fastrand"
+version = "2.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "37909eebbb50d72f9059c3b6d82c0463f2ff062c9e95845c43a6c9c0355411be"
+
+[[package]]
+name = "feature-probe"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "835a3dc7d1ec9e75e2b5fb4ba75396837112d2060b03f7d43bc1897c7f7211da"
+
+[[package]]
+name = "fiat-crypto"
+version = "0.2.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "28dea519a9695b9977216879a3ebfddf92f1c08c05d984f8996aecd6ecdc811d"
+
+[[package]]
+name = "filetime"
+version = "0.2.25"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "35c0522e981e68cbfa8c3f978441a5f34b30b96e146b33cd3359176b50fe8586"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "libredox",
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "five8_const"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72b4f62f0f8ca357f93ae90c8c2dd1041a1f665fde2f889ea9b1787903829015"
+dependencies = [
+ "five8_core",
+]
+
+[[package]]
+name = "five8_core"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94474d15a76982be62ca8a39570dccce148d98c238ebb7408b0a21b2c4bdddc4"
+
+[[package]]
+name = "flate2"
+version = "1.0.35"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c936bfdafb507ebbf50b8074c54fa31c5be9a1e7e5f467dd659697041407d07c"
+dependencies = [
+ "crc32fast",
+ "miniz_oxide",
+]
+
+[[package]]
+name = "float-cmp"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "98de4bbd547a563b716d8dfa9aad1cb19bfab00f4fa09a6a4ed21dbcf44ce9c4"
+dependencies = [
+ "num-traits",
+]
+
+[[package]]
+name = "fnv"
+version = "1.0.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
+
+[[package]]
+name = "foreign-types"
+version = "0.3.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f6f339eb8adc052cd2ca78910fda869aefa38d22d5cb648e6485e4d3fc06f3b1"
+dependencies = [
+ "foreign-types-shared",
+]
+
+[[package]]
+name = "foreign-types-shared"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b"
+
+[[package]]
+name = "form_urlencoded"
+version = "1.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e13624c2627564efccf4934284bdd98cbaa14e79b0b5a141218e507b3a823456"
+dependencies = [
+ "percent-encoding",
+]
+
+[[package]]
+name = "fragile"
+version = "2.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6c2141d6d6c8512188a7891b4b01590a45f6dac67afb4f255c4124dbb86d4eaa"
+
+[[package]]
+name = "futures"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "65bc07b1a8bc7c85c5f2e110c476c7389b4554ba72af57d8445ea63a576b0876"
+dependencies = [
+ "futures-channel",
+ "futures-core",
+ "futures-executor",
+ "futures-io",
+ "futures-sink",
+ "futures-task",
+ "futures-util",
+]
+
+[[package]]
+name = "futures-channel"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2dff15bf788c671c1934e366d07e30c1814a8ef514e1af724a602e8a2fbe1b10"
+dependencies = [
+ "futures-core",
+ "futures-sink",
+]
+
+[[package]]
+name = "futures-core"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "05f29059c0c2090612e8d742178b0580d2dc940c837851ad723096f87af6663e"
+
+[[package]]
+name = "futures-executor"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e28d1d997f585e54aebc3f97d39e72338912123a67330d723fdbb564d646c9f"
+dependencies = [
+ "futures-core",
+ "futures-task",
+ "futures-util",
+]
+
+[[package]]
+name = "futures-io"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9e5c1b78ca4aae1ac06c48a526a655760685149f0d465d21f37abfe57ce075c6"
+
+[[package]]
+name = "futures-macro"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "futures-sink"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e575fab7d1e0dcb8d0c7bcf9a63ee213816ab51902e6d244a95819acacf1d4f7"
+
+[[package]]
+name = "futures-task"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f90f7dce0722e95104fcb095585910c0977252f286e354b5e3bd38902cd99988"
+
+[[package]]
+name = "futures-timer"
+version = "3.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f288b0a4f20f9a56b5d1da57e2227c661b7b16168e2f72365f57b63326e29b24"
+
+[[package]]
+name = "futures-util"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9fa08315bb612088cc391249efdc3bc77536f16c91f6cf495e6fbe85b20a4a81"
+dependencies = [
+ "futures-channel",
+ "futures-core",
+ "futures-io",
+ "futures-macro",
+ "futures-sink",
+ "futures-task",
+ "memchr",
+ "pin-project-lite",
+ "pin-utils",
+ "slab",
+]
+
+[[package]]
+name = "generic-array"
+version = "0.14.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a"
+dependencies = [
+ "serde",
+ "typenum",
+ "version_check",
+]
+
+[[package]]
+name = "gethostname"
+version = "0.2.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c1ebd34e35c46e00bb73e81363248d627782724609fe1b6396f553f68fe3862e"
+dependencies = [
+ "libc",
+ "winapi",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.1.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce"
+dependencies = [
+ "cfg-if",
+ "js-sys",
+ "libc",
+ "wasi 0.9.0+wasi-snapshot-preview1",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.2.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7"
+dependencies = [
+ "cfg-if",
+ "js-sys",
+ "libc",
+ "wasi 0.11.0+wasi-snapshot-preview1",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "43a49c392881ce6d5c3b8cb70f98717b7c07aabbdff06687b9030dbfbe2725f8"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "wasi 0.13.3+wasi-0.2.2",
+ "windows-targets 0.52.6",
+]
+
+[[package]]
+name = "gimli"
+version = "0.31.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "07e28edb80900c19c28f1072f2e8aeca7fa06b23cd4169cefe1af5aa3260783f"
+
+[[package]]
+name = "governor"
+version = "0.6.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "68a7f542ee6b35af73b06abc0dad1c1bae89964e4e253bc4b587b91c9637867b"
+dependencies = [
+ "cfg-if",
+ "dashmap",
+ "futures",
+ "futures-timer",
+ "no-std-compat",
+ "nonzero_ext",
+ "parking_lot",
+ "portable-atomic",
+ "quanta",
+ "rand 0.8.5",
+ "smallvec",
+ "spinning_top",
+]
+
+[[package]]
+name = "h2"
+version = "0.3.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "81fe527a889e1532da5c525686d96d4c2e74cdd345badf8dfef9f6b39dd5f5e8"
+dependencies = [
+ "bytes",
+ "fnv",
+ "futures-core",
+ "futures-sink",
+ "futures-util",
+ "http",
+ "indexmap",
+ "slab",
+ "tokio",
+ "tokio-util 0.7.13",
+ "tracing",
+]
+
+[[package]]
+name = "hash32"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b0c35f58762feb77d74ebe43bdbc3210f09be9fe6742234d573bacc26ed92b67"
+dependencies = [
+ "byteorder",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.13.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "43a3c133739dddd0d2990f9a4bdf8eb4b21ef50e4851ca85ab661199821d510e"
+dependencies = [
+ "ahash",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.14.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e5274423e17b7c9fc20b6e7e208532f9b19825d82dfd615708b70edd83df41f1"
+
+[[package]]
+name = "hashbrown"
+version = "0.15.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bf151400ff0baff5465007dd2f3e717f3fe502074ca563069ce3a6629d07b289"
+
+[[package]]
+name = "heck"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "95505c38b4572b2d910cecb0281560f54b440a19336cbbcb27bf6ce6adc6f5a8"
+
+[[package]]
+name = "hermit-abi"
+version = "0.1.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "hermit-abi"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d231dfb89cfffdbc30e7fc41579ed6066ad03abda9e567ccafae602b97ec5024"
+
+[[package]]
+name = "histogram"
+version = "0.6.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "12cb882ccb290b8646e554b157ab0b71e64e8d5bef775cd66b6531e52d302669"
+
+[[package]]
+name = "hmac"
+version = "0.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "126888268dcc288495a26bf004b38c5fdbb31682f992c84ceb046a1f0fe38840"
+dependencies = [
+ "crypto-mac",
+ "digest 0.9.0",
+]
+
+[[package]]
+name = "hmac"
+version = "0.12.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e"
+dependencies = [
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "hmac-drbg"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "17ea0a1394df5b6574da6e0c1ade9e78868c9fb0a4e5ef4428e32da4676b85b1"
+dependencies = [
+ "digest 0.9.0",
+ "generic-array",
+ "hmac 0.8.1",
+]
+
+[[package]]
+name = "http"
+version = "0.2.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "601cbb57e577e2f5ef5be8e7b83f0f63994f25aa94d673e54a92d5c516d101f1"
+dependencies = [
+ "bytes",
+ "fnv",
+ "itoa",
+]
+
+[[package]]
+name = "http-body"
+version = "0.4.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7ceab25649e9960c0311ea418d17bee82c0dcec1bd053b5f9a66e265a693bed2"
+dependencies = [
+ "bytes",
+ "http",
+ "pin-project-lite",
+]
+
+[[package]]
+name = "httparse"
+version = "1.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f2d708df4e7140240a16cd6ab0ab65c972d7433ab77819ea693fde9c43811e2a"
+
+[[package]]
+name = "httpdate"
+version = "1.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9"
+
+[[package]]
+name = "humantime"
+version = "2.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4"
+
+[[package]]
+name = "hyper"
+version = "0.14.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "41dfc780fdec9373c01bae43289ea34c972e40ee3c9f6b3c8801a35f35586ce7"
+dependencies = [
+ "bytes",
+ "futures-channel",
+ "futures-core",
+ "futures-util",
+ "h2",
+ "http",
+ "http-body",
+ "httparse",
+ "httpdate",
+ "itoa",
+ "pin-project-lite",
+ "socket2",
+ "tokio",
+ "tower-service",
+ "tracing",
+ "want",
+]
+
+[[package]]
+name = "hyper-rustls"
+version = "0.24.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ec3efd23720e2049821a693cbc7e65ea87c72f1c58ff2f9522ff332b1491e590"
+dependencies = [
+ "futures-util",
+ "http",
+ "hyper",
+ "rustls 0.21.12",
+ "tokio",
+ "tokio-rustls",
+]
+
+[[package]]
+name = "iana-time-zone"
+version = "0.1.61"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "235e081f3925a06703c2d0117ea8b91f042756fd6e7a6e5d901e8ca1a996b220"
+dependencies = [
+ "android_system_properties",
+ "core-foundation-sys",
+ "iana-time-zone-haiku",
+ "js-sys",
+ "wasm-bindgen",
+ "windows-core",
+]
+
+[[package]]
+name = "iana-time-zone-haiku"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f31827a206f56af32e590ba56d5d2d085f558508192593743f16b2306495269f"
+dependencies = [
+ "cc",
+]
+
+[[package]]
+name = "icu_collections"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "db2fa452206ebee18c4b5c2274dbf1de17008e874b4dc4f0aea9d01ca79e4526"
+dependencies = [
+ "displaydoc",
+ "yoke",
+ "zerofrom",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_locid"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "13acbb8371917fc971be86fc8057c41a64b521c184808a698c02acc242dbf637"
+dependencies = [
+ "displaydoc",
+ "litemap",
+ "tinystr",
+ "writeable",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_locid_transform"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "01d11ac35de8e40fdeda00d9e1e9d92525f3f9d887cdd7aa81d727596788b54e"
+dependencies = [
+ "displaydoc",
+ "icu_locid",
+ "icu_locid_transform_data",
+ "icu_provider",
+ "tinystr",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_locid_transform_data"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fdc8ff3388f852bede6b579ad4e978ab004f139284d7b28715f773507b946f6e"
+
+[[package]]
+name = "icu_normalizer"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "19ce3e0da2ec68599d193c93d088142efd7f9c5d6fc9b803774855747dc6a84f"
+dependencies = [
+ "displaydoc",
+ "icu_collections",
+ "icu_normalizer_data",
+ "icu_properties",
+ "icu_provider",
+ "smallvec",
+ "utf16_iter",
+ "utf8_iter",
+ "write16",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_normalizer_data"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f8cafbf7aa791e9b22bec55a167906f9e1215fd475cd22adfcf660e03e989516"
+
+[[package]]
+name = "icu_properties"
+version = "1.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "93d6020766cfc6302c15dbbc9c8778c37e62c14427cb7f6e601d849e092aeef5"
+dependencies = [
+ "displaydoc",
+ "icu_collections",
+ "icu_locid_transform",
+ "icu_properties_data",
+ "icu_provider",
+ "tinystr",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_properties_data"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "67a8effbc3dd3e4ba1afa8ad918d5684b8868b3b26500753effea8d2eed19569"
+
+[[package]]
+name = "icu_provider"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6ed421c8a8ef78d3e2dbc98a973be2f3770cb42b606e3ab18d6237c4dfde68d9"
+dependencies = [
+ "displaydoc",
+ "icu_locid",
+ "icu_provider_macros",
+ "stable_deref_trait",
+ "tinystr",
+ "writeable",
+ "yoke",
+ "zerofrom",
+ "zerovec",
+]
+
+[[package]]
+name = "icu_provider_macros"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1ec89e9337638ecdc08744df490b221a7399bf8d164eb52a665454e60e075ad6"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "ident_case"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39"
+
+[[package]]
+name = "idna"
+version = "1.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "686f825264d630750a544639377bae737628043f20d38bbc029e8f29ea968a7e"
+dependencies = [
+ "idna_adapter",
+ "smallvec",
+ "utf8_iter",
+]
+
+[[package]]
+name = "idna_adapter"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "daca1df1c957320b2cf139ac61e7bd64fed304c5040df000a745aa1de3b4ef71"
+dependencies = [
+ "icu_normalizer",
+ "icu_properties",
+]
+
+[[package]]
+name = "im"
+version = "15.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0acd33ff0285af998aaf9b57342af478078f53492322fafc47450e09397e0e9"
+dependencies = [
+ "bitmaps",
+ "rand_core 0.6.4",
+ "rand_xoshiro",
+ "rayon",
+ "serde",
+ "sized-chunks",
+ "typenum",
+ "version_check",
+]
+
+[[package]]
+name = "include_dir"
+version = "0.7.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "923d117408f1e49d914f1a379a309cffe4f18c05cf4e3d12e613a15fc81bd0dd"
+dependencies = [
+ "include_dir_macros",
+]
+
+[[package]]
+name = "include_dir_macros"
+version = "0.7.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7cab85a7ed0bd5f0e76d93846e0147172bed2e2d3f859bcc33a8d9699cad1a75"
+dependencies = [
+ "proc-macro2",
+ "quote",
+]
+
+[[package]]
+name = "index_list"
+version = "0.2.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fa38453685e5fe724fd23ff6c1a158c1e2ca21ce0c2718fa11e96e70e99fd4de"
+
+[[package]]
+name = "indexmap"
+version = "2.7.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8c9c992b02b5b4c94ea26e32fe5bccb7aa7d9f390ab5c1221ff895bc7ea8b652"
+dependencies = [
+ "equivalent",
+ "hashbrown 0.15.2",
+]
+
+[[package]]
+name = "indicatif"
+version = "0.17.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "183b3088984b400f4cfac3620d5e076c84da5364016b4f49473de574b2586235"
+dependencies = [
+ "console",
+ "number_prefix",
+ "portable-atomic",
+ "unicode-width",
+ "web-time",
+]
+
+[[package]]
+name = "inout"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a0c10553d664a4d0bcff9f4215d0aac67a639cc68ef660840afe309b807bc9f5"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "ipnet"
+version = "2.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "469fb0b9cefa57e3ef31275ee7cacb78f2fdca44e4765491884a2b119d4eb130"
+
+[[package]]
+name = "itertools"
+version = "0.10.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b0fd2260e829bddf4cb6ea802289de2f86d6a7a690192fbe91b3f46e0f2c8473"
+dependencies = [
+ "either",
+]
+
+[[package]]
+name = "itertools"
+version = "0.12.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ba291022dbbd398a455acf126c1e341954079855bc60dfdda641363bd6922569"
+dependencies = [
+ "either",
+]
+
+[[package]]
+name = "itoa"
+version = "1.0.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d75a2a4b1b190afb6f5425f10f6a8f959d2ea0b9c2b1d79553551850539e4674"
+
+[[package]]
+name = "jni"
+version = "0.19.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c6df18c2e3db7e453d3c6ac5b3e9d5182664d28788126d39b91f2d1e22b017ec"
+dependencies = [
+ "cesu8",
+ "combine 4.6.7",
+ "jni-sys",
+ "log",
+ "thiserror 1.0.69",
+ "walkdir",
+]
+
+[[package]]
+name = "jni-sys"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8eaf4bc02d17cbdd7ff4c7438cafcdf7fb9a4613313ad11b4f8fefe7d3fa0130"
+
+[[package]]
+name = "jobserver"
+version = "0.1.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "48d1dbcbbeb6a7fec7e059840aa538bd62aaccf972c7346c4d9d2059312853d0"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "js-sys"
+version = "0.3.77"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1cfaf33c695fc6e08064efbc1f72ec937429614f25eef83af942d0e227c3a28f"
+dependencies = [
+ "once_cell",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "jsonrpc-core"
+version = "18.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "14f7f76aef2d054868398427f6c54943cf3d1caa9a7ec7d0c38d69df97a965eb"
+dependencies = [
+ "futures",
+ "futures-executor",
+ "futures-util",
+ "log",
+ "serde",
+ "serde_derive",
+ "serde_json",
+]
+
+[[package]]
+name = "keccak"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ecc2af9a1119c51f12a14607e783cb977bde58bc069ff0c3da1095e635d70654"
+dependencies = [
+ "cpufeatures",
+]
+
+[[package]]
+name = "lazy_static"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe"
+
+[[package]]
+name = "libc"
+version = "0.2.169"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b5aba8db14291edd000dfcc4d620c7ebfb122c613afb886ca8803fa4e128a20a"
+
+[[package]]
+name = "libredox"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c0ff37bd590ca25063e35af745c343cb7a0271906fb7b37e4813e8f79f00268d"
+dependencies = [
+ "bitflags 2.8.0",
+ "libc",
+ "redox_syscall",
+]
+
+[[package]]
+name = "libsecp256k1"
+version = "0.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c9d220bc1feda2ac231cb78c3d26f27676b8cf82c96971f7aeef3d0cf2797c73"
+dependencies = [
+ "arrayref",
+ "base64 0.12.3",
+ "digest 0.9.0",
+ "hmac-drbg",
+ "libsecp256k1-core",
+ "libsecp256k1-gen-ecmult",
+ "libsecp256k1-gen-genmult",
+ "rand 0.7.3",
+ "serde",
+ "sha2 0.9.9",
+ "typenum",
+]
+
+[[package]]
+name = "libsecp256k1-core"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0f6ab710cec28cef759c5f18671a27dae2a5f952cdaaee1d8e2908cb2478a80"
+dependencies = [
+ "crunchy",
+ "digest 0.9.0",
+ "subtle",
+]
+
+[[package]]
+name = "libsecp256k1-gen-ecmult"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ccab96b584d38fac86a83f07e659f0deafd0253dc096dab5a36d53efe653c5c3"
+dependencies = [
+ "libsecp256k1-core",
+]
+
+[[package]]
+name = "libsecp256k1-gen-genmult"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "67abfe149395e3aa1c48a2beb32b068e2334402df8181f818d3aee2b304c4f5d"
+dependencies = [
+ "libsecp256k1-core",
+]
+
+[[package]]
+name = "light-poseidon"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3c9a85a9752c549ceb7578064b4ed891179d20acd85f27318573b64d2d7ee7ee"
+dependencies = [
+ "ark-bn254",
+ "ark-ff",
+ "num-bigint 0.4.6",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "linux-raw-sys"
+version = "0.4.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d26c52dbd32dccf2d10cac7725f8eae5296885fb5703b261f7d0a0739ec807ab"
+
+[[package]]
+name = "litemap"
+version = "0.7.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4ee93343901ab17bd981295f2cf0026d4ad018c7c31ba84549a4ddbb47a45104"
+
+[[package]]
+name = "lock_api"
+version = "0.4.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "07af8b9cdd281b7915f413fa73f29ebd5d55d0d3f0155584dade1ff18cea1b17"
+dependencies = [
+ "autocfg",
+ "scopeguard",
+]
+
+[[package]]
+name = "log"
+version = "0.4.25"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "04cbf5b083de1c7e0222a7a51dbfdba1cbe1c6ab0b15e29fff3f6c077fd9cd9f"
+
+[[package]]
+name = "lz4"
+version = "1.28.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a20b523e860d03443e98350ceaac5e71c6ba89aea7d960769ec3ce37f4de5af4"
+dependencies = [
+ "lz4-sys",
+]
+
+[[package]]
+name = "lz4-sys"
+version = "1.11.1+lz4-1.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6bd8c0d6c6ed0cd30b3652886bb8711dc4bb01d637a68105a3d5158039b418e6"
+dependencies = [
+ "cc",
+ "libc",
+]
+
+[[package]]
+name = "memchr"
+version = "2.7.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3"
+
+[[package]]
+name = "memmap2"
+version = "0.5.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "83faa42c0a078c393f6b29d5db232d8be22776a891f8f56e5284faee4a20b327"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "memoffset"
+version = "0.9.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "488016bfae457b036d996092f6cb448677611ce4449e970ceaf42695203f218a"
+dependencies = [
+ "autocfg",
+]
+
+[[package]]
+name = "merlin"
+version = "3.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "58c38e2799fc0978b65dfff8023ec7843e2330bb462f19198840b34b6582397d"
+dependencies = [
+ "byteorder",
+ "keccak",
+ "rand_core 0.6.4",
+ "zeroize",
+]
+
+[[package]]
+name = "mime"
+version = "0.3.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6877bb514081ee2a7ff5ef9de3281f14a4dd4bceac4c09388074a6b5df8a139a"
+
+[[package]]
+name = "mime_guess"
+version = "2.0.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f7c44f8e672c00fe5308fa235f821cb4198414e1c77935c1ab6948d3fd78550e"
+dependencies = [
+ "mime",
+ "unicase",
+]
+
+[[package]]
+name = "minimal-lexical"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "68354c5c6bd36d73ff3feceb05efa59b6acb7626617f4962be322a825e61f79a"
+
+[[package]]
+name = "miniz_oxide"
+version = "0.8.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b8402cab7aefae129c6977bb0ff1b8fd9a04eb5b51efc50a70bea51cda0c7924"
+dependencies = [
+ "adler2",
+]
+
+[[package]]
+name = "mio"
+version = "1.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2886843bf800fba2e3377cff24abf6379b4c4d5c6681eaf9ea5b0d15090450bd"
+dependencies = [
+ "libc",
+ "wasi 0.11.0+wasi-snapshot-preview1",
+ "windows-sys 0.52.0",
+]
+
+[[package]]
+name = "mockall"
+version = "0.11.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4c84490118f2ee2d74570d114f3d0493cbf02790df303d2707606c3e14e07c96"
+dependencies = [
+ "cfg-if",
+ "downcast",
+ "fragile",
+ "lazy_static",
+ "mockall_derive",
+ "predicates",
+ "predicates-tree",
+]
+
+[[package]]
+name = "mockall_derive"
+version = "0.11.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "22ce75669015c4f47b289fd4d4f56e894e4c96003ffdf3ac51313126f94c6cbb"
+dependencies = [
+ "cfg-if",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "modular-bitfield"
+version = "0.11.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a53d79ba8304ac1c4f9eb3b9d281f21f7be9d4626f72ce7df4ad8fbde4f38a74"
+dependencies = [
+ "modular-bitfield-impl",
+ "static_assertions",
+]
+
+[[package]]
+name = "modular-bitfield-impl"
+version = "0.11.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5a7d5f7076603ebc68de2dc6a650ec331a062a13abaa346975be747bbfa4b789"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "nix"
+version = "0.29.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "71e2746dc3a24dd78b3cfcb7be93368c6de9963d30f43a6a73998a9cf4b17b46"
+dependencies = [
+ "bitflags 2.8.0",
+ "cfg-if",
+ "cfg_aliases",
+ "libc",
+ "memoffset",
+]
+
+[[package]]
+name = "no-std-compat"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b93853da6d84c2e3c7d730d6473e8817692dd89be387eb01b94d7f108ecb5b8c"
+
+[[package]]
+name = "nom"
+version = "7.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d273983c5a657a70a3e8f2a01329822f3b8c8172b73826411a55751e404a0a4a"
+dependencies = [
+ "memchr",
+ "minimal-lexical",
+]
+
+[[package]]
+name = "nonzero_ext"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "38bf9645c8b145698bb0b18a4637dcacbc421ea49bef2317e4fd8065a387cf21"
+
+[[package]]
+name = "normalize-line-endings"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "61807f77802ff30975e01f4f071c8ba10c022052f98b3294119f3e615d13e5be"
+
+[[package]]
+name = "num"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b8536030f9fea7127f841b45bb6243b27255787fb4eb83958aa1ef9d2fdc0c36"
+dependencies = [
+ "num-bigint 0.2.6",
+ "num-complex",
+ "num-integer",
+ "num-iter",
+ "num-rational",
+ "num-traits",
+]
+
+[[package]]
+name = "num-bigint"
+version = "0.2.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "090c7f9998ee0ff65aa5b723e4009f7b217707f1fb5ea551329cc4d6231fb304"
+dependencies = [
+ "autocfg",
+ "num-integer",
+ "num-traits",
+]
+
+[[package]]
+name = "num-bigint"
+version = "0.4.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a5e44f723f1133c9deac646763579fdb3ac745e418f2a7af9cd0c431da1f20b9"
+dependencies = [
+ "num-integer",
+ "num-traits",
+]
+
+[[package]]
+name = "num-complex"
+version = "0.2.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b6b19411a9719e753aff12e5187b74d60d3dc449ec3f4dc21e3989c3f554bc95"
+dependencies = [
+ "autocfg",
+ "num-traits",
+]
+
+[[package]]
+name = "num-conv"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "51d515d32fb182ee37cda2ccdcb92950d6a3c2893aa280e540671c2cd0f3b1d9"
+
+[[package]]
+name = "num-derive"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ed3955f1a9c7c0c15e092f9c887db08b1fc683305fdf6eb6684f22555355e202"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "num-integer"
+version = "0.1.46"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7969661fd2958a5cb096e56c8e1ad0444ac2bbcd0061bd28660485a44879858f"
+dependencies = [
+ "num-traits",
+]
+
+[[package]]
+name = "num-iter"
+version = "0.1.45"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1429034a0490724d0075ebb2bc9e875d6503c3cf69e235a8941aa757d83ef5bf"
+dependencies = [
+ "autocfg",
+ "num-integer",
+ "num-traits",
+]
+
+[[package]]
+name = "num-rational"
+version = "0.2.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5c000134b5dbf44adc5cb772486d335293351644b801551abe8f75c84cfa4aef"
+dependencies = [
+ "autocfg",
+ "num-bigint 0.2.6",
+ "num-integer",
+ "num-traits",
+]
+
+[[package]]
+name = "num-traits"
+version = "0.2.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841"
+dependencies = [
+ "autocfg",
+]
+
+[[package]]
+name = "num_cpus"
+version = "1.16.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43"
+dependencies = [
+ "hermit-abi 0.3.9",
+ "libc",
+]
+
+[[package]]
+name = "num_enum"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4e613fc340b2220f734a8595782c551f1250e969d87d3be1ae0579e8d4065179"
+dependencies = [
+ "num_enum_derive",
+]
+
+[[package]]
+name = "num_enum_derive"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "af1844ef2428cc3e1cb900be36181049ef3d3193c63e43026cfe202983b27a56"
+dependencies = [
+ "proc-macro-crate 3.2.0",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "number_prefix"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "830b246a0e5f20af87141b25c173cd1b609bd7779a4617d6ec582abaf90870f3"
+
+[[package]]
+name = "object"
+version = "0.36.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "62948e14d923ea95ea2c7c86c71013138b66525b86bdc08d2dcc262bdb497b87"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "oid-registry"
+version = "0.6.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9bedf36ffb6ba96c2eb7144ef6270557b52e54b20c0a8e1eb2ff99a6c6959bff"
+dependencies = [
+ "asn1-rs",
+]
+
+[[package]]
+name = "once_cell"
+version = "1.20.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775"
+
+[[package]]
+name = "opaque-debug"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c08d65885ee38876c4f86fa503fb49d7b507c2b62552df7c70b2fce627e06381"
+
+[[package]]
+name = "openssl"
+version = "0.10.70"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "61cfb4e166a8bb8c9b55c500bc2308550148ece889be90f609377e58140f42c6"
+dependencies = [
+ "bitflags 2.8.0",
+ "cfg-if",
+ "foreign-types",
+ "libc",
+ "once_cell",
+ "openssl-macros",
+ "openssl-sys",
+]
+
+[[package]]
+name = "openssl-macros"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a948666b637a0f465e8564c73e89d4dde00d72d4d473cc972f390fc3dcee7d9c"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "openssl-probe"
+version = "0.1.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d05e27ee213611ffe7d6348b942e8f942b37114c00cc03cec254295a4a17852e"
+
+[[package]]
+name = "openssl-src"
+version = "300.4.1+3.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "faa4eac4138c62414b5622d1b31c5c304f34b406b013c079c2bbc652fdd6678c"
+dependencies = [
+ "cc",
+]
+
+[[package]]
+name = "openssl-sys"
+version = "0.9.105"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8b22d5b84be05a8d6947c7cb71f7c849aa0f112acd4bf51c2a7c1c988ac0a9dc"
+dependencies = [
+ "cc",
+ "libc",
+ "openssl-src",
+ "pkg-config",
+ "vcpkg",
+]
+
+[[package]]
+name = "opentelemetry"
+version = "0.17.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6105e89802af13fdf48c49d7646d3b533a70e536d818aae7e78ba0433d01acb8"
+dependencies = [
+ "async-trait",
+ "crossbeam-channel",
+ "futures-channel",
+ "futures-executor",
+ "futures-util",
+ "js-sys",
+ "lazy_static",
+ "percent-encoding",
+ "pin-project",
+ "rand 0.8.5",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "parking"
+version = "2.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f38d5652c16fde515bb1ecef450ab0f6a219d619a7274976324d5e377f7dceba"
+
+[[package]]
+name = "parking_lot"
+version = "0.12.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f1bf18183cf54e8d6059647fc3063646a1801cf30896933ec2311622cc4b9a27"
+dependencies = [
+ "lock_api",
+ "parking_lot_core",
+]
+
+[[package]]
+name = "parking_lot_core"
+version = "0.9.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "redox_syscall",
+ "smallvec",
+ "windows-targets 0.52.6",
+]
+
+[[package]]
+name = "paste"
+version = "1.0.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a"
+
+[[package]]
+name = "pbkdf2"
+version = "0.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "83a0692ec44e4cf1ef28ca317f14f8f07da2d95ec3fa01f86e4467b725e60917"
+dependencies = [
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "pem"
+version = "1.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a8835c273a76a90455d7344889b0964598e3316e2a79ede8e36f16bdcf2228b8"
+dependencies = [
+ "base64 0.13.1",
+]
+
+[[package]]
+name = "percent-encoding"
+version = "2.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e"
+
+[[package]]
+name = "percentage"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2fd23b938276f14057220b707937bcb42fa76dda7560e57a2da30cb52d557937"
+dependencies = [
+ "num",
+]
+
+[[package]]
+name = "pin-project"
+version = "1.1.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dfe2e71e1471fe07709406bf725f710b02927c9c54b2b5b2ec0e8087d97c327d"
+dependencies = [
+ "pin-project-internal",
+]
+
+[[package]]
+name = "pin-project-internal"
+version = "1.1.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f6e859e6e5bd50440ab63c47e3ebabc90f26251f7c73c3d3e837b74a1cc3fa67"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "pin-project-lite"
+version = "0.2.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3b3cff922bd51709b605d9ead9aa71031d81447142d828eb4a6eba76fe619f9b"
+
+[[package]]
+name = "pin-utils"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184"
+
+[[package]]
+name = "pkg-config"
+version = "0.3.31"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "953ec861398dccce10c670dfeaf3ec4911ca479e9c02154b3a215178c5f566f2"
+
+[[package]]
+name = "polyval"
+version = "0.6.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d1fe60d06143b2430aa532c94cfe9e29783047f06c0d7fd359a9a51b729fa25"
+dependencies = [
+ "cfg-if",
+ "cpufeatures",
+ "opaque-debug",
+ "universal-hash",
+]
+
+[[package]]
+name = "portable-atomic"
+version = "1.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "280dc24453071f1b63954171985a0b0d30058d287960968b9b2aca264c8d4ee6"
+
+[[package]]
+name = "powerfmt"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "439ee305def115ba05938db6eb1644ff94165c5ab5e9420d1c1bcedbba909391"
+
+[[package]]
+name = "ppv-lite86"
+version = "0.2.20"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04"
+dependencies = [
+ "zerocopy",
+]
+
+[[package]]
+name = "predicates"
+version = "2.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "59230a63c37f3e18569bdb90e4a89cbf5bf8b06fea0b84e65ea10cc4df47addd"
+dependencies = [
+ "difflib",
+ "float-cmp",
+ "itertools 0.10.5",
+ "normalize-line-endings",
+ "predicates-core",
+ "regex",
+]
+
+[[package]]
+name = "predicates-core"
+version = "1.0.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "727e462b119fe9c93fd0eb1429a5f7647394014cf3c04ab2c0350eeb09095ffa"
+
+[[package]]
+name = "predicates-tree"
+version = "1.0.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72dd2d6d381dfb73a193c7fca536518d7caee39fc8503f74e7dc0be0531b425c"
+dependencies = [
+ "predicates-core",
+ "termtree",
+]
+
+[[package]]
+name = "proc-macro-crate"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1d6ea3c4595b96363c13943497db34af4460fb474a95c43f4446ad341b8c9785"
+dependencies = [
+ "toml",
+]
+
+[[package]]
+name = "proc-macro-crate"
+version = "3.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8ecf48c7ca261d60b74ab1a7b20da18bede46776b2e55535cb958eb595c5fa7b"
+dependencies = [
+ "toml_edit",
+]
+
+[[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.93"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "60946a68e5f9d28b0dc1c21bb8a97ee7d018a8b322fa57838ba31cc878e22d99"
+dependencies = [
+ "unicode-ident",
+]
+
+[[package]]
+name = "qstring"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d464fae65fff2680baf48019211ce37aaec0c78e9264c84a3e484717f965104e"
+dependencies = [
+ "percent-encoding",
+]
+
+[[package]]
+name = "qualifier_attr"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9e2e25ee72f5b24d773cae88422baddefff7714f97aab68d96fe2b6fc4a28fb2"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "quanta"
+version = "0.12.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3bd1fe6824cea6538803de3ff1bc0cf3949024db3d43c9643024bfb33a807c0e"
+dependencies = [
+ "crossbeam-utils",
+ "libc",
+ "once_cell",
+ "raw-cpuid",
+ "wasi 0.11.0+wasi-snapshot-preview1",
+ "web-sys",
+ "winapi",
+]
+
+[[package]]
+name = "quinn"
+version = "0.11.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "62e96808277ec6f97351a2380e6c25114bc9e67037775464979f3037c92d05ef"
+dependencies = [
+ "bytes",
+ "pin-project-lite",
+ "quinn-proto",
+ "quinn-udp",
+ "rustc-hash",
+ "rustls 0.23.22",
+ "socket2",
+ "thiserror 2.0.11",
+ "tokio",
+ "tracing",
+]
+
+[[package]]
+name = "quinn-proto"
+version = "0.11.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a2fe5ef3495d7d2e377ff17b1a8ce2ee2ec2a18cde8b6ad6619d65d0701c135d"
+dependencies = [
+ "bytes",
+ "getrandom 0.2.15",
+ "rand 0.8.5",
+ "ring",
+ "rustc-hash",
+ "rustls 0.23.22",
+ "rustls-pki-types",
+ "rustls-platform-verifier",
+ "slab",
+ "thiserror 2.0.11",
+ "tinyvec",
+ "tracing",
+ "web-time",
+]
+
+[[package]]
+name = "quinn-udp"
+version = "0.5.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1c40286217b4ba3a71d644d752e6a0b71f13f1b6a2c5311acfcbe0c2418ed904"
+dependencies = [
+ "cfg_aliases",
+ "libc",
+ "once_cell",
+ "socket2",
+ "tracing",
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "quote"
+version = "1.0.38"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0e4dccaaaf89514f546c693ddc140f729f958c247918a13380cccc6078391acc"
+dependencies = [
+ "proc-macro2",
+]
+
+[[package]]
+name = "rand"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03"
+dependencies = [
+ "getrandom 0.1.16",
+ "libc",
+ "rand_chacha 0.2.2",
+ "rand_core 0.5.1",
+ "rand_hc",
+]
+
+[[package]]
+name = "rand"
+version = "0.8.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
+dependencies = [
+ "libc",
+ "rand_chacha 0.3.1",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19"
+dependencies = [
+ "getrandom 0.1.16",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c"
+dependencies = [
+ "getrandom 0.2.15",
+]
+
+[[package]]
+name = "rand_hc"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c"
+dependencies = [
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "rand_xoshiro"
+version = "0.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6f97cdb2a36ed4183de61b2f824cc45c9f1037f28afe0a322e9fff4c108b5aaa"
+dependencies = [
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "raw-cpuid"
+version = "11.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c6928fa44c097620b706542d428957635951bade7143269085389d42c8a4927e"
+dependencies = [
+ "bitflags 2.8.0",
+]
+
+[[package]]
+name = "rayon"
+version = "1.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b418a60154510ca1a002a752ca9714984e21e4241e804d32555251faf8b78ffa"
+dependencies = [
+ "either",
+ "rayon-core",
+]
+
+[[package]]
+name = "rayon-core"
+version = "1.12.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1465873a3dfdaa8ae7cb14b4383657caab0b3e8a0aa9ae8e04b044854c8dfce2"
+dependencies = [
+ "crossbeam-deque",
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "redox_syscall"
+version = "0.5.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "03a862b389f93e68874fbf580b9de08dd02facb9a788ebadaf4a3fd33cf58834"
+dependencies = [
+ "bitflags 2.8.0",
+]
+
+[[package]]
+name = "regex"
+version = "1.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b544ef1b4eac5dc2db33ea63606ae9ffcfac26c1416a2806ae0bf5f56b201191"
+dependencies = [
+ "aho-corasick",
+ "memchr",
+ "regex-automata",
+ "regex-syntax",
+]
+
+[[package]]
+name = "regex-automata"
+version = "0.4.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "809e8dc61f6de73b46c85f4c96486310fe304c434cfa43669d7b40f711150908"
+dependencies = [
+ "aho-corasick",
+ "memchr",
+ "regex-syntax",
+]
+
+[[package]]
+name = "regex-syntax"
+version = "0.8.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c"
+
+[[package]]
+name = "reqwest"
+version = "0.11.27"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dd67538700a17451e7cba03ac727fb961abb7607553461627b97de0b89cf4a62"
+dependencies = [
+ "async-compression",
+ "base64 0.21.7",
+ "bytes",
+ "encoding_rs",
+ "futures-core",
+ "futures-util",
+ "h2",
+ "http",
+ "http-body",
+ "hyper",
+ "hyper-rustls",
+ "ipnet",
+ "js-sys",
+ "log",
+ "mime",
+ "mime_guess",
+ "once_cell",
+ "percent-encoding",
+ "pin-project-lite",
+ "rustls 0.21.12",
+ "rustls-pemfile 1.0.4",
+ "serde",
+ "serde_json",
+ "serde_urlencoded",
+ "sync_wrapper",
+ "system-configuration",
+ "tokio",
+ "tokio-rustls",
+ "tokio-util 0.7.13",
+ "tower-service",
+ "url",
+ "wasm-bindgen",
+ "wasm-bindgen-futures",
+ "web-sys",
+ "webpki-roots 0.25.4",
+ "winreg",
+]
+
+[[package]]
+name = "reqwest-middleware"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5a735987236a8e238bf0296c7e351b999c188ccc11477f311b82b55c93984216"
+dependencies = [
+ "anyhow",
+ "async-trait",
+ "http",
+ "reqwest",
+ "serde",
+ "task-local-extensions",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "ring"
+version = "0.17.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c17fa4cb658e3583423e915b9f3acc01cceaee1860e33d59ebae66adc3a2dc0d"
+dependencies = [
+ "cc",
+ "cfg-if",
+ "getrandom 0.2.15",
+ "libc",
+ "spin",
+ "untrusted",
+ "windows-sys 0.52.0",
+]
+
+[[package]]
+name = "rustc-demangle"
+version = "0.1.24"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "719b953e2095829ee67db738b3bfa9fa368c94900df327b3f07fe6e794d2fe1f"
+
+[[package]]
+name = "rustc-hash"
+version = "2.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c7fb8039b3032c191086b10f11f319a6e99e1e82889c5cc6046f515c9db1d497"
+
+[[package]]
+name = "rustc_version"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cfcb3a22ef46e85b45de6ee7e79d063319ebb6594faafcf1c225ea92ab6e9b92"
+dependencies = [
+ "semver",
+]
+
+[[package]]
+name = "rusticata-macros"
+version = "4.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "faf0c4a6ece9950b9abdb62b1cfcf2a68b3b67a10ba445b3bb85be2a293d0632"
+dependencies = [
+ "nom",
+]
+
+[[package]]
+name = "rustix"
+version = "0.38.44"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fdb5bc1ae2baa591800df16c9ca78619bf65c0488b41b96ccec5d11220d8c154"
+dependencies = [
+ "bitflags 2.8.0",
+ "errno",
+ "libc",
+ "linux-raw-sys",
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "rustls"
+version = "0.21.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3f56a14d1f48b391359b22f731fd4bd7e43c97f3c50eee276f3aa09c94784d3e"
+dependencies = [
+ "log",
+ "ring",
+ "rustls-webpki 0.101.7",
+ "sct",
+]
+
+[[package]]
+name = "rustls"
+version = "0.23.22"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9fb9263ab4eb695e42321db096e3b8fbd715a59b154d5c88d82db2175b681ba7"
+dependencies = [
+ "once_cell",
+ "ring",
+ "rustls-pki-types",
+ "rustls-webpki 0.102.8",
+ "subtle",
+ "zeroize",
+]
+
+[[package]]
+name = "rustls-native-certs"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e5bfb394eeed242e909609f56089eecfe5fda225042e8b171791b9c95f5931e5"
+dependencies = [
+ "openssl-probe",
+ "rustls-pemfile 2.2.0",
+ "rustls-pki-types",
+ "schannel",
+ "security-framework",
+]
+
+[[package]]
+name = "rustls-pemfile"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1c74cae0a4cf6ccbbf5f359f08efdf8ee7e1dc532573bf0db71968cb56b1448c"
+dependencies = [
+ "base64 0.21.7",
+]
+
+[[package]]
+name = "rustls-pemfile"
+version = "2.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dce314e5fee3f39953d46bb63bb8a46d40c2f8fb7cc5a3b6cab2bde9721d6e50"
+dependencies = [
+ "rustls-pki-types",
+]
+
+[[package]]
+name = "rustls-pki-types"
+version = "1.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "917ce264624a4b4db1c364dcc35bfca9ded014d0a958cd47ad3e960e988ea51c"
+dependencies = [
+ "web-time",
+]
+
+[[package]]
+name = "rustls-platform-verifier"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a4c7dc240fec5517e6c4eab3310438636cfe6391dfc345ba013109909a90d136"
+dependencies = [
+ "core-foundation",
+ "core-foundation-sys",
+ "jni",
+ "log",
+ "once_cell",
+ "rustls 0.23.22",
+ "rustls-native-certs",
+ "rustls-platform-verifier-android",
+ "rustls-webpki 0.102.8",
+ "security-framework",
+ "security-framework-sys",
+ "webpki-root-certs",
+ "windows-sys 0.52.0",
+]
+
+[[package]]
+name = "rustls-platform-verifier-android"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f87165f0995f63a9fbeea62b64d10b4d9d8e78ec6d7d51fb2125fda7bb36788f"
+
+[[package]]
+name = "rustls-webpki"
+version = "0.101.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8b6275d1ee7a1cd780b64aca7726599a1dbc893b1e64144529e55c3c2f745765"
+dependencies = [
+ "ring",
+ "untrusted",
+]
+
+[[package]]
+name = "rustls-webpki"
+version = "0.102.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "64ca1bc8749bd4cf37b5ce386cc146580777b4e8572c7b97baf22c83f444bee9"
+dependencies = [
+ "ring",
+ "rustls-pki-types",
+ "untrusted",
+]
+
+[[package]]
+name = "rustversion"
+version = "1.0.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f7c45b9784283f1b2e7fb61b42047c2fd678ef0960d4f6f1eba131594cc369d4"
+
+[[package]]
+name = "ryu"
+version = "1.0.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6ea1a2d0a644769cc99faa24c3ad26b379b786fe7c36fd3c546254801650e6dd"
+
+[[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.27"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1f29ebaa345f945cec9fbbc532eb307f0fdad8161f281b6369539c8d84876b3d"
+dependencies = [
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "scopeguard"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49"
+
+[[package]]
+name = "scroll"
+version = "0.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "04c565b551bafbef4157586fa379538366e4385d42082f255bfd96e4fe8519da"
+
+[[package]]
+name = "sct"
+version = "0.7.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "da046153aa2352493d6cb7da4b6e5c0c057d8a1d0a9aa8560baffdd945acd414"
+dependencies = [
+ "ring",
+ "untrusted",
+]
+
+[[package]]
+name = "security-framework"
+version = "2.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "897b2245f0b511c87893af39b033e5ca9cce68824c4d7e7630b5a1d339658d02"
+dependencies = [
+ "bitflags 2.8.0",
+ "core-foundation",
+ "core-foundation-sys",
+ "libc",
+ "num-bigint 0.4.6",
+ "security-framework-sys",
+]
+
+[[package]]
+name = "security-framework-sys"
+version = "2.14.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "49db231d56a190491cb4aeda9527f1ad45345af50b0851622a7adb8c03b01c32"
+dependencies = [
+ "core-foundation-sys",
+ "libc",
+]
+
+[[package]]
+name = "semver"
+version = "1.0.25"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f79dfe2d285b0488816f30e700a7438c5a73d816b5b7d3ac72fbc48b0d185e03"
+
+[[package]]
+name = "seqlock"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b5c67b6f14ecc5b86c66fa63d76b5092352678545a8a3cdae80aef5128371910"
+dependencies = [
+ "parking_lot",
+]
+
+[[package]]
+name = "serde"
+version = "1.0.217"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "02fc4265df13d6fa1d00ecff087228cc0a2b5f3c0e87e258d8b94a156e984c70"
+dependencies = [
+ "serde_derive",
+]
+
+[[package]]
+name = "serde_bytes"
+version = "0.11.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "387cc504cb06bb40a96c8e04e951fe01854cf6bc921053c954e4a606d9675c6a"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "serde_derive"
+version = "1.0.217"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5a9bf7cf98d04a2b28aead066b7496853d4779c9cc183c440dbac457641e19a0"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "serde_json"
+version = "1.0.138"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d434192e7da787e94a6ea7e9670b26a036d0ca41e0b7efb2676dd32bae872949"
+dependencies = [
+ "itoa",
+ "memchr",
+ "ryu",
+ "serde",
+]
+
+[[package]]
+name = "serde_urlencoded"
+version = "0.7.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd"
+dependencies = [
+ "form_urlencoded",
+ "itoa",
+ "ryu",
+ "serde",
+]
+
+[[package]]
+name = "serde_with"
+version = "3.12.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d6b6f7f2fcb69f747921f79f3926bd1e203fce4fef62c268dd3abfb6d86029aa"
+dependencies = [
+ "serde",
+ "serde_derive",
+ "serde_with_macros",
+]
+
+[[package]]
+name = "serde_with_macros"
+version = "3.12.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8d00caa5193a3c8362ac2b73be6b9e768aa5a4b2f721d8f4b339600c3cb51f8e"
+dependencies = [
+ "darling",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "sha1"
+version = "0.10.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba"
+dependencies = [
+ "cfg-if",
+ "cpufeatures",
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "sha2"
+version = "0.9.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800"
+dependencies = [
+ "block-buffer 0.9.0",
+ "cfg-if",
+ "cpufeatures",
+ "digest 0.9.0",
+ "opaque-debug",
+]
+
+[[package]]
+name = "sha2"
+version = "0.10.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8"
+dependencies = [
+ "cfg-if",
+ "cpufeatures",
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "sha2-const-stable"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5f179d4e11094a893b82fff208f74d448a7512f99f5a0acbd5c679b705f83ed9"
+
+[[package]]
+name = "sha3"
+version = "0.10.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "75872d278a8f37ef87fa0ddbda7802605cb18344497949862c0d4dcb291eba60"
+dependencies = [
+ "digest 0.10.7",
+ "keccak",
+]
+
+[[package]]
+name = "sharded-slab"
+version = "0.1.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f40ca3c46823713e0d4209592e8d6e826aa57e928f09752619fc696c499637f6"
+dependencies = [
+ "lazy_static",
+]
+
+[[package]]
+name = "shlex"
+version = "1.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64"
+
+[[package]]
+name = "signal-hook-registry"
+version = "1.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a9e9e0b4211b72e7b8b6e85c807d36c212bdb33ea8587f7569562a84df5465b1"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "signature"
+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 = "sized-chunks"
+version = "0.6.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "16d69225bde7a69b235da73377861095455d298f2b970996eec25ddbb42b3d1e"
+dependencies = [
+ "bitmaps",
+ "typenum",
+]
+
+[[package]]
+name = "slab"
+version = "0.4.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8f92a496fb766b417c996b9c5e57daf2f7ad3b0bebe1ccfca4856390e3d3bb67"
+dependencies = [
+ "autocfg",
+]
+
+[[package]]
+name = "smallvec"
+version = "1.13.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67"
+
+[[package]]
+name = "socket2"
+version = "0.5.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c970269d99b64e60ec3bd6ad27270092a5394c4e309314b18ae3fe575695fbe8"
+dependencies = [
+ "libc",
+ "windows-sys 0.52.0",
+]
+
+[[package]]
+name = "solana-account"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "af199e43e00e46fd74c825a5a815a62c105b69fe24aeea67cce34167cb3c8246"
+dependencies = [
+ "bincode",
+ "serde",
+ "serde_bytes",
+ "serde_derive",
+ "solana-instruction",
+ "solana-program",
+]
+
+[[package]]
+name = "solana-account-decoder"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "62aebffd03315ce52610db64368738a93b76e2e10045eae1e816b75f4789096d"
+dependencies = [
+ "Inflector",
+ "base64 0.22.1",
+ "bincode",
+ "bs58",
+ "bv",
+ "lazy_static",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "solana-account-decoder-client-types",
+ "solana-config-program",
+ "solana-sdk",
+ "spl-token",
+ "spl-token-2022",
+ "spl-token-group-interface",
+ "spl-token-metadata-interface",
+ "thiserror 1.0.69",
+ "zstd",
+]
+
+[[package]]
+name = "solana-account-decoder-client-types"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "387a8d7161b5643d2e7be8c58ab5ebbb1819f1fc8ed7e225d1379f0b44886d25"
+dependencies = [
+ "base64 0.22.1",
+ "bs58",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "solana-account",
+ "solana-pubkey",
+ "zstd",
+]
+
+[[package]]
+name = "solana-account-info"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0108d43d0dc6cb62f2fb5c5c6dddbbd2944cd41d674e78cfd7ab25505112fa83"
+dependencies = [
+ "bincode",
+ "serde",
+ "solana-program-error",
+ "solana-program-memory",
+ "solana-pubkey",
+]
+
+[[package]]
+name = "solana-accounts-db"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2a787a384cd2286bf62485e81c306a60815eeaca25fb641fb9a7dedd372131c6"
+dependencies = [
+ "ahash",
+ "bincode",
+ "blake3",
+ "bv",
+ "bytemuck",
+ "bytemuck_derive",
+ "bzip2",
+ "crossbeam-channel",
+ "dashmap",
+ "index_list",
+ "indexmap",
+ "itertools 0.12.1",
+ "lazy_static",
+ "log",
+ "lz4",
+ "memmap2",
+ "modular-bitfield",
+ "num_cpus",
+ "num_enum",
+ "rand 0.8.5",
+ "rayon",
+ "seqlock",
+ "serde",
+ "serde_derive",
+ "smallvec",
+ "solana-bucket-map",
+ "solana-inline-spl",
+ "solana-lattice-hash",
+ "solana-measure",
+ "solana-metrics",
+ "solana-nohash-hasher",
+ "solana-rayon-threadlimit",
+ "solana-sdk",
+ "solana-svm-transaction",
+ "static_assertions",
+ "tar",
+ "tempfile",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-address-lookup-table-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f00dd5f6df644e150e398c35839b6d216f05b42f37516311f1f25ae345d02749"
+dependencies = [
+ "bincode",
+ "bytemuck",
+ "log",
+ "num-derive",
+ "num-traits",
+ "solana-feature-set",
+ "solana-log-collector",
+ "solana-program",
+ "solana-program-runtime",
+ "solana-sdk",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-atomic-u64"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fb870569c01e96a8bada5423e1e2499f65c83fc23e18d15290c921eaa7464562"
+dependencies = [
+ "parking_lot",
+]
+
+[[package]]
+name = "solana-banks-client"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8c6c37a5757134ea9818fa1ad50c57f3c0a8a2320eaca1b3b40d07d0d25097b4"
+dependencies = [
+ "borsh 1.5.5",
+ "futures",
+ "solana-banks-interface",
+ "solana-program",
+ "solana-sdk",
+ "tarpc",
+ "thiserror 1.0.69",
+ "tokio",
+ "tokio-serde",
+]
+
+[[package]]
+name = "solana-banks-interface"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cee35cee7b970e10a98106b2eaf0a24a1bd03cc0408d875a052b65f1aa5fa6b9"
+dependencies = [
+ "serde",
+ "serde_derive",
+ "solana-sdk",
+ "tarpc",
+]
+
+[[package]]
+name = "solana-banks-server"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c2535d2c3dbebf0cddb48da4c94741322ffc35b893e4ad60f6e6f475b5cf86ee"
+dependencies = [
+ "bincode",
+ "crossbeam-channel",
+ "futures",
+ "solana-banks-interface",
+ "solana-client",
+ "solana-feature-set",
+ "solana-runtime",
+ "solana-sdk",
+ "solana-send-transaction-service",
+ "solana-svm",
+ "tarpc",
+ "tokio",
+ "tokio-serde",
+]
+
+[[package]]
+name = "solana-bincode"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "eeb4dc964902dbe248bcfc57d6c5e7e814d300e0a8fee1fb5e9e517dacda0c12"
+dependencies = [
+ "bincode",
+ "serde",
+ "solana-instruction",
+]
+
+[[package]]
+name = "solana-bn254"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a5f3acfa2be716cce5ae92a0e6919e48ea84505a81791f1f4966d2cabd06abdf"
+dependencies = [
+ "ark-bn254",
+ "ark-ec",
+ "ark-ff",
+ "ark-serialize",
+ "bytemuck",
+ "solana-program",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-borsh"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9616b7f3914d8b630d8bc61bed5340c05ec2d932e7c9694615c1c37f312529c6"
+dependencies = [
+ "borsh 0.10.4",
+ "borsh 1.5.5",
+]
+
+[[package]]
+name = "solana-bpf-loader-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a72f4ec4df7ac88310560ec9e03358017176d38d6947f0de118925b31bfcc57c"
+dependencies = [
+ "bincode",
+ "byteorder",
+ "libsecp256k1",
+ "log",
+ "scopeguard",
+ "solana-bn254",
+ "solana-compute-budget",
+ "solana-curve25519",
+ "solana-feature-set",
+ "solana-log-collector",
+ "solana-measure",
+ "solana-poseidon",
+ "solana-program-memory",
+ "solana-program-runtime",
+ "solana-sdk",
+ "solana-timings",
+ "solana-type-overrides",
+ "solana_rbpf",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-bucket-map"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0cd2b01f8e3db75b6087747c62a064cbe6392a110a3766d01d8c6286280b981a"
+dependencies = [
+ "bv",
+ "bytemuck",
+ "bytemuck_derive",
+ "log",
+ "memmap2",
+ "modular-bitfield",
+ "num_enum",
+ "rand 0.8.5",
+ "solana-measure",
+ "solana-sdk",
+ "tempfile",
+]
+
+[[package]]
+name = "solana-builtins-default-costs"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "32ac719f0730c4371f20f8fc37963b639907f2a0587a616968ee701a80ff1199"
+dependencies = [
+ "ahash",
+ "lazy_static",
+ "log",
+ "solana-address-lookup-table-program",
+ "solana-bpf-loader-program",
+ "solana-compute-budget-program",
+ "solana-config-program",
+ "solana-loader-v4-program",
+ "solana-sdk",
+ "solana-stake-program",
+ "solana-system-program",
+ "solana-vote-program",
+]
+
+[[package]]
+name = "solana-client"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "55bd69782e1a5ac38beb8621970318f8a3c242cb6ed48d06b7862465b9c624e2"
+dependencies = [
+ "async-trait",
+ "bincode",
+ "dashmap",
+ "futures",
+ "futures-util",
+ "indexmap",
+ "indicatif",
+ "log",
+ "quinn",
+ "rayon",
+ "solana-connection-cache",
+ "solana-measure",
+ "solana-pubsub-client",
+ "solana-quic-client",
+ "solana-rpc-client",
+ "solana-rpc-client-api",
+ "solana-rpc-client-nonce-utils",
+ "solana-sdk",
+ "solana-streamer",
+ "solana-thin-client",
+ "solana-tpu-client",
+ "solana-udp-client",
+ "thiserror 1.0.69",
+ "tokio",
+]
+
+[[package]]
+name = "solana-clock"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "27b7f7803b61bc339c57d7b3beed7c93d9757dcd2fdc90bf74660670378ed307"
+dependencies = [
+ "serde",
+ "serde_derive",
+ "solana-sdk-macro",
+ "solana-sysvar-id",
+]
+
+[[package]]
+name = "solana-compute-budget"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9ec695847457a262bd5fc5a834bb00215ad39ebc4111a9db4003bc0df86ca86b"
+dependencies = [
+ "solana-sdk",
+]
+
+[[package]]
+name = "solana-compute-budget-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b0d748e714ed38f97d0bd2c2b5374dea962f2056c3c1335a77932fbd35310598"
+dependencies = [
+ "solana-program-runtime",
+ "solana-sdk",
+]
+
+[[package]]
+name = "solana-config-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1dd59294e3aab93e718ba128c8022869ae2ac8a76762ab4c9cafad721dd449a4"
+dependencies = [
+ "bincode",
+ "chrono",
+ "serde",
+ "serde_derive",
+ "solana-log-collector",
+ "solana-program-runtime",
+ "solana-sdk",
+ "solana-short-vec",
+]
+
+[[package]]
+name = "solana-connection-cache"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ddd194666e1ddc8123229510d73572c9221d1a420786903ea3d3f1fb31ac5d50"
+dependencies = [
+ "async-trait",
+ "bincode",
+ "crossbeam-channel",
+ "futures-util",
+ "indexmap",
+ "log",
+ "rand 0.8.5",
+ "rayon",
+ "solana-measure",
+ "solana-metrics",
+ "solana-sdk",
+ "thiserror 1.0.69",
+ "tokio",
+]
+
+[[package]]
+name = "solana-cost-model"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4728f200bafab770831a1d8777374bdb900ec60b17332d5894f264da65bb2c29"
+dependencies = [
+ "ahash",
+ "lazy_static",
+ "log",
+ "solana-builtins-default-costs",
+ "solana-compute-budget",
+ "solana-feature-set",
+ "solana-metrics",
+ "solana-runtime-transaction",
+ "solana-sdk",
+ "solana-svm-transaction",
+ "solana-vote-program",
+]
+
+[[package]]
+name = "solana-cpi"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "faece18bae6d818d5121b40f10acd5a0f1f0d1ee2283e79ad00825a43a4dfaaa"
+dependencies = [
+ "solana-account-info",
+ "solana-define-syscall",
+ "solana-instruction",
+ "solana-program-error",
+ "solana-pubkey",
+ "solana-stable-layout",
+]
+
+[[package]]
+name = "solana-curve25519"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ddd364b9c90c05406baa604743a266a4526fd7ac1e7a048c40971ca28200851b"
+dependencies = [
+ "bytemuck",
+ "bytemuck_derive",
+ "curve25519-dalek 4.1.3",
+ "solana-program",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-decode-error"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e97eddcef283b84b0bc040550796de561dd52d5b2599d3791d2d8d20121dc6dc"
+dependencies = [
+ "num-traits",
+]
+
+[[package]]
+name = "solana-define-syscall"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "366739af4e4606111a1c62ab1c91376d7d42864f79b9b863b5544fc56e140c27"
+
+[[package]]
+name = "solana-derivation-path"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "549c98194387fac54f13558c701653bca55a14f2fbfa007ecc4fae601832cbc5"
+dependencies = [
+ "derivation-path",
+ "qstring",
+ "uriparse",
+]
+
+[[package]]
+name = "solana-epoch-schedule"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fd5c9e8f4a155d29cb8e6aa2e4856e1200c6d1b41d63c189823a074a55e31aff"
+dependencies = [
+ "serde",
+ "serde_derive",
+ "solana-sdk-macro",
+ "solana-sysvar-id",
+]
+
+[[package]]
+name = "solana-feature-set"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6b782063b475df133508569bc7b4bb490b067a45da35c7aa2eea0f758c159c79"
+dependencies = [
+ "lazy_static",
+ "solana-clock",
+ "solana-epoch-schedule",
+ "solana-hash",
+ "solana-pubkey",
+ "solana-sha256-hasher",
+]
+
+[[package]]
+name = "solana-fee"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "226b0f10ae1d4f032f0a1191e92f9c3540586bb9f1d23d65bd7282b4885dabea"
+dependencies = [
+ "solana-sdk",
+ "solana-svm-transaction",
+]
+
+[[package]]
+name = "solana-fee-calculator"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8959bafc64a69bc622b8ac8079f665aefcad14b234629ee742f34deef2d3167f"
+dependencies = [
+ "log",
+ "serde",
+ "serde_derive",
+]
+
+[[package]]
+name = "solana-hash"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d7983e14fd607a13fa6ef89ab2ddad91fd3e65fd24b2b37c142ddc94ac8cb27"
+dependencies = [
+ "borsh 1.5.5",
+ "bs58",
+ "bytemuck",
+ "bytemuck_derive",
+ "js-sys",
+ "serde",
+ "serde_derive",
+ "solana-atomic-u64",
+ "solana-sanitize",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "solana-inflation"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c4250c724fafb1b30c39eb5af6f45ce918611d148712e3aa9b55470cc15069d7"
+dependencies = [
+ "serde",
+ "serde_derive",
+]
+
+[[package]]
+name = "solana-inline-spl"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1c2e58fd15eed588914072f91ceaf19f4c9d59aab287a8bd5a5a97a820bc9c0b"
+dependencies = [
+ "bytemuck",
+ "solana-pubkey",
+]
+
+[[package]]
+name = "solana-instruction"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "34de1bf4e8abcc182bf90888b84b9656eee5fef7a252a44a20fed7c778abfbda"
+dependencies = [
+ "bincode",
+ "borsh 1.5.5",
+ "getrandom 0.2.15",
+ "js-sys",
+ "num-traits",
+ "serde",
+ "serde_derive",
+ "solana-define-syscall",
+ "solana-pubkey",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "solana-last-restart-slot"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "21db3a4a50b400bb03d32971bb9fefafb421e4210f86081d3c3217d8d98c1a77"
+dependencies = [
+ "serde",
+ "serde_derive",
+ "solana-sdk-macro",
+ "solana-sysvar-id",
+]
+
+[[package]]
+name = "solana-lattice-hash"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "011d73028747245389f366f6e9805d2565110be1f31e3bd5bf7f571d92605760"
+dependencies = [
+ "base64 0.22.1",
+ "blake3",
+ "bs58",
+ "bytemuck",
+]
+
+[[package]]
+name = "solana-loader-v4-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7521d33996e367b8c434c92ce3a3fa70ec76c2d6b2f5712f4e8ce206ff6a4ba8"
+dependencies = [
+ "log",
+ "solana-bpf-loader-program",
+ "solana-compute-budget",
+ "solana-log-collector",
+ "solana-measure",
+ "solana-program-runtime",
+ "solana-sdk",
+ "solana-type-overrides",
+ "solana_rbpf",
+]
+
+[[package]]
+name = "solana-log-collector"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b0af672ade9beba20d59eb0b6e3a9a1800b8c772138bcadb96bf6894bc552821"
+dependencies = [
+ "log",
+]
+
+[[package]]
+name = "solana-logger"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0e607485a4b8b66f7d3592e4a79355073cb7e6a101c244e5fa0aa52bfdce506"
+dependencies = [
+ "env_logger",
+ "lazy_static",
+ "log",
+]
+
+[[package]]
+name = "solana-measure"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6f3f0a8b37208c237452f2612049f338711c8d28242cb989c4d755452d84a6e0"
+
+[[package]]
+name = "solana-metrics"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d2f52947d3edea1d8924dd9f8b9898fb9c828887198871b5ba9af0e0597757bc"
+dependencies = [
+ "crossbeam-channel",
+ "gethostname",
+ "lazy_static",
+ "log",
+ "reqwest",
+ "solana-sdk",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-msg"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b772497c73933dc8f0123f1fc1c8ccae1e6edee194606e7fd5cb0a2f074f0cc6"
+dependencies = [
+ "solana-define-syscall",
+]
+
+[[package]]
+name = "solana-native-token"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "880eea211f489f04d9aa8d436ea3dd172b1188dee97bdd7ec64acb05bef29a32"
+
+[[package]]
+name = "solana-net-utils"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a1e278dffb6a0a99f5b152764196c1d3a5f3b7e0d59cb8c158b4e31b70cf89c3"
+dependencies = [
+ "bincode",
+ "crossbeam-channel",
+ "log",
+ "nix",
+ "rand 0.8.5",
+ "serde",
+ "serde_derive",
+ "socket2",
+ "solana-sdk",
+ "tokio",
+ "url",
+]
+
+[[package]]
+name = "solana-nohash-hasher"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8b8a731ed60e89177c8a7ab05fe0f1511cedd3e70e773f288f9de33a9cfdc21e"
+
+[[package]]
+name = "solana-packet"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bf06ea5042a29391f26bfbeeb30cd4709de4f47642f8a0ec2e15c5faa099cf44"
+dependencies = [
+ "bincode",
+ "bitflags 2.8.0",
+ "cfg_eval",
+ "serde",
+ "serde_derive",
+ "serde_with",
+]
+
+[[package]]
+name = "solana-perf"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1048cc942d584b263124076e42970ef1f0b1104a7cfa703e1badd28fafb2dd21"
+dependencies = [
+ "ahash",
+ "bincode",
+ "bv",
+ "caps",
+ "curve25519-dalek 4.1.3",
+ "dlopen2",
+ "fnv",
+ "lazy_static",
+ "libc",
+ "log",
+ "nix",
+ "rand 0.8.5",
+ "rayon",
+ "serde",
+ "solana-metrics",
+ "solana-rayon-threadlimit",
+ "solana-sdk",
+ "solana-short-vec",
+ "solana-vote-program",
+]
+
+[[package]]
+name = "solana-poseidon"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9cb8f71202f04231fbc0869c94da00e7198abc9117100c62e4efb3cdf533f39d"
+dependencies = [
+ "ark-bn254",
+ "light-poseidon",
+ "solana-define-syscall",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-precompile-error"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "307485e198c542e975a916c354764c2ced285b6df520793a2f10beb39d0d7f1c"
+dependencies = [
+ "num-traits",
+ "solana-decode-error",
+]
+
+[[package]]
+name = "solana-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0a2334209564c9fa7673b01a5a5dac16c0a474bea95e218135ef1430c20ecba5"
+dependencies = [
+ "base64 0.22.1",
+ "bincode",
+ "bitflags 2.8.0",
+ "blake3",
+ "borsh 0.10.4",
+ "borsh 1.5.5",
+ "bs58",
+ "bv",
+ "bytemuck",
+ "bytemuck_derive",
+ "console_error_panic_hook",
+ "console_log",
+ "curve25519-dalek 4.1.3",
+ "five8_const",
+ "getrandom 0.2.15",
+ "js-sys",
+ "lazy_static",
+ "log",
+ "memoffset",
+ "num-bigint 0.4.6",
+ "num-derive",
+ "num-traits",
+ "parking_lot",
+ "rand 0.8.5",
+ "serde",
+ "serde_bytes",
+ "serde_derive",
+ "sha2 0.10.8",
+ "sha3",
+ "solana-account-info",
+ "solana-atomic-u64",
+ "solana-bincode",
+ "solana-borsh",
+ "solana-clock",
+ "solana-cpi",
+ "solana-decode-error",
+ "solana-define-syscall",
+ "solana-epoch-schedule",
+ "solana-fee-calculator",
+ "solana-hash",
+ "solana-instruction",
+ "solana-last-restart-slot",
+ "solana-msg",
+ "solana-native-token",
+ "solana-program-entrypoint",
+ "solana-program-error",
+ "solana-program-memory",
+ "solana-program-option",
+ "solana-program-pack",
+ "solana-pubkey",
+ "solana-rent",
+ "solana-sanitize",
+ "solana-sdk-macro",
+ "solana-secp256k1-recover",
+ "solana-serde-varint",
+ "solana-serialize-utils",
+ "solana-sha256-hasher",
+ "solana-short-vec",
+ "solana-slot-hashes",
+ "solana-slot-history",
+ "solana-stable-layout",
+ "solana-sysvar-id",
+ "solana-transaction-error",
+ "thiserror 1.0.69",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "solana-program-entrypoint"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "39fdb97781dc0f4093f0a5ebe0b5c353da217d8c8a13fb7ded3c62a4774f7d13"
+dependencies = [
+ "solana-account-info",
+ "solana-msg",
+ "solana-program-error",
+ "solana-pubkey",
+]
+
+[[package]]
+name = "solana-program-error"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a18ee0aac9905389f424598fd3656c981ed09268c04e160d08a468b25ecc7b1"
+dependencies = [
+ "borsh 1.5.5",
+ "num-traits",
+ "serde",
+ "serde_derive",
+ "solana-decode-error",
+ "solana-instruction",
+ "solana-msg",
+ "solana-pubkey",
+]
+
+[[package]]
+name = "solana-program-memory"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d4473c6315820301c873c3df0a110a4ded9874579be27362a1a7cc4b8c904d37"
+dependencies = [
+ "num-traits",
+ "solana-define-syscall",
+]
+
+[[package]]
+name = "solana-program-option"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c68158410c335eeb28e95520ab3628c6d887ea6acb07f20dfce55780e127cf87"
+
+[[package]]
+name = "solana-program-pack"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "31d03d7ad1c427cf50af1c51f82abe330c5dec09a0a4bace05fb303788fe58d8"
+dependencies = [
+ "solana-program-error",
+]
+
+[[package]]
+name = "solana-program-runtime"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "18c14d71bc78f0b09c3dd3f4cb06e1dab74354b2a96c79e86441e04cfeabebd7"
+dependencies = [
+ "base64 0.22.1",
+ "bincode",
+ "enum-iterator",
+ "itertools 0.12.1",
+ "libc",
+ "log",
+ "num-derive",
+ "num-traits",
+ "percentage",
+ "rand 0.8.5",
+ "serde",
+ "solana-compute-budget",
+ "solana-feature-set",
+ "solana-log-collector",
+ "solana-measure",
+ "solana-metrics",
+ "solana-sdk",
+ "solana-timings",
+ "solana-type-overrides",
+ "solana-vote",
+ "solana_rbpf",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-program-test"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "02031132d3d65b1193d2a5ac0fed4a58203badefc225df811db5b4cc434c7a4e"
+dependencies = [
+ "assert_matches",
+ "async-trait",
+ "base64 0.22.1",
+ "bincode",
+ "chrono-humanize",
+ "crossbeam-channel",
+ "log",
+ "serde",
+ "solana-accounts-db",
+ "solana-banks-client",
+ "solana-banks-interface",
+ "solana-banks-server",
+ "solana-bpf-loader-program",
+ "solana-compute-budget",
+ "solana-feature-set",
+ "solana-inline-spl",
+ "solana-instruction",
+ "solana-log-collector",
+ "solana-logger",
+ "solana-program-runtime",
+ "solana-runtime",
+ "solana-sdk",
+ "solana-svm",
+ "solana-timings",
+ "solana-vote-program",
+ "solana_rbpf",
+ "thiserror 1.0.69",
+ "tokio",
+]
+
+[[package]]
+name = "solana-pubkey"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d87a0e1282ea13d3fab42d1c5df8f00283f8f739b2e5a9a5b530f76c4292e2b5"
+dependencies = [
+ "borsh 0.10.4",
+ "borsh 1.5.5",
+ "bs58",
+ "bytemuck",
+ "bytemuck_derive",
+ "curve25519-dalek 4.1.3",
+ "five8_const",
+ "getrandom 0.2.15",
+ "js-sys",
+ "num-traits",
+ "rand 0.8.5",
+ "serde",
+ "serde_derive",
+ "solana-atomic-u64",
+ "solana-decode-error",
+ "solana-define-syscall",
+ "solana-sanitize",
+ "solana-sha256-hasher",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "solana-pubsub-client"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "350983b022bab3228143099f90cc422a2a18423c3321527be9e8365a47dae8e3"
+dependencies = [
+ "crossbeam-channel",
+ "futures-util",
+ "log",
+ "reqwest",
+ "semver",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "solana-account-decoder",
+ "solana-rpc-client-api",
+ "solana-sdk",
+ "thiserror 1.0.69",
+ "tokio",
+ "tokio-stream",
+ "tokio-tungstenite",
+ "tungstenite",
+ "url",
+]
+
+[[package]]
+name = "solana-quic-client"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e92ae085cdccd05290c8ed630a9a3fe8f69c5cb7d94fb274c2ffd627c95e128"
+dependencies = [
+ "async-lock",
+ "async-trait",
+ "futures",
+ "itertools 0.12.1",
+ "lazy_static",
+ "log",
+ "quinn",
+ "quinn-proto",
+ "rustls 0.23.22",
+ "solana-connection-cache",
+ "solana-measure",
+ "solana-metrics",
+ "solana-net-utils",
+ "solana-rpc-client-api",
+ "solana-sdk",
+ "solana-streamer",
+ "thiserror 1.0.69",
+ "tokio",
+]
+
+[[package]]
+name = "solana-rayon-threadlimit"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "028863b869a84a064967c5b9d174f5424af66cdf6be6cfc38f31f68e7e94b655"
+dependencies = [
+ "lazy_static",
+ "num_cpus",
+]
+
+[[package]]
+name = "solana-rent"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0ae7a8fb4d869c6134d9ebdde2bb68e4cfd6034fe947932ade7721e236b66bb"
+dependencies = [
+ "serde",
+ "serde_derive",
+ "solana-sdk-macro",
+ "solana-sysvar-id",
+]
+
+[[package]]
+name = "solana-rpc-client"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ea4f8b9b337acd7e3ac0376bc79397bd64a292d9a839a58889ffad0ccf0b3300"
+dependencies = [
+ "async-trait",
+ "base64 0.22.1",
+ "bincode",
+ "bs58",
+ "indicatif",
+ "log",
+ "reqwest",
+ "reqwest-middleware",
+ "semver",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "solana-account-decoder-client-types",
+ "solana-rpc-client-api",
+ "solana-sdk",
+ "solana-transaction-status-client-types",
+ "solana-version",
+ "solana-vote-program",
+ "tokio",
+]
+
+[[package]]
+name = "solana-rpc-client-api"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "abe8a2d84814609dcb41e3a6953e8ef1631d3a75e0e2c553dd2292fb3137d26f"
+dependencies = [
+ "anyhow",
+ "base64 0.22.1",
+ "bs58",
+ "jsonrpc-core",
+ "reqwest",
+ "reqwest-middleware",
+ "semver",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "solana-account-decoder-client-types",
+ "solana-inline-spl",
+ "solana-sdk",
+ "solana-transaction-status-client-types",
+ "solana-version",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-rpc-client-nonce-utils"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c65fafc816a26d2c4df9551e66f7b088f36549d04b5f17e99ecc7b08e5d15c02"
+dependencies = [
+ "solana-rpc-client",
+ "solana-sdk",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-runtime"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c7fb6646ddbaf08011d35d7925962c6797826ee9b39a8a38910c9000e78b83ea"
+dependencies = [
+ "ahash",
+ "aquamarine",
+ "arrayref",
+ "base64 0.22.1",
+ "bincode",
+ "blake3",
+ "bv",
+ "bytemuck",
+ "byteorder",
+ "bzip2",
+ "crossbeam-channel",
+ "dashmap",
+ "dir-diff",
+ "flate2",
+ "fnv",
+ "im",
+ "index_list",
+ "itertools 0.12.1",
+ "lazy_static",
+ "libc",
+ "log",
+ "lz4",
+ "memmap2",
+ "mockall",
+ "modular-bitfield",
+ "num-derive",
+ "num-traits",
+ "num_cpus",
+ "num_enum",
+ "percentage",
+ "qualifier_attr",
+ "rand 0.8.5",
+ "rayon",
+ "regex",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "serde_with",
+ "solana-accounts-db",
+ "solana-address-lookup-table-program",
+ "solana-bpf-loader-program",
+ "solana-bucket-map",
+ "solana-compute-budget",
+ "solana-compute-budget-program",
+ "solana-config-program",
+ "solana-cost-model",
+ "solana-feature-set",
+ "solana-fee",
+ "solana-inline-spl",
+ "solana-lattice-hash",
+ "solana-loader-v4-program",
+ "solana-measure",
+ "solana-metrics",
+ "solana-perf",
+ "solana-program",
+ "solana-program-runtime",
+ "solana-rayon-threadlimit",
+ "solana-runtime-transaction",
+ "solana-sdk",
+ "solana-stake-program",
+ "solana-svm",
+ "solana-svm-rent-collector",
+ "solana-svm-transaction",
+ "solana-system-program",
+ "solana-timings",
+ "solana-transaction-status",
+ "solana-version",
+ "solana-vote",
+ "solana-vote-program",
+ "solana-zk-elgamal-proof-program",
+ "solana-zk-sdk",
+ "solana-zk-token-proof-program",
+ "solana-zk-token-sdk",
+ "static_assertions",
+ "strum",
+ "strum_macros",
+ "symlink",
+ "tar",
+ "tempfile",
+ "thiserror 1.0.69",
+ "zstd",
+]
+
+[[package]]
+name = "solana-runtime-transaction"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b48bbdc09f0b2d445c1a36b2fa0739b0c30e4c6b4349899762e71280e0c68182"
+dependencies = [
+ "agave-transaction-view",
+ "log",
+ "solana-builtins-default-costs",
+ "solana-compute-budget",
+ "solana-pubkey",
+ "solana-sdk",
+ "solana-svm-transaction",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-sanitize"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f83981cbb16d532c96a7911e810bf3302fd14fd2651096130c3d3cc4e2b3410e"
+
+[[package]]
+name = "solana-sdk"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "54e2e78e01f045ae38199cacd980773e711ac8c8a3af6d9d21424184aa4dae1a"
+dependencies = [
+ "bincode",
+ "bitflags 2.8.0",
+ "borsh 1.5.5",
+ "bs58",
+ "bytemuck",
+ "bytemuck_derive",
+ "byteorder",
+ "chrono",
+ "digest 0.10.7",
+ "ed25519-dalek",
+ "ed25519-dalek-bip32",
+ "getrandom 0.1.16",
+ "hmac 0.12.1",
+ "itertools 0.12.1",
+ "js-sys",
+ "lazy_static",
+ "libsecp256k1",
+ "log",
+ "memmap2",
+ "num-derive",
+ "num-traits",
+ "num_enum",
+ "pbkdf2",
+ "rand 0.7.3",
+ "rand 0.8.5",
+ "serde",
+ "serde_bytes",
+ "serde_derive",
+ "serde_json",
+ "serde_with",
+ "sha2 0.10.8",
+ "sha3",
+ "siphasher",
+ "solana-account",
+ "solana-bn254",
+ "solana-decode-error",
+ "solana-derivation-path",
+ "solana-feature-set",
+ "solana-inflation",
+ "solana-instruction",
+ "solana-native-token",
+ "solana-packet",
+ "solana-precompile-error",
+ "solana-program",
+ "solana-program-memory",
+ "solana-pubkey",
+ "solana-sanitize",
+ "solana-sdk-macro",
+ "solana-secp256k1-recover",
+ "solana-secp256r1-program",
+ "solana-serde-varint",
+ "solana-short-vec",
+ "solana-signature",
+ "solana-transaction-error",
+ "thiserror 1.0.69",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "solana-sdk-macro"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6822c0596b48342e7e6acebce6b0a4aa6bb57c9bef03ebf11405b5f5c7124627"
+dependencies = [
+ "bs58",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "solana-secp256k1-recover"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a4e19dfea19083c99baecbf09dc96f0c6abf56e39a610cb1c817935e54e03eaf"
+dependencies = [
+ "borsh 1.5.5",
+ "libsecp256k1",
+ "solana-define-syscall",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-secp256r1-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "12a60175bd0890161815bf5a1b8bec6f1db2fe5c4cd2b6d0a8eea3db6b362930"
+dependencies = [
+ "bytemuck",
+ "openssl",
+ "solana-feature-set",
+ "solana-instruction",
+ "solana-precompile-error",
+ "solana-pubkey",
+]
+
+[[package]]
+name = "solana-security-txt"
+version = "1.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "468aa43b7edb1f9b7b7b686d5c3aeb6630dc1708e86e31343499dd5c4d775183"
+
+[[package]]
+name = "solana-send-transaction-service"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "abc3dd272c12fcc043f90bd1546183bd5ac60b8dbf508055c32b9eb5b51e9fac"
+dependencies = [
+ "crossbeam-channel",
+ "log",
+ "solana-client",
+ "solana-connection-cache",
+ "solana-measure",
+ "solana-metrics",
+ "solana-runtime",
+ "solana-sdk",
+ "solana-tpu-client",
+]
+
+[[package]]
+name = "solana-serde-varint"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "75c3085b541f48338be527a5bb55bab216dd5b3715184473239efa09719e935d"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "solana-serialize-utils"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "410bd8c5aab8a447fcb925e59e03753333f1a43483cf2445d700ba72d5c133a6"
+dependencies = [
+ "solana-instruction",
+ "solana-pubkey",
+ "solana-sanitize",
+]
+
+[[package]]
+name = "solana-sha256-hasher"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8ef58345b6473a4045299ce77a90a1e6c46e360ed94899c9178fe2d03ea3e5e4"
+dependencies = [
+ "sha2 0.10.8",
+ "solana-define-syscall",
+ "solana-hash",
+]
+
+[[package]]
+name = "solana-short-vec"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5a27e2842e02e138a8a4156f7d71c318539f8aae65b80bc7dc311e0ee350eefa"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "solana-signature"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7da51452851bdbee15558f23f23579b3dcdf93aefdbcb911bca41bbf1e875f0a"
+dependencies = [
+ "bs58",
+ "ed25519-dalek",
+ "generic-array",
+ "rand 0.8.5",
+ "serde",
+ "serde_derive",
+ "solana-sanitize",
+]
+
+[[package]]
+name = "solana-slot-hashes"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2f7c1e1ea9b27d21b094162462b026f47de83102fafabf9ed2b6221d59e6f28a"
+dependencies = [
+ "serde",
+ "serde_derive",
+ "solana-hash",
+ "solana-sysvar-id",
+]
+
+[[package]]
+name = "solana-slot-history"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0882d70244b024ae16463c7c8ad547f8ee4e42c4331a0641f7e817c9da3fac49"
+dependencies = [
+ "bv",
+ "serde",
+ "serde_derive",
+ "solana-sysvar-id",
+]
+
+[[package]]
+name = "solana-stable-layout"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7db19a6a2bab0e421b73ebd2930e990c391ac34ccb14b464108b1d071fb44fa7"
+dependencies = [
+ "solana-instruction",
+ "solana-pubkey",
+]
+
+[[package]]
+name = "solana-stake-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4743926d81b270d81a8c8e65ae9670bb71f0013d3075396c449856bd0d5b0c99"
+dependencies = [
+ "bincode",
+ "log",
+ "solana-config-program",
+ "solana-feature-set",
+ "solana-log-collector",
+ "solana-program-runtime",
+ "solana-sdk",
+ "solana-type-overrides",
+ "solana-vote-program",
+]
+
+[[package]]
+name = "solana-streamer"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bb417ef7fae4238b28ded6679a1ccb6376f9bf7416ab615ed11a136924ff4390"
+dependencies = [
+ "async-channel",
+ "bytes",
+ "crossbeam-channel",
+ "dashmap",
+ "futures",
+ "futures-util",
+ "governor",
+ "histogram",
+ "indexmap",
+ "itertools 0.12.1",
+ "libc",
+ "log",
+ "nix",
+ "pem",
+ "percentage",
+ "quinn",
+ "quinn-proto",
+ "rand 0.8.5",
+ "rustls 0.23.22",
+ "smallvec",
+ "socket2",
+ "solana-measure",
+ "solana-metrics",
+ "solana-perf",
+ "solana-sdk",
+ "solana-transaction-metrics-tracker",
+ "thiserror 1.0.69",
+ "tokio",
+ "tokio-util 0.7.13",
+ "x509-parser",
+]
+
+[[package]]
+name = "solana-svm"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "01f53424b9d9b897a8d6352b744d843f145cf513b71b677b6326502bff5c7661"
+dependencies = [
+ "itertools 0.12.1",
+ "log",
+ "percentage",
+ "serde",
+ "serde_derive",
+ "solana-bpf-loader-program",
+ "solana-compute-budget",
+ "solana-feature-set",
+ "solana-fee",
+ "solana-loader-v4-program",
+ "solana-log-collector",
+ "solana-measure",
+ "solana-program-runtime",
+ "solana-runtime-transaction",
+ "solana-sdk",
+ "solana-svm-rent-collector",
+ "solana-svm-transaction",
+ "solana-system-program",
+ "solana-timings",
+ "solana-type-overrides",
+ "solana-vote",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-svm-rent-collector"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0cd3aa6296089bdf08f74d4b63170823e0c60e7957089160ac2444f76dad54f5"
+dependencies = [
+ "solana-sdk",
+]
+
+[[package]]
+name = "solana-svm-transaction"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "67274d9edb34c7182c8c31b29474b2b6f0156fef5369265f6e61e01b9d8f05fc"
+dependencies = [
+ "solana-sdk",
+]
+
+[[package]]
+name = "solana-system-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "58e1e57bab6f597a95790ce6a1133a8984dd5e4939bffe87ac08b7b0fa1f545f"
+dependencies = [
+ "bincode",
+ "log",
+ "serde",
+ "serde_derive",
+ "solana-log-collector",
+ "solana-program-runtime",
+ "solana-sdk",
+ "solana-type-overrides",
+]
+
+[[package]]
+name = "solana-sysvar-id"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "803ec4576594f2c633f670be4cfe2e8febc6e8fb387a0f3f2555972b7309bdb6"
+dependencies = [
+ "solana-pubkey",
+]
+
+[[package]]
+name = "solana-thin-client"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "86df1ee1cb5378f10efec879e266761f060b4140f1e57c0a384e141cfa3048ad"
+dependencies = [
+ "bincode",
+ "log",
+ "rayon",
+ "solana-connection-cache",
+ "solana-rpc-client",
+ "solana-rpc-client-api",
+ "solana-sdk",
+]
+
+[[package]]
+name = "solana-timings"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2a9d1bb674024ec73911c7f86020b270d982ee39888598549938800d11bb510d"
+dependencies = [
+ "eager",
+ "enum-iterator",
+ "solana-sdk",
+]
+
+[[package]]
+name = "solana-tpu-client"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "331c9627d2bf56260c2efe4d8e4db012c7e0610fc5381299f2712342442d3e46"
+dependencies = [
+ "async-trait",
+ "bincode",
+ "futures-util",
+ "indexmap",
+ "indicatif",
+ "log",
+ "rayon",
+ "solana-connection-cache",
+ "solana-measure",
+ "solana-pubsub-client",
+ "solana-rpc-client",
+ "solana-rpc-client-api",
+ "solana-sdk",
+ "thiserror 1.0.69",
+ "tokio",
+]
+
+[[package]]
+name = "solana-transaction-error"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1a7f870884ac7127bfdc9b68c16f4d5236e7300ea06f22028638e01b3fd5a7b9"
+dependencies = [
+ "serde",
+ "serde_derive",
+ "solana-instruction",
+ "solana-sanitize",
+]
+
+[[package]]
+name = "solana-transaction-metrics-tracker"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f32859feecdf70241dcf0b95a61cc94458d4bd7f9e61fcf259a986776eaa32d5"
+dependencies = [
+ "base64 0.22.1",
+ "bincode",
+ "lazy_static",
+ "log",
+ "rand 0.8.5",
+ "solana-perf",
+ "solana-sdk",
+ "solana-short-vec",
+]
+
+[[package]]
+name = "solana-transaction-status"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8fdb4f5d42a130ee25287f08f94a299584143d026b7fed3b90ba7a47e954e232"
+dependencies = [
+ "Inflector",
+ "base64 0.22.1",
+ "bincode",
+ "borsh 1.5.5",
+ "bs58",
+ "lazy_static",
+ "log",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "solana-account-decoder",
+ "solana-sdk",
+ "solana-transaction-status-client-types",
+ "spl-associated-token-account",
+ "spl-memo",
+ "spl-token",
+ "spl-token-2022",
+ "spl-token-group-interface",
+ "spl-token-metadata-interface",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-transaction-status-client-types"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7b710871be4ea981115a51fcbcd5666d7ec11139fdbe7831783d97da4afe4f74"
+dependencies = [
+ "base64 0.22.1",
+ "bincode",
+ "bs58",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "solana-account-decoder-client-types",
+ "solana-sdk",
+ "solana-signature",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-type-overrides"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9dde9f11d4ae3d701ef2cfc5eb40dbdb9d42977de3d74a11949406fb9e75d6a1"
+dependencies = [
+ "lazy_static",
+ "rand 0.8.5",
+]
+
+[[package]]
+name = "solana-udp-client"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6ad8075ebcb118e9c626445e01dc8f56b2d247c0c85feb40b23cae475a245c96"
+dependencies = [
+ "async-trait",
+ "solana-connection-cache",
+ "solana-net-utils",
+ "solana-sdk",
+ "solana-streamer",
+ "thiserror 1.0.69",
+ "tokio",
+]
+
+[[package]]
+name = "solana-version"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "295094544c3487280e32dfd323887b8e2df6810cd6681c10f2397a97abfa58e7"
+dependencies = [
+ "semver",
+ "serde",
+ "serde_derive",
+ "solana-feature-set",
+ "solana-sanitize",
+ "solana-serde-varint",
+]
+
+[[package]]
+name = "solana-vote"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e36de277406038484902b3ec5909d3a7e967f321800c9b7e2fe21e373074978e"
+dependencies = [
+ "itertools 0.12.1",
+ "log",
+ "serde",
+ "serde_derive",
+ "solana-sdk",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-vote-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e5f5be40dbb7a8592e1968995907bf03d47007ba29d9bdea4e2c07562479abc"
+dependencies = [
+ "bincode",
+ "log",
+ "num-derive",
+ "num-traits",
+ "serde",
+ "serde_derive",
+ "solana-feature-set",
+ "solana-metrics",
+ "solana-program",
+ "solana-program-runtime",
+ "solana-sdk",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "solana-zk-elgamal-proof-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cab847cc2b6a4ea361dd4b18032ff5b4fd3232bd39e620cc0fdd80e479100d7a"
+dependencies = [
+ "bytemuck",
+ "num-derive",
+ "num-traits",
+ "solana-log-collector",
+ "solana-program-runtime",
+ "solana-sdk",
+ "solana-zk-sdk",
+]
+
+[[package]]
+name = "solana-zk-sdk"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b2aab701561fac2d5124e1e2eb5ab3962723e5b4a7a2e57bdf528374c8ee7831"
+dependencies = [
+ "aes-gcm-siv",
+ "base64 0.22.1",
+ "bincode",
+ "bytemuck",
+ "bytemuck_derive",
+ "curve25519-dalek 4.1.3",
+ "itertools 0.12.1",
+ "js-sys",
+ "lazy_static",
+ "merlin",
+ "num-derive",
+ "num-traits",
+ "rand 0.8.5",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "sha3",
+ "solana-derivation-path",
+ "solana-program",
+ "solana-sdk",
+ "subtle",
+ "thiserror 1.0.69",
+ "wasm-bindgen",
+ "zeroize",
+]
+
+[[package]]
+name = "solana-zk-token-proof-program"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8c73f91561b4166b62ea6d3561aad8a5347ab2efd72a6973c469efce81767b1e"
+dependencies = [
+ "bytemuck",
+ "num-derive",
+ "num-traits",
+ "solana-feature-set",
+ "solana-log-collector",
+ "solana-program-runtime",
+ "solana-sdk",
+ "solana-zk-token-sdk",
+]
+
+[[package]]
+name = "solana-zk-token-sdk"
+version = "2.1.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c518ceed28518ac563f57bde2d2126c2adfe7c3339439496a3acbf8ca9024f3b"
+dependencies = [
+ "aes-gcm-siv",
+ "base64 0.22.1",
+ "bincode",
+ "bytemuck",
+ "bytemuck_derive",
+ "byteorder",
+ "curve25519-dalek 4.1.3",
+ "itertools 0.12.1",
+ "lazy_static",
+ "merlin",
+ "num-derive",
+ "num-traits",
+ "rand 0.8.5",
+ "serde",
+ "serde_derive",
+ "serde_json",
+ "sha3",
+ "solana-curve25519",
+ "solana-derivation-path",
+ "solana-program",
+ "solana-sdk",
+ "subtle",
+ "thiserror 1.0.69",
+ "zeroize",
+]
+
+[[package]]
+name = "solana_rbpf"
+version = "0.8.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1c1941b5ef0c3ce8f2ac5dd984d0fb1a97423c4ff2a02eec81e3913f02e2ac2b"
+dependencies = [
+ "byteorder",
+ "combine 3.8.1",
+ "hash32",
+ "libc",
+ "log",
+ "rand 0.8.5",
+ "rustc-demangle",
+ "scroll",
+ "thiserror 1.0.69",
+ "winapi",
+]
+
+[[package]]
+name = "spin"
+version = "0.9.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67"
+
+[[package]]
+name = "spinning_top"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d96d2d1d716fb500937168cc09353ffdc7a012be8475ac7308e1bdf0e3923300"
+dependencies = [
+ "lock_api",
+]
+
+[[package]]
+name = "spl-associated-token-account"
+version = "4.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "68034596cf4804880d265f834af1ff2f821ad5293e41fa0f8f59086c181fc38e"
+dependencies = [
+ "assert_matches",
+ "borsh 1.5.5",
+ "num-derive",
+ "num-traits",
+ "solana-program",
+ "spl-token",
+ "spl-token-2022",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "spl-discriminator"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a38ea8b6dedb7065887f12d62ed62c1743aa70749e8558f963609793f6fb12bc"
+dependencies = [
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator-derive",
+]
+
+[[package]]
+name = "spl-discriminator-derive"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d9e8418ea6269dcfb01c712f0444d2c75542c04448b480e87de59d2865edc750"
+dependencies = [
+ "quote",
+ "spl-discriminator-syn",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "spl-discriminator-syn"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8c1f05593b7ca9eac7caca309720f2eafb96355e037e6d373b909a80fe7b69b9"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "sha2 0.10.8",
+ "syn 2.0.98",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "spl-memo"
+version = "5.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a0dba2f2bb6419523405d21c301a32c9f9568354d4742552e7972af801f4bdb3"
+dependencies = [
+ "solana-program",
+]
+
+[[package]]
+name = "spl-pod"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c704c88fc457fa649ba3aabe195c79d885c3f26709efaddc453c8de352c90b87"
+dependencies = [
+ "borsh 1.5.5",
+ "bytemuck",
+ "bytemuck_derive",
+ "solana-program",
+ "solana-zk-token-sdk",
+ "spl-program-error",
+]
+
+[[package]]
+name = "spl-program-error"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d7b28bed65356558133751cc32b48a7a5ddfc59ac4e941314630bbed1ac10532"
+dependencies = [
+ "num-derive",
+ "num-traits",
+ "solana-program",
+ "spl-program-error-derive",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "spl-program-error-derive"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6d375dd76c517836353e093c2dbb490938ff72821ab568b545fd30ab3256b3e"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "sha2 0.10.8",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "spl-tlv-account-resolution"
+version = "0.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "37a75a5f0fcc58126693ed78a17042e9dc53f07e357d6be91789f7d62aff61a4"
+dependencies = [
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+ "spl-type-length-value",
+]
+
+[[package]]
+name = "spl-token"
+version = "6.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "70a0f06ac7f23dc0984931b1fe309468f14ea58e32660439c1cef19456f5d0e3"
+dependencies = [
+ "arrayref",
+ "bytemuck",
+ "num-derive",
+ "num-traits",
+ "num_enum",
+ "solana-program",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "spl-token-2022"
+version = "4.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d9c10f3483e48679619c76598d4e4aebb955bc49b0a5cc63323afbf44135c9bf"
+dependencies = [
+ "arrayref",
+ "bytemuck",
+ "num-derive",
+ "num-traits",
+ "num_enum",
+ "solana-program",
+ "solana-security-txt",
+ "solana-zk-token-sdk",
+ "spl-memo",
+ "spl-pod",
+ "spl-token",
+ "spl-token-group-interface",
+ "spl-token-metadata-interface",
+ "spl-transfer-hook-interface",
+ "spl-type-length-value",
+ "thiserror 1.0.69",
+]
+
+[[package]]
+name = "spl-token-group-interface"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "df8752b85a5ecc1d9f3a43bce3dd9a6a053673aacf5deb513d1cbb88d3534ffd"
+dependencies = [
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+]
+
+[[package]]
+name = "spl-token-metadata-interface"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c6c2318ddff97e006ed9b1291ebec0750a78547f870f62a69c56fe3b46a5d8fc"
+dependencies = [
+ "borsh 1.5.5",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+ "spl-type-length-value",
+]
+
+[[package]]
+name = "spl-transfer-hook-interface"
+version = "0.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a110f33d941275d9f868b96daaa993f1e73b6806cc8836e43075b4d3ad8338a7"
+dependencies = [
+ "arrayref",
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+ "spl-tlv-account-resolution",
+ "spl-type-length-value",
+]
+
+[[package]]
+name = "spl-type-length-value"
+version = "0.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bdcd73ec187bc409464c60759232e309f83b52a18a9c5610bf281c9c6432918c"
+dependencies = [
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+]
+
+[[package]]
+name = "stable_deref_trait"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3"
+
+[[package]]
+name = "static_assertions"
+version = "1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a2eb9349b6444b326872e140eb1cf5e7c522154d69e7a0ffb0fb81c06b37543f"
+
+[[package]]
+name = "strsim"
+version = "0.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f"
+
+[[package]]
+name = "strum"
+version = "0.24.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "063e6045c0e62079840579a7e47a355ae92f60eb74daaf156fb1e84ba164e63f"
+dependencies = [
+ "strum_macros",
+]
+
+[[package]]
+name = "strum_macros"
+version = "0.24.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e385be0d24f186b4ce2f9982191e7101bb737312ad61c1f2f984f34bcf85d59"
+dependencies = [
+ "heck",
+ "proc-macro2",
+ "quote",
+ "rustversion",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "subtle"
+version = "2.6.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292"
+
+[[package]]
+name = "symlink"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a7973cce6668464ea31f176d85b13c7ab3bba2cb3b77a2ed26abd7801688010a"
+
+[[package]]
+name = "syn"
+version = "1.0.109"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unicode-ident",
+]
+
+[[package]]
+name = "syn"
+version = "2.0.98"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "36147f1a48ae0ec2b5b3bc5b537d267457555a10dc06f3dbc8cb11ba3006d3b1"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unicode-ident",
+]
+
+[[package]]
+name = "sync_wrapper"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2047c6ded9c721764247e62cd3b03c09ffc529b2ba5b10ec482ae507a4a70160"
+
+[[package]]
+name = "synstructure"
+version = "0.12.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f36bdaa60a83aca3921b5259d5400cbf5e90fc51931376a9bd4a0eb79aa7210f"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+ "unicode-xid",
+]
+
+[[package]]
+name = "synstructure"
+version = "0.13.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c8af7666ab7b6390ab78131fb5b0fce11d6b7a6951602017c35fa82800708971"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[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 = "tar"
+version = "0.4.43"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c65998313f8e17d0d553d28f91a0df93e4dbbbf770279c7bc21ca0f09ea1a1f6"
+dependencies = [
+ "filetime",
+ "libc",
+ "xattr",
+]
+
+[[package]]
+name = "tarpc"
+version = "0.29.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1c38a012bed6fb9681d3bf71ffaa4f88f3b4b9ed3198cda6e4c8462d24d4bb80"
+dependencies = [
+ "anyhow",
+ "fnv",
+ "futures",
+ "humantime",
+ "opentelemetry",
+ "pin-project",
+ "rand 0.8.5",
+ "serde",
+ "static_assertions",
+ "tarpc-plugins",
+ "thiserror 1.0.69",
+ "tokio",
+ "tokio-serde",
+ "tokio-util 0.6.10",
+ "tracing",
+ "tracing-opentelemetry",
+]
+
+[[package]]
+name = "tarpc-plugins"
+version = "0.12.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0ee42b4e559f17bce0385ebf511a7beb67d5cc33c12c96b7f4e9789919d9c10f"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "task-local-extensions"
+version = "0.1.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ba323866e5d033818e3240feeb9f7db2c4296674e4d9e16b97b7bf8f490434e8"
+dependencies = [
+ "pin-utils",
+]
+
+[[package]]
+name = "tempfile"
+version = "3.16.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "38c246215d7d24f48ae091a2902398798e05d978b24315d6efbc00ede9a8bb91"
+dependencies = [
+ "cfg-if",
+ "fastrand",
+ "getrandom 0.3.1",
+ "once_cell",
+ "rustix",
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "termcolor"
+version = "1.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "06794f8f6c5c898b3275aebefa6b8a1cb24cd2c6c79397ab15774837a0bc5755"
+dependencies = [
+ "winapi-util",
+]
+
+[[package]]
+name = "termtree"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8f50febec83f5ee1df3015341d8bd429f2d1cc62bcba7ea2076759d315084683"
+
+[[package]]
+name = "thiserror"
+version = "1.0.69"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b6aaf5339b578ea85b50e080feb250a3e8ae8cfcdff9a461c9ec2904bc923f52"
+dependencies = [
+ "thiserror-impl 1.0.69",
+]
+
+[[package]]
+name = "thiserror"
+version = "2.0.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d452f284b73e6d76dd36758a0c8684b1d5be31f92b89d07fd5822175732206fc"
+dependencies = [
+ "thiserror-impl 2.0.11",
+]
+
+[[package]]
+name = "thiserror-impl"
+version = "1.0.69"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "thiserror-impl"
+version = "2.0.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "26afc1baea8a989337eeb52b6e72a039780ce45c3edfcc9c5b9d112feeb173c2"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "thread_local"
+version = "1.1.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8b9ef9bad013ada3808854ceac7b46812a6465ba368859a37e2100283d2d719c"
+dependencies = [
+ "cfg-if",
+ "once_cell",
+]
+
+[[package]]
+name = "time"
+version = "0.3.37"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "35e7868883861bd0e56d9ac6efcaaca0d6d5d82a2a7ec8209ff492c07cf37b21"
+dependencies = [
+ "deranged",
+ "itoa",
+ "num-conv",
+ "powerfmt",
+ "serde",
+ "time-core",
+ "time-macros",
+]
+
+[[package]]
+name = "time-core"
+version = "0.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ef927ca75afb808a4d64dd374f00a2adf8d0fcff8e7b184af886c3c87ec4a3f3"
+
+[[package]]
+name = "time-macros"
+version = "0.2.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2834e6017e3e5e4b9834939793b282bc03b37a3336245fa820e35e233e2a85de"
+dependencies = [
+ "num-conv",
+ "time-core",
+]
+
+[[package]]
+name = "tinystr"
+version = "0.7.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9117f5d4db391c1cf6927e7bea3db74b9a1c1add8f7eda9ffd5364f40f57b82f"
+dependencies = [
+ "displaydoc",
+ "zerovec",
+]
+
+[[package]]
+name = "tinyvec"
+version = "1.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "022db8904dfa342efe721985167e9fcd16c29b226db4397ed752a761cfce81e8"
+dependencies = [
+ "tinyvec_macros",
+]
+
+[[package]]
+name = "tinyvec_macros"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
+
+[[package]]
+name = "tokio"
+version = "1.43.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3d61fa4ffa3de412bfea335c6ecff681de2b609ba3c77ef3e00e521813a9ed9e"
+dependencies = [
+ "backtrace",
+ "bytes",
+ "libc",
+ "mio",
+ "parking_lot",
+ "pin-project-lite",
+ "signal-hook-registry",
+ "socket2",
+ "tokio-macros",
+ "windows-sys 0.52.0",
+]
+
+[[package]]
+name = "tokio-macros"
+version = "2.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6e06d43f1345a3bcd39f6a56dbb7dcab2ba47e68e8ac134855e7e2bdbaf8cab8"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "tokio-rustls"
+version = "0.24.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c28327cf380ac148141087fbfb9de9d7bd4e84ab5d2c28fbc911d753de8a7081"
+dependencies = [
+ "rustls 0.21.12",
+ "tokio",
+]
+
+[[package]]
+name = "tokio-serde"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "911a61637386b789af998ee23f50aa30d5fd7edcec8d6d3dedae5e5815205466"
+dependencies = [
+ "bincode",
+ "bytes",
+ "educe",
+ "futures-core",
+ "futures-sink",
+ "pin-project",
+ "serde",
+ "serde_json",
+]
+
+[[package]]
+name = "tokio-stream"
+version = "0.1.17"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "eca58d7bba4a75707817a2c44174253f9236b2d5fbd055602e9d5c07c139a047"
+dependencies = [
+ "futures-core",
+ "pin-project-lite",
+ "tokio",
+]
+
+[[package]]
+name = "tokio-tungstenite"
+version = "0.20.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "212d5dcb2a1ce06d81107c3d0ffa3121fe974b73f068c8282cb1c32328113b6c"
+dependencies = [
+ "futures-util",
+ "log",
+ "rustls 0.21.12",
+ "tokio",
+ "tokio-rustls",
+ "tungstenite",
+ "webpki-roots 0.25.4",
+]
+
+[[package]]
+name = "tokio-util"
+version = "0.6.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "36943ee01a6d67977dd3f84a5a1d2efeb4ada3a1ae771cadfaa535d9d9fc6507"
+dependencies = [
+ "bytes",
+ "futures-core",
+ "futures-sink",
+ "log",
+ "pin-project-lite",
+ "slab",
+ "tokio",
+]
+
+[[package]]
+name = "tokio-util"
+version = "0.7.13"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d7fcaa8d55a2bdd6b83ace262b016eca0d79ee02818c5c1bcdf0305114081078"
+dependencies = [
+ "bytes",
+ "futures-core",
+ "futures-sink",
+ "pin-project-lite",
+ "tokio",
+]
+
+[[package]]
+name = "toml"
+version = "0.5.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4f7f0dd8d50a853a531c426359045b1998f04219d88799810762cd4ad314234"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "toml_datetime"
+version = "0.6.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0dd7358ecb8fc2f8d014bf86f6f638ce72ba252a2c3a2572f2a795f1d23efb41"
+
+[[package]]
+name = "toml_edit"
+version = "0.22.23"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "02a8b472d1a3d7c18e2d61a489aee3453fd9031c33e4f55bd533f4a7adca1bee"
+dependencies = [
+ "indexmap",
+ "toml_datetime",
+ "winnow",
+]
+
+[[package]]
+name = "tower-service"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8df9b6e13f2d32c91b9bd719c00d1958837bc7dec474d94952798cc8e69eeec3"
+
+[[package]]
+name = "tracing"
+version = "0.1.41"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "784e0ac535deb450455cbfa28a6f0df145ea1bb7ae51b821cf5e7927fdcfbdd0"
+dependencies = [
+ "log",
+ "pin-project-lite",
+ "tracing-attributes",
+ "tracing-core",
+]
+
+[[package]]
+name = "tracing-attributes"
+version = "0.1.28"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "395ae124c09f9e6918a2310af6038fba074bcf474ac352496d5910dd59a2226d"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "tracing-core"
+version = "0.1.33"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e672c95779cf947c5311f83787af4fa8fffd12fb27e4993211a84bdfd9610f9c"
+dependencies = [
+ "once_cell",
+ "valuable",
+]
+
+[[package]]
+name = "tracing-opentelemetry"
+version = "0.17.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fbbe89715c1dbbb790059e2565353978564924ee85017b5fff365c872ff6721f"
+dependencies = [
+ "once_cell",
+ "opentelemetry",
+ "tracing",
+ "tracing-core",
+ "tracing-subscriber",
+]
+
+[[package]]
+name = "tracing-subscriber"
+version = "0.3.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e8189decb5ac0fa7bc8b96b7cb9b2701d60d48805aca84a238004d665fcc4008"
+dependencies = [
+ "sharded-slab",
+ "thread_local",
+ "tracing-core",
+]
+
+[[package]]
+name = "try-lock"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e421abadd41a4225275504ea4d6566923418b7f05506fbc9c0fe86ba7396114b"
+
+[[package]]
+name = "tungstenite"
+version = "0.20.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9e3dac10fd62eaf6617d3a904ae222845979aec67c615d1c842b4002c7666fb9"
+dependencies = [
+ "byteorder",
+ "bytes",
+ "data-encoding",
+ "http",
+ "httparse",
+ "log",
+ "rand 0.8.5",
+ "rustls 0.21.12",
+ "sha1",
+ "thiserror 1.0.69",
+ "url",
+ "utf-8",
+ "webpki-roots 0.24.0",
+]
+
+[[package]]
+name = "typenum"
+version = "1.17.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825"
+
+[[package]]
+name = "unicase"
+version = "2.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "75b844d17643ee918803943289730bec8aac480150456169e647ed0b576ba539"
+
+[[package]]
+name = "unicode-ident"
+version = "1.0.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a210d160f08b701c8721ba1c726c11662f877ea6b7094007e1ca9a1041945034"
+
+[[package]]
+name = "unicode-width"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1fc81956842c57dac11422a97c3b8195a1ff727f06e85c84ed2e8aa277c9a0fd"
+
+[[package]]
+name = "unicode-xid"
+version = "0.2.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ebc1c04c71510c7f702b52b7c350734c9ff1295c464a03335b00bb84fc54f853"
+
+[[package]]
+name = "universal-hash"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fc1de2c688dc15305988b563c3854064043356019f97a4b46276fe734c4f07ea"
+dependencies = [
+ "crypto-common",
+ "subtle",
+]
+
+[[package]]
+name = "unreachable"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "382810877fe448991dfc7f0dd6e3ae5d58088fd0ea5e35189655f84e6814fa56"
+dependencies = [
+ "void",
+]
+
+[[package]]
+name = "untrusted"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1"
+
+[[package]]
+name = "uriparse"
+version = "0.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0200d0fc04d809396c2ad43f3c95da3582a2556eba8d453c1087f4120ee352ff"
+dependencies = [
+ "fnv",
+ "lazy_static",
+]
+
+[[package]]
+name = "url"
+version = "2.5.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "32f8b686cadd1473f4bd0117a5d28d36b1ade384ea9b5069a1c40aefed7fda60"
+dependencies = [
+ "form_urlencoded",
+ "idna",
+ "percent-encoding",
+]
+
+[[package]]
+name = "utf-8"
+version = "0.7.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9"
+
+[[package]]
+name = "utf16_iter"
+version = "1.0.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c8232dd3cdaed5356e0f716d285e4b40b932ac434100fe9b7e0e8e935b9e6246"
+
+[[package]]
+name = "utf8_iter"
+version = "1.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b6c140620e7ffbb22c2dee59cafe6084a59b5ffc27a8859a5f0d494b5d52b6be"
+
+[[package]]
+name = "valuable"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ba73ea9cf16a25df0c8caa16c51acb937d5712a8429db78a3ee29d5dcacd3a65"
+
+[[package]]
+name = "vcpkg"
+version = "0.2.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426"
+
+[[package]]
+name = "version_check"
+version = "0.9.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a"
+
+[[package]]
+name = "void"
+version = "1.0.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a02e4885ed3bc0f2de90ea6dd45ebcbb66dacffe03547fadbb0eeae2770887d"
+
+[[package]]
+name = "walkdir"
+version = "2.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "29790946404f91d9c5d06f9874efddea1dc06c5efe94541a7d6863108e3a5e4b"
+dependencies = [
+ "same-file",
+ "winapi-util",
+]
+
+[[package]]
+name = "want"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bfa7760aed19e106de2c7c0b581b509f2f25d3dacaf737cb82ac61bc6d760b0e"
+dependencies = [
+ "try-lock",
+]
+
+[[package]]
+name = "wasi"
+version = "0.9.0+wasi-snapshot-preview1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519"
+
+[[package]]
+name = "wasi"
+version = "0.11.0+wasi-snapshot-preview1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
+
+[[package]]
+name = "wasi"
+version = "0.13.3+wasi-0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "26816d2e1a4a36a2940b96c5296ce403917633dff8f3440e9b236ed6f6bacad2"
+dependencies = [
+ "wit-bindgen-rt",
+]
+
+[[package]]
+name = "wasm-bindgen"
+version = "0.2.100"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1edc8929d7499fc4e8f0be2262a241556cfc54a0bea223790e71446f2aab1ef5"
+dependencies = [
+ "cfg-if",
+ "once_cell",
+ "rustversion",
+ "wasm-bindgen-macro",
+]
+
+[[package]]
+name = "wasm-bindgen-backend"
+version = "0.2.100"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2f0a0651a5c2bc21487bde11ee802ccaf4c51935d0d3d42a6101f98161700bc6"
+dependencies = [
+ "bumpalo",
+ "log",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-futures"
+version = "0.4.50"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "555d470ec0bc3bb57890405e5d4322cc9ea83cebb085523ced7be4144dac1e61"
+dependencies = [
+ "cfg-if",
+ "js-sys",
+ "once_cell",
+ "wasm-bindgen",
+ "web-sys",
+]
+
+[[package]]
+name = "wasm-bindgen-macro"
+version = "0.2.100"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7fe63fc6d09ed3792bd0897b314f53de8e16568c2b3f7982f468c0bf9bd0b407"
+dependencies = [
+ "quote",
+ "wasm-bindgen-macro-support",
+]
+
+[[package]]
+name = "wasm-bindgen-macro-support"
+version = "0.2.100"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8ae87ea40c9f689fc23f209965b6fb8a99ad69aeeb0231408be24920604395de"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+ "wasm-bindgen-backend",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-shared"
+version = "0.2.100"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1a05d73b933a847d6cccdda8f838a22ff101ad9bf93e33684f39c1f5f0eece3d"
+dependencies = [
+ "unicode-ident",
+]
+
+[[package]]
+name = "web-sys"
+version = "0.3.77"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "33b6dd2ef9186f1f2072e409e99cd22a975331a6b3591b12c764e0e55c60d5d2"
+dependencies = [
+ "js-sys",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "web-time"
+version = "1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5a6580f308b1fad9207618087a65c04e7a10bc77e02c8e84e9b00dd4b12fa0bb"
+dependencies = [
+ "js-sys",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "webpki-root-certs"
+version = "0.26.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "09aed61f5e8d2c18344b3faa33a4c837855fe56642757754775548fee21386c4"
+dependencies = [
+ "rustls-pki-types",
+]
+
+[[package]]
+name = "webpki-roots"
+version = "0.24.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b291546d5d9d1eab74f069c77749f2cb8504a12caa20f0f2de93ddbf6f411888"
+dependencies = [
+ "rustls-webpki 0.101.7",
+]
+
+[[package]]
+name = "webpki-roots"
+version = "0.25.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5f20c57d8d7db6d3b86154206ae5d8fba62dd39573114de97c2cb0578251f8e1"
+
+[[package]]
+name = "winapi"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419"
+dependencies = [
+ "winapi-i686-pc-windows-gnu",
+ "winapi-x86_64-pc-windows-gnu",
+]
+
+[[package]]
+name = "winapi-i686-pc-windows-gnu"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
+
+[[package]]
+name = "winapi-util"
+version = "0.1.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb"
+dependencies = [
+ "windows-sys 0.59.0",
+]
+
+[[package]]
+name = "winapi-x86_64-pc-windows-gnu"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
+
+[[package]]
+name = "windows-core"
+version = "0.52.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "33ab640c8d7e35bf8ba19b884ba838ceb4fba93a4e8c65a9059d08afcfc683d9"
+dependencies = [
+ "windows-targets 0.52.6",
+]
+
+[[package]]
+name = "windows-sys"
+version = "0.48.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "677d2418bec65e3338edb076e806bc1ec15693c5d0104683f2efe857f61056a9"
+dependencies = [
+ "windows-targets 0.48.5",
+]
+
+[[package]]
+name = "windows-sys"
+version = "0.52.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "282be5f36a8ce781fad8c8ae18fa3f9beff57ec1b52cb3de0789201425d9a33d"
+dependencies = [
+ "windows-targets 0.52.6",
+]
+
+[[package]]
+name = "windows-sys"
+version = "0.59.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b"
+dependencies = [
+ "windows-targets 0.52.6",
+]
+
+[[package]]
+name = "windows-targets"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9a2fa6e2155d7247be68c096456083145c183cbbbc2764150dda45a87197940c"
+dependencies = [
+ "windows_aarch64_gnullvm 0.48.5",
+ "windows_aarch64_msvc 0.48.5",
+ "windows_i686_gnu 0.48.5",
+ "windows_i686_msvc 0.48.5",
+ "windows_x86_64_gnu 0.48.5",
+ "windows_x86_64_gnullvm 0.48.5",
+ "windows_x86_64_msvc 0.48.5",
+]
+
+[[package]]
+name = "windows-targets"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b724f72796e036ab90c1021d4780d4d3d648aca59e491e6b98e725b84e99973"
+dependencies = [
+ "windows_aarch64_gnullvm 0.52.6",
+ "windows_aarch64_msvc 0.52.6",
+ "windows_i686_gnu 0.52.6",
+ "windows_i686_gnullvm",
+ "windows_i686_msvc 0.52.6",
+ "windows_x86_64_gnu 0.52.6",
+ "windows_x86_64_gnullvm 0.52.6",
+ "windows_x86_64_msvc 0.52.6",
+]
+
+[[package]]
+name = "windows_aarch64_gnullvm"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2b38e32f0abccf9987a4e3079dfb67dcd799fb61361e53e2882c3cbaf0d905d8"
+
+[[package]]
+name = "windows_aarch64_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3"
+
+[[package]]
+name = "windows_aarch64_msvc"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "dc35310971f3b2dbbf3f0690a219f40e2d9afcf64f9ab7cc1be722937c26b4bc"
+
+[[package]]
+name = "windows_aarch64_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469"
+
+[[package]]
+name = "windows_i686_gnu"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a75915e7def60c94dcef72200b9a8e58e5091744960da64ec734a6c6e9b3743e"
+
+[[package]]
+name = "windows_i686_gnu"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b"
+
+[[package]]
+name = "windows_i686_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66"
+
+[[package]]
+name = "windows_i686_msvc"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8f55c233f70c4b27f66c523580f78f1004e8b5a8b659e05a4eb49d4166cca406"
+
+[[package]]
+name = "windows_i686_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66"
+
+[[package]]
+name = "windows_x86_64_gnu"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "53d40abd2583d23e4718fddf1ebec84dbff8381c07cae67ff7768bbf19c6718e"
+
+[[package]]
+name = "windows_x86_64_gnu"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78"
+
+[[package]]
+name = "windows_x86_64_gnullvm"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b7b52767868a23d5bab768e390dc5f5c55825b6d30b86c844ff2dc7414044cc"
+
+[[package]]
+name = "windows_x86_64_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d"
+
+[[package]]
+name = "windows_x86_64_msvc"
+version = "0.48.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ed94fce61571a4006852b7389a063ab983c02eb1bb37b47f8272ce92d06d9538"
+
+[[package]]
+name = "windows_x86_64_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec"
+
+[[package]]
+name = "winnow"
+version = "0.7.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "86e376c75f4f43f44db463cf729e0d3acbf954d13e22c51e26e4c264b4ab545f"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "winreg"
+version = "0.50.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "524e57b2c537c0f9b1e69f1965311ec12182b4122e45035b1508cd24d2adadb1"
+dependencies = [
+ "cfg-if",
+ "windows-sys 0.48.0",
+]
+
+[[package]]
+name = "wit-bindgen-rt"
+version = "0.33.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3268f3d866458b787f390cf61f4bbb563b922d091359f9608842999eaee3943c"
+dependencies = [
+ "bitflags 2.8.0",
+]
+
+[[package]]
+name = "wormhole-post-message-shim"
+version = "0.1.0"
+dependencies = [
+ "solana-program",
+ "solana-program-test",
+ "solana-sdk",
+ "wormhole-svm-definitions",
+ "wormhole-svm-shim",
+]
+
+[[package]]
+name = "wormhole-svm-definitions"
+version = "0.1.0"
+dependencies = [
+ "base64 0.22.1",
+ "borsh 1.5.5",
+ "cfg-if",
+ "sha2-const-stable",
+ "solana-program",
+]
+
+[[package]]
+name = "wormhole-svm-shim"
+version = "0.1.0"
+dependencies = [
+ "solana-program",
+ "wormhole-svm-definitions",
+]
+
+[[package]]
+name = "wormhole-verify-vaa-shim"
+version = "0.1.0"
+dependencies = [
+ "base64 0.22.1",
+ "borsh 1.5.5",
+ "solana-program",
+ "solana-program-test",
+ "solana-sdk",
+ "wormhole-svm-definitions",
+ "wormhole-svm-shim",
+]
+
+[[package]]
+name = "write16"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d1890f4022759daae28ed4fe62859b1236caebfc61ede2f63ed4e695f3f6d936"
+
+[[package]]
+name = "writeable"
+version = "0.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e9df38ee2d2c3c5948ea468a8406ff0db0b29ae1ffde1bcf20ef305bcc95c51"
+
+[[package]]
+name = "x509-parser"
+version = "0.14.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e0ecbeb7b67ce215e40e3cc7f2ff902f94a223acf44995934763467e7b1febc8"
+dependencies = [
+ "asn1-rs",
+ "base64 0.13.1",
+ "data-encoding",
+ "der-parser",
+ "lazy_static",
+ "nom",
+ "oid-registry",
+ "rusticata-macros",
+ "thiserror 1.0.69",
+ "time",
+]
+
+[[package]]
+name = "xattr"
+version = "1.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e105d177a3871454f754b33bb0ee637ecaaac997446375fd3e5d43a2ed00c909"
+dependencies = [
+ "libc",
+ "linux-raw-sys",
+ "rustix",
+]
+
+[[package]]
+name = "yoke"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "120e6aef9aa629e3d4f52dc8cc43a015c7724194c97dfaf45180d2daf2b77f40"
+dependencies = [
+ "serde",
+ "stable_deref_trait",
+ "yoke-derive",
+ "zerofrom",
+]
+
+[[package]]
+name = "yoke-derive"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2380878cad4ac9aac1e2435f3eb4020e8374b5f13c296cb75b4620ff8e229154"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+ "synstructure 0.13.1",
+]
+
+[[package]]
+name = "zerocopy"
+version = "0.7.35"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0"
+dependencies = [
+ "byteorder",
+ "zerocopy-derive",
+]
+
+[[package]]
+name = "zerocopy-derive"
+version = "0.7.35"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "zerofrom"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cff3ee08c995dee1859d998dea82f7374f2826091dd9cd47def953cae446cd2e"
+dependencies = [
+ "zerofrom-derive",
+]
+
+[[package]]
+name = "zerofrom-derive"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "595eed982f7d355beb85837f651fa22e90b3c044842dc7f2c2842c086f295808"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+ "synstructure 0.13.1",
+]
+
+[[package]]
+name = "zeroize"
+version = "1.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde"
+dependencies = [
+ "zeroize_derive",
+]
+
+[[package]]
+name = "zeroize_derive"
+version = "1.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "zerovec"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "aa2b893d79df23bfb12d5461018d408ea19dfafe76c2c7ef6d4eba614f8ff079"
+dependencies = [
+ "yoke",
+ "zerofrom",
+ "zerovec-derive",
+]
+
+[[package]]
+name = "zerovec-derive"
+version = "0.10.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6eafa6dfb17584ea3e2bd6e76e0cc15ad7af12b09abdd1ca55961bed9b1063c6"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.98",
+]
+
+[[package]]
+name = "zstd"
+version = "0.13.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fcf2b778a664581e31e389454a7072dab1647606d44f7feea22cd5abb9c9f3f9"
+dependencies = [
+ "zstd-safe",
+]
+
+[[package]]
+name = "zstd-safe"
+version = "7.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "54a3ab4db68cea366acc5c897c7b4d4d1b8994a9cd6e6f841f8964566a419059"
+dependencies = [
+ "zstd-sys",
+]
+
+[[package]]
+name = "zstd-sys"
+version = "2.0.13+zstd.1.5.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "38ff0f21cfee8f97d94cef41359e0c89aa6113028ab0291aa8ca0038995a95aa"
+dependencies = [
+ "cc",
+ "pkg-config",
+]

+ 34 - 0
svm/wormhole-core-shims/Cargo.toml

@@ -0,0 +1,34 @@
+[workspace]
+members = [
+    "crates/*",
+    "programs/*"
+]
+resolver = "2"
+
+[workspace.package]
+edition = "2021"
+license = "Apache-2.0"
+repository = "https://github.com/wormhole-foundation/wormhole"
+rust-version = "1.75"
+version = "0.1.0"
+
+[workspace.dependencies]
+base64 = "0.22.1"
+borsh = "1.5"
+cfg-if = "1.0"
+sha2-const-stable = "0.1.0"
+solana-program = ">=1.18,<=2"
+solana-program-test = ">=1.18,<=2"
+solana-sdk = ">=1.18,<=2"
+wormhole-svm-definitions = { path = "crates/definitions" }
+wormhole-svm-shim = { path = "crates/shim" }
+
+[profile.release]
+overflow-checks = false
+lto = "fat"
+codegen-units = 1
+
+[profile.release.build-override]
+opt-level = 3
+incremental = false
+codegen-units = 1

+ 63 - 0
svm/wormhole-core-shims/Makefile

@@ -0,0 +1,63 @@
+SOLANA_VERSION := 2.1.11
+
+NETWORK ?= mainnet
+
+ifeq ($(NETWORK),mainnet)
+    BUILD_FEATURES = $(SVM)
+else
+    BUILD_FEATURES = $(SVM),$(NETWORK)
+endif
+
+POST_MESSAGE_TEST_FIXTURES_PATH = programs/post-message/tests/fixtures
+
+.PHONY: clean
+clean:
+	cargo clean
+	rm -rf $(POST_MESSAGE_TEST_FIXTURES_PATH) artifacts-mainnet artifacts-testnet artifacts-localnet
+	cd anchor && $(MAKE) clean
+
+.PHONY: build
+build: solana-install
+	@if [ -z "$(SVM)" ]; then \
+		echo "Error: SVM must be defined" >&2; \
+		exit 1; \
+	fi
+	cargo build-sbf --features "$(BUILD_FEATURES)"
+
+artifacts-$(NETWORK):
+	@if [ ! -d "$@" ]; then \
+		$(MAKE) build && \
+		mkdir -p "$@" && \
+		cp target/deploy/*.so "$@/"; \
+	else \
+		echo "Directory $@ already exists, skipping"; \
+		exit 1; \
+	fi
+
+.PHONY: build-artifacts
+build-artifacts: artifacts-$(NETWORK)
+
+.PHONY: test
+test: solana-install $(POST_MESSAGE_TEST_FIXTURES_PATH)/core_bridge.so
+	cargo test --lib --features solana
+	cargo test --doc --features solana
+	cargo test-sbf --features solana
+
+.PHONY: clean-test
+clean-test: clean test
+
+.PHONY: clippy
+clippy:
+	cargo clippy --all-targets --all-features -- -D warnings
+
+$(POST_MESSAGE_TEST_FIXTURES_PATH)/core_bridge.so:
+	mkdir -p $(POST_MESSAGE_TEST_FIXTURES_PATH)
+	solana program dump -u m worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth $(POST_MESSAGE_TEST_FIXTURES_PATH)/core_bridge.so
+
+.PHONY: solana-install
+solana-install:
+	@if ! solana --version | grep -q "$(SOLANA_VERSION)"; then \
+		agave-install init $(SOLANA_VERSION); \
+	else \
+		echo "Solana $(SOLANA_VERSION) is already installed"; \
+	fi

+ 56 - 0
svm/wormhole-core-shims/README.md

@@ -0,0 +1,56 @@
+# Wormhole Core Shims
+
+The intent of the following programs is to reduce the cost of Core Bridge
+message emission and verification on Solana without making changes to the
+existing core bridge.
+
+- [Wormhole Post Message Shim]
+- [Wormhole Verify VAA Shim]
+
+The following are provided for example purposes only
+
+- [Wormhole Integrator Example]
+
+## Tests
+
+To perform unit, doc and integration tests, run:
+
+```sh
+make test
+```
+
+Integration tests are run using `cargo test-sbf`, but this requires having the
+Solana toolchain installed via [agave-install].
+
+Programs are built using Solana version 2.1.11, which is the current CLI
+available at the time these programs were written.
+
+**The `make` command above will initialize the Solana CLI version needed to
+build and test. After running the tests, your CLI will still be configured to
+this version. Please note your Solana CLI version before running this command.**
+
+There are separate Anchor tests found in the [anchor directory].
+
+For initial end-to-end (e2e) testing the post message shim with the guardian,
+the programs were built with the following:
+
+```sh
+NETWORK=localnet SVM=solana make build
+```
+
+Please see the [anchor directory] to build the examples. The resulting program
+binaries were then
+
+- Copied to [../../solana/tests/artifacts]
+- Copied into the test validator dockerfile in
+  [../../solana/Dockerfile.test-validator]
+- Loaded into the test validator at startup in [../../devnet/solana-devnet.yaml]
+
+[../../devnet/solana-devnet.yaml]: ../../devnet/solana-devnet.yaml
+[../../solana/Dockerfile.test-validator]: ../../solana/Dockerfile.test-validator
+[../../solana/tests/artifacts]: ../../solana/tests/artifacts
+[agave-install]: https://docs.anza.xyz/cli/install#use-the-solana-install-tool
+[anchor directory]: anchor
+[Wormhole Integrator Example]: anchor/programs/wormhole-integrator-example/src/lib.rs
+[Wormhole Post Message Shim]: programs/post-message/README.md
+[Wormhole Verify VAA Shim]: programs/verify-vaa/README.md

+ 7 - 0
svm/wormhole-core-shims/anchor/.prettierignore

@@ -0,0 +1,7 @@
+.anchor
+.DS_Store
+target
+node_modules
+dist
+build
+test-ledger

+ 81 - 0
svm/wormhole-core-shims/anchor/Anchor.toml

@@ -0,0 +1,81 @@
+[toolchain]
+anchor_version = "0.30.1"
+
+[workspace]
+members = [
+    "interfaces/*",
+    "programs/*"
+]
+
+[features]
+resolution = true
+skip-lint = true
+
+[programs.localnet]
+wormhole_integrator_example = "AEwubmehHNvkMXoH2C5MgDSemZgQ3HUSYpeaF3UrNZdQ"
+wormhole_post_message_shim = "EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX"
+wormhole_vaa_verification_comparison = "m7qfXCwkwvpB4RayZoVsNxm4X2iE2a34rCAsjbNvrRs"
+wormhole_verify_vaa_shim = "EFaNWErqAtVWufdNb7yofSHHfWFos843DFpu4JBw24at"
+
+[registry]
+url = "https://api.apr.dev"
+
+[provider]
+cluster = "Localnet"
+wallet = "~/.config/solana/id.json"
+
+[scripts]
+test = "yarn run ts-mocha -p ./tsconfig.json -t 1000000 tests/wormhole-*.ts"
+
+[test]
+startup_wait = 5000
+shutdown_wait = 2000
+upgradeable = true
+
+[test.validator]
+bind_address = "0.0.0.0"
+ledger = ".anchor/test-ledger"
+rpc_port = 8599
+
+## CORE BRIDGE DEVNET
+### Wormhole Core Bridge Program (Devnet)
+[[test.genesis]]
+address = "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o"
+program = "tests/artifacts/devnet_core_bridge.so"
+upgradeable = true
+
+### Wormhole Core Bridge (Devnet) -- Config
+[[test.validator.account]]
+address = "FKoMTctsC7vJbEqyRiiPskPnuQx2tX1kurmvWByq5uZP"
+filename = "tests/accounts/core_bridge_devnet/bridge_config.json"
+
+### Wormhole Core Bridge (Devnet) -- Fee Collector
+[[test.validator.account]]
+address = "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs"
+filename = "tests/accounts/core_bridge_devnet/fee_collector.json"
+
+### Wormhole Core Bridge (Devnet) -- Guardian Set 0
+[[test.validator.account]]
+address = "6MxkvoEwgB9EqQRLNhvYaPGhfcLtBtpBqdQugr3AZUgD"
+filename = "tests/accounts/core_bridge_devnet/guardian_set.json"
+
+## CORE BRIDGE MAINNET
+### Wormhole Core Bridge Program (Mainnet)
+[[test.genesis]]
+address = "worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth"
+program = "tests/artifacts/mainnet_core_bridge.so"
+
+### Wormhole Core Bridge (Mainnet) -- Config
+[[test.validator.account]]
+address = "2yVjuQwpsvdsrywzsJJVs9Ueh4zayyo5DYJbBNc3DDpn"
+filename = "tests/accounts/core_bridge_mainnet/config.json"
+
+### Wormhole Core Bridge (Mainnet) -- Fee Collector
+[[test.validator.account]]
+address = "9bFNrXNb2WTx8fMHXCheaZqkLZ3YCCaiqTftHxeintHy"
+filename = "tests/accounts/core_bridge_mainnet/fee_collector.json"
+
+### Wormhole Core Bridge (Mainnet) -- Guardian Set 4 (current)
+[[test.validator.account]]
+address = "AFEXK4A1BU7BZfi8niAmker98LH9EARB544wKGPXwMyy"
+filename = "tests/accounts/core_bridge_mainnet/guardian_set_4.json"

+ 2767 - 0
svm/wormhole-core-shims/anchor/Cargo.lock

@@ -0,0 +1,2767 @@
+# This file is automatically @generated by Cargo.
+# It is not intended for manual editing.
+version = 3
+
+[[package]]
+name = "aead"
+version = "0.4.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b613b8e1e3cf911a086f53f03bf286f52fd7a7258e4fa606f0ef220d39d8877"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "aes"
+version = "0.7.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9e8b47f52ea9bae42228d07ec09eb676433d7c4ed1ebdf0f1d1c29ed446f1ab8"
+dependencies = [
+ "cfg-if",
+ "cipher",
+ "cpufeatures",
+ "opaque-debug",
+]
+
+[[package]]
+name = "aes-gcm-siv"
+version = "0.10.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "589c637f0e68c877bbd59a4599bbe849cac8e5f3e4b5a3ebae8f528cd218dcdc"
+dependencies = [
+ "aead",
+ "aes",
+ "cipher",
+ "ctr",
+ "polyval",
+ "subtle",
+ "zeroize",
+]
+
+[[package]]
+name = "ahash"
+version = "0.7.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "891477e0c6a8957309ee5c45a6368af3ae14bb510732d2684ffa19af310920f9"
+dependencies = [
+ "getrandom 0.2.15",
+ "once_cell",
+ "version_check",
+]
+
+[[package]]
+name = "ahash"
+version = "0.8.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e89da841a80418a9b391ebaea17f5c112ffaaa96f621d2c285b5174da76b9011"
+dependencies = [
+ "cfg-if",
+ "once_cell",
+ "version_check",
+ "zerocopy",
+]
+
+[[package]]
+name = "aho-corasick"
+version = "1.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8e60d3430d3a69478ad0993f19238d2df97c507009a52b3c10addcd7f6bcb916"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "anchor-attribute-access-control"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "47fe28365b33e8334dd70ae2f34a43892363012fe239cf37d2ee91693575b1f8"
+dependencies = [
+ "anchor-syn",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-attribute-account"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3c288d496168268d198d9b53ee9f4f9d260a55ba4df9877ea1d4486ad6109e0f"
+dependencies = [
+ "anchor-syn",
+ "bs58 0.5.1",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-attribute-constant"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "49b77b6948d0eeaaa129ce79eea5bbbb9937375a9241d909ca8fb9e006bb6e90"
+dependencies = [
+ "anchor-syn",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-attribute-error"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4d20bb569c5a557c86101b944721d865e1fd0a4c67c381d31a44a84f07f84828"
+dependencies = [
+ "anchor-syn",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-attribute-event"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4cebd8d0671a3a9dc3160c48598d652c34c77de6be4d44345b8b514323284d57"
+dependencies = [
+ "anchor-syn",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-attribute-program"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "efb2a5eb0860e661ab31aff7bb5e0288357b176380e985bade4ccb395981b42d"
+dependencies = [
+ "anchor-lang-idl",
+ "anchor-syn",
+ "anyhow",
+ "bs58 0.5.1",
+ "heck",
+ "proc-macro2",
+ "quote",
+ "serde_json",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-derive-accounts"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "04368b5abef4266250ca8d1d12f4dff860242681e4ec22b885dcfe354fd35aa1"
+dependencies = [
+ "anchor-syn",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-derive-serde"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e0bb0e0911ad4a70cab880cdd6287fe1e880a1a9d8e4e6defa8e9044b9796a6c"
+dependencies = [
+ "anchor-syn",
+ "borsh-derive-internal 0.10.4",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-derive-space"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5ef415ff156dc82e9ecb943189b0cb241b3a6bfc26a180234dc21bd3ef3ce0cb"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "anchor-lang"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6620c9486d9d36a4389cab5e37dc34a42ed0bfaa62e6a75a2999ce98f8f2e373"
+dependencies = [
+ "anchor-attribute-access-control",
+ "anchor-attribute-account",
+ "anchor-attribute-constant",
+ "anchor-attribute-error",
+ "anchor-attribute-event",
+ "anchor-attribute-program",
+ "anchor-derive-accounts",
+ "anchor-derive-serde",
+ "anchor-derive-space",
+ "anchor-lang-idl",
+ "arrayref",
+ "base64 0.21.7",
+ "bincode",
+ "borsh 0.10.4",
+ "bytemuck",
+ "getrandom 0.2.15",
+ "solana-program",
+ "thiserror",
+]
+
+[[package]]
+name = "anchor-lang-idl"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "31cf97b4e6f7d6144a05e435660fcf757dbc3446d38d0e2b851d11ed13625bba"
+dependencies = [
+ "anchor-lang-idl-spec",
+ "anyhow",
+ "heck",
+ "regex",
+ "serde",
+ "serde_json",
+ "sha2 0.10.8",
+]
+
+[[package]]
+name = "anchor-lang-idl-spec"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2bdf143115440fe621bdac3a29a1f7472e09f6cd82b2aa569429a0c13f103838"
+dependencies = [
+ "anyhow",
+ "serde",
+]
+
+[[package]]
+name = "anchor-spl"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "04bd077c34449319a1e4e0bc21cea572960c9ae0d0fefda0dd7c52fcc3c647a3"
+dependencies = [
+ "anchor-lang",
+ "mpl-token-metadata",
+ "spl-associated-token-account",
+ "spl-pod",
+ "spl-token",
+ "spl-token-2022",
+ "spl-token-group-interface",
+ "spl-token-metadata-interface",
+]
+
+[[package]]
+name = "anchor-syn"
+version = "0.30.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f99daacb53b55cfd37ce14d6c9905929721137fd4c67bbab44a19802aecb622f"
+dependencies = [
+ "anyhow",
+ "bs58 0.5.1",
+ "cargo_toml",
+ "heck",
+ "proc-macro2",
+ "quote",
+ "serde",
+ "serde_json",
+ "sha2 0.10.8",
+ "syn 1.0.109",
+ "thiserror",
+]
+
+[[package]]
+name = "anyhow"
+version = "1.0.95"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "34ac096ce696dc2fcabef30516bb13c0a68a11d30131d3df6f04711467681b04"
+
+[[package]]
+name = "ark-bn254"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a22f4561524cd949590d78d7d4c5df8f592430d221f7f3c9497bbafd8972120f"
+dependencies = [
+ "ark-ec",
+ "ark-ff",
+ "ark-std",
+]
+
+[[package]]
+name = "ark-ec"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "defd9a439d56ac24968cca0571f598a61bc8c55f71d50a89cda591cb750670ba"
+dependencies = [
+ "ark-ff",
+ "ark-poly",
+ "ark-serialize",
+ "ark-std",
+ "derivative",
+ "hashbrown 0.13.2",
+ "itertools",
+ "num-traits",
+ "zeroize",
+]
+
+[[package]]
+name = "ark-ff"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ec847af850f44ad29048935519032c33da8aa03340876d351dfab5660d2966ba"
+dependencies = [
+ "ark-ff-asm",
+ "ark-ff-macros",
+ "ark-serialize",
+ "ark-std",
+ "derivative",
+ "digest 0.10.7",
+ "itertools",
+ "num-bigint",
+ "num-traits",
+ "paste",
+ "rustc_version",
+ "zeroize",
+]
+
+[[package]]
+name = "ark-ff-asm"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3ed4aa4fe255d0bc6d79373f7e31d2ea147bcf486cba1be5ba7ea85abdb92348"
+dependencies = [
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "ark-ff-macros"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7abe79b0e4288889c4574159ab790824d0033b9fdcb2a112a3182fac2e514565"
+dependencies = [
+ "num-bigint",
+ "num-traits",
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "ark-poly"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d320bfc44ee185d899ccbadfa8bc31aab923ce1558716e1997a1e74057fe86bf"
+dependencies = [
+ "ark-ff",
+ "ark-serialize",
+ "ark-std",
+ "derivative",
+ "hashbrown 0.13.2",
+]
+
+[[package]]
+name = "ark-serialize"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "adb7b85a02b83d2f22f89bd5cac66c9c89474240cb6207cb1efc16d098e822a5"
+dependencies = [
+ "ark-serialize-derive",
+ "ark-std",
+ "digest 0.10.7",
+ "num-bigint",
+]
+
+[[package]]
+name = "ark-serialize-derive"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ae3281bc6d0fd7e549af32b52511e1302185bd688fd3359fa36423346ff682ea"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "ark-std"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94893f1e0c6eeab764ade8dc4c0db24caf4fe7cbbaafc0eba0a9030f447b5185"
+dependencies = [
+ "num-traits",
+ "rand 0.8.5",
+]
+
+[[package]]
+name = "arrayref"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "76a2e8124351fda1ef8aaaa3bbd7ebbcb486bbcd4225aca0aa0d84bb2db8fecb"
+
+[[package]]
+name = "arrayvec"
+version = "0.7.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7c02d123df017efcdfbd739ef81735b36c5ba83ec3c59c80a9d7ecc718f92e50"
+
+[[package]]
+name = "assert_matches"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b34d609dfbaf33d6889b2b7106d3ca345eacad44200913df5ba02bfd31d2ba9"
+
+[[package]]
+name = "atty"
+version = "0.2.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d9b39be18770d11421cdb1b9947a45dd3f37e93092cbf377614828a319d5fee8"
+dependencies = [
+ "hermit-abi",
+ "libc",
+ "winapi",
+]
+
+[[package]]
+name = "autocfg"
+version = "1.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ace50bade8e6234aa140d9a2f552bbee1db4d353f69b8217bc503490fc1a9f26"
+
+[[package]]
+name = "base64"
+version = "0.12.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3441f0f7b02788e948e47f457ca01f1d7e6d92c693bc132c22b087d3141c03ff"
+
+[[package]]
+name = "base64"
+version = "0.21.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d297deb1925b89f2ccc13d7635fa0714f12c87adce1c75356b39ca9b7178567"
+
+[[package]]
+name = "bincode"
+version = "1.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b1f45e9417d87227c7a56d22e471c6206462cba514c7590c09aff4cf6d1ddcad"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "bitflags"
+version = "2.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b048fb63fd8b5923fc5aa7b340d8e156aec7ec02f0c78fa8a6ddc2613f6f71de"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "bitmaps"
+version = "2.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "031043d04099746d8db04daf1fa424b2bc8bd69d92b25962dcde24da39ab64a2"
+dependencies = [
+ "typenum",
+]
+
+[[package]]
+name = "blake3"
+version = "1.5.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b8ee0c1824c4dea5b5f81736aff91bae041d2c07ee1192bec91054e10e3e601e"
+dependencies = [
+ "arrayref",
+ "arrayvec",
+ "cc",
+ "cfg-if",
+ "constant_time_eq",
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "block-buffer"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4"
+dependencies = [
+ "block-padding",
+ "generic-array",
+]
+
+[[package]]
+name = "block-buffer"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "block-padding"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8d696c370c750c948ada61c69a0ee2cbbb9c50b1019ddb86d9317157a99c2cae"
+
+[[package]]
+name = "borsh"
+version = "0.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "15bf3650200d8bffa99015595e10f1fbd17de07abbc25bb067da79e769939bfa"
+dependencies = [
+ "borsh-derive 0.9.3",
+ "hashbrown 0.11.2",
+]
+
+[[package]]
+name = "borsh"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "115e54d64eb62cdebad391c19efc9dce4981c690c85a33a12199d99bb9546fee"
+dependencies = [
+ "borsh-derive 0.10.4",
+ "hashbrown 0.13.2",
+]
+
+[[package]]
+name = "borsh"
+version = "1.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2506947f73ad44e344215ccd6403ac2ae18cd8e046e581a441bf8d199f257f03"
+dependencies = [
+ "borsh-derive 1.5.3",
+ "cfg_aliases",
+]
+
+[[package]]
+name = "borsh-derive"
+version = "0.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6441c552f230375d18e3cc377677914d2ca2b0d36e52129fe15450a2dce46775"
+dependencies = [
+ "borsh-derive-internal 0.9.3",
+ "borsh-schema-derive-internal 0.9.3",
+ "proc-macro-crate 0.1.5",
+ "proc-macro2",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "borsh-derive"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "831213f80d9423998dd696e2c5345aba6be7a0bd8cd19e31c5243e13df1cef89"
+dependencies = [
+ "borsh-derive-internal 0.10.4",
+ "borsh-schema-derive-internal 0.10.4",
+ "proc-macro-crate 0.1.5",
+ "proc-macro2",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "borsh-derive"
+version = "1.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c2593a3b8b938bd68373196c9832f516be11fa487ef4ae745eb282e6a56a7244"
+dependencies = [
+ "once_cell",
+ "proc-macro-crate 3.2.0",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "borsh-derive-internal"
+version = "0.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5449c28a7b352f2d1e592a8a28bf139bc71afb0764a14f3c02500935d8c44065"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "borsh-derive-internal"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "65d6ba50644c98714aa2a70d13d7df3cd75cd2b523a2b452bf010443800976b3"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "borsh-schema-derive-internal"
+version = "0.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cdbd5696d8bfa21d53d9fe39a714a18538bad11492a42d066dbbc395fb1951c0"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "borsh-schema-derive-internal"
+version = "0.10.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "276691d96f063427be83e6692b86148e488ebba9f48f77788724ca027ba3b6d4"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "bs58"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "771fe0050b883fcc3ea2359b1a96bcfbc090b7116eae7c3c512c7a083fdf23d3"
+
+[[package]]
+name = "bs58"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bf88ba1141d185c399bee5288d850d63b8369520c1eafc32a0430b5b6c287bf4"
+dependencies = [
+ "tinyvec",
+]
+
+[[package]]
+name = "bumpalo"
+version = "3.16.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "79296716171880943b8470b5f8d03aa55eb2e645a4874bdbb28adb49162e012c"
+
+[[package]]
+name = "bv"
+version = "0.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8834bb1d8ee5dc048ee3124f2c7c1afcc6bc9aed03f11e9dfd8c69470a5db340"
+dependencies = [
+ "feature-probe",
+ "serde",
+]
+
+[[package]]
+name = "bytemuck"
+version = "1.21.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ef657dfab802224e671f5818e9a4935f9b1957ed18e58292690cc39e7a4092a3"
+dependencies = [
+ "bytemuck_derive",
+]
+
+[[package]]
+name = "bytemuck_derive"
+version = "1.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3fa76293b4f7bb636ab88fd78228235b5248b4d05cc589aed610f954af5d7c7a"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "byteorder"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b"
+
+[[package]]
+name = "cargo_toml"
+version = "0.19.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a98356df42a2eb1bd8f1793ae4ee4de48e384dd974ce5eac8eee802edb7492be"
+dependencies = [
+ "serde",
+ "toml 0.8.19",
+]
+
+[[package]]
+name = "cc"
+version = "1.2.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8d6dbb628b8f8555f86d0323c2eb39e3ec81901f4b83e091db8a6a76d316a333"
+dependencies = [
+ "jobserver",
+ "libc",
+ "shlex",
+]
+
+[[package]]
+name = "cfg-if"
+version = "1.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd"
+
+[[package]]
+name = "cfg_aliases"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "613afe47fcd5fac7ccf1db93babcb082c5994d996f20b8b159f2ad1658eb5724"
+
+[[package]]
+name = "chrono"
+version = "0.4.39"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7e36cc9d416881d2e24f9a963be5fb1cd90966419ac844274161d10488b3e825"
+dependencies = [
+ "num-traits",
+]
+
+[[package]]
+name = "cipher"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7ee52072ec15386f770805afd189a01c8841be8696bed250fa2f13c4c0d6dfb7"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "console_error_panic_hook"
+version = "0.1.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a06aeb73f470f66dcdbf7223caeebb85984942f22f1adb2a088cf9668146bbbc"
+dependencies = [
+ "cfg-if",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "console_log"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e89f72f65e8501878b8a004d5a1afb780987e2ce2b4532c562e367a72c57499f"
+dependencies = [
+ "log",
+ "web-sys",
+]
+
+[[package]]
+name = "constant_time_eq"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7c74b8349d32d297c9134b8c88677813a227df8f779daa29bfc29c183fe3dca6"
+
+[[package]]
+name = "cpufeatures"
+version = "0.2.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "16b80225097f2e5ae4e7179dd2266824648f3e2f49d9134d584b76389d31c4c3"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "crossbeam-deque"
+version = "0.8.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9dd111b7b7f7d55b72c0a6ae361660ee5853c9af73f70c3c2ef6858b950e2e51"
+dependencies = [
+ "crossbeam-epoch",
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "crossbeam-epoch"
+version = "0.9.18"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e"
+dependencies = [
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "crossbeam-utils"
+version = "0.8.21"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0a5c400df2834b80a4c3327b3aad3a4c4cd4de0629063962b03235697506a28"
+
+[[package]]
+name = "crunchy"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7"
+
+[[package]]
+name = "crypto-common"
+version = "0.1.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3"
+dependencies = [
+ "generic-array",
+ "typenum",
+]
+
+[[package]]
+name = "crypto-mac"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b584a330336237c1eecd3e94266efb216c56ed91225d634cb2991c5f3fd1aeab"
+dependencies = [
+ "generic-array",
+ "subtle",
+]
+
+[[package]]
+name = "ctr"
+version = "0.8.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "049bb91fb4aaf0e3c7efa6cd5ef877dbbbd15b39dad06d9948de4ec8a75761ea"
+dependencies = [
+ "cipher",
+]
+
+[[package]]
+name = "curve25519-dalek"
+version = "3.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "90f9d052967f590a76e62eb387bd0bbb1b000182c3cefe5364db6b7211651bc0"
+dependencies = [
+ "byteorder",
+ "digest 0.9.0",
+ "rand_core 0.5.1",
+ "serde",
+ "subtle",
+ "zeroize",
+]
+
+[[package]]
+name = "darling"
+version = "0.20.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6f63b86c8a8826a49b8c21f08a2d07338eec8d900540f8630dc76284be802989"
+dependencies = [
+ "darling_core",
+ "darling_macro",
+]
+
+[[package]]
+name = "darling_core"
+version = "0.20.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "95133861a8032aaea082871032f5815eb9e98cef03fa916ab4500513994df9e5"
+dependencies = [
+ "fnv",
+ "ident_case",
+ "proc-macro2",
+ "quote",
+ "strsim",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "darling_macro"
+version = "0.20.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d336a2a514f6ccccaa3e09b02d41d35330c07ddf03a62165fcec10bb561c7806"
+dependencies = [
+ "darling_core",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "derivation-path"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6e5c37193a1db1d8ed868c03ec7b152175f26160a5b740e5e484143877e0adf0"
+
+[[package]]
+name = "derivative"
+version = "2.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fcc3dd5e9e9c0b295d6e1e4d811fb6f157d5ffd784b8d202fc62eac8035a770b"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "digest"
+version = "0.9.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d3dd60d1080a57a05ab032377049e0591415d2b31afd7028356dbf3cc6dcb066"
+dependencies = [
+ "generic-array",
+]
+
+[[package]]
+name = "digest"
+version = "0.10.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292"
+dependencies = [
+ "block-buffer 0.10.4",
+ "crypto-common",
+ "subtle",
+]
+
+[[package]]
+name = "ed25519"
+version = "1.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "91cff35c70bba8a626e3185d8cd48cc11b5437e1a5bcd15b9b5fa3c64b6dfee7"
+dependencies = [
+ "signature",
+]
+
+[[package]]
+name = "ed25519-dalek"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c762bae6dcaf24c4c84667b8579785430908723d5c889f469d76a41d59cc7a9d"
+dependencies = [
+ "curve25519-dalek",
+ "ed25519",
+ "rand 0.7.3",
+ "serde",
+ "sha2 0.9.9",
+ "zeroize",
+]
+
+[[package]]
+name = "ed25519-dalek-bip32"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9d2be62a4061b872c8c0873ee4fc6f101ce7b889d039f019c5fa2af471a59908"
+dependencies = [
+ "derivation-path",
+ "ed25519-dalek",
+ "hmac 0.12.1",
+ "sha2 0.10.8",
+]
+
+[[package]]
+name = "either"
+version = "1.13.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "60b1af1c220855b6ceac025d3f6ecdd2b7c4894bfe9cd9bda4fbb4bc7c0d4cf0"
+
+[[package]]
+name = "env_logger"
+version = "0.9.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a12e6657c4c97ebab115a42dcee77225f7f482cdd841cf7088c657a42e9e00e7"
+dependencies = [
+ "atty",
+ "humantime",
+ "log",
+ "regex",
+ "termcolor",
+]
+
+[[package]]
+name = "equivalent"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5443807d6dff69373d433ab9ef5378ad8df50ca6298caf15de6e52e24aaf54d5"
+
+[[package]]
+name = "feature-probe"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "835a3dc7d1ec9e75e2b5fb4ba75396837112d2060b03f7d43bc1897c7f7211da"
+
+[[package]]
+name = "fnv"
+version = "1.0.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1"
+
+[[package]]
+name = "generic-array"
+version = "0.14.7"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a"
+dependencies = [
+ "serde",
+ "typenum",
+ "version_check",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.1.16"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce"
+dependencies = [
+ "cfg-if",
+ "js-sys",
+ "libc",
+ "wasi 0.9.0+wasi-snapshot-preview1",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "getrandom"
+version = "0.2.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7"
+dependencies = [
+ "cfg-if",
+ "js-sys",
+ "libc",
+ "wasi 0.11.0+wasi-snapshot-preview1",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.11.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e"
+dependencies = [
+ "ahash 0.7.8",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.13.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "43a3c133739dddd0d2990f9a4bdf8eb4b21ef50e4851ca85ab661199821d510e"
+dependencies = [
+ "ahash 0.8.11",
+]
+
+[[package]]
+name = "hashbrown"
+version = "0.15.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bf151400ff0baff5465007dd2f3e717f3fe502074ca563069ce3a6629d07b289"
+
+[[package]]
+name = "heck"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6d621efb26863f0e9924c6ac577e8275e5e6b77455db64ffa6c65c904e9e132c"
+dependencies = [
+ "unicode-segmentation",
+]
+
+[[package]]
+name = "hermit-abi"
+version = "0.1.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "hmac"
+version = "0.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "126888268dcc288495a26bf004b38c5fdbb31682f992c84ceb046a1f0fe38840"
+dependencies = [
+ "crypto-mac",
+ "digest 0.9.0",
+]
+
+[[package]]
+name = "hmac"
+version = "0.12.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e"
+dependencies = [
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "hmac-drbg"
+version = "0.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "17ea0a1394df5b6574da6e0c1ade9e78868c9fb0a4e5ef4428e32da4676b85b1"
+dependencies = [
+ "digest 0.9.0",
+ "generic-array",
+ "hmac 0.8.1",
+]
+
+[[package]]
+name = "humantime"
+version = "2.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9a3a5bfb195931eeb336b2a7b4d761daec841b97f947d34394601737a7bba5e4"
+
+[[package]]
+name = "ident_case"
+version = "1.0.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39"
+
+[[package]]
+name = "im"
+version = "15.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0acd33ff0285af998aaf9b57342af478078f53492322fafc47450e09397e0e9"
+dependencies = [
+ "bitmaps",
+ "rand_core 0.6.4",
+ "rand_xoshiro",
+ "rayon",
+ "serde",
+ "sized-chunks",
+ "typenum",
+ "version_check",
+]
+
+[[package]]
+name = "indexmap"
+version = "2.7.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "62f822373a4fe84d4bb149bf54e584a7f4abec90e072ed49cda0edea5b95471f"
+dependencies = [
+ "equivalent",
+ "hashbrown 0.15.2",
+]
+
+[[package]]
+name = "itertools"
+version = "0.10.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b0fd2260e829bddf4cb6ea802289de2f86d6a7a690192fbe91b3f46e0f2c8473"
+dependencies = [
+ "either",
+]
+
+[[package]]
+name = "itoa"
+version = "1.0.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d75a2a4b1b190afb6f5425f10f6a8f959d2ea0b9c2b1d79553551850539e4674"
+
+[[package]]
+name = "jobserver"
+version = "0.1.32"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "48d1dbcbbeb6a7fec7e059840aa538bd62aaccf972c7346c4d9d2059312853d0"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "js-sys"
+version = "0.3.76"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6717b6b5b077764fb5966237269cb3c64edddde4b14ce42647430a78ced9e7b7"
+dependencies = [
+ "once_cell",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "keccak"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ecc2af9a1119c51f12a14607e783cb977bde58bc069ff0c3da1095e635d70654"
+dependencies = [
+ "cpufeatures",
+]
+
+[[package]]
+name = "lazy_static"
+version = "1.5.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "bbd2bcb4c963f2ddae06a2efc7e9f3591312473c50c6685e1f298068316e66fe"
+
+[[package]]
+name = "libc"
+version = "0.2.169"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b5aba8db14291edd000dfcc4d620c7ebfb122c613afb886ca8803fa4e128a20a"
+
+[[package]]
+name = "libsecp256k1"
+version = "0.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c9d220bc1feda2ac231cb78c3d26f27676b8cf82c96971f7aeef3d0cf2797c73"
+dependencies = [
+ "arrayref",
+ "base64 0.12.3",
+ "digest 0.9.0",
+ "hmac-drbg",
+ "libsecp256k1-core",
+ "libsecp256k1-gen-ecmult",
+ "libsecp256k1-gen-genmult",
+ "rand 0.7.3",
+ "serde",
+ "sha2 0.9.9",
+ "typenum",
+]
+
+[[package]]
+name = "libsecp256k1-core"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d0f6ab710cec28cef759c5f18671a27dae2a5f952cdaaee1d8e2908cb2478a80"
+dependencies = [
+ "crunchy",
+ "digest 0.9.0",
+ "subtle",
+]
+
+[[package]]
+name = "libsecp256k1-gen-ecmult"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ccab96b584d38fac86a83f07e659f0deafd0253dc096dab5a36d53efe653c5c3"
+dependencies = [
+ "libsecp256k1-core",
+]
+
+[[package]]
+name = "libsecp256k1-gen-genmult"
+version = "0.2.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "67abfe149395e3aa1c48a2beb32b068e2334402df8181f818d3aee2b304c4f5d"
+dependencies = [
+ "libsecp256k1-core",
+]
+
+[[package]]
+name = "light-poseidon"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3c9a85a9752c549ceb7578064b4ed891179d20acd85f27318573b64d2d7ee7ee"
+dependencies = [
+ "ark-bn254",
+ "ark-ff",
+ "num-bigint",
+ "thiserror",
+]
+
+[[package]]
+name = "lock_api"
+version = "0.4.12"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "07af8b9cdd281b7915f413fa73f29ebd5d55d0d3f0155584dade1ff18cea1b17"
+dependencies = [
+ "autocfg",
+ "scopeguard",
+]
+
+[[package]]
+name = "log"
+version = "0.4.22"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a7a70ba024b9dc04c27ea2f0c0548feb474ec5c54bba33a7f72f873a39d07b24"
+
+[[package]]
+name = "memchr"
+version = "2.7.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3"
+
+[[package]]
+name = "memmap2"
+version = "0.5.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "83faa42c0a078c393f6b29d5db232d8be22776a891f8f56e5284faee4a20b327"
+dependencies = [
+ "libc",
+]
+
+[[package]]
+name = "memoffset"
+version = "0.9.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "488016bfae457b036d996092f6cb448677611ce4449e970ceaf42695203f218a"
+dependencies = [
+ "autocfg",
+]
+
+[[package]]
+name = "merlin"
+version = "3.0.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "58c38e2799fc0978b65dfff8023ec7843e2330bb462f19198840b34b6582397d"
+dependencies = [
+ "byteorder",
+ "keccak",
+ "rand_core 0.6.4",
+ "zeroize",
+]
+
+[[package]]
+name = "mpl-token-metadata"
+version = "4.1.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "caf0f61b553e424a6234af1268456972ee66c2222e1da89079242251fa7479e5"
+dependencies = [
+ "borsh 0.10.4",
+ "num-derive 0.3.3",
+ "num-traits",
+ "solana-program",
+ "thiserror",
+]
+
+[[package]]
+name = "num-bigint"
+version = "0.4.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a5e44f723f1133c9deac646763579fdb3ac745e418f2a7af9cd0c431da1f20b9"
+dependencies = [
+ "num-integer",
+ "num-traits",
+]
+
+[[package]]
+name = "num-derive"
+version = "0.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "876a53fff98e03a936a674b29568b0e605f06b29372c2489ff4de23f1949743d"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 1.0.109",
+]
+
+[[package]]
+name = "num-derive"
+version = "0.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ed3955f1a9c7c0c15e092f9c887db08b1fc683305fdf6eb6684f22555355e202"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "num-integer"
+version = "0.1.46"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7969661fd2958a5cb096e56c8e1ad0444ac2bbcd0061bd28660485a44879858f"
+dependencies = [
+ "num-traits",
+]
+
+[[package]]
+name = "num-traits"
+version = "0.2.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841"
+dependencies = [
+ "autocfg",
+]
+
+[[package]]
+name = "num_enum"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4e613fc340b2220f734a8595782c551f1250e969d87d3be1ae0579e8d4065179"
+dependencies = [
+ "num_enum_derive",
+]
+
+[[package]]
+name = "num_enum_derive"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "af1844ef2428cc3e1cb900be36181049ef3d3193c63e43026cfe202983b27a56"
+dependencies = [
+ "proc-macro-crate 3.2.0",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "once_cell"
+version = "1.20.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1261fe7e33c73b354eab43b1273a57c8f967d0391e80353e51f764ac02cf6775"
+
+[[package]]
+name = "opaque-debug"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c08d65885ee38876c4f86fa503fb49d7b507c2b62552df7c70b2fce627e06381"
+
+[[package]]
+name = "parking_lot"
+version = "0.12.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f1bf18183cf54e8d6059647fc3063646a1801cf30896933ec2311622cc4b9a27"
+dependencies = [
+ "lock_api",
+ "parking_lot_core",
+]
+
+[[package]]
+name = "parking_lot_core"
+version = "0.9.10"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8"
+dependencies = [
+ "cfg-if",
+ "libc",
+ "redox_syscall",
+ "smallvec",
+ "windows-targets",
+]
+
+[[package]]
+name = "paste"
+version = "1.0.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a"
+
+[[package]]
+name = "pbkdf2"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "216eaa586a190f0a738f2f918511eecfa90f13295abec0e457cdebcceda80cbd"
+dependencies = [
+ "crypto-mac",
+]
+
+[[package]]
+name = "pbkdf2"
+version = "0.11.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "83a0692ec44e4cf1ef28ca317f14f8f07da2d95ec3fa01f86e4467b725e60917"
+dependencies = [
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "percent-encoding"
+version = "2.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e3148f5046208a5d56bcfc03053e3ca6334e51da8dfb19b6cdc8b306fae3283e"
+
+[[package]]
+name = "polyval"
+version = "0.5.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8419d2b623c7c0896ff2d5d96e2cb4ede590fed28fcc34934f4c33c036e620a1"
+dependencies = [
+ "cfg-if",
+ "cpufeatures",
+ "opaque-debug",
+ "universal-hash",
+]
+
+[[package]]
+name = "ppv-lite86"
+version = "0.2.20"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "77957b295656769bb8ad2b6a6b09d897d94f05c41b069aede1fcdaa675eaea04"
+dependencies = [
+ "zerocopy",
+]
+
+[[package]]
+name = "proc-macro-crate"
+version = "0.1.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1d6ea3c4595b96363c13943497db34af4460fb474a95c43f4446ad341b8c9785"
+dependencies = [
+ "toml 0.5.11",
+]
+
+[[package]]
+name = "proc-macro-crate"
+version = "3.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8ecf48c7ca261d60b74ab1a7b20da18bede46776b2e55535cb958eb595c5fa7b"
+dependencies = [
+ "toml_edit",
+]
+
+[[package]]
+name = "proc-macro2"
+version = "1.0.92"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "37d3544b3f2748c54e147655edb5025752e2303145b5aefb3c3ea2c78b973bb0"
+dependencies = [
+ "unicode-ident",
+]
+
+[[package]]
+name = "qstring"
+version = "0.7.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d464fae65fff2680baf48019211ce37aaec0c78e9264c84a3e484717f965104e"
+dependencies = [
+ "percent-encoding",
+]
+
+[[package]]
+name = "qualifier_attr"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9e2e25ee72f5b24d773cae88422baddefff7714f97aab68d96fe2b6fc4a28fb2"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "quote"
+version = "1.0.38"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0e4dccaaaf89514f546c693ddc140f729f958c247918a13380cccc6078391acc"
+dependencies = [
+ "proc-macro2",
+]
+
+[[package]]
+name = "rand"
+version = "0.7.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03"
+dependencies = [
+ "getrandom 0.1.16",
+ "libc",
+ "rand_chacha 0.2.2",
+ "rand_core 0.5.1",
+ "rand_hc",
+]
+
+[[package]]
+name = "rand"
+version = "0.8.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404"
+dependencies = [
+ "libc",
+ "rand_chacha 0.3.1",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.2.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "rand_chacha"
+version = "0.3.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88"
+dependencies = [
+ "ppv-lite86",
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.5.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19"
+dependencies = [
+ "getrandom 0.1.16",
+]
+
+[[package]]
+name = "rand_core"
+version = "0.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c"
+dependencies = [
+ "getrandom 0.2.15",
+]
+
+[[package]]
+name = "rand_hc"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c"
+dependencies = [
+ "rand_core 0.5.1",
+]
+
+[[package]]
+name = "rand_xoshiro"
+version = "0.6.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6f97cdb2a36ed4183de61b2f824cc45c9f1037f28afe0a322e9fff4c108b5aaa"
+dependencies = [
+ "rand_core 0.6.4",
+]
+
+[[package]]
+name = "rayon"
+version = "1.10.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b418a60154510ca1a002a752ca9714984e21e4241e804d32555251faf8b78ffa"
+dependencies = [
+ "either",
+ "rayon-core",
+]
+
+[[package]]
+name = "rayon-core"
+version = "1.12.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1465873a3dfdaa8ae7cb14b4383657caab0b3e8a0aa9ae8e04b044854c8dfce2"
+dependencies = [
+ "crossbeam-deque",
+ "crossbeam-utils",
+]
+
+[[package]]
+name = "redox_syscall"
+version = "0.5.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "03a862b389f93e68874fbf580b9de08dd02facb9a788ebadaf4a3fd33cf58834"
+dependencies = [
+ "bitflags",
+]
+
+[[package]]
+name = "regex"
+version = "1.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b544ef1b4eac5dc2db33ea63606ae9ffcfac26c1416a2806ae0bf5f56b201191"
+dependencies = [
+ "aho-corasick",
+ "memchr",
+ "regex-automata",
+ "regex-syntax",
+]
+
+[[package]]
+name = "regex-automata"
+version = "0.4.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "809e8dc61f6de73b46c85f4c96486310fe304c434cfa43669d7b40f711150908"
+dependencies = [
+ "aho-corasick",
+ "memchr",
+ "regex-syntax",
+]
+
+[[package]]
+name = "regex-syntax"
+version = "0.8.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c"
+
+[[package]]
+name = "rustc-hash"
+version = "1.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2"
+
+[[package]]
+name = "rustc_version"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cfcb3a22ef46e85b45de6ee7e79d063319ebb6594faafcf1c225ea92ab6e9b92"
+dependencies = [
+ "semver",
+]
+
+[[package]]
+name = "rustversion"
+version = "1.0.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f7c45b9784283f1b2e7fb61b42047c2fd678ef0960d4f6f1eba131594cc369d4"
+
+[[package]]
+name = "ryu"
+version = "1.0.18"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f3cb5ba0dc43242ce17de99c180e96db90b235b8a9fdc9543c96d2209116bd9f"
+
+[[package]]
+name = "scopeguard"
+version = "1.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "94143f37725109f92c262ed2cf5e59bce7498c01bcc1502d7b9afe439a4e9f49"
+
+[[package]]
+name = "semver"
+version = "1.0.24"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3cb6eb87a131f756572d7fb904f6e7b68633f09cca868c5df1c4b8d1a694bbba"
+
+[[package]]
+name = "serde"
+version = "1.0.217"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "02fc4265df13d6fa1d00ecff087228cc0a2b5f3c0e87e258d8b94a156e984c70"
+dependencies = [
+ "serde_derive",
+]
+
+[[package]]
+name = "serde_bytes"
+version = "0.11.15"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "387cc504cb06bb40a96c8e04e951fe01854cf6bc921053c954e4a606d9675c6a"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "serde_derive"
+version = "1.0.217"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5a9bf7cf98d04a2b28aead066b7496853d4779c9cc183c440dbac457641e19a0"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "serde_json"
+version = "1.0.134"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d00f4175c42ee48b15416f6193a959ba3a0d67fc699a0db9ad12df9f83991c7d"
+dependencies = [
+ "itoa",
+ "memchr",
+ "ryu",
+ "serde",
+]
+
+[[package]]
+name = "serde_spanned"
+version = "0.6.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "87607cb1398ed59d48732e575a4c28a7a8ebf2454b964fe3f224f2afc07909e1"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "serde_with"
+version = "2.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "07ff71d2c147a7b57362cead5e22f772cd52f6ab31cfcd9edcd7f6aeb2a0afbe"
+dependencies = [
+ "serde",
+ "serde_with_macros",
+]
+
+[[package]]
+name = "serde_with_macros"
+version = "2.3.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "881b6f881b17d13214e5d494c939ebab463d01264ce1811e9d4ac3a882e7695f"
+dependencies = [
+ "darling",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "sha2"
+version = "0.9.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800"
+dependencies = [
+ "block-buffer 0.9.0",
+ "cfg-if",
+ "cpufeatures",
+ "digest 0.9.0",
+ "opaque-debug",
+]
+
+[[package]]
+name = "sha2"
+version = "0.10.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8"
+dependencies = [
+ "cfg-if",
+ "cpufeatures",
+ "digest 0.10.7",
+]
+
+[[package]]
+name = "sha2-const-stable"
+version = "0.1.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5f179d4e11094a893b82fff208f74d448a7512f99f5a0acbd5c679b705f83ed9"
+
+[[package]]
+name = "sha3"
+version = "0.9.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f81199417d4e5de3f04b1e871023acea7389672c4135918f05aa9cbf2f2fa809"
+dependencies = [
+ "block-buffer 0.9.0",
+ "digest 0.9.0",
+ "keccak",
+ "opaque-debug",
+]
+
+[[package]]
+name = "sha3"
+version = "0.10.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "75872d278a8f37ef87fa0ddbda7802605cb18344497949862c0d4dcb291eba60"
+dependencies = [
+ "digest 0.10.7",
+ "keccak",
+]
+
+[[package]]
+name = "shlex"
+version = "1.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0fda2ff0d084019ba4d7c6f371c95d8fd75ce3524c3cb8fb653a3023f6323e64"
+
+[[package]]
+name = "signature"
+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 = "sized-chunks"
+version = "0.6.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "16d69225bde7a69b235da73377861095455d298f2b970996eec25ddbb42b3d1e"
+dependencies = [
+ "bitmaps",
+ "typenum",
+]
+
+[[package]]
+name = "smallvec"
+version = "1.13.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67"
+
+[[package]]
+name = "solana-frozen-abi"
+version = "1.18.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "03ab2c30c15311b511c0d1151e4ab6bc9a3e080a37e7c6e7c2d96f5784cf9434"
+dependencies = [
+ "block-buffer 0.10.4",
+ "bs58 0.4.0",
+ "bv",
+ "either",
+ "generic-array",
+ "im",
+ "lazy_static",
+ "log",
+ "memmap2",
+ "rustc_version",
+ "serde",
+ "serde_bytes",
+ "serde_derive",
+ "sha2 0.10.8",
+ "solana-frozen-abi-macro",
+ "subtle",
+ "thiserror",
+]
+
+[[package]]
+name = "solana-frozen-abi-macro"
+version = "1.18.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c142f779c3633ac83c84d04ff06c70e1f558c876f13358bed77ba629c7417932"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "rustc_version",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "solana-logger"
+version = "1.18.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "121d36ffb3c6b958763312cbc697fbccba46ee837d3a0aa4fc0e90fcb3b884f3"
+dependencies = [
+ "env_logger",
+ "lazy_static",
+ "log",
+]
+
+[[package]]
+name = "solana-program"
+version = "1.18.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c10f4588cefd716b24a1a40dd32c278e43a560ab8ce4de6b5805c9d113afdfa1"
+dependencies = [
+ "ark-bn254",
+ "ark-ec",
+ "ark-ff",
+ "ark-serialize",
+ "base64 0.21.7",
+ "bincode",
+ "bitflags",
+ "blake3",
+ "borsh 0.10.4",
+ "borsh 0.9.3",
+ "borsh 1.5.3",
+ "bs58 0.4.0",
+ "bv",
+ "bytemuck",
+ "cc",
+ "console_error_panic_hook",
+ "console_log",
+ "curve25519-dalek",
+ "getrandom 0.2.15",
+ "itertools",
+ "js-sys",
+ "lazy_static",
+ "libc",
+ "libsecp256k1",
+ "light-poseidon",
+ "log",
+ "memoffset",
+ "num-bigint",
+ "num-derive 0.4.2",
+ "num-traits",
+ "parking_lot",
+ "rand 0.8.5",
+ "rustc_version",
+ "rustversion",
+ "serde",
+ "serde_bytes",
+ "serde_derive",
+ "serde_json",
+ "sha2 0.10.8",
+ "sha3 0.10.8",
+ "solana-frozen-abi",
+ "solana-frozen-abi-macro",
+ "solana-sdk-macro",
+ "thiserror",
+ "tiny-bip39",
+ "wasm-bindgen",
+ "zeroize",
+]
+
+[[package]]
+name = "solana-sdk"
+version = "1.18.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "580ad66c2f7a4c3cb3244fe21440546bd500f5ecb955ad9826e92a78dded8009"
+dependencies = [
+ "assert_matches",
+ "base64 0.21.7",
+ "bincode",
+ "bitflags",
+ "borsh 1.5.3",
+ "bs58 0.4.0",
+ "bytemuck",
+ "byteorder",
+ "chrono",
+ "derivation-path",
+ "digest 0.10.7",
+ "ed25519-dalek",
+ "ed25519-dalek-bip32",
+ "generic-array",
+ "hmac 0.12.1",
+ "itertools",
+ "js-sys",
+ "lazy_static",
+ "libsecp256k1",
+ "log",
+ "memmap2",
+ "num-derive 0.4.2",
+ "num-traits",
+ "num_enum",
+ "pbkdf2 0.11.0",
+ "qstring",
+ "qualifier_attr",
+ "rand 0.7.3",
+ "rand 0.8.5",
+ "rustc_version",
+ "rustversion",
+ "serde",
+ "serde_bytes",
+ "serde_derive",
+ "serde_json",
+ "serde_with",
+ "sha2 0.10.8",
+ "sha3 0.10.8",
+ "siphasher",
+ "solana-frozen-abi",
+ "solana-frozen-abi-macro",
+ "solana-logger",
+ "solana-program",
+ "solana-sdk-macro",
+ "thiserror",
+ "uriparse",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "solana-sdk-macro"
+version = "1.18.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1b75d0f193a27719257af19144fdaebec0415d1c9e9226ae4bd29b791be5e9bd"
+dependencies = [
+ "bs58 0.4.0",
+ "proc-macro2",
+ "quote",
+ "rustversion",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "solana-security-txt"
+version = "1.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "468aa43b7edb1f9b7b7b686d5c3aeb6630dc1708e86e31343499dd5c4d775183"
+
+[[package]]
+name = "solana-zk-token-sdk"
+version = "1.18.26"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7cbdf4249b6dfcbba7d84e2b53313698043f60f8e22ce48286e6fbe8a17c8d16"
+dependencies = [
+ "aes-gcm-siv",
+ "base64 0.21.7",
+ "bincode",
+ "bytemuck",
+ "byteorder",
+ "curve25519-dalek",
+ "getrandom 0.1.16",
+ "itertools",
+ "lazy_static",
+ "merlin",
+ "num-derive 0.4.2",
+ "num-traits",
+ "rand 0.7.3",
+ "serde",
+ "serde_json",
+ "sha3 0.9.1",
+ "solana-program",
+ "solana-sdk",
+ "subtle",
+ "thiserror",
+ "zeroize",
+]
+
+[[package]]
+name = "spl-associated-token-account"
+version = "3.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "143109d789171379e6143ef23191786dfaac54289ad6e7917cfb26b36c432b10"
+dependencies = [
+ "assert_matches",
+ "borsh 1.5.3",
+ "num-derive 0.4.2",
+ "num-traits",
+ "solana-program",
+ "spl-token",
+ "spl-token-2022",
+ "thiserror",
+]
+
+[[package]]
+name = "spl-discriminator"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "210101376962bb22bb13be6daea34656ea1cbc248fce2164b146e39203b55e03"
+dependencies = [
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator-derive",
+]
+
+[[package]]
+name = "spl-discriminator-derive"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "d9e8418ea6269dcfb01c712f0444d2c75542c04448b480e87de59d2865edc750"
+dependencies = [
+ "quote",
+ "spl-discriminator-syn",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "spl-discriminator-syn"
+version = "0.2.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8c1f05593b7ca9eac7caca309720f2eafb96355e037e6d373b909a80fe7b69b9"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "sha2 0.10.8",
+ "syn 2.0.94",
+ "thiserror",
+]
+
+[[package]]
+name = "spl-memo"
+version = "4.0.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a49f49f95f2d02111ded31696ab38a081fab623d4c76bd4cb074286db4560836"
+dependencies = [
+ "solana-program",
+]
+
+[[package]]
+name = "spl-pod"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c52d84c55efeef8edcc226743dc089d7e3888b8e3474569aa3eff152b37b9996"
+dependencies = [
+ "borsh 1.5.3",
+ "bytemuck",
+ "solana-program",
+ "solana-zk-token-sdk",
+ "spl-program-error",
+]
+
+[[package]]
+name = "spl-program-error"
+version = "0.4.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e45a49acb925db68aa501b926096b2164adbdcade7a0c24152af9f0742d0a602"
+dependencies = [
+ "num-derive 0.4.2",
+ "num-traits",
+ "solana-program",
+ "spl-program-error-derive",
+ "thiserror",
+]
+
+[[package]]
+name = "spl-program-error-derive"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6d375dd76c517836353e093c2dbb490938ff72821ab568b545fd30ab3256b3e"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "sha2 0.10.8",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "spl-tlv-account-resolution"
+version = "0.6.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fab8edfd37be5fa17c9e42c1bff86abbbaf0494b031b37957f2728ad2ff842ba"
+dependencies = [
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+ "spl-type-length-value",
+]
+
+[[package]]
+name = "spl-token"
+version = "4.0.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b9eb465e4bf5ce1d498f05204c8089378c1ba34ef2777ea95852fc53a1fd4fb2"
+dependencies = [
+ "arrayref",
+ "bytemuck",
+ "num-derive 0.4.2",
+ "num-traits",
+ "num_enum",
+ "solana-program",
+ "thiserror",
+]
+
+[[package]]
+name = "spl-token-2022"
+version = "3.0.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4c39e416aeb1ea0b22f3b2bbecaf7e38a92a1aa8f4a0c5785c94179694e846a0"
+dependencies = [
+ "arrayref",
+ "bytemuck",
+ "num-derive 0.4.2",
+ "num-traits",
+ "num_enum",
+ "solana-program",
+ "solana-security-txt",
+ "solana-zk-token-sdk",
+ "spl-memo",
+ "spl-pod",
+ "spl-token",
+ "spl-token-group-interface",
+ "spl-token-metadata-interface",
+ "spl-transfer-hook-interface",
+ "spl-type-length-value",
+ "thiserror",
+]
+
+[[package]]
+name = "spl-token-group-interface"
+version = "0.2.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "014817d6324b1e20c4bbc883e8ee30a5faa13e59d91d1b2b95df98b920150c17"
+dependencies = [
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+]
+
+[[package]]
+name = "spl-token-metadata-interface"
+version = "0.3.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f3da00495b602ebcf5d8ba8b3ecff1ee454ce4c125c9077747be49c2d62335ba"
+dependencies = [
+ "borsh 1.5.3",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+ "spl-type-length-value",
+]
+
+[[package]]
+name = "spl-transfer-hook-interface"
+version = "0.6.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a9b5c08a89838e5a2931f79b17f611857f281a14a2100968a3ccef352cb7414b"
+dependencies = [
+ "arrayref",
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+ "spl-tlv-account-resolution",
+ "spl-type-length-value",
+]
+
+[[package]]
+name = "spl-type-length-value"
+version = "0.4.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "c872f93d0600e743116501eba2d53460e73a12c9a496875a42a7d70e034fe06d"
+dependencies = [
+ "bytemuck",
+ "solana-program",
+ "spl-discriminator",
+ "spl-pod",
+ "spl-program-error",
+]
+
+[[package]]
+name = "strsim"
+version = "0.11.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f"
+
+[[package]]
+name = "subtle"
+version = "2.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "6bdef32e8150c2a081110b42772ffe7d7c9032b606bc226c8260fd97e0976601"
+
+[[package]]
+name = "syn"
+version = "1.0.109"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "72b64191b275b66ffe2469e8af2c1cfe3bafa67b529ead792a6d0160888b4237"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unicode-ident",
+]
+
+[[package]]
+name = "syn"
+version = "2.0.94"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "987bc0be1cdea8b10216bd06e2ca407d40b9543468fafd3ddfb02f36e77f71f3"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "unicode-ident",
+]
+
+[[package]]
+name = "termcolor"
+version = "1.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "06794f8f6c5c898b3275aebefa6b8a1cb24cd2c6c79397ab15774837a0bc5755"
+dependencies = [
+ "winapi-util",
+]
+
+[[package]]
+name = "thiserror"
+version = "1.0.69"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b6aaf5339b578ea85b50e080feb250a3e8ae8cfcdff9a461c9ec2904bc923f52"
+dependencies = [
+ "thiserror-impl",
+]
+
+[[package]]
+name = "thiserror-impl"
+version = "1.0.69"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "tiny-bip39"
+version = "0.8.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ffc59cb9dfc85bb312c3a78fd6aa8a8582e310b0fa885d5bb877f6dcc601839d"
+dependencies = [
+ "anyhow",
+ "hmac 0.8.1",
+ "once_cell",
+ "pbkdf2 0.4.0",
+ "rand 0.7.3",
+ "rustc-hash",
+ "sha2 0.9.9",
+ "thiserror",
+ "unicode-normalization",
+ "wasm-bindgen",
+ "zeroize",
+]
+
+[[package]]
+name = "tinyvec"
+version = "1.8.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "022db8904dfa342efe721985167e9fcd16c29b226db4397ed752a761cfce81e8"
+dependencies = [
+ "tinyvec_macros",
+]
+
+[[package]]
+name = "tinyvec_macros"
+version = "0.1.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
+
+[[package]]
+name = "toml"
+version = "0.5.11"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f4f7f0dd8d50a853a531c426359045b1998f04219d88799810762cd4ad314234"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "toml"
+version = "0.8.19"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a1ed1f98e3fdc28d6d910e6737ae6ab1a93bf1985935a1193e68f93eeb68d24e"
+dependencies = [
+ "serde",
+ "serde_spanned",
+ "toml_datetime",
+ "toml_edit",
+]
+
+[[package]]
+name = "toml_datetime"
+version = "0.6.8"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0dd7358ecb8fc2f8d014bf86f6f638ce72ba252a2c3a2572f2a795f1d23efb41"
+dependencies = [
+ "serde",
+]
+
+[[package]]
+name = "toml_edit"
+version = "0.22.22"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4ae48d6208a266e853d946088ed816055e556cc6028c5e8e2b84d9fa5dd7c7f5"
+dependencies = [
+ "indexmap",
+ "serde",
+ "serde_spanned",
+ "toml_datetime",
+ "winnow",
+]
+
+[[package]]
+name = "typenum"
+version = "1.17.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "42ff0bf0c66b8238c6f3b578df37d0b7848e55df8577b3f74f92a69acceeb825"
+
+[[package]]
+name = "unicode-ident"
+version = "1.0.14"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "adb9e6ca4f869e1180728b7950e35922a7fc6397f7b641499e8f3ef06e50dc83"
+
+[[package]]
+name = "unicode-normalization"
+version = "0.1.24"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5033c97c4262335cded6d6fc3e5c18ab755e1a3dc96376350f3d8e9f009ad956"
+dependencies = [
+ "tinyvec",
+]
+
+[[package]]
+name = "unicode-segmentation"
+version = "1.12.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "f6ccf251212114b54433ec949fd6a7841275f9ada20dddd2f29e9ceea4501493"
+
+[[package]]
+name = "universal-hash"
+version = "0.4.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9f214e8f697e925001e66ec2c6e37a4ef93f0f78c2eed7814394e10c62025b05"
+dependencies = [
+ "generic-array",
+ "subtle",
+]
+
+[[package]]
+name = "uriparse"
+version = "0.6.4"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0200d0fc04d809396c2ad43f3c95da3582a2556eba8d453c1087f4120ee352ff"
+dependencies = [
+ "fnv",
+ "lazy_static",
+]
+
+[[package]]
+name = "version_check"
+version = "0.9.5"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0b928f33d975fc6ad9f86c8f283853ad26bdd5b10b7f1542aa2fa15e2289105a"
+
+[[package]]
+name = "wasi"
+version = "0.9.0+wasi-snapshot-preview1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519"
+
+[[package]]
+name = "wasi"
+version = "0.11.0+wasi-snapshot-preview1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423"
+
+[[package]]
+name = "wasm-bindgen"
+version = "0.2.99"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "a474f6281d1d70c17ae7aa6a613c87fce69a127e2624002df63dcb39d6cf6396"
+dependencies = [
+ "cfg-if",
+ "once_cell",
+ "wasm-bindgen-macro",
+]
+
+[[package]]
+name = "wasm-bindgen-backend"
+version = "0.2.99"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5f89bb38646b4f81674e8f5c3fb81b562be1fd936d84320f3264486418519c79"
+dependencies = [
+ "bumpalo",
+ "log",
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-macro"
+version = "0.2.99"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "2cc6181fd9a7492eef6fef1f33961e3695e4579b9872a6f7c83aee556666d4fe"
+dependencies = [
+ "quote",
+ "wasm-bindgen-macro-support",
+]
+
+[[package]]
+name = "wasm-bindgen-macro-support"
+version = "0.2.99"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "30d7a95b763d3c45903ed6c81f156801839e5ee968bb07e534c44df0fcd330c2"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+ "wasm-bindgen-backend",
+ "wasm-bindgen-shared",
+]
+
+[[package]]
+name = "wasm-bindgen-shared"
+version = "0.2.99"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "943aab3fdaaa029a6e0271b35ea10b72b943135afe9bffca82384098ad0e06a6"
+
+[[package]]
+name = "web-sys"
+version = "0.3.76"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "04dd7223427d52553d3702c004d3b2fe07c148165faa56313cb00211e31c12bc"
+dependencies = [
+ "js-sys",
+ "wasm-bindgen",
+]
+
+[[package]]
+name = "winapi"
+version = "0.3.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419"
+dependencies = [
+ "winapi-i686-pc-windows-gnu",
+ "winapi-x86_64-pc-windows-gnu",
+]
+
+[[package]]
+name = "winapi-i686-pc-windows-gnu"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6"
+
+[[package]]
+name = "winapi-util"
+version = "0.1.9"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb"
+dependencies = [
+ "windows-sys",
+]
+
+[[package]]
+name = "winapi-x86_64-pc-windows-gnu"
+version = "0.4.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f"
+
+[[package]]
+name = "windows-sys"
+version = "0.59.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1e38bc4d79ed67fd075bcc251a1c39b32a1776bbe92e5bef1f0bf1f8c531853b"
+dependencies = [
+ "windows-targets",
+]
+
+[[package]]
+name = "windows-targets"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "9b724f72796e036ab90c1021d4780d4d3d648aca59e491e6b98e725b84e99973"
+dependencies = [
+ "windows_aarch64_gnullvm",
+ "windows_aarch64_msvc",
+ "windows_i686_gnu",
+ "windows_i686_gnullvm",
+ "windows_i686_msvc",
+ "windows_x86_64_gnu",
+ "windows_x86_64_gnullvm",
+ "windows_x86_64_msvc",
+]
+
+[[package]]
+name = "windows_aarch64_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "32a4622180e7a0ec044bb555404c800bc9fd9ec262ec147edd5989ccd0c02cd3"
+
+[[package]]
+name = "windows_aarch64_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "09ec2a7bb152e2252b53fa7803150007879548bc709c039df7627cabbd05d469"
+
+[[package]]
+name = "windows_i686_gnu"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "8e9b5ad5ab802e97eb8e295ac6720e509ee4c243f69d781394014ebfe8bbfa0b"
+
+[[package]]
+name = "windows_i686_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "0eee52d38c090b3caa76c563b86c3a4bd71ef1a819287c19d586d7334ae8ed66"
+
+[[package]]
+name = "windows_i686_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "240948bc05c5e7c6dabba28bf89d89ffce3e303022809e73deaefe4f6ec56c66"
+
+[[package]]
+name = "windows_x86_64_gnu"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "147a5c80aabfbf0c7d901cb5895d1de30ef2907eb21fbbab29ca94c5b08b1a78"
+
+[[package]]
+name = "windows_x86_64_gnullvm"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "24d5b23dc417412679681396f2b49f3de8c1473deb516bd34410872eff51ed0d"
+
+[[package]]
+name = "windows_x86_64_msvc"
+version = "0.52.6"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "589f6da84c646204747d1270a2a5661ea66ed1cced2631d546fdfb155959f9ec"
+
+[[package]]
+name = "winnow"
+version = "0.6.21"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "e6f5bb5257f2407a5425c6e749bfd9692192a73e70a6060516ac04f889087d68"
+dependencies = [
+ "memchr",
+]
+
+[[package]]
+name = "wormhole-anchor-sdk"
+version = "0.30.1-alpha.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "99f519ac46a3e5ae1883eb4bb1036421e12e2d21ea917d24aac8df59e82334ee"
+dependencies = [
+ "anchor-lang",
+ "anchor-spl",
+ "cfg-if",
+ "wormhole-io",
+]
+
+[[package]]
+name = "wormhole-integrator-example"
+version = "0.1.0"
+dependencies = [
+ "anchor-lang",
+ "cfg-if",
+ "wormhole-raw-vaas",
+ "wormhole-solana-consts",
+]
+
+[[package]]
+name = "wormhole-io"
+version = "0.1.3"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "b021a14ea7bcef9517ed9f81d4466c4a663dd90e726c5724707a976fa83ad8f3"
+
+[[package]]
+name = "wormhole-post-message-shim"
+version = "0.0.0"
+dependencies = [
+ "anchor-lang",
+]
+
+[[package]]
+name = "wormhole-raw-vaas"
+version = "0.3.0-alpha.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "045a3cc929189ffc0df110e4dc04421ed3226543f47a302088e6175b97b7d75f"
+
+[[package]]
+name = "wormhole-solana-consts"
+version = "0.3.0-alpha.1"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "95ae70b3233a1d2e799276528ad7ec0a55c9929f8eae41683a5d6f0e8ba37649"
+dependencies = [
+ "cfg-if",
+ "solana-program",
+]
+
+[[package]]
+name = "wormhole-svm-definitions"
+version = "0.1.0"
+dependencies = [
+ "cfg-if",
+ "sha2-const-stable",
+ "solana-program",
+]
+
+[[package]]
+name = "wormhole-vaa-verification-comparison"
+version = "0.0.0"
+dependencies = [
+ "anchor-lang",
+ "wormhole-anchor-sdk",
+ "wormhole-raw-vaas",
+ "wormhole-solana-consts",
+]
+
+[[package]]
+name = "wormhole-verify-vaa-shim"
+version = "0.0.0"
+dependencies = [
+ "anchor-lang",
+ "wormhole-svm-definitions",
+]
+
+[[package]]
+name = "zerocopy"
+version = "0.7.35"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "1b9b4fd18abc82b8136838da5d50bae7bdea537c574d8dc1a34ed098d6c166f0"
+dependencies = [
+ "byteorder",
+ "zerocopy-derive",
+]
+
+[[package]]
+name = "zerocopy-derive"
+version = "0.7.35"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]
+
+[[package]]
+name = "zeroize"
+version = "1.3.0"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "4756f7db3f7b5574938c3eb1c117038b8e07f95ee6718c0efad4ac21508f1efd"
+dependencies = [
+ "zeroize_derive",
+]
+
+[[package]]
+name = "zeroize_derive"
+version = "1.4.2"
+source = "registry+https://github.com/rust-lang/crates.io-index"
+checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69"
+dependencies = [
+ "proc-macro2",
+ "quote",
+ "syn 2.0.94",
+]

+ 24 - 0
svm/wormhole-core-shims/anchor/Cargo.toml

@@ -0,0 +1,24 @@
+[workspace]
+members = [
+    "interfaces/*",
+    "programs/*"
+]
+resolver = "2"
+
+[workspace.package]
+edition = "2021"
+publish = false
+rust-version = "1.75"
+version = "0.0.0"
+
+[workspace.dependencies]
+anchor-lang = "0.30.1"
+cfg-if = "1.0"
+wormhole-raw-vaas = "0.3.0-alpha.1"
+wormhole-solana-consts = "0.3.0-alpha.1"
+wormhole-svm-definitions = { path = "../crates/definitions" }
+wormhole-svm-shim = { path = "../crates/shim" }
+
+# `anchor build` requires overflow-checks to be enabled.
+[profile.release]
+overflow-checks = true

+ 50 - 0
svm/wormhole-core-shims/anchor/Makefile

@@ -0,0 +1,50 @@
+SOLANA_VERSION := 1.18.26
+
+INTERFACES := wormhole_post_message_shim wormhole_verify_vaa_shim
+EXAMPLES := wormhole_integrator_example wormhole_vaa_verification_comparison
+
+.PHONY: clean
+clean:
+	cargo clean
+
+.PHONY: test
+test: build-examples
+# Force rebuild of the programs.
+	cd .. && SVM=solana $(MAKE) build
+	cp -R ../target/deploy/*.so target/deploy
+# Copy IDLs and Typescript types to target directory.
+	@for interface in $(INTERFACES); do \
+		cp idls/$$interface.json target/idl; \
+		cp idls/$$interface.ts target/types; \
+	done
+# Finally run Typescript tests.
+	$(MAKE) solana-install
+	anchor test --skip-build
+
+.PHONY: build-examples
+build-examples: solana-install build-interfaces
+	@for example in $(EXAMPLES); do \
+		anchor build -p $$example; \
+	done
+
+.PHONY: build-interfaces
+build-interfaces: solana-install 
+	@for interface in $(INTERFACES); do \
+		anchor idl build -p $$interface -o idls/$$interface.json; \
+		anchor idl type idls/$$interface.json > idls/$$interface.ts; \
+	done
+
+.PHONY: clean-test
+clean-test: clean test
+
+.PHONY: check-idl
+check-idl: build-interfaces
+	git diff --exit-code idls/
+
+.PHONY: solana-install
+solana-install:
+	@if ! solana --version | grep -q "$(SOLANA_VERSION)"; then \
+		agave-install init $(SOLANA_VERSION); \
+	else \
+		echo "Solana $(SOLANA_VERSION) is already installed"; \
+	fi

+ 23 - 0
svm/wormhole-core-shims/anchor/README.md

@@ -0,0 +1,23 @@
+# Shim Interfaces and Examples
+
+For those integrators that use Anchor and IDLs, this directory defines Anchor
+interfaces for the Wormhole Shim programs and example programs that demonstrate
+how to use these interfaces.
+
+## Tests
+
+To perform the Anchor Typescript tests, run:
+
+```sh
+make test
+```
+
+Shim programs are built from the parent directory and moved to this directory
+for the Anchor test. Interfaces are built to generate IDLs so the tests can
+build instructions to interact with these programs in a local validator.
+
+The e2e tests can then be run with `npx tsx tests/e2e.ts` against a running Tilt
+environment with at least `tilt up -- --solana --manual`.
+
+These are _not_ currently run within Tilt. It would be prudent to both build and
+run these within Tilt once the shim approach has been finalized.

+ 314 - 0
svm/wormhole-core-shims/anchor/idls/wormhole_post_message_shim.json

@@ -0,0 +1,314 @@
+{
+  "address": "EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX",
+  "metadata": {
+    "name": "wormhole_post_message_shim",
+    "version": "0.0.0",
+    "spec": "0.1.0",
+    "description": "Anchor Interface for Wormhole Post Message Shim"
+  },
+  "instructions": [
+    {
+      "name": "post_message",
+      "docs": [
+        "This instruction is intended to be a significantly cheaper alternative",
+        "to the post message instruction on Wormhole Core Bridge program. It",
+        "achieves this by reusing the message account (per emitter) via the post",
+        "message unreliable instruction and emitting data via self-CPI (Anchor",
+        "event) for the guardian to observe. This instruction data contains",
+        "information previously found only in the resulting message account.",
+        "",
+        "Because this instruction passes through the emitter and calls the post",
+        "message unreliable instruction on the Wormhole Core Bridge, it can be",
+        "used without disruption.",
+        "",
+        "NOTE: In the initial message publication for a new emitter, this will",
+        "require one additional CPI call depth when compared to using the",
+        "Wormhole Core Bridge directly. If this initial call depth is an issue,",
+        "emit an empty message on initialization (or migration) in order to",
+        "instantiate the message account. Posting a message will result in a VAA",
+        "from your emitter, so be careful to avoid any issues that may result",
+        "from this first message.",
+        "",
+        "Call depth of direct case:",
+        "1. post message (Wormhole Post Message Shim)",
+        "2. multiple CPI",
+        "- post message unreliable (Wormhole Core Bridge)",
+        "- Anchor event of `MesssageEvent` (Wormhole Post Message Shim)",
+        "",
+        "Call depth of integrator case:",
+        "1. integrator instruction",
+        "2. CPI post message (Wormhole Post Message Shim)",
+        "3. multiple CPI",
+        "- post message unreliable (Wormhole Core Bridge)",
+        "- Anchor event of `MesssageEvent` (Wormhole Post Message Shim)"
+      ],
+      "discriminator": [
+        214,
+        50,
+        100,
+        209,
+        38,
+        34,
+        7,
+        76
+      ],
+      "accounts": [
+        {
+          "name": "bridge",
+          "docs": [
+            "Wormhole Core Bridge config. The Wormhole Core Bridge program's post",
+            "message instruction requires this account to be mutable."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  66,
+                  114,
+                  105,
+                  100,
+                  103,
+                  101
+                ]
+              }
+            ],
+            "program": {
+              "kind": "account",
+              "path": "wormhole_program"
+            }
+          }
+        },
+        {
+          "name": "message",
+          "docs": [
+            "Wormhole Message. The Wormhole Core Bridge program's post message",
+            "instruction requires this account to be a mutable signer.",
+            "",
+            "This program uses a PDA per emitter. Messages are already bottle-necked",
+            "by emitter sequence and the Wormhole Core Bridge program enforces that",
+            "emitter must be identical for reused accounts. While this could be",
+            "managed by the integrator, it seems more effective to have this Shim",
+            "program manage these accounts."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "account",
+                "path": "emitter"
+              }
+            ]
+          }
+        },
+        {
+          "name": "emitter",
+          "docs": [
+            "Emitter of the Wormhole Core Bridge message. Wormhole Core Bridge",
+            "program's post message instruction requires this account to be a signer."
+          ],
+          "signer": true
+        },
+        {
+          "name": "sequence",
+          "docs": [
+            "Emitter's sequence account. Wormhole Core Bridge program's post message",
+            "instruction requires this account to be mutable."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  83,
+                  101,
+                  113,
+                  117,
+                  101,
+                  110,
+                  99,
+                  101
+                ]
+              },
+              {
+                "kind": "account",
+                "path": "emitter"
+              }
+            ],
+            "program": {
+              "kind": "account",
+              "path": "wormhole_program"
+            }
+          }
+        },
+        {
+          "name": "payer",
+          "docs": [
+            "Payer will pay the rent for the Wormhole Core Bridge emitter sequence",
+            "and message on the first post message call. Subsequent calls will not",
+            "require more lamports for rent."
+          ],
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "fee_collector",
+          "docs": [
+            "Wormhole Core Bridge fee collector. Wormhole Core Bridge program's post",
+            "message instruction requires this account to be mutable."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  102,
+                  101,
+                  101,
+                  95,
+                  99,
+                  111,
+                  108,
+                  108,
+                  101,
+                  99,
+                  116,
+                  111,
+                  114
+                ]
+              }
+            ],
+            "program": {
+              "kind": "account",
+              "path": "wormhole_program"
+            }
+          }
+        },
+        {
+          "name": "clock",
+          "docs": [
+            "Clock sysvar."
+          ],
+          "address": "SysvarC1ock11111111111111111111111111111111"
+        },
+        {
+          "name": "system_program",
+          "docs": [
+            "System program."
+          ],
+          "address": "11111111111111111111111111111111"
+        },
+        {
+          "name": "wormhole_program",
+          "docs": [
+            "Wormhole Core Bridge program."
+          ]
+        },
+        {
+          "name": "event_authority",
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  95,
+                  95,
+                  101,
+                  118,
+                  101,
+                  110,
+                  116,
+                  95,
+                  97,
+                  117,
+                  116,
+                  104,
+                  111,
+                  114,
+                  105,
+                  116,
+                  121
+                ]
+              }
+            ]
+          }
+        },
+        {
+          "name": "program"
+        }
+      ],
+      "args": [
+        {
+          "name": "nonce",
+          "type": "u32"
+        },
+        {
+          "name": "consistency_level",
+          "type": {
+            "defined": {
+              "name": "Finality"
+            }
+          }
+        },
+        {
+          "name": "payload",
+          "type": "bytes"
+        }
+      ]
+    }
+  ],
+  "events": [
+    {
+      "name": "MessageEvent",
+      "discriminator": [
+        68,
+        27,
+        143,
+        0,
+        77,
+        76,
+        137,
+        112
+      ]
+    }
+  ],
+  "types": [
+    {
+      "name": "Finality",
+      "type": {
+        "kind": "enum",
+        "variants": [
+          {
+            "name": "Confirmed"
+          },
+          {
+            "name": "Finalized"
+          }
+        ]
+      }
+    },
+    {
+      "name": "MessageEvent",
+      "type": {
+        "kind": "struct",
+        "fields": [
+          {
+            "name": "emitter",
+            "type": "pubkey"
+          },
+          {
+            "name": "sequence",
+            "type": "u64"
+          },
+          {
+            "name": "submission_time",
+            "type": "u32"
+          }
+        ]
+      }
+    }
+  ]
+}

+ 321 - 0
svm/wormhole-core-shims/anchor/idls/wormhole_post_message_shim.ts

@@ -0,0 +1,321 @@
+/**
+ * Program IDL in camelCase format in order to be used in JS/TS.
+ *
+ * Note that this is only a type helper and is not the actual IDL. The original
+ * IDL can be found at `target/idl/wormhole_post_message_shim.json`.
+ */
+export type WormholePostMessageShim = {
+  "address": "EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX",
+  "metadata": {
+    "name": "wormholePostMessageShim",
+    "version": "0.0.0",
+    "spec": "0.1.0",
+    "description": "Anchor Interface for Wormhole Post Message Shim"
+  },
+  "instructions": [
+    {
+      "name": "postMessage",
+      "docs": [
+        "This instruction is intended to be a significantly cheaper alternative",
+        "to the post message instruction on Wormhole Core Bridge program. It",
+        "achieves this by reusing the message account (per emitter) via the post",
+        "message unreliable instruction and emitting data via self-CPI (Anchor",
+        "event) for the guardian to observe. This instruction data contains",
+        "information previously found only in the resulting message account.",
+        "",
+        "Because this instruction passes through the emitter and calls the post",
+        "message unreliable instruction on the Wormhole Core Bridge, it can be",
+        "used without disruption.",
+        "",
+        "NOTE: In the initial message publication for a new emitter, this will",
+        "require one additional CPI call depth when compared to using the",
+        "Wormhole Core Bridge directly. If this initial call depth is an issue,",
+        "emit an empty message on initialization (or migration) in order to",
+        "instantiate the message account. Posting a message will result in a VAA",
+        "from your emitter, so be careful to avoid any issues that may result",
+        "from this first message.",
+        "",
+        "Call depth of direct case:",
+        "1. post message (Wormhole Post Message Shim)",
+        "2. multiple CPI",
+        "- post message unreliable (Wormhole Core Bridge)",
+        "- Anchor event of `MesssageEvent` (Wormhole Post Message Shim)",
+        "",
+        "Call depth of integrator case:",
+        "1. integrator instruction",
+        "2. CPI post message (Wormhole Post Message Shim)",
+        "3. multiple CPI",
+        "- post message unreliable (Wormhole Core Bridge)",
+        "- Anchor event of `MesssageEvent` (Wormhole Post Message Shim)"
+      ],
+      "discriminator": [
+        214,
+        50,
+        100,
+        209,
+        38,
+        34,
+        7,
+        76
+      ],
+      "accounts": [
+        {
+          "name": "bridge",
+          "docs": [
+            "Wormhole Core Bridge config. The Wormhole Core Bridge program's post",
+            "message instruction requires this account to be mutable."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  66,
+                  114,
+                  105,
+                  100,
+                  103,
+                  101
+                ]
+              }
+            ],
+            "program": {
+              "kind": "account",
+              "path": "wormholeProgram"
+            }
+          }
+        },
+        {
+          "name": "message",
+          "docs": [
+            "Wormhole Message. The Wormhole Core Bridge program's post message",
+            "instruction requires this account to be a mutable signer.",
+            "",
+            "This program uses a PDA per emitter. Messages are already bottle-necked",
+            "by emitter sequence and the Wormhole Core Bridge program enforces that",
+            "emitter must be identical for reused accounts. While this could be",
+            "managed by the integrator, it seems more effective to have this Shim",
+            "program manage these accounts."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "account",
+                "path": "emitter"
+              }
+            ]
+          }
+        },
+        {
+          "name": "emitter",
+          "docs": [
+            "Emitter of the Wormhole Core Bridge message. Wormhole Core Bridge",
+            "program's post message instruction requires this account to be a signer."
+          ],
+          "signer": true
+        },
+        {
+          "name": "sequence",
+          "docs": [
+            "Emitter's sequence account. Wormhole Core Bridge program's post message",
+            "instruction requires this account to be mutable."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  83,
+                  101,
+                  113,
+                  117,
+                  101,
+                  110,
+                  99,
+                  101
+                ]
+              },
+              {
+                "kind": "account",
+                "path": "emitter"
+              }
+            ],
+            "program": {
+              "kind": "account",
+              "path": "wormholeProgram"
+            }
+          }
+        },
+        {
+          "name": "payer",
+          "docs": [
+            "Payer will pay the rent for the Wormhole Core Bridge emitter sequence",
+            "and message on the first post message call. Subsequent calls will not",
+            "require more lamports for rent."
+          ],
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "feeCollector",
+          "docs": [
+            "Wormhole Core Bridge fee collector. Wormhole Core Bridge program's post",
+            "message instruction requires this account to be mutable."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  102,
+                  101,
+                  101,
+                  95,
+                  99,
+                  111,
+                  108,
+                  108,
+                  101,
+                  99,
+                  116,
+                  111,
+                  114
+                ]
+              }
+            ],
+            "program": {
+              "kind": "account",
+              "path": "wormholeProgram"
+            }
+          }
+        },
+        {
+          "name": "clock",
+          "docs": [
+            "Clock sysvar."
+          ],
+          "address": "SysvarC1ock11111111111111111111111111111111"
+        },
+        {
+          "name": "systemProgram",
+          "docs": [
+            "System program."
+          ],
+          "address": "11111111111111111111111111111111"
+        },
+        {
+          "name": "wormholeProgram",
+          "docs": [
+            "Wormhole Core Bridge program."
+          ]
+        },
+        {
+          "name": "eventAuthority",
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  95,
+                  95,
+                  101,
+                  118,
+                  101,
+                  110,
+                  116,
+                  95,
+                  97,
+                  117,
+                  116,
+                  104,
+                  111,
+                  114,
+                  105,
+                  116,
+                  121
+                ]
+              }
+            ]
+          }
+        },
+        {
+          "name": "program"
+        }
+      ],
+      "args": [
+        {
+          "name": "nonce",
+          "type": "u32"
+        },
+        {
+          "name": "consistencyLevel",
+          "type": {
+            "defined": {
+              "name": "finality"
+            }
+          }
+        },
+        {
+          "name": "payload",
+          "type": "bytes"
+        }
+      ]
+    }
+  ],
+  "events": [
+    {
+      "name": "messageEvent",
+      "discriminator": [
+        68,
+        27,
+        143,
+        0,
+        77,
+        76,
+        137,
+        112
+      ]
+    }
+  ],
+  "types": [
+    {
+      "name": "finality",
+      "type": {
+        "kind": "enum",
+        "variants": [
+          {
+            "name": "confirmed"
+          },
+          {
+            "name": "finalized"
+          }
+        ]
+      }
+    },
+    {
+      "name": "messageEvent",
+      "type": {
+        "kind": "struct",
+        "fields": [
+          {
+            "name": "emitter",
+            "type": "pubkey"
+          },
+          {
+            "name": "sequence",
+            "type": "u64"
+          },
+          {
+            "name": "submissionTime",
+            "type": "u32"
+          }
+        ]
+      }
+    }
+  ]
+};
+

+ 257 - 0
svm/wormhole-core-shims/anchor/idls/wormhole_verify_vaa_shim.json

@@ -0,0 +1,257 @@
+{
+  "address": "EFaNWErqAtVWufdNb7yofSHHfWFos843DFpu4JBw24at",
+  "metadata": {
+    "name": "wormhole_verify_vaa_shim",
+    "version": "0.0.0",
+    "spec": "0.1.0",
+    "description": "Anchor Interface for Wormhole Verify VAA Shim"
+  },
+  "instructions": [
+    {
+      "name": "close_signatures",
+      "docs": [
+        "Allows the initial payer to close the signature account, reclaiming the",
+        "rent taken by the post signatures instruction."
+      ],
+      "discriminator": [
+        192,
+        65,
+        63,
+        117,
+        213,
+        138,
+        179,
+        190
+      ],
+      "accounts": [
+        {
+          "name": "guardian_signatures",
+          "writable": true
+        },
+        {
+          "name": "refund_recipient",
+          "writable": true,
+          "signer": true,
+          "relations": [
+            "guardian_signatures"
+          ]
+        }
+      ],
+      "args": []
+    },
+    {
+      "name": "post_signatures",
+      "docs": [
+        "Creates or appends to a guardian signatures account for subsequent use",
+        "by the verify hash instruction.",
+        "",
+        "This instruction is necessary due to the Wormhole VAA body, which has an",
+        "arbitrary size, and 13 guardian signatures (a quorum of the current 19",
+        "mainnet guardians, 66 bytes each) alongside the required accounts is",
+        "likely larger than the transaction size limit on Solana (1232 bytes).",
+        "",
+        "This instruction will also allow for the verification of other messages",
+        "which guardians sign, such as query results.",
+        "",
+        "This instruction allows for the initial payer to append additional",
+        "signatures to the account by calling the instruction again. Subsequent",
+        "calls may be necessary if a quorum of signatures from the current guardian",
+        "set grows larger than can fit into a single transaction.",
+        "",
+        "The guardian signatures account can be closed by the initial payer via",
+        "the close signatures instruction, which will refund this payer."
+      ],
+      "discriminator": [
+        138,
+        2,
+        53,
+        166,
+        45,
+        77,
+        137,
+        51
+      ],
+      "accounts": [
+        {
+          "name": "payer",
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "guardian_signatures",
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "system_program",
+          "address": "11111111111111111111111111111111"
+        }
+      ],
+      "args": [
+        {
+          "name": "guardian_set_index",
+          "type": "u32"
+        },
+        {
+          "name": "total_signatures",
+          "type": "u8"
+        },
+        {
+          "name": "guardian_signatures",
+          "type": {
+            "vec": {
+              "array": [
+                "u8",
+                66
+              ]
+            }
+          }
+        }
+      ]
+    },
+    {
+      "name": "verify_hash",
+      "docs": [
+        "This instruction is intended to be invoked via CPI call. It verifies a",
+        "digest against a guardian signatures account and a Wormhole Core Bridge",
+        "guardian set account.",
+        "",
+        "Prior to this call (and likely in a separate transaction), call the post",
+        "signatures instruction to create the guardian signatures account.",
+        "",
+        "Immediately after this verify call, call the close signatures",
+        "instruction to reclaim the rent paid to create the guardian signatures",
+        "account.",
+        "",
+        "A v1 VAA digest can be computed as follows:",
+        "```rust",
+        "use wormhole_svm_definitions::compute_keccak_digest;",
+        "",
+        "// `vec_body` is the encoded body of the VAA.",
+        "# let vaa_body = vec![];",
+        "let digest = compute_keccak_digest(",
+        "solana_program::keccak::hash(&vaa_body),",
+        "None, // there is no prefix for V1 messages",
+        ");",
+        "```",
+        "",
+        "A QueryResponse digest can be computed as follows:",
+        "```rust",
+        "# mod wormhole_query_sdk {",
+        "#    pub const MESSAGE_PREFIX: &'static [u8] = b\"ruh roh\";",
+        "# }",
+        "use wormhole_query_sdk::MESSAGE_PREFIX;",
+        "use wormhole_svm_definitions::compute_keccak_digest;",
+        "",
+        "# let query_response_bytes = vec![];",
+        "let digest = compute_keccak_digest(",
+        "solana_program::keccak::hash(&query_response_bytes),",
+        "Some(MESSAGE_PREFIX)",
+        ");",
+        "```"
+      ],
+      "discriminator": [
+        22,
+        152,
+        160,
+        69,
+        241,
+        148,
+        14,
+        124
+      ],
+      "accounts": [
+        {
+          "name": "guardian_set",
+          "docs": [
+            "Guardian set used for signature verification."
+          ]
+        },
+        {
+          "name": "guardian_signatures",
+          "docs": [
+            "Stores unverified guardian signatures as they are too large to fit in",
+            "the instruction data."
+          ]
+        }
+      ],
+      "args": [
+        {
+          "name": "guardian_set_bump",
+          "type": "u8"
+        },
+        {
+          "name": "digest",
+          "type": {
+            "array": [
+              "u8",
+              32
+            ]
+          }
+        }
+      ]
+    }
+  ],
+  "accounts": [
+    {
+      "name": "GuardianSignatures",
+      "discriminator": [
+        203,
+        184,
+        130,
+        157,
+        113,
+        14,
+        184,
+        83
+      ]
+    }
+  ],
+  "types": [
+    {
+      "name": "GuardianSignatures",
+      "type": {
+        "kind": "struct",
+        "fields": [
+          {
+            "name": "refund_recipient",
+            "docs": [
+              "Payer of this guardian signatures account.",
+              "Only they may amend signatures.",
+              "Used for reimbursements upon cleanup."
+            ],
+            "type": "pubkey"
+          },
+          {
+            "name": "guardian_set_index_be",
+            "docs": [
+              "Guardian set index that these signatures correspond to.",
+              "Storing this simplifies the integrator data.",
+              "Using big-endian to match the derivation used by the core bridge."
+            ],
+            "type": {
+              "array": [
+                "u8",
+                4
+              ]
+            }
+          },
+          {
+            "name": "guardian_signatures",
+            "docs": [
+              "Unverified guardian signatures."
+            ],
+            "type": {
+              "vec": {
+                "array": [
+                  "u8",
+                  66
+                ]
+              }
+            }
+          }
+        ]
+      }
+    }
+  ]
+}

+ 264 - 0
svm/wormhole-core-shims/anchor/idls/wormhole_verify_vaa_shim.ts

@@ -0,0 +1,264 @@
+/**
+ * Program IDL in camelCase format in order to be used in JS/TS.
+ *
+ * Note that this is only a type helper and is not the actual IDL. The original
+ * IDL can be found at `target/idl/wormhole_verify_vaa_shim.json`.
+ */
+export type WormholeVerifyVaaShim = {
+  "address": "EFaNWErqAtVWufdNb7yofSHHfWFos843DFpu4JBw24at",
+  "metadata": {
+    "name": "wormholeVerifyVaaShim",
+    "version": "0.0.0",
+    "spec": "0.1.0",
+    "description": "Anchor Interface for Wormhole Verify VAA Shim"
+  },
+  "instructions": [
+    {
+      "name": "closeSignatures",
+      "docs": [
+        "Allows the initial payer to close the signature account, reclaiming the",
+        "rent taken by the post signatures instruction."
+      ],
+      "discriminator": [
+        192,
+        65,
+        63,
+        117,
+        213,
+        138,
+        179,
+        190
+      ],
+      "accounts": [
+        {
+          "name": "guardianSignatures",
+          "writable": true
+        },
+        {
+          "name": "refundRecipient",
+          "writable": true,
+          "signer": true,
+          "relations": [
+            "guardianSignatures"
+          ]
+        }
+      ],
+      "args": []
+    },
+    {
+      "name": "postSignatures",
+      "docs": [
+        "Creates or appends to a guardian signatures account for subsequent use",
+        "by the verify hash instruction.",
+        "",
+        "This instruction is necessary due to the Wormhole VAA body, which has an",
+        "arbitrary size, and 13 guardian signatures (a quorum of the current 19",
+        "mainnet guardians, 66 bytes each) alongside the required accounts is",
+        "likely larger than the transaction size limit on Solana (1232 bytes).",
+        "",
+        "This instruction will also allow for the verification of other messages",
+        "which guardians sign, such as query results.",
+        "",
+        "This instruction allows for the initial payer to append additional",
+        "signatures to the account by calling the instruction again. Subsequent",
+        "calls may be necessary if a quorum of signatures from the current guardian",
+        "set grows larger than can fit into a single transaction.",
+        "",
+        "The guardian signatures account can be closed by the initial payer via",
+        "the close signatures instruction, which will refund this payer."
+      ],
+      "discriminator": [
+        138,
+        2,
+        53,
+        166,
+        45,
+        77,
+        137,
+        51
+      ],
+      "accounts": [
+        {
+          "name": "payer",
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "guardianSignatures",
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "systemProgram",
+          "address": "11111111111111111111111111111111"
+        }
+      ],
+      "args": [
+        {
+          "name": "guardianSetIndex",
+          "type": "u32"
+        },
+        {
+          "name": "totalSignatures",
+          "type": "u8"
+        },
+        {
+          "name": "guardianSignatures",
+          "type": {
+            "vec": {
+              "array": [
+                "u8",
+                66
+              ]
+            }
+          }
+        }
+      ]
+    },
+    {
+      "name": "verifyHash",
+      "docs": [
+        "This instruction is intended to be invoked via CPI call. It verifies a",
+        "digest against a guardian signatures account and a Wormhole Core Bridge",
+        "guardian set account.",
+        "",
+        "Prior to this call (and likely in a separate transaction), call the post",
+        "signatures instruction to create the guardian signatures account.",
+        "",
+        "Immediately after this verify call, call the close signatures",
+        "instruction to reclaim the rent paid to create the guardian signatures",
+        "account.",
+        "",
+        "A v1 VAA digest can be computed as follows:",
+        "```rust",
+        "use wormhole_svm_definitions::compute_keccak_digest;",
+        "",
+        "// `vec_body` is the encoded body of the VAA.",
+        "# let vaa_body = vec![];",
+        "let digest = compute_keccak_digest(",
+        "solana_program::keccak::hash(&vaa_body),",
+        "None, // there is no prefix for V1 messages",
+        ");",
+        "```",
+        "",
+        "A QueryResponse digest can be computed as follows:",
+        "```rust",
+        "# mod wormhole_query_sdk {",
+        "#    pub const MESSAGE_PREFIX: &'static [u8] = b\"ruh roh\";",
+        "# }",
+        "use wormhole_query_sdk::MESSAGE_PREFIX;",
+        "use wormhole_svm_definitions::compute_keccak_digest;",
+        "",
+        "# let query_response_bytes = vec![];",
+        "let digest = compute_keccak_digest(",
+        "solana_program::keccak::hash(&query_response_bytes),",
+        "Some(MESSAGE_PREFIX)",
+        ");",
+        "```"
+      ],
+      "discriminator": [
+        22,
+        152,
+        160,
+        69,
+        241,
+        148,
+        14,
+        124
+      ],
+      "accounts": [
+        {
+          "name": "guardianSet",
+          "docs": [
+            "Guardian set used for signature verification."
+          ]
+        },
+        {
+          "name": "guardianSignatures",
+          "docs": [
+            "Stores unverified guardian signatures as they are too large to fit in",
+            "the instruction data."
+          ]
+        }
+      ],
+      "args": [
+        {
+          "name": "guardianSetBump",
+          "type": "u8"
+        },
+        {
+          "name": "digest",
+          "type": {
+            "array": [
+              "u8",
+              32
+            ]
+          }
+        }
+      ]
+    }
+  ],
+  "accounts": [
+    {
+      "name": "guardianSignatures",
+      "discriminator": [
+        203,
+        184,
+        130,
+        157,
+        113,
+        14,
+        184,
+        83
+      ]
+    }
+  ],
+  "types": [
+    {
+      "name": "guardianSignatures",
+      "type": {
+        "kind": "struct",
+        "fields": [
+          {
+            "name": "refundRecipient",
+            "docs": [
+              "Payer of this guardian signatures account.",
+              "Only they may amend signatures.",
+              "Used for reimbursements upon cleanup."
+            ],
+            "type": "pubkey"
+          },
+          {
+            "name": "guardianSetIndexBe",
+            "docs": [
+              "Guardian set index that these signatures correspond to.",
+              "Storing this simplifies the integrator data.",
+              "Using big-endian to match the derivation used by the core bridge."
+            ],
+            "type": {
+              "array": [
+                "u8",
+                4
+              ]
+            }
+          },
+          {
+            "name": "guardianSignatures",
+            "docs": [
+              "Unverified guardian signatures."
+            ],
+            "type": {
+              "vec": {
+                "array": [
+                  "u8",
+                  66
+                ]
+              }
+            }
+          }
+        ]
+      }
+    }
+  ]
+};
+

+ 21 - 0
svm/wormhole-core-shims/anchor/interfaces/wormhole-post-message-shim/Cargo.toml

@@ -0,0 +1,21 @@
+[package]
+name = "wormhole-post-message-shim"
+description = "Anchor Interface for Wormhole Post Message Shim"
+
+edition.workspace = true
+publish.workspace = true
+rust-version.workspace = true
+version.workspace = true
+
+[lib]
+crate-type = ["cdylib", "lib"]
+
+[features]
+cpi = ["no-entrypoint"]
+no-entrypoint = []
+no-idl = []
+no-log-ix-name = []
+idl-build = ["anchor-lang/idl-build"]
+
+[dependencies]
+anchor-lang = { workspace = true, features = ["event-cpi"] }

+ 9 - 0
svm/wormhole-core-shims/anchor/interfaces/wormhole-post-message-shim/README.md

@@ -0,0 +1,9 @@
+# Anchor Interface for Post Message Shim
+
+Anchor interface for the Wormhole Post Message Shim program. This interface is
+used to generate an IDL that composing programs can use and to provide
+convenient Typescript instruction builders.
+
+See the [program document] for more information.
+
+[program document]: ../../../programs/post-message/README.md

+ 133 - 0
svm/wormhole-core-shims/anchor/interfaces/wormhole-post-message-shim/src/lib.rs

@@ -0,0 +1,133 @@
+use anchor_lang::prelude::*;
+
+declare_id!("EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX");
+
+#[program]
+pub mod wormhole_post_message_shim {
+    use super::*;
+
+    /// This instruction is intended to be a significantly cheaper alternative
+    /// to the post message instruction on Wormhole Core Bridge program. It
+    /// achieves this by reusing the message account (per emitter) via the post
+    /// message unreliable instruction and emitting data via self-CPI (Anchor
+    /// event) for the guardian to observe. This instruction data contains
+    /// information previously found only in the resulting message account.
+    ///
+    /// Because this instruction passes through the emitter and calls the post
+    /// message unreliable instruction on the Wormhole Core Bridge, it can be
+    /// used without disruption.
+    ///
+    /// NOTE: In the initial message publication for a new emitter, this will
+    /// require one additional CPI call depth when compared to using the
+    /// Wormhole Core Bridge directly. If this initial call depth is an issue,
+    /// emit an empty message on initialization (or migration) in order to
+    /// instantiate the message account. Posting a message will result in a VAA
+    /// from your emitter, so be careful to avoid any issues that may result
+    /// from this first message.
+    ///
+    /// Call depth of direct case:
+    /// 1. post message (Wormhole Post Message Shim)
+    /// 2. multiple CPI
+    ///     - post message unreliable (Wormhole Core Bridge)
+    ///     - Anchor event of `MesssageEvent` (Wormhole Post Message Shim)
+    ///
+    /// Call depth of integrator case:
+    /// 1. integrator instruction
+    /// 2. CPI post message (Wormhole Post Message Shim)
+    /// 3. multiple CPI
+    ///    - post message unreliable (Wormhole Core Bridge)
+    ///    - Anchor event of `MesssageEvent` (Wormhole Post Message Shim)
+    pub fn post_message(
+        _ctx: Context<PostMessage>,
+        nonce: u32,
+        consistency_level: Finality,
+        payload: Vec<u8>,
+    ) -> Result<()> {
+        let _ = (nonce, consistency_level, payload);
+        err!(ErrorCode::InstructionMissing)
+    }
+}
+
+#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, AnchorDeserialize, AnchorSerialize)]
+pub enum Finality {
+    Confirmed,
+    Finalized,
+}
+
+#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
+#[event]
+pub struct MessageEvent {
+    pub emitter: Pubkey,
+    pub sequence: u64,
+    pub submission_time: u32,
+}
+
+/// The accounts are ordered and named the same as the Wormhole Core Bridge
+/// program's post message unreliable instruction.
+#[derive(Accounts)]
+#[event_cpi]
+pub struct PostMessage<'info> {
+    /// Wormhole Core Bridge config. The Wormhole Core Bridge program's post
+    /// message instruction requires this account to be mutable.
+    #[account(
+        mut,
+        seeds = [b"Bridge"],
+        bump,
+        seeds::program = wormhole_program.key()
+    )]
+    bridge: UncheckedAccount<'info>,
+
+    /// Wormhole Message. The Wormhole Core Bridge program's post message
+    /// instruction requires this account to be a mutable signer.
+    ///
+    /// This program uses a PDA per emitter. Messages are already bottle-necked
+    /// by emitter sequence and the Wormhole Core Bridge program enforces that
+    /// emitter must be identical for reused accounts. While this could be
+    /// managed by the integrator, it seems more effective to have this Shim
+    /// program manage these accounts.
+    #[account(
+        mut,
+        seeds = [&emitter.key.to_bytes()],
+        bump
+    )]
+    message: UncheckedAccount<'info>,
+
+    /// Emitter of the Wormhole Core Bridge message. Wormhole Core Bridge
+    /// program's post message instruction requires this account to be a signer.
+    emitter: Signer<'info>,
+
+    /// Emitter's sequence account. Wormhole Core Bridge program's post message
+    /// instruction requires this account to be mutable.
+    #[account(
+        mut,
+        seeds = [b"Sequence", &emitter.key.to_bytes()],
+        bump,
+        seeds::program = wormhole_program.key()
+    )]
+    sequence: UncheckedAccount<'info>,
+
+    /// Payer will pay the rent for the Wormhole Core Bridge emitter sequence
+    /// and message on the first post message call. Subsequent calls will not
+    /// require more lamports for rent.
+    #[account(mut)]
+    payer: Signer<'info>,
+
+    /// Wormhole Core Bridge fee collector. Wormhole Core Bridge program's post
+    /// message instruction requires this account to be mutable.
+    #[account(
+        mut,
+        seeds = [b"fee_collector"],
+        bump,
+        seeds::program = wormhole_program.key()
+    )]
+    fee_collector: UncheckedAccount<'info>,
+
+    /// Clock sysvar.
+    clock: Sysvar<'info, Clock>,
+
+    /// System program.
+    system_program: Program<'info, System>,
+
+    /// Wormhole Core Bridge program.
+    wormhole_program: UncheckedAccount<'info>,
+}

+ 22 - 0
svm/wormhole-core-shims/anchor/interfaces/wormhole-verify-vaa-shim/Cargo.toml

@@ -0,0 +1,22 @@
+[package]
+name = "wormhole-verify-vaa-shim"
+description = "Anchor Interface for Wormhole Verify VAA Shim"
+
+edition.workspace = true
+publish.workspace = true
+rust-version.workspace = true
+version.workspace = true
+
+[lib]
+crate-type = ["cdylib", "lib"]
+
+[features]
+cpi = ["no-entrypoint"]
+no-entrypoint = []
+no-idl = []
+no-log-ix-name = []
+idl-build = ["anchor-lang/idl-build"]
+
+[dependencies]
+anchor-lang.workspace = true
+wormhole-svm-definitions.workspace = true

+ 9 - 0
svm/wormhole-core-shims/anchor/interfaces/wormhole-verify-vaa-shim/README.md

@@ -0,0 +1,9 @@
+# Anchor Interface for Verify VAA Shim
+
+Anchor interface for the Wormhole Verify VAA Shim program. This interface is
+used to generate an IDL that composing programs can use and to provide
+convenient Typescript instruction builders.
+
+See the [program document] for more information.
+
+[program document]: ../../../programs/verify-vaa/README.md

+ 136 - 0
svm/wormhole-core-shims/anchor/interfaces/wormhole-verify-vaa-shim/src/lib.rs

@@ -0,0 +1,136 @@
+use anchor_lang::prelude::*;
+use wormhole_svm_definitions::HASH_BYTES;
+
+declare_id!("EFaNWErqAtVWufdNb7yofSHHfWFos843DFpu4JBw24at");
+
+#[program]
+pub mod wormhole_verify_vaa_shim {
+    use super::*;
+
+    /// Allows the initial payer to close the signature account, reclaiming the
+    /// rent taken by the post signatures instruction.
+    pub fn close_signatures(_ctx: Context<CloseSignatures>) -> Result<()> {
+        err!(ErrorCode::InstructionMissing)
+    }
+
+    /// Creates or appends to a guardian signatures account for subsequent use
+    /// by the verify hash instruction.
+    ///
+    /// This instruction is necessary due to the Wormhole VAA body, which has an
+    /// arbitrary size, and 13 guardian signatures (a quorum of the current 19
+    /// mainnet guardians, 66 bytes each) alongside the required accounts is
+    /// likely larger than the transaction size limit on Solana (1232 bytes).
+    ///
+    /// This instruction will also allow for the verification of other messages
+    /// which guardians sign, such as query results.
+    ///
+    /// This instruction allows for the initial payer to append additional
+    /// signatures to the account by calling the instruction again. Subsequent
+    /// calls may be necessary if a quorum of signatures from the current guardian
+    /// set grows larger than can fit into a single transaction.
+    ///
+    /// The guardian signatures account can be closed by the initial payer via
+    /// the close signatures instruction, which will refund this payer.
+    pub fn post_signatures(
+        _ctx: Context<PostSignatures>,
+        guardian_set_index: u32,
+        total_signatures: u8,
+        guardian_signatures: Vec<[u8; 66]>,
+    ) -> Result<()> {
+        let _ = (guardian_set_index, total_signatures, guardian_signatures);
+        err!(ErrorCode::InstructionMissing)
+    }
+
+    /// This instruction is intended to be invoked via CPI call. It verifies a
+    /// digest against a guardian signatures account and a Wormhole Core Bridge
+    /// guardian set account.
+    ///
+    /// Prior to this call (and likely in a separate transaction), call the post
+    /// signatures instruction to create the guardian signatures account.
+    ///
+    /// Immediately after this verify call, call the close signatures
+    /// instruction to reclaim the rent paid to create the guardian signatures
+    /// account.
+    ///
+    /// A v1 VAA digest can be computed as follows:
+    /// ```rust
+    /// use wormhole_svm_definitions::compute_keccak_digest;
+    ///
+    /// // `vec_body` is the encoded body of the VAA.
+    /// # let vaa_body = vec![];
+    /// let digest = compute_keccak_digest(
+    ///     solana_program::keccak::hash(&vaa_body),
+    ///     None, // there is no prefix for V1 messages
+    /// );
+    /// ```
+    ///
+    /// A QueryResponse digest can be computed as follows:
+    /// ```rust
+    /// # mod wormhole_query_sdk {
+    /// #    pub const MESSAGE_PREFIX: &'static [u8] = b"ruh roh";
+    /// # }
+    /// use wormhole_query_sdk::MESSAGE_PREFIX;
+    /// use wormhole_svm_definitions::compute_keccak_digest;
+    ///
+    /// # let query_response_bytes = vec![];
+    /// let digest = compute_keccak_digest(
+    ///     solana_program::keccak::hash(&query_response_bytes),
+    ///     Some(MESSAGE_PREFIX)
+    /// );
+    /// ```
+    pub fn verify_hash(
+        _ctx: Context<VerifyHash>,
+        guardian_set_bump: u8,
+        digest: [u8; HASH_BYTES],
+    ) -> Result<()> {
+        let _ = (guardian_set_bump, digest);
+        err!(ErrorCode::InstructionMissing)
+    }
+}
+
+#[derive(Accounts)]
+pub struct CloseSignatures<'info> {
+    #[account(mut, has_one = refund_recipient)]
+    guardian_signatures: Account<'info, GuardianSignatures>,
+
+    #[account(mut, address = guardian_signatures.refund_recipient)]
+    refund_recipient: Signer<'info>,
+}
+
+#[derive(Accounts)]
+pub struct PostSignatures<'info> {
+    #[account(mut)]
+    payer: Signer<'info>,
+
+    #[account(mut, signer)]
+    guardian_signatures: Account<'info, GuardianSignatures>,
+
+    system_program: Program<'info, System>,
+}
+
+#[derive(Accounts)]
+pub struct VerifyHash<'info> {
+    /// Guardian set used for signature verification.
+    guardian_set: UncheckedAccount<'info>,
+
+    /// Stores unverified guardian signatures as they are too large to fit in
+    /// the instruction data.
+    guardian_signatures: Account<'info, GuardianSignatures>,
+}
+
+#[account]
+#[derive(Debug, PartialEq, Eq)]
+pub struct GuardianSignatures {
+    /// Payer of this guardian signatures account.
+    /// Only they may amend signatures.
+    /// Used for reimbursements upon cleanup.
+    pub refund_recipient: Pubkey,
+
+    /// Guardian set index that these signatures correspond to.
+    /// Storing this simplifies the integrator data.
+    /// Using big-endian to match the derivation used by the core bridge.
+    pub guardian_set_index_be: [u8; 4],
+
+    /// Unverified guardian signatures.
+    pub guardian_signatures: Vec<[u8; 66]>,
+}

+ 23 - 0
svm/wormhole-core-shims/anchor/package.json

@@ -0,0 +1,23 @@
+{
+  "license": "ISC",
+  "scripts": {
+    "lint:fix": "prettier */*.js \"*/**/*{.js,.ts}\" -w",
+    "lint": "prettier */*.js \"*/**/*{.js,.ts}\" --check"
+  },
+  "dependencies": {
+    "@certusone/wormhole-sdk": "^0.10.18",
+    "@coral-xyz/anchor": "^0.30.1",
+    "@improbable-eng/grpc-web-node-http-transport": "0.15.0"
+  },
+  "devDependencies": {
+    "@types/bn.js": "^5.1.0",
+    "@types/chai": "^4.3.0",
+    "@types/mocha": "^9.0.0",
+    "chai": "^4.3.4",
+    "mocha": "^9.0.3",
+    "prettier": "^2.6.2",
+    "ts-mocha": "^10.0.0",
+    "tsx": "^4.19.2",
+    "typescript": "^4.3.5"
+  }
+}

+ 26 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/Cargo.toml

@@ -0,0 +1,26 @@
+[package]
+name = "wormhole-integrator-example"
+version = "0.1.0"
+description = "Created with Anchor"
+edition = "2021"
+
+[lib]
+crate-type = ["cdylib", "lib"]
+name = "wormhole_integrator_example"
+
+[features]
+default = ["mainnet"]
+mainnet = ["wormhole-solana-consts/mainnet"]
+testnet = ["wormhole-solana-consts/testnet"]
+localnet = ["wormhole-solana-consts/localnet"]
+cpi = ["no-entrypoint"]
+no-entrypoint = []
+no-idl = []
+no-log-ix-name = []
+idl-build = ["anchor-lang/idl-build"]
+
+[dependencies]
+anchor-lang = "0.30.1"
+cfg-if = "1.0"
+wormhole-raw-vaas = "0.3.0-alpha.1"
+wormhole-solana-consts = "0.3.0-alpha.1"

+ 2 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/Xargo.toml

@@ -0,0 +1,2 @@
+[target.bpfel-unknown-unknown.dependencies.std]
+features = []

+ 45 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/instructions/consume_vaa.rs

@@ -0,0 +1,45 @@
+declare_program!(wormhole_verify_vaa_shim);
+
+use anchor_lang::{
+    prelude::*,
+    solana_program::{self, keccak},
+};
+use wormhole_verify_vaa_shim::cpi::accounts::VerifyHash;
+use wormhole_verify_vaa_shim::program::WormholeVerifyVaaShim;
+
+#[derive(Accounts)]
+pub struct ConsumeVaa<'info> {
+    /// CHECK: Guardian set used for signature verification by shim.
+    /// Derivation is checked by the shim.
+    guardian_set: UncheckedAccount<'info>,
+
+    /// CHECK: Stored guardian signatures to be verified by shim.
+    /// Ownership ownership and discriminator is checked by the shim.
+    guardian_signatures: UncheckedAccount<'info>,
+
+    wormhole_verify_vaa_shim: Program<'info, WormholeVerifyVaaShim>,
+}
+
+pub fn consume_vaa(
+    ctx: Context<ConsumeVaa>,
+    guardian_set_bump: u8,
+    vaa_body: Vec<u8>,
+) -> Result<()> {
+    // Compute the message hash.
+    let message_hash = &solana_program::keccak::hashv(&[&vaa_body]).to_bytes();
+    let digest = keccak::hash(message_hash.as_slice()).to_bytes();
+    // Verify the hash against the signatures.
+    wormhole_verify_vaa_shim::cpi::verify_hash(
+        CpiContext::new(
+            ctx.accounts.wormhole_verify_vaa_shim.to_account_info(),
+            VerifyHash {
+                guardian_set: ctx.accounts.guardian_set.to_account_info(),
+                guardian_signatures: ctx.accounts.guardian_signatures.to_account_info(),
+            },
+        ),
+        guardian_set_bump,
+        digest,
+    )?;
+    // Decode vaa_body, perform security checks, and do your thing.
+    Ok(())
+}

+ 94 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/instructions/initialize.rs

@@ -0,0 +1,94 @@
+declare_program!(wormhole_post_message_shim);
+
+use anchor_lang::{prelude::*, solana_program::bpf_loader_upgradeable};
+
+use wormhole_post_message_shim::{
+    cpi::accounts::PostMessage, program::WormholePostMessageShim, types::Finality,
+};
+use wormhole_solana_consts::{
+    CORE_BRIDGE_CONFIG, CORE_BRIDGE_FEE_COLLECTOR, CORE_BRIDGE_PROGRAM_ID,
+};
+
+#[derive(Accounts)]
+pub struct Initialize<'info> {
+    #[account(mut)]
+    payer: Signer<'info>,
+
+    #[account(constraint = deployer.key() == program_data.upgrade_authority_address.unwrap_or_default())]
+    deployer: Signer<'info>,
+
+    #[account(
+        seeds = [crate::ID.as_ref()],
+        bump,
+        seeds::program = bpf_loader_upgradeable::id(),
+    )]
+    program_data: Account<'info, ProgramData>,
+
+    wormhole_post_message_shim: Program<'info, WormholePostMessageShim>,
+
+    #[account(mut, address = CORE_BRIDGE_CONFIG)]
+    /// CHECK: Wormhole bridge config. [`wormhole::post_message`] requires this account be mutable.
+    pub bridge: UncheckedAccount<'info>,
+
+    #[account(mut, seeds = [&emitter.key.to_bytes()], bump, seeds::program = wormhole_post_message_shim::ID)]
+    /// CHECK: Wormhole Message. [`wormhole::post_message`] requires this account be signer and mutable.
+    /// This program uses a PDA per emitter, since these are already bottle-necked by sequence and
+    /// the bridge enforces that emitter must be identical for reused accounts.
+    /// While this could be managed by the integrator, it seems more effective to have the shim manage these accounts.
+    /// Bonus, this also allows Anchor to automatically handle deriving the address.
+    pub message: UncheckedAccount<'info>,
+
+    #[account(seeds = [b"emitter"], bump)]
+    /// CHECK: Our emitter
+    pub emitter: UncheckedAccount<'info>,
+
+    #[account(mut)]
+    /// CHECK: Emitter's sequence account. [`wormhole::post_message`] requires this account be mutable.
+    /// Explicitly do not re-derive this account. The core bridge verifies the derivation anyway and
+    /// as of Anchor 0.30.1, auto-derivation for other programs' accounts via IDL doesn't work.
+    pub sequence: UncheckedAccount<'info>,
+
+    #[account(mut, address = CORE_BRIDGE_FEE_COLLECTOR)]
+    /// CHECK: Wormhole fee collector. [`wormhole::post_message`] requires this account be mutable.
+    pub fee_collector: UncheckedAccount<'info>,
+
+    /// Clock sysvar.
+    pub clock: Sysvar<'info, Clock>,
+
+    /// System program.
+    pub system_program: Program<'info, System>,
+
+    #[account(address = CORE_BRIDGE_PROGRAM_ID)]
+    /// CHECK: Wormhole program.
+    pub wormhole_program: UncheckedAccount<'info>,
+
+    /// CHECK: Shim event authority
+    pub wormhole_post_message_shim_ea: UncheckedAccount<'info>,
+}
+
+pub fn initialize(ctx: Context<Initialize>) -> Result<()> {
+    wormhole_post_message_shim::cpi::post_message(
+        CpiContext::new_with_signer(
+            ctx.accounts.wormhole_post_message_shim.to_account_info(),
+            PostMessage {
+                payer: ctx.accounts.payer.to_account_info(),
+                bridge: ctx.accounts.bridge.to_account_info(),
+                message: ctx.accounts.message.to_account_info(),
+                emitter: ctx.accounts.emitter.to_account_info(),
+                sequence: ctx.accounts.sequence.to_account_info(),
+                fee_collector: ctx.accounts.fee_collector.to_account_info(),
+                clock: ctx.accounts.clock.to_account_info(),
+                system_program: ctx.accounts.system_program.to_account_info(),
+                wormhole_program: ctx.accounts.wormhole_program.to_account_info(),
+                program: ctx.accounts.wormhole_post_message_shim.to_account_info(),
+                event_authority: ctx.accounts.wormhole_post_message_shim_ea.to_account_info(),
+            },
+            &[&[b"emitter", &[ctx.bumps.emitter]]],
+        ),
+        0,
+        Finality::Finalized,
+        vec![],
+    )?;
+
+    Ok(())
+}

+ 8 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/instructions/mod.rs

@@ -0,0 +1,8 @@
+mod consume_vaa;
+pub use consume_vaa::*;
+
+mod initialize;
+pub use initialize::*;
+
+mod post_message;
+pub use post_message::*;

+ 103 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/instructions/post_message.rs

@@ -0,0 +1,103 @@
+declare_program!(wormhole_post_message_shim);
+
+use anchor_lang::prelude::*;
+
+use wormhole_post_message_shim::{program::WormholePostMessageShim, types::Finality};
+use wormhole_solana_consts::{
+    CORE_BRIDGE_CONFIG, CORE_BRIDGE_FEE_COLLECTOR, CORE_BRIDGE_PROGRAM_ID,
+};
+
+#[derive(Accounts)]
+pub struct PostMessage<'info> {
+    #[account(mut)]
+    payer: Signer<'info>,
+
+    wormhole_post_message_shim: Program<'info, WormholePostMessageShim>,
+
+    #[account(mut, address = CORE_BRIDGE_CONFIG)]
+    /// CHECK: Wormhole bridge config. [`wormhole::post_message`] requires this account be mutable.
+    /// Address constraint added for IDL generation / convenience, it will be enforced by the core bridge.
+    pub bridge: UncheckedAccount<'info>,
+
+    #[account(mut, seeds = [&emitter.key.to_bytes()], bump, seeds::program = wormhole_post_message_shim::ID)]
+    /// CHECK: Wormhole Message. [`wormhole::post_message`] requires this account be signer and mutable.
+    /// Seeds constraint added for IDL generation / convenience, it will be enforced by the shim.
+    pub message: UncheckedAccount<'info>,
+
+    #[account(seeds = [b"emitter"], bump)]
+    /// CHECK: Our emitter
+    /// Seeds constraint added for IDL generation / convenience, it will be enforced to match the signer used in the CPI call.
+    pub emitter: UncheckedAccount<'info>,
+
+    #[account(mut)]
+    /// CHECK: Emitter's sequence account. [`wormhole::post_message`] requires this account be mutable.
+    /// Explicitly do not re-derive this account. The core bridge verifies the derivation anyway and
+    /// as of Anchor 0.30.1, auto-derivation for other programs' accounts via IDL doesn't work.
+    pub sequence: UncheckedAccount<'info>,
+
+    #[account(mut, address = CORE_BRIDGE_FEE_COLLECTOR)]
+    /// CHECK: Wormhole fee collector. [`wormhole::post_message`] requires this account be mutable.
+    /// Address constraint added for IDL generation / convenience, it will be enforced by the core bridge.
+    pub fee_collector: UncheckedAccount<'info>,
+
+    /// Clock sysvar.
+    /// Type added for IDL generation / convenience, it will be enforced by the core bridge.
+    pub clock: Sysvar<'info, Clock>,
+
+    /// System program.
+    /// Type for IDL generation / convenience, it will be enforced by the core bridge.
+    pub system_program: Program<'info, System>,
+
+    #[account(address = CORE_BRIDGE_PROGRAM_ID)]
+    /// CHECK: Wormhole program.
+    /// Address constraint added for IDL generation / convenience, it will be enforced by the shim.
+    pub wormhole_program: UncheckedAccount<'info>,
+
+    /// CHECK: Shim event authority
+    /// TODO: An address constraint could be included if this address was published to wormhole_solana_consts
+    /// Address will be enforced by the shim.
+    pub wormhole_post_message_shim_ea: UncheckedAccount<'info>,
+}
+
+pub fn post_message(ctx: Context<PostMessage>) -> Result<()> {
+    // wormhole::post_message may require that a fee be sent to the fee_collector account of the core bridge.
+    // The following code could be used to handle this via CPI call.
+    // However, this example handles this complexity on the client side using a `preInstruction`
+    //
+    // let fee = ctx.accounts.wormhole_bridge.fee();
+    // if fee > 0 {
+    //     solana_program::program::invoke(
+    //         &solana_program::system_instruction::transfer(
+    //             &ctx.accounts.payer.key(),
+    //             &ctx.accounts.fee_collector.key(),
+    //             fee,
+    //         ),
+    //         &ctx.accounts.to_account_infos(),
+    //     )?;
+    // }
+
+    wormhole_post_message_shim::cpi::post_message(
+        CpiContext::new_with_signer(
+            ctx.accounts.wormhole_post_message_shim.to_account_info(),
+            wormhole_post_message_shim::cpi::accounts::PostMessage {
+                payer: ctx.accounts.payer.to_account_info(),
+                bridge: ctx.accounts.bridge.to_account_info(),
+                message: ctx.accounts.message.to_account_info(),
+                emitter: ctx.accounts.emitter.to_account_info(),
+                sequence: ctx.accounts.sequence.to_account_info(),
+                fee_collector: ctx.accounts.fee_collector.to_account_info(),
+                clock: ctx.accounts.clock.to_account_info(),
+                system_program: ctx.accounts.system_program.to_account_info(),
+                wormhole_program: ctx.accounts.wormhole_program.to_account_info(),
+                program: ctx.accounts.wormhole_post_message_shim.to_account_info(),
+                event_authority: ctx.accounts.wormhole_post_message_shim_ea.to_account_info(),
+            },
+            &[&[b"emitter", &[ctx.bumps.emitter]]],
+        ),
+        0,
+        Finality::Finalized,
+        b"your message goes here!".to_vec(),
+    )?;
+
+    Ok(())
+}

+ 37 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-integrator-example/src/lib.rs

@@ -0,0 +1,37 @@
+use anchor_lang::prelude::*;
+
+declare_id!("AEwubmehHNvkMXoH2C5MgDSemZgQ3HUSYpeaF3UrNZdQ");
+
+mod instructions;
+pub(crate) use instructions::*;
+
+#[program]
+pub mod wormhole_integrator_example {
+    use super::*;
+
+    /// This example instruction posts an empty message during initialize in order to have one less
+    /// CPI depth on subsequent posts.
+    ///
+    /// NOTE: this example does not replay protect the call. Typically, this may be done with an
+    /// `init` constraint on an account that was used to store config that is also set up during initialization.
+    pub fn initialize(ctx: Context<Initialize>) -> Result<()> {
+        instructions::initialize(ctx)
+    }
+
+    /// This example instruction takes the guardian signatures account previously posted to the verify vaa shim
+    /// with `post_signatures` and the corresponding guardian set from the core bridge and verifies a
+    /// provided VAA body against it. `close_signatures` on the shim should be called immediately
+    /// afterwards in order to reclaim the rent lamports taken by `post_signatures`.
+    pub fn consume_vaa(
+        ctx: Context<ConsumeVaa>,
+        guardian_set_bump: u8,
+        vaa_body: Vec<u8>,
+    ) -> Result<()> {
+        instructions::consume_vaa(ctx, guardian_set_bump, vaa_body)
+    }
+
+    /// This example instruction posts a message via the post message shim.
+    pub fn post_message(ctx: Context<PostMessage>) -> Result<()> {
+        instructions::post_message(ctx)
+    }
+}

+ 28 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/Cargo.toml

@@ -0,0 +1,28 @@
+[package]
+name = "wormhole-vaa-verification-comparison"
+description = "Created with Anchor"
+
+edition.workspace = true
+publish.workspace = true
+rust-version.workspace = true
+version.workspace = true
+
+[lib]
+crate-type = ["cdylib", "lib"]
+
+[features]
+default = ["mainnet"]
+mainnet = ["wormhole-solana-consts/mainnet"]
+testnet = ["wormhole-solana-consts/testnet"]
+localnet = ["wormhole-solana-consts/localnet"]
+cpi = ["no-entrypoint"]
+no-entrypoint = []
+no-idl = []
+no-log-ix-name = []
+idl-build = ["anchor-lang/idl-build"]
+
+[dependencies]
+anchor-lang = { workspace = true, features = ["init-if-needed"] }
+wormhole-anchor-sdk = "=0.30.1-alpha.3"
+wormhole-raw-vaas.workspace = true
+wormhole-solana-consts.workspace = true

+ 2 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/Xargo.toml

@@ -0,0 +1,2 @@
+[target.bpfel-unknown-unknown.dependencies.std]
+features = []

+ 28 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/error.rs

@@ -0,0 +1,28 @@
+use anchor_lang::prelude::error_code;
+
+#[error_code]
+pub enum WormholeVaaVerificationComparisonError {
+    #[msg("EmptyGuardianSignatures")]
+    EmptyGuardianSignatures = 0x100,
+
+    #[msg("WriteAuthorityMismatch")]
+    WriteAuthorityMismatch = 0x101,
+
+    #[msg("GuardianSetExpired")]
+    GuardianSetExpired = 0x102,
+
+    #[msg("InvalidGuardianKeyRecovery")]
+    InvalidGuardianKeyRecovery = 0x103,
+
+    #[msg("NoQuorum")]
+    NoQuorum = 0x104,
+
+    #[msg("InvalidSignature")]
+    InvalidSignature = 0x105,
+
+    #[msg("InvalidGuardianIndexNonIncreasing")]
+    InvalidGuardianIndexNonIncreasing = 0x106,
+
+    #[msg("InvalidGuardianIndexOutOfRange")]
+    InvalidGuardianIndexOutOfRange = 0x107,
+}

+ 27 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/consume_core_posted_vaa.rs

@@ -0,0 +1,27 @@
+use anchor_lang::prelude::*;
+use wormhole_anchor_sdk::wormhole;
+use wormhole_solana_consts::CORE_BRIDGE_PROGRAM_ID;
+
+#[derive(Accounts)]
+#[instruction(vaa_hash: [u8; 32])]
+pub struct ConsumeCorePostedVaa<'info> {
+    #[account(
+        seeds = [
+            wormhole::SEED_PREFIX_POSTED_VAA,
+            &vaa_hash
+        ],
+        bump,
+        seeds::program = CORE_BRIDGE_PROGRAM_ID
+    )]
+    /// CHECK: Verified Wormhole message account. The Wormhole program verified
+    /// signatures and posted the account data here. Read-only.
+    pub posted: UncheckedAccount<'info>,
+}
+
+pub fn consume_core_posted_vaa(
+    ctx: Context<ConsumeCorePostedVaa>,
+    _vaa_hash: [u8; 32],
+) -> Result<()> {
+    ctx.accounts.posted.try_borrow_data()?;
+    Ok(())
+}

+ 141 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/consume_vaa.rs

@@ -0,0 +1,141 @@
+use anchor_lang::{
+    prelude::*,
+    solana_program::{
+        self, keccak, program_memory::sol_memcpy, secp256k1_recover::secp256k1_recover,
+    },
+};
+use wormhole_raw_vaas::{utils::quorum, GuardianSetSig};
+use wormhole_solana_consts::CORE_BRIDGE_PROGRAM_ID;
+
+use crate::{
+    error::WormholeVaaVerificationComparisonError,
+    state::{GuardianSignatures, WormholeGuardianSet},
+};
+
+#[derive(Accounts)]
+#[instruction(vaa_body: Vec<u8>, guardian_set_index: u32)]
+pub struct ConsumeVaa<'info> {
+    /// Guardian set used for signature verification.
+    #[account(
+        seeds = [
+            WormholeGuardianSet::SEED_PREFIX,
+            guardian_set_index.to_be_bytes().as_ref()
+        ],
+        bump,
+        seeds::program = CORE_BRIDGE_PROGRAM_ID
+    )]
+    guardian_set: Account<'info, WormholeGuardianSet>,
+
+    /// Stores unverified guardian signatures as they are too large to fit in the instruction data.
+    #[account(mut, has_one = refund_recipient, close = refund_recipient)]
+    guardian_signatures: Account<'info, GuardianSignatures>,
+
+    /// CHECK: This account is the refund recipient for the above signature_set
+    #[account(mut, address = guardian_signatures.refund_recipient)]
+    refund_recipient: AccountInfo<'info>,
+}
+
+impl<'info> ConsumeVaa<'info> {
+    pub fn constraints(ctx: &Context<Self>, vaa_body: &Vec<u8>) -> Result<()> {
+        let guardian_set = &ctx.accounts.guardian_set;
+
+        // Check that the guardian set is still active.
+        let timestamp = Clock::get()?
+            .unix_timestamp
+            .try_into()
+            .expect("timestamp overflow");
+        require!(
+            guardian_set.is_active(&timestamp),
+            WormholeVaaVerificationComparisonError::GuardianSetExpired
+        );
+
+        let guardian_signatures = &ctx.accounts.guardian_signatures.guardian_signatures;
+
+        // This section is borrowed from https://github.com/wormhole-foundation/wormhole/blob/wen/solana-rewrite/solana/programs/core-bridge/src/processor/parse_and_verify_vaa/verify_encoded_vaa_v1.rs#L72-L103
+        // Also similarly used here https://github.com/pyth-network/pyth-crosschain/blob/6771c2c6998f53effee9247347cb0ac71612b3dc/target_chains/solana/programs/pyth-solana-receiver/src/lib.rs#L121-L159
+        // Do we have enough signatures for quorum?
+        let guardian_keys = &guardian_set.keys;
+        let quorum = quorum(guardian_keys.len());
+        require!(
+            guardian_signatures.len() >= quorum,
+            WormholeVaaVerificationComparisonError::NoQuorum
+        );
+
+        // Compute the message hash.
+        let message_hash = &solana_program::keccak::hashv(&[&vaa_body]).to_bytes();
+        let digest = keccak::hash(message_hash.as_slice());
+
+        // Verify signatures
+        let mut last_guardian_index = None;
+        for sig_bytes in guardian_signatures {
+            let sig = GuardianSetSig::try_from(sig_bytes.as_slice())
+                .map_err(|_| WormholeVaaVerificationComparisonError::InvalidSignature)?;
+            // We do not allow for non-increasing guardian signature indices.
+            let index = usize::from(sig.guardian_index());
+            if let Some(last_index) = last_guardian_index {
+                require!(
+                    index > last_index,
+                    WormholeVaaVerificationComparisonError::InvalidGuardianIndexNonIncreasing
+                );
+            }
+
+            // Does this guardian index exist in this guardian set?
+            let guardian_pubkey = guardian_keys.get(index).ok_or_else(|| {
+                error!(WormholeVaaVerificationComparisonError::InvalidGuardianIndexOutOfRange)
+            })?;
+
+            // Now verify that the signature agrees with the expected Guardian's pubkey.
+            verify_guardian_signature(&sig, guardian_pubkey, digest.as_ref())?;
+
+            last_guardian_index = Some(index);
+        }
+        // End borrowed section
+
+        // Done.
+        Ok(())
+    }
+}
+
+#[access_control(ConsumeVaa::constraints(&ctx, &vaa_body))]
+pub fn consume_vaa(
+    ctx: Context<ConsumeVaa>,
+    vaa_body: Vec<u8>,
+    _guardian_set_index: u32,
+) -> Result<()> {
+    Ok(())
+}
+
+/**
+ * Borrowed from https://github.com/wormhole-foundation/wormhole/blob/wen/solana-rewrite/solana/programs/core-bridge/src/processor/parse_and_verify_vaa/verify_encoded_vaa_v1.rs#L121
+ * Also used here https://github.com/pyth-network/pyth-crosschain/blob/6771c2c6998f53effee9247347cb0ac71612b3dc/target_chains/solana/programs/pyth-solana-receiver/src/lib.rs#L432
+ */
+fn verify_guardian_signature(
+    sig: &GuardianSetSig,
+    guardian_pubkey: &[u8; 20],
+    digest: &[u8],
+) -> Result<()> {
+    // Recover using `solana_program::secp256k1_recover`. Public key recovery costs 25k compute
+    // units. And hashing this public key to recover the Ethereum public key costs about 13k.
+    let recovered = {
+        // Recover EC public key (64 bytes).
+        let pubkey = secp256k1_recover(digest, sig.recovery_id(), &sig.rs())
+            .map_err(|_| WormholeVaaVerificationComparisonError::InvalidSignature)?;
+
+        // The Ethereum public key is the last 20 bytes of keccak hashed public key above.
+        let hashed = keccak::hash(&pubkey.to_bytes());
+
+        let mut eth_pubkey = [0; 20];
+        sol_memcpy(&mut eth_pubkey, &hashed.0[12..], 20);
+
+        eth_pubkey
+    };
+
+    // The recovered public key should agree with the Guardian's public key at this index.
+    require!(
+        recovered == *guardian_pubkey,
+        WormholeVaaVerificationComparisonError::InvalidGuardianKeyRecovery
+    );
+
+    // Done.
+    Ok(())
+}

+ 42 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/consume_vaa_via_shim.rs

@@ -0,0 +1,42 @@
+declare_program!(wormhole_verify_vaa_shim);
+
+use anchor_lang::{
+    prelude::*,
+    solana_program::{self, keccak},
+};
+use wormhole_verify_vaa_shim::cpi::accounts::VerifyHash;
+use wormhole_verify_vaa_shim::program::WormholeVerifyVaaShim;
+
+#[derive(Accounts)]
+pub struct ConsumeVaaViaShim<'info> {
+    /// CHECK: Guardian set used for signature verification by shim.
+    /// Derivation is checked by the shim.
+    guardian_set: UncheckedAccount<'info>,
+
+    /// CHECK: Stores guardian signatures to be verified by shim.
+    guardian_signatures: UncheckedAccount<'info>,
+
+    wormhole_verify_vaa_shim: Program<'info, WormholeVerifyVaaShim>,
+}
+
+pub fn consume_vaa_via_shim(
+    ctx: Context<ConsumeVaaViaShim>,
+    guardian_set_bump: u8,
+    vaa_body: Vec<u8>,
+) -> Result<()> {
+    // Compute the message hash.
+    let message_hash = &solana_program::keccak::hashv(&[&vaa_body]).to_bytes();
+    let digest = keccak::hash(message_hash.as_slice()).to_bytes();
+    wormhole_verify_vaa_shim::cpi::verify_hash(
+        CpiContext::new(
+            ctx.accounts.wormhole_verify_vaa_shim.to_account_info(),
+            VerifyHash {
+                guardian_set: ctx.accounts.guardian_set.to_account_info(),
+                guardian_signatures: ctx.accounts.guardian_signatures.to_account_info(),
+            },
+        ),
+        guardian_set_bump,
+        digest,
+    )?;
+    Ok(())
+}

+ 11 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/mod.rs

@@ -0,0 +1,11 @@
+mod consume_core_posted_vaa;
+pub use consume_core_posted_vaa::*;
+
+mod post_signatures;
+pub use post_signatures::*;
+
+mod consume_vaa;
+pub use consume_vaa::*;
+
+mod consume_vaa_via_shim;
+pub use consume_vaa_via_shim::*;

+ 70 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/instructions/post_signatures.rs

@@ -0,0 +1,70 @@
+use crate::{error::WormholeVaaVerificationComparisonError, state::GuardianSignatures};
+use anchor_lang::prelude::*;
+
+#[derive(Accounts)]
+#[instruction(_guardian_signatures: Vec<[u8; 66]>, total_signatures: u8)]
+pub struct PostSignatures<'info> {
+    #[account(mut)]
+    payer: Signer<'info>,
+
+    #[account(
+        init_if_needed,
+        payer = payer,
+        space = 8 + GuardianSignatures::compute_size(usize::from(total_signatures))
+    )]
+    guardian_signatures: Account<'info, GuardianSignatures>,
+
+    system_program: Program<'info, System>,
+}
+
+impl<'info> PostSignatures<'info> {
+    pub fn constraints(guardian_signatures: &Vec<[u8; 66]>) -> Result<()> {
+        // Signatures should not be empty, since this is used by is_initialized.
+        // Additionally, there is no reason for it to be.
+        require!(
+            !guardian_signatures.is_empty(),
+            WormholeVaaVerificationComparisonError::EmptyGuardianSignatures
+        );
+
+        // Done.
+        Ok(())
+    }
+}
+
+/// Creates or appends to a GuardianSignatures account for subsequent use by consume_vaa.
+/// This is necessary as the Wormhole query response (220 bytes)
+/// and 13 guardian signatures (a quorum of the current 19 mainnet guardians, 66 bytes each)
+/// alongside the required accounts is larger than the transaction size limit on Solana (1232 bytes).
+///
+/// This instruction allows for the initial payer to append additional signatures to the account by calling the instruction again.
+/// This may be necessary if a quorum of signatures from the current guardian set grows larger than can fit into a single transaction.
+///
+/// The GuardianSignatures account can be closed by anyone with a successful consume_vaa instruction
+/// or by the initial payer via close_signatures, either of which will refund the initial payer.
+#[access_control(PostSignatures::constraints(&guardian_signatures))]
+pub fn post_signatures(
+    ctx: Context<PostSignatures>,
+    mut guardian_signatures: Vec<[u8; 66]>,
+    _total_signatures: u8,
+) -> Result<()> {
+    if ctx.accounts.guardian_signatures.is_initialized() {
+        require_eq!(
+            ctx.accounts.guardian_signatures.refund_recipient,
+            ctx.accounts.payer.key(),
+            WormholeVaaVerificationComparisonError::WriteAuthorityMismatch
+        );
+        ctx.accounts
+            .guardian_signatures
+            .guardian_signatures
+            .append(&mut guardian_signatures);
+    } else {
+        ctx.accounts
+            .guardian_signatures
+            .set_inner(GuardianSignatures {
+                refund_recipient: ctx.accounts.payer.key(),
+                guardian_signatures,
+            });
+    }
+    // Done.
+    Ok(())
+}

+ 46 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/lib.rs

@@ -0,0 +1,46 @@
+use anchor_lang::prelude::*;
+
+declare_id!("m7qfXCwkwvpB4RayZoVsNxm4X2iE2a34rCAsjbNvrRs");
+
+mod instructions;
+pub(crate) use instructions::*;
+
+pub mod state;
+
+pub mod error;
+
+#[program]
+pub mod wormhole_vaa_verification_comparison {
+    use super::*;
+
+    pub fn consume_core_posted_vaa(
+        ctx: Context<ConsumeCorePostedVaa>,
+        vaa_hash: [u8; 32],
+    ) -> Result<()> {
+        instructions::consume_core_posted_vaa(ctx, vaa_hash)
+    }
+
+    pub fn post_signatures(
+        ctx: Context<PostSignatures>,
+        guardian_signatures: Vec<[u8; 66]>,
+        total_signatures: u8,
+    ) -> Result<()> {
+        instructions::post_signatures(ctx, guardian_signatures, total_signatures)
+    }
+
+    pub fn consume_vaa(
+        ctx: Context<ConsumeVaa>,
+        vaa_body: Vec<u8>,
+        guardian_set_index: u32,
+    ) -> Result<()> {
+        instructions::consume_vaa(ctx, vaa_body, guardian_set_index)
+    }
+
+    pub fn consume_vaa_via_shim(
+        ctx: Context<ConsumeVaaViaShim>,
+        guardian_set_bump: u8,
+        vaa_body: Vec<u8>,
+    ) -> Result<()> {
+        instructions::consume_vaa_via_shim(ctx, guardian_set_bump, vaa_body)
+    }
+}

+ 52 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/state/ext/mod.rs

@@ -0,0 +1,52 @@
+pub use anchor_lang::prelude::*;
+use wormhole_solana_consts::CORE_BRIDGE_PROGRAM_ID;
+
+#[derive(Debug, AnchorSerialize, AnchorDeserialize, Clone)]
+pub struct WormholeGuardianSet {
+    /// Index representing an incrementing version number for this guardian set.
+    pub index: u32,
+
+    /// Ethereum-style public keys.
+    pub keys: Vec<[u8; 20]>,
+
+    /// Timestamp representing the time this guardian became active.
+    pub creation_time: u32,
+
+    /// Expiration time when VAAs issued by this set are no longer valid.
+    pub expiration_time: u32,
+}
+
+// TODO: does there need to be some sort of seed check as well?
+impl Owner for WormholeGuardianSet {
+    fn owner() -> Pubkey {
+        CORE_BRIDGE_PROGRAM_ID
+    }
+}
+
+// workaround for anchor 0.30.1
+// https://github.com/coral-xyz/anchor/blob/e6d7dafe12da661a36ad1b4f3b5970e8986e5321/spl/src/idl_build.rs#L11
+impl anchor_lang::Discriminator for WormholeGuardianSet {
+    const DISCRIMINATOR: [u8; 8] = [0; 8];
+}
+
+impl AccountSerialize for WormholeGuardianSet {}
+
+impl AccountDeserialize for WormholeGuardianSet {
+    fn try_deserialize_unchecked(buf: &mut &[u8]) -> Result<Self> {
+        Self::deserialize(buf).map_err(Into::into)
+    }
+}
+
+impl WormholeGuardianSet {
+    pub const SEED_PREFIX: &'static [u8] = b"GuardianSet";
+
+    pub fn is_active(&self, timestamp: &u32) -> bool {
+        // Note: This is a fix for Wormhole on mainnet.  The initial guardian set was never expired
+        // so we block it here.
+        if self.index == 0 && self.creation_time == 1628099186 {
+            false
+        } else {
+            self.expiration_time == 0 || self.expiration_time >= *timestamp
+        }
+    }
+}

+ 24 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/state/guardian_signatures.rs

@@ -0,0 +1,24 @@
+use anchor_lang::prelude::*;
+
+#[account]
+#[derive(Debug)]
+pub struct GuardianSignatures {
+    /// Payer of this guardian signatures account.
+    /// Only they may amend signatures.
+    /// Used for reimbursements upon cleanup.
+    pub refund_recipient: Pubkey,
+
+    /// Unverified guardian signatures.
+    pub guardian_signatures: Vec<[u8; 66]>,
+}
+
+impl GuardianSignatures {
+    pub(crate) fn compute_size(num_guardians: usize) -> usize {
+        32 // refund_recipient
+        + 4 + num_guardians * 66 // signatures
+    }
+
+    pub fn is_initialized(&self) -> bool {
+        !self.guardian_signatures.is_empty()
+    }
+}

+ 5 - 0
svm/wormhole-core-shims/anchor/programs/wormhole-vaa-verification-comparison/src/state/mod.rs

@@ -0,0 +1,5 @@
+mod guardian_signatures;
+pub use guardian_signatures::*;
+
+mod ext;
+pub use ext::*;

+ 3 - 0
svm/wormhole-core-shims/anchor/rust-toolchain.toml

@@ -0,0 +1,3 @@
+[toolchain]
+channel = "1.75.0"
+components = [ "clippy", "rustc", "rustfmt" ]

+ 13 - 0
svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_devnet/bridge_config.json

@@ -0,0 +1,13 @@
+{
+  "pubkey": "FKoMTctsC7vJbEqyRiiPskPnuQx2tX1kurmvWByq5uZP",
+  "account": {
+    "lamports": 1057920,
+    "data": [
+      "AAAAAACYDQAAAAAAgFEBAGQAAAAAAAAA",
+      "base64"
+    ],
+    "owner": "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o",
+    "executable": false,
+    "rentEpoch": 0
+  }
+}

+ 13 - 0
svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_devnet/fee_collector.json

@@ -0,0 +1,13 @@
+{
+    "pubkey": "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs",
+    "account": {
+      "lamports": 890880,
+      "data": [
+        "",
+        "base64"
+      ],
+      "owner": "11111111111111111111111111111111",
+      "executable": false,
+      "rentEpoch": 0
+    }
+  }

+ 13 - 0
svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_devnet/guardian_set.json

@@ -0,0 +1,13 @@
+{
+  "pubkey": "6MxkvoEwgB9EqQRLNhvYaPGhfcLtBtpBqdQugr3AZUgD",
+  "account": {
+    "lamports": 1141440,
+    "data": [
+      "AAAAAAEAAAC++kKdV80Yt/ik2RotqatK8F0PvoX2jWIAAAAA",
+      "base64"
+    ],
+    "owner": "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o",
+    "executable": false,
+    "rentEpoch": 0
+  }
+}

+ 11 - 0
svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_mainnet/config.json

@@ -0,0 +1,11 @@
+{
+  "pubkey": "2yVjuQwpsvdsrywzsJJVs9Ueh4zayyo5DYJbBNc3DDpn",
+  "account": {
+    "lamports": 1057920,
+    "data": ["BAAAAAQYDQ0AAAAAgFEBAGQAAAAAAAAA", "base64"],
+    "owner": "worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth",
+    "executable": false,
+    "rentEpoch": 18446744073709551615,
+    "space": 24
+  }
+}

+ 14 - 0
svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_mainnet/fee_collector.json

@@ -0,0 +1,14 @@
+{
+  "pubkey": "9bFNrXNb2WTx8fMHXCheaZqkLZ3YCCaiqTftHxeintHy",
+  "account": {
+    "lamports": 2350640070,
+    "data": [
+      "",
+      "base64"
+    ],
+    "owner": "11111111111111111111111111111111",
+    "executable": false,
+    "rentEpoch": 18446744073709551615,
+    "space": 0
+  }
+}

+ 14 - 0
svm/wormhole-core-shims/anchor/tests/accounts/core_bridge_mainnet/guardian_set_4.json

@@ -0,0 +1,14 @@
+{
+  "pubkey": "AFEXK4A1BU7BZfi8niAmker98LH9EARB544wKGPXwMyy",
+  "account": {
+    "lamports": 3647040,
+    "data": [
+      "BAAAABMAAABYk7WnbD9zlkVkiIW9zMBs1wo80/9suVJYm96GLCXvQ5ITL7nUpCFXEU3oRgGTvfOi/PgfhqCXZfR2L9EQegCGsy16CXeSaiBRMdhzHTnL64yCsv2C+u0nEdWa8PJJnRbnJvayEbOXVsBCRBvm2GULabVOvnFeI0NUzltNNI+3S5WOiWbi7D29SVinzRXnyvB8Tj3I58Rp+SyM2I+4AFogdKO/kTlT1pUmDYi8GqJaTu42PvAACsAHZyezX76i2sKP7lzLD+p2jq9FztE2udniSQNGSuiJ9cinI/wU+TEkt8c4hDy7iehkyGLDjN3Mz5XSzDek3ANqjSMrSPYs3UcxQS9IkNp5j2iWozMfZLSMEtHVf9nL5wgRcaob4dNsr+OGeRD5nAnjR4mcGcOBkrbnOHzNdoJ3wX2rG3pQJ8CzzxeOIa0ud64GcRVJz7sfnHqdgJboXhSH81UV0CqSdTUEqNdUcbn0nttvvryJj0A+R3PpX+sV6Ayamcg0jXiZHmYAAAAA",
+      "base64"
+    ],
+    "owner": "worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth",
+    "executable": false,
+    "rentEpoch": 18446744073709551615,
+    "space": 396
+  }
+}

BIN
svm/wormhole-core-shims/anchor/tests/artifacts/devnet_core_bridge.so


BIN
svm/wormhole-core-shims/anchor/tests/artifacts/mainnet_core_bridge.so


+ 253 - 0
svm/wormhole-core-shims/anchor/tests/e2e.ts

@@ -0,0 +1,253 @@
+import {
+  AnchorProvider,
+  Program,
+  setProvider,
+  Wallet,
+  web3,
+} from "@coral-xyz/anchor";
+import WormholePostMessageShimIdl from "../idls/wormhole_post_message_shim.json";
+import { WormholePostMessageShim } from "../idls/wormhole_post_message_shim";
+import WormholeIntegratorExampleIdl from "./idls/devnet/wormhole_integrator_example.json";
+import { WormholeIntegratorExample } from "./idls/devnet/wormhole_integrator_example";
+import { getSequenceFromTx } from "./helpers";
+import { getSignedVAAWithRetry } from "@certusone/wormhole-sdk/lib/cjs/rpc";
+import { parseVaa } from "@certusone/wormhole-sdk/lib/cjs/vaa/wormhole";
+import { NodeHttpTransport } from "@improbable-eng/grpc-web-node-http-transport";
+import { getSequenceTracker } from "@certusone/wormhole-sdk/lib/cjs/solana/wormhole";
+
+(async () => {
+  const SOLANA_RPC_URL = "http://127.0.0.1:8899";
+  const GUARDIAN_URL = "http://127.0.0.1:7071";
+
+  const CORE_BRIDGE_PROGRAM_ID = new web3.PublicKey(
+    "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o"
+  );
+
+  const connection = new web3.Connection(SOLANA_RPC_URL, "confirmed");
+
+  const getShimProgramWithNewSigner = async () => {
+    const payer = web3.Keypair.generate();
+    {
+      const tx = await connection.requestAirdrop(payer.publicKey, 10000000000);
+      await connection.confirmTransaction({
+        ...(await connection.getLatestBlockhash()),
+        signature: tx,
+      });
+    }
+    const provider = new AnchorProvider(connection, new Wallet(payer));
+    setProvider(provider);
+
+    const program = new Program<WormholePostMessageShim>(
+      WormholePostMessageShimIdl as WormholePostMessageShim
+    );
+    return program;
+  };
+
+  const getIntegrationProgramWithNewSigner = async () => {
+    const payer = web3.Keypair.generate();
+    {
+      const tx = await connection.requestAirdrop(payer.publicKey, 10000000000);
+      await connection.confirmTransaction({
+        ...(await connection.getLatestBlockhash()),
+        signature: tx,
+      });
+    }
+    const provider = new AnchorProvider(connection, new Wallet(payer));
+    setProvider(provider);
+
+    const program = new Program<WormholeIntegratorExample>(
+      WormholeIntegratorExampleIdl as WormholeIntegratorExample
+    );
+    return program;
+  };
+
+  const postMessage = async (
+    program: Program<WormholePostMessageShim>,
+    msg: string,
+    consistencyLevel: number
+  ): Promise<string> =>
+    await program.methods
+      .postMessage(
+        0,
+        consistencyLevel === 0 ? { confirmed: {} } : { finalized: {} },
+        Buffer.from(msg, "ascii")
+      )
+      .accounts({
+        emitter: program.provider.publicKey,
+        wormholeProgram: CORE_BRIDGE_PROGRAM_ID,
+      })
+      .preInstructions([
+        // gotta pay the fee
+        web3.SystemProgram.transfer({
+          fromPubkey: program.provider.publicKey,
+          toPubkey: new web3.PublicKey(
+            "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs"
+          ), // fee collector
+          lamports: 100, // hardcoded for tilt in devnet_setup.sh
+        }),
+      ])
+      .rpc();
+
+  const testTopLevel = async (
+    program: Program<WormholePostMessageShim>,
+    msg: string,
+    expectedSeq: bigint,
+    consistencyLevel: number,
+    testDescription: string
+  ) => {
+    const tx = await postMessage(program, msg, consistencyLevel);
+    const emitter = program.provider.publicKey.toBuffer().toString("hex");
+    const seq = await getSequenceFromTx(tx).then((evt) =>
+      evt.sequence.toString()
+    );
+
+    const { vaaBytes } = await getSignedVAAWithRetry(
+      [GUARDIAN_URL],
+      1,
+      emitter,
+      seq,
+      {
+        transport: NodeHttpTransport(),
+      },
+      500, // every .5 secs
+      100 // 100 times, or 50 seconds
+    );
+
+    const vaa = parseVaa(vaaBytes);
+    if (
+      vaa.sequence === expectedSeq &&
+      vaa.consistencyLevel === consistencyLevel &&
+      vaa.payload.equals(Buffer.from(msg, "ascii"))
+    ) {
+      console.log(`✅ ${testDescription} success!`);
+    } else {
+      throw new Error(`❌ ${testDescription} failed!`);
+    }
+  };
+
+  const testInner = async (
+    program: Program<WormholeIntegratorExample>,
+    expectedSeq: bigint,
+    testDescription: string
+  ) => {
+    const postShimProgram = new Program<WormholePostMessageShim>(
+      WormholePostMessageShimIdl as WormholePostMessageShim
+    );
+    const emitterBuf = web3.PublicKey.findProgramAddressSync(
+      [Buffer.from("emitter")],
+      program.programId
+    )[0].toBuffer();
+    const tx = await program.methods
+      .postMessage()
+      .accounts({
+        // sequence: web3.PublicKey.findProgramAddressSync(
+        //   [Buffer.from("Sequence"), emitterBuf],
+        //   new web3.PublicKey("Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o")
+        // )[0],
+        wormholePostMessageShimEa: web3.PublicKey.findProgramAddressSync(
+          [Buffer.from("__event_authority")],
+          postShimProgram.programId
+        )[0],
+      })
+      .preInstructions([
+        // gotta pay the fee
+        web3.SystemProgram.transfer({
+          fromPubkey: program.provider.publicKey,
+          toPubkey: new web3.PublicKey(
+            "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs"
+          ), // fee collector
+          lamports: 100, // hardcoded for tilt in devnet_setup.sh
+        }),
+      ])
+      .rpc();
+    const emitter = emitterBuf.toString("hex");
+    const seq = await getSequenceFromTx(tx).then((evt) =>
+      evt.sequence.toString()
+    );
+
+    const { vaaBytes } = await getSignedVAAWithRetry(
+      [GUARDIAN_URL],
+      1,
+      emitter,
+      seq,
+      {
+        transport: NodeHttpTransport(),
+      },
+      500, // every .5 secs
+      100 // 100 times, or 50 seconds
+    );
+
+    const vaa = parseVaa(vaaBytes);
+    if (
+      vaa.sequence === expectedSeq &&
+      // ../programs/wormhole-integrator-example/src/instructions/post_message.rs
+      vaa.consistencyLevel === 1 &&
+      vaa.payload.equals(Buffer.from("your message goes here!", "ascii"))
+    ) {
+      console.log(`✅ ${testDescription} success!`);
+    } else {
+      throw new Error(`❌ ${testDescription} failed!`);
+    }
+  };
+
+  {
+    const program = await getShimProgramWithNewSigner();
+    await testTopLevel(
+      program,
+      "hello world",
+      BigInt(0),
+      0,
+      "Top level initial message, confirmed"
+    );
+    await testTopLevel(
+      program,
+      "hello everyone",
+      BigInt(1),
+      0,
+      "Top level subsequent message, confirmed"
+    );
+  }
+  {
+    const program = await getShimProgramWithNewSigner();
+    await testTopLevel(
+      program,
+      "hello here",
+      BigInt(0),
+      1,
+      "Top level initial message, finalized"
+    );
+    await testTopLevel(
+      program,
+      "hello there",
+      BigInt(1),
+      1,
+      "Top level subsequent message, finalized"
+    );
+  }
+  {
+    const program = await getIntegrationProgramWithNewSigner();
+    let currentSequence = BigInt(0);
+    try {
+      currentSequence = (
+        await getSequenceTracker(
+          connection,
+          web3.PublicKey.findProgramAddressSync(
+            [Buffer.from("emitter")],
+            program.programId
+          )[0],
+          new web3.PublicKey("Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o")
+        )
+      ).sequence;
+    } catch (e) {}
+    await testInner(
+      program,
+      currentSequence,
+      "Integration initial message, finalized"
+    );
+    await testInner(
+      program,
+      currentSequence + BigInt(1),
+      "Integration subsequent message, finalized"
+    );
+  }
+})();

+ 61 - 0
svm/wormhole-core-shims/anchor/tests/helpers.ts

@@ -0,0 +1,61 @@
+import * as anchor from "@coral-xyz/anchor";
+import { bs58 } from "@coral-xyz/anchor/dist/cjs/utils/bytes";
+import { PublicKey } from "@solana/web3.js";
+import { expect } from "chai";
+import WormholePostMessageShimIdl from "../idls/wormhole_post_message_shim.json";
+
+export async function getSequenceFromTx(
+  tx: string
+): Promise<{ emitter: PublicKey; sequence: bigint }> {
+  const txDetails = await getTransactionDetails(tx);
+
+  const borshEventCoder = new anchor.BorshEventCoder(
+    WormholePostMessageShimIdl as any
+  );
+
+  const innerInstructions = txDetails.meta.innerInstructions[0].instructions;
+
+  // Get the last instruction from the inner instructions
+  const lastInstruction = innerInstructions[innerInstructions.length - 1];
+
+  // Decode the Base58 encoded data
+  const decodedData = bs58.decode(lastInstruction.data);
+
+  // Remove the instruction discriminator and re-encode the rest as Base58
+  const eventData = Buffer.from(decodedData.subarray(8)).toString("base64");
+
+  const borshEvents = borshEventCoder.decode(eventData);
+  expect(txDetails.blockTime).is.not.undefined;
+  expect(borshEvents.data.submission_time).to.equal(txDetails.blockTime);
+
+  return {
+    emitter: borshEvents.data.emitter,
+    sequence: BigInt(borshEvents.data.sequence.toString()),
+  };
+}
+
+export async function getTransactionDetails(
+  tx: string
+): Promise<anchor.web3.VersionedTransactionResponse> {
+  let txDetails: anchor.web3.VersionedTransactionResponse | null = null;
+  while (!txDetails) {
+    txDetails = await anchor.getProvider().connection.getTransaction(tx, {
+      maxSupportedTransactionVersion: 0,
+      commitment: "confirmed",
+    });
+  }
+  return txDetails;
+}
+
+export async function logCostAndCompute(method: string, tx: string) {
+  const SOL_PRICE = 217.54; // 2025-01-03
+  const txDetails = await getTransactionDetails(tx);
+  const lamports =
+    txDetails.meta.preBalances[0] - txDetails.meta.postBalances[0];
+  const sol = lamports / 1_000_000_000;
+  console.log(
+    `${method}: lamports ${lamports} SOL ${sol}, $${sol * SOL_PRICE}, CU ${
+      txDetails.meta.computeUnitsConsumed
+    }, tx https://explorer.solana.com/tx/${tx}?cluster=custom&customUrl=http%3A%2F%2Flocalhost%3A8899`
+  );
+}

+ 407 - 0
svm/wormhole-core-shims/anchor/tests/idls/devnet/wormhole_integrator_example.json

@@ -0,0 +1,407 @@
+{
+  "address": "AEwubmehHNvkMXoH2C5MgDSemZgQ3HUSYpeaF3UrNZdQ",
+  "metadata": {
+    "name": "wormhole_integrator_example",
+    "version": "0.1.0",
+    "spec": "0.1.0",
+    "description": "Created with Anchor"
+  },
+  "instructions": [
+    {
+      "name": "consume_vaa",
+      "docs": [
+        "This example instruction takes the guardian signatures account previously posted to the verify vaa shim",
+        "with `post_signatures` and the corresponding guardian set from the core bridge and verifies a",
+        "provided VAA body against it. `close_signatures` on the shim should be called immediately",
+        "afterwards in order to reclaim the rent lamports taken by `post_signatures`."
+      ],
+      "discriminator": [
+        224,
+        143,
+        180,
+        192,
+        139,
+        120,
+        177,
+        63
+      ],
+      "accounts": [
+        {
+          "name": "guardian_set",
+          "docs": [
+            "Derivation is checked by the shim."
+          ]
+        },
+        {
+          "name": "guardian_signatures",
+          "docs": [
+            "Ownership ownership and discriminator is checked by the shim."
+          ]
+        },
+        {
+          "name": "wormhole_verify_vaa_shim",
+          "address": "EFaNWErqAtVWufdNb7yofSHHfWFos843DFpu4JBw24at"
+        }
+      ],
+      "args": [
+        {
+          "name": "guardian_set_bump",
+          "type": "u8"
+        },
+        {
+          "name": "vaa_body",
+          "type": "bytes"
+        }
+      ]
+    },
+    {
+      "name": "initialize",
+      "docs": [
+        "This example instruction posts an empty message during initialize in order to have one less",
+        "CPI depth on subsequent posts.",
+        "",
+        "NOTE: this example does not replay protect the call. Typically, this may be done with an",
+        "`init` constraint on an account that was used to store config that is also set up during initialization."
+      ],
+      "discriminator": [
+        175,
+        175,
+        109,
+        31,
+        13,
+        152,
+        155,
+        237
+      ],
+      "accounts": [
+        {
+          "name": "payer",
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "deployer",
+          "signer": true
+        },
+        {
+          "name": "program_data",
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  137,
+                  75,
+                  208,
+                  247,
+                  32,
+                  141,
+                  13,
+                  249,
+                  158,
+                  80,
+                  9,
+                  219,
+                  168,
+                  206,
+                  79,
+                  113,
+                  223,
+                  183,
+                  165,
+                  86,
+                  124,
+                  110,
+                  87,
+                  91,
+                  73,
+                  217,
+                  240,
+                  163,
+                  59,
+                  193,
+                  96,
+                  119
+                ]
+              }
+            ]
+          }
+        },
+        {
+          "name": "wormhole_post_message_shim",
+          "address": "EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX"
+        },
+        {
+          "name": "bridge",
+          "writable": true,
+          "address": "FKoMTctsC7vJbEqyRiiPskPnuQx2tX1kurmvWByq5uZP"
+        },
+        {
+          "name": "message",
+          "docs": [
+            "This program uses a PDA per emitter, since these are already bottle-necked by sequence and",
+            "the bridge enforces that emitter must be identical for reused accounts.",
+            "While this could be managed by the integrator, it seems more effective to have the shim manage these accounts.",
+            "Bonus, this also allows Anchor to automatically handle deriving the address."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "account",
+                "path": "emitter"
+              }
+            ],
+            "program": {
+              "kind": "const",
+              "value": [
+                206,
+                93,
+                34,
+                116,
+                131,
+                143,
+                202,
+                41,
+                198,
+                209,
+                143,
+                152,
+                10,
+                211,
+                213,
+                245,
+                235,
+                78,
+                129,
+                210,
+                121,
+                29,
+                243,
+                98,
+                128,
+                136,
+                144,
+                147,
+                38,
+                68,
+                208,
+                24
+              ]
+            }
+          }
+        },
+        {
+          "name": "emitter",
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  101,
+                  109,
+                  105,
+                  116,
+                  116,
+                  101,
+                  114
+                ]
+              }
+            ]
+          }
+        },
+        {
+          "name": "sequence",
+          "docs": [
+            "Explicitly do not re-derive this account. The core bridge verifies the derivation anyway and",
+            "as of Anchor 0.30.1, auto-derivation for other programs' accounts via IDL doesn't work."
+          ],
+          "writable": true
+        },
+        {
+          "name": "fee_collector",
+          "writable": true,
+          "address": "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs"
+        },
+        {
+          "name": "clock",
+          "docs": [
+            "Clock sysvar."
+          ],
+          "address": "SysvarC1ock11111111111111111111111111111111"
+        },
+        {
+          "name": "system_program",
+          "docs": [
+            "System program."
+          ],
+          "address": "11111111111111111111111111111111"
+        },
+        {
+          "name": "wormhole_program",
+          "address": "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o"
+        },
+        {
+          "name": "wormhole_post_message_shim_ea"
+        }
+      ],
+      "args": []
+    },
+    {
+      "name": "post_message",
+      "docs": [
+        "This example instruction posts a message via the post message shim."
+      ],
+      "discriminator": [
+        214,
+        50,
+        100,
+        209,
+        38,
+        34,
+        7,
+        76
+      ],
+      "accounts": [
+        {
+          "name": "payer",
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "wormhole_post_message_shim",
+          "address": "EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX"
+        },
+        {
+          "name": "bridge",
+          "docs": [
+            "Address constraint added for IDL generation / convenience, it will be enforced by the core bridge."
+          ],
+          "writable": true,
+          "address": "FKoMTctsC7vJbEqyRiiPskPnuQx2tX1kurmvWByq5uZP"
+        },
+        {
+          "name": "message",
+          "docs": [
+            "Seeds constraint added for IDL generation / convenience, it will be enforced by the shim."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "account",
+                "path": "emitter"
+              }
+            ],
+            "program": {
+              "kind": "const",
+              "value": [
+                206,
+                93,
+                34,
+                116,
+                131,
+                143,
+                202,
+                41,
+                198,
+                209,
+                143,
+                152,
+                10,
+                211,
+                213,
+                245,
+                235,
+                78,
+                129,
+                210,
+                121,
+                29,
+                243,
+                98,
+                128,
+                136,
+                144,
+                147,
+                38,
+                68,
+                208,
+                24
+              ]
+            }
+          }
+        },
+        {
+          "name": "emitter",
+          "docs": [
+            "Seeds constraint added for IDL generation / convenience, it will be enforced to match the signer used in the CPI call."
+          ],
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  101,
+                  109,
+                  105,
+                  116,
+                  116,
+                  101,
+                  114
+                ]
+              }
+            ]
+          }
+        },
+        {
+          "name": "sequence",
+          "docs": [
+            "Explicitly do not re-derive this account. The core bridge verifies the derivation anyway and",
+            "as of Anchor 0.30.1, auto-derivation for other programs' accounts via IDL doesn't work."
+          ],
+          "writable": true
+        },
+        {
+          "name": "fee_collector",
+          "docs": [
+            "Address constraint added for IDL generation / convenience, it will be enforced by the core bridge."
+          ],
+          "writable": true,
+          "address": "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs"
+        },
+        {
+          "name": "clock",
+          "docs": [
+            "Clock sysvar.",
+            "Type added for IDL generation / convenience, it will be enforced by the core bridge."
+          ],
+          "address": "SysvarC1ock11111111111111111111111111111111"
+        },
+        {
+          "name": "system_program",
+          "docs": [
+            "System program.",
+            "Type for IDL generation / convenience, it will be enforced by the core bridge."
+          ],
+          "address": "11111111111111111111111111111111"
+        },
+        {
+          "name": "wormhole_program",
+          "docs": [
+            "Address constraint added for IDL generation / convenience, it will be enforced by the shim."
+          ],
+          "address": "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o"
+        },
+        {
+          "name": "wormhole_post_message_shim_ea",
+          "docs": [
+            "TODO: An address constraint could be included if this address was published to wormhole_solana_consts",
+            "Address will be enforced by the shim."
+          ]
+        }
+      ],
+      "args": []
+    }
+  ]
+}

+ 414 - 0
svm/wormhole-core-shims/anchor/tests/idls/devnet/wormhole_integrator_example.ts

@@ -0,0 +1,414 @@
+/**
+ * Program IDL in camelCase format in order to be used in JS/TS.
+ *
+ * Note that this is only a type helper and is not the actual IDL. The original
+ * IDL can be found at `target/idl/wormhole_integrator_example.json`.
+ */
+export type WormholeIntegratorExample = {
+  "address": "AEwubmehHNvkMXoH2C5MgDSemZgQ3HUSYpeaF3UrNZdQ",
+  "metadata": {
+    "name": "wormholeIntegratorExample",
+    "version": "0.1.0",
+    "spec": "0.1.0",
+    "description": "Created with Anchor"
+  },
+  "instructions": [
+    {
+      "name": "consumeVaa",
+      "docs": [
+        "This example instruction takes the guardian signatures account previously posted to the verify vaa shim",
+        "with `post_signatures` and the corresponding guardian set from the core bridge and verifies a",
+        "provided VAA body against it. `close_signatures` on the shim should be called immediately",
+        "afterwards in order to reclaim the rent lamports taken by `post_signatures`."
+      ],
+      "discriminator": [
+        224,
+        143,
+        180,
+        192,
+        139,
+        120,
+        177,
+        63
+      ],
+      "accounts": [
+        {
+          "name": "guardianSet",
+          "docs": [
+            "Derivation is checked by the shim."
+          ]
+        },
+        {
+          "name": "guardianSignatures",
+          "docs": [
+            "Ownership ownership and discriminator is checked by the shim."
+          ]
+        },
+        {
+          "name": "wormholeVerifyVaaShim",
+          "address": "EFaNWErqAtVWufdNb7yofSHHfWFos843DFpu4JBw24at"
+        }
+      ],
+      "args": [
+        {
+          "name": "guardianSetBump",
+          "type": "u8"
+        },
+        {
+          "name": "vaaBody",
+          "type": "bytes"
+        }
+      ]
+    },
+    {
+      "name": "initialize",
+      "docs": [
+        "This example instruction posts an empty message during initialize in order to have one less",
+        "CPI depth on subsequent posts.",
+        "",
+        "NOTE: this example does not replay protect the call. Typically, this may be done with an",
+        "`init` constraint on an account that was used to store config that is also set up during initialization."
+      ],
+      "discriminator": [
+        175,
+        175,
+        109,
+        31,
+        13,
+        152,
+        155,
+        237
+      ],
+      "accounts": [
+        {
+          "name": "payer",
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "deployer",
+          "signer": true
+        },
+        {
+          "name": "programData",
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  137,
+                  75,
+                  208,
+                  247,
+                  32,
+                  141,
+                  13,
+                  249,
+                  158,
+                  80,
+                  9,
+                  219,
+                  168,
+                  206,
+                  79,
+                  113,
+                  223,
+                  183,
+                  165,
+                  86,
+                  124,
+                  110,
+                  87,
+                  91,
+                  73,
+                  217,
+                  240,
+                  163,
+                  59,
+                  193,
+                  96,
+                  119
+                ]
+              }
+            ]
+          }
+        },
+        {
+          "name": "wormholePostMessageShim",
+          "address": "EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX"
+        },
+        {
+          "name": "bridge",
+          "writable": true,
+          "address": "FKoMTctsC7vJbEqyRiiPskPnuQx2tX1kurmvWByq5uZP"
+        },
+        {
+          "name": "message",
+          "docs": [
+            "This program uses a PDA per emitter, since these are already bottle-necked by sequence and",
+            "the bridge enforces that emitter must be identical for reused accounts.",
+            "While this could be managed by the integrator, it seems more effective to have the shim manage these accounts.",
+            "Bonus, this also allows Anchor to automatically handle deriving the address."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "account",
+                "path": "emitter"
+              }
+            ],
+            "program": {
+              "kind": "const",
+              "value": [
+                206,
+                93,
+                34,
+                116,
+                131,
+                143,
+                202,
+                41,
+                198,
+                209,
+                143,
+                152,
+                10,
+                211,
+                213,
+                245,
+                235,
+                78,
+                129,
+                210,
+                121,
+                29,
+                243,
+                98,
+                128,
+                136,
+                144,
+                147,
+                38,
+                68,
+                208,
+                24
+              ]
+            }
+          }
+        },
+        {
+          "name": "emitter",
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  101,
+                  109,
+                  105,
+                  116,
+                  116,
+                  101,
+                  114
+                ]
+              }
+            ]
+          }
+        },
+        {
+          "name": "sequence",
+          "docs": [
+            "Explicitly do not re-derive this account. The core bridge verifies the derivation anyway and",
+            "as of Anchor 0.30.1, auto-derivation for other programs' accounts via IDL doesn't work."
+          ],
+          "writable": true
+        },
+        {
+          "name": "feeCollector",
+          "writable": true,
+          "address": "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs"
+        },
+        {
+          "name": "clock",
+          "docs": [
+            "Clock sysvar."
+          ],
+          "address": "SysvarC1ock11111111111111111111111111111111"
+        },
+        {
+          "name": "systemProgram",
+          "docs": [
+            "System program."
+          ],
+          "address": "11111111111111111111111111111111"
+        },
+        {
+          "name": "wormholeProgram",
+          "address": "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o"
+        },
+        {
+          "name": "wormholePostMessageShimEa"
+        }
+      ],
+      "args": []
+    },
+    {
+      "name": "postMessage",
+      "docs": [
+        "This example instruction posts a message via the post message shim."
+      ],
+      "discriminator": [
+        214,
+        50,
+        100,
+        209,
+        38,
+        34,
+        7,
+        76
+      ],
+      "accounts": [
+        {
+          "name": "payer",
+          "writable": true,
+          "signer": true
+        },
+        {
+          "name": "wormholePostMessageShim",
+          "address": "EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX"
+        },
+        {
+          "name": "bridge",
+          "docs": [
+            "Address constraint added for IDL generation / convenience, it will be enforced by the core bridge."
+          ],
+          "writable": true,
+          "address": "FKoMTctsC7vJbEqyRiiPskPnuQx2tX1kurmvWByq5uZP"
+        },
+        {
+          "name": "message",
+          "docs": [
+            "Seeds constraint added for IDL generation / convenience, it will be enforced by the shim."
+          ],
+          "writable": true,
+          "pda": {
+            "seeds": [
+              {
+                "kind": "account",
+                "path": "emitter"
+              }
+            ],
+            "program": {
+              "kind": "const",
+              "value": [
+                206,
+                93,
+                34,
+                116,
+                131,
+                143,
+                202,
+                41,
+                198,
+                209,
+                143,
+                152,
+                10,
+                211,
+                213,
+                245,
+                235,
+                78,
+                129,
+                210,
+                121,
+                29,
+                243,
+                98,
+                128,
+                136,
+                144,
+                147,
+                38,
+                68,
+                208,
+                24
+              ]
+            }
+          }
+        },
+        {
+          "name": "emitter",
+          "docs": [
+            "Seeds constraint added for IDL generation / convenience, it will be enforced to match the signer used in the CPI call."
+          ],
+          "pda": {
+            "seeds": [
+              {
+                "kind": "const",
+                "value": [
+                  101,
+                  109,
+                  105,
+                  116,
+                  116,
+                  101,
+                  114
+                ]
+              }
+            ]
+          }
+        },
+        {
+          "name": "sequence",
+          "docs": [
+            "Explicitly do not re-derive this account. The core bridge verifies the derivation anyway and",
+            "as of Anchor 0.30.1, auto-derivation for other programs' accounts via IDL doesn't work."
+          ],
+          "writable": true
+        },
+        {
+          "name": "feeCollector",
+          "docs": [
+            "Address constraint added for IDL generation / convenience, it will be enforced by the core bridge."
+          ],
+          "writable": true,
+          "address": "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs"
+        },
+        {
+          "name": "clock",
+          "docs": [
+            "Clock sysvar.",
+            "Type added for IDL generation / convenience, it will be enforced by the core bridge."
+          ],
+          "address": "SysvarC1ock11111111111111111111111111111111"
+        },
+        {
+          "name": "systemProgram",
+          "docs": [
+            "System program.",
+            "Type for IDL generation / convenience, it will be enforced by the core bridge."
+          ],
+          "address": "11111111111111111111111111111111"
+        },
+        {
+          "name": "wormholeProgram",
+          "docs": [
+            "Address constraint added for IDL generation / convenience, it will be enforced by the shim."
+          ],
+          "address": "Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o"
+        },
+        {
+          "name": "wormholePostMessageShimEa",
+          "docs": [
+            "TODO: An address constraint could be included if this address was published to wormhole_solana_consts",
+            "Address will be enforced by the shim."
+          ]
+        }
+      ],
+      "args": []
+    }
+  ]
+};
+

+ 155 - 0
svm/wormhole-core-shims/anchor/tests/wormhole-integrator-example.ts

@@ -0,0 +1,155 @@
+import {
+  parseVaa,
+  SignedVaa,
+} from "@certusone/wormhole-sdk/lib/cjs/vaa/wormhole";
+import * as anchor from "@coral-xyz/anchor";
+import { Program } from "@coral-xyz/anchor";
+import { WormholeIntegratorExample } from "../target/types/wormhole_integrator_example";
+import { WormholePostMessageShim } from "../idls/wormhole_post_message_shim";
+import { WormholeVerifyVaaShim } from "../idls/wormhole_verify_vaa_shim";
+import { logCostAndCompute } from "./helpers";
+
+// VAA from https://wormholescan.io/#/tx/AEa98mf68bcwUmT8Yheidw4C4KUVSG9732SVg5kqnfmH?view=advanced
+const VAA =
+  "AQAAAAQNAL1qji7v9KnngyX0VxK+3fCMVscWTLoYX8L48NWquq2WGrcHd4H0wYc0KF4ZOWjLD2okXoBjGQIDJzx4qIrbSzQBAQq69h+neXGb58VfhZgraPVCxJmnTj8JIDq5jqi3Qav1e+IW51mIJlOhSAdCRbEyQLzf6Z3C19WJJqSyt/z1XF0AAvFgDHkseyMZTE5vQjflu4tc5OLPJe2VYCxTJT15LA02YPrWgOM6HhfUhXDhFoG5AI/s2ApjK8jaqi7LGJILAUMBA6cp4vfko8hYyRvogqQWsdk9e20g0O6s60h4ewweapXCQHerQpoJYdDxlCehN4fuYnuudEhW+6FaXLjwNJBdqsoABDg9qXjXB47nBVCZAGns2eosVqpjkyDaCfo/p1x8AEjBA80CyC1/QlbG9L4zlnnDIfZWylsf3keJqx28+fZNC5oABi6XegfozgE8JKqvZLvd7apDhrJ6Qv+fMiynaXASkafeVJOqgFOFbCMXdMKehD38JXvz3JrlnZ92E+I5xOJaDVgABzDSte4mxUMBMJB9UUgJBeAVsokFvK4DOfvh6G3CVqqDJplLwmjUqFB7fAgRfGcA8PWNStRc+YDZiG66YxPnptwACe84S31Kh9voz2xRk1THMpqHQ4fqE7DizXPNWz6Z6ebEXGcd7UP9PBXoNNvjkLWZJZOdbkZyZqztaIiAo4dgWUABCobiuQP92WjTxOZz0KhfWVJ3YBVfsXUwaVQH4/p6khX0HCEVHR9VHmjvrAAGDMdJGWW+zu8mFQc4gPU6m4PZ6swADO7voA5GWZZPiztz22pftwxKINGvOjCPlLpM1Y2+Vq6AQuez/mlUAmaL0NKgs+5VYcM1SGBz0TL3ABRhKQAhUEMADWmiMo0J1Qaj8gElb+9711ZjvAY663GIyG/E6EdPW+nPKJI9iZE180sLct+krHj0J7PlC9BjDiO2y149oCOJ6FgAEcaVkYK43EpN7XqxrdpanX6R6TaqECgZTjvtN3L6AP2ceQr8mJJraYq+qY8pTfFvPKEqmW9CBYvnA5gIMpX59WsAEjIL9Hdnx+zFY0qSPB1hB9AhqWeBP/QfJjqzqafsczaeCN/rWUf6iNBgXI050ywtEp8JQ36rCn8w6dRhUusn+MEAZ32XyAAAAAAAFczO6yk0j3G90i/+9DoqGcH1teF8XMpUEVKRIBgmcq3lAAAAAAAC/1wAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAC6Q7dAAAAAAAAAAAAAAAAAAoLhpkcYhizbB0Z1KLp6wzjYG60gAAgAAAAAAAAAAAAAAAInNTEvk5b/1WVF+JawF1smtAdicABAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA==";
+const CORE_BRIDGE_PROGRAM_ID = new anchor.web3.PublicKey(
+  "worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth"
+);
+const GUARDIAN_SET_SEED = "GuardianSet";
+
+describe("wormhole-integrator-example", () => {
+  // Configure the client to use the local cluster.
+  anchor.setProvider(anchor.AnchorProvider.env());
+
+  const program = anchor.workspace
+    .WormholeIntegratorExample as Program<WormholeIntegratorExample>;
+  const verifyShimProgram = anchor.workspace
+    .WormholeVerifyVaaShim as Program<WormholeVerifyVaaShim>;
+  const postShimProgram = anchor.workspace
+    .WormholePostMessageShim as Program<WormholePostMessageShim>;
+
+  it("Initializes!", async () => {
+    const tx = await program.methods
+      .initialize()
+      .accountsPartial({
+        sequence: anchor.web3.PublicKey.findProgramAddressSync(
+          [
+            Buffer.from("Sequence"),
+            anchor.web3.PublicKey.findProgramAddressSync(
+              [Buffer.from("emitter")],
+              program.programId
+            )[0].toBuffer(),
+          ],
+          CORE_BRIDGE_PROGRAM_ID
+        )[0],
+        wormholePostMessageShimEa: anchor.web3.PublicKey.findProgramAddressSync(
+          [Buffer.from("__event_authority")],
+          postShimProgram.programId
+        )[0],
+        programData: anchor.web3.PublicKey.findProgramAddressSync(
+          [program.programId.toBuffer()],
+          new anchor.web3.PublicKey(
+            "BPFLoaderUpgradeab1e11111111111111111111111"
+          )
+        )[0],
+      })
+      .preInstructions([
+        // gotta pay the fee
+        anchor.web3.SystemProgram.transfer({
+          fromPubkey: program.provider.publicKey,
+          toPubkey: new anchor.web3.PublicKey(
+            "9bFNrXNb2WTx8fMHXCheaZqkLZ3YCCaiqTftHxeintHy"
+          ), // fee collector
+          lamports: 100, // hardcoded for tilt in devnet_setup.sh
+        }),
+      ])
+      .rpc();
+    await logCostAndCompute("init", tx);
+  });
+
+  it("Posts a message!", async () => {
+    const tx = await program.methods
+      .postMessage()
+      .accounts({
+        sequence: anchor.web3.PublicKey.findProgramAddressSync(
+          [
+            Buffer.from("Sequence"),
+            anchor.web3.PublicKey.findProgramAddressSync(
+              [Buffer.from("emitter")],
+              program.programId
+            )[0].toBuffer(),
+          ],
+          CORE_BRIDGE_PROGRAM_ID
+        )[0],
+        wormholePostMessageShimEa: anchor.web3.PublicKey.findProgramAddressSync(
+          [Buffer.from("__event_authority")],
+          postShimProgram.programId
+        )[0],
+      })
+      .preInstructions([
+        // gotta pay the fee
+        anchor.web3.SystemProgram.transfer({
+          fromPubkey: program.provider.publicKey,
+          toPubkey: new anchor.web3.PublicKey(
+            "9bFNrXNb2WTx8fMHXCheaZqkLZ3YCCaiqTftHxeintHy"
+          ), // fee collector
+          lamports: 100, // hardcoded for tilt in devnet_setup.sh
+        }),
+      ])
+      .rpc();
+    await logCostAndCompute("shim post", tx);
+  });
+
+  it("Consumes a VAA!", async () => {
+    const signatureKeypair = anchor.web3.Keypair.generate();
+    const buf = Buffer.from(VAA, "base64");
+    const vaa = parseVaa(buf);
+    const tx = await verifyShimProgram.methods
+      .postSignatures(
+        vaa.guardianSetIndex,
+        vaa.guardianSignatures.length,
+        vaa.guardianSignatures.map((s) => [s.index, ...s.signature])
+      )
+      .accounts({ guardianSignatures: signatureKeypair.publicKey })
+      .signers([signatureKeypair])
+      .rpc();
+    await logCostAndCompute("shim verify (1/2)", tx);
+
+    // Convert guardian_set_index to big-endian bytes
+    const guardianSetIndex = vaa.guardianSetIndex;
+    const indexBuffer = Buffer.alloc(4); // guardian_set_index is a u32
+    indexBuffer.writeUInt32BE(guardianSetIndex);
+    const [guardianSet, guardianSetBump] =
+      anchor.web3.PublicKey.findProgramAddressSync(
+        [Buffer.from(GUARDIAN_SET_SEED), indexBuffer],
+        CORE_BRIDGE_PROGRAM_ID
+      );
+    const tx2 = await program.methods
+      .consumeVaa(guardianSetBump, vaaBody(buf))
+      .accountsPartial({
+        guardianSignatures: signatureKeypair.publicKey,
+        guardianSet,
+      })
+      .preInstructions([
+        anchor.web3.ComputeBudgetProgram.setComputeUnitLimit({
+          units: 420_000,
+        }),
+      ])
+      .postInstructions([
+        await verifyShimProgram.methods
+          .closeSignatures()
+          .accounts({ guardianSignatures: signatureKeypair.publicKey })
+          .instruction(),
+      ])
+      .rpc();
+    await logCostAndCompute("shim verify (2/2)", tx2);
+  });
+});
+
+function vaaBody(vaa: SignedVaa): Buffer {
+  const signedVaa = Buffer.isBuffer(vaa) ? vaa : Buffer.from(vaa as Uint8Array);
+  const sigStart = 6;
+  const numSigners = signedVaa[5];
+  const sigLength = 66;
+  return signedVaa.subarray(sigStart + sigLength * numSigners);
+}

+ 168 - 0
svm/wormhole-core-shims/anchor/tests/wormhole-post-message-shim.ts

@@ -0,0 +1,168 @@
+import * as anchor from "@coral-xyz/anchor";
+import { Program } from "@coral-xyz/anchor";
+import { expect } from "chai";
+import { WormholePostMessageShim } from "../idls/wormhole_post_message_shim";
+import { getSequenceFromTx, logCostAndCompute } from "./helpers";
+
+const CORE_BRIDGE_PROGRAM_ID = new anchor.web3.PublicKey(
+  "worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth"
+);
+
+describe("wormhole-post-message-shim", () => {
+  // Configure the client to use the local cluster.
+  anchor.setProvider(anchor.AnchorProvider.env());
+
+  const program = anchor.workspace
+    .WormholePostMessageShim as Program<WormholePostMessageShim>;
+
+  const postMessage = async (msg: string): Promise<string> =>
+    await program.methods
+      .postMessage(0, { confirmed: {} }, Buffer.from(msg, "ascii"))
+      .accounts({
+        emitter: program.provider.publicKey,
+        wormholeProgram: CORE_BRIDGE_PROGRAM_ID,
+        // sequence: anchor.web3.PublicKey.findProgramAddressSync(
+        //   [Buffer.from("Sequence"), program.provider.publicKey.toBuffer()],
+        //   new anchor.web3.PublicKey(
+        //     "worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth"
+        //   )
+        // )[0],
+        // these are needed if removing the address checks
+        // bridge: new anchor.web3.PublicKey(
+        //   "FKoMTctsC7vJbEqyRiiPskPnuQx2tX1kurmvWByq5uZP"
+        // ),
+        // feeCollector: new anchor.web3.PublicKey(
+        //   "GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs"
+        // ),
+        // clock: new anchor.web3.PublicKey(
+        //   "SysvarC1ock11111111111111111111111111111111"
+        // ),
+      })
+      .preInstructions([
+        // gotta pay the fee
+        anchor.web3.SystemProgram.transfer({
+          fromPubkey: program.provider.publicKey,
+          toPubkey: new anchor.web3.PublicKey(
+            "9bFNrXNb2WTx8fMHXCheaZqkLZ3YCCaiqTftHxeintHy"
+          ), // fee collector
+          lamports: 100, // hardcoded for tilt in devnet_setup.sh
+        }),
+      ])
+      .rpc();
+
+  it("Posts a message!", async () => {
+    const tx = await postMessage("hello world");
+    console.log("Your transaction signature", tx);
+    const evt = await getSequenceFromTx(tx);
+    expect(evt).to.eql({
+      emitter: program.provider.publicKey,
+      sequence: BigInt(0),
+    });
+  });
+  it("Posts a second message!", async () => {
+    const tx = await postMessage("hello world");
+    console.log("Your transaction signature", tx);
+    const evt = await getSequenceFromTx(tx);
+    expect(evt).to.eql({
+      emitter: program.provider.publicKey,
+      sequence: BigInt(1),
+    });
+  });
+  it("Compares core post_message to shim post_message!", async () => {
+    {
+      const acct = new anchor.web3.Keypair();
+      const data = Buffer.from(
+        "01000000000b00000068656c6c6f20776f726c6400",
+        "hex"
+      );
+      const transaction = new anchor.web3.Transaction();
+      transaction.add(
+        anchor.web3.SystemProgram.transfer({
+          fromPubkey: program.provider.publicKey,
+          toPubkey: new anchor.web3.PublicKey(
+            "9bFNrXNb2WTx8fMHXCheaZqkLZ3YCCaiqTftHxeintHy"
+          ), // fee collector
+          lamports: 100, // hardcoded for tilt in devnet_setup.sh
+        })
+      );
+      transaction.add(
+        new anchor.web3.TransactionInstruction({
+          keys: [
+            {
+              // config
+              isSigner: false,
+              isWritable: true,
+              pubkey: new anchor.web3.PublicKey(
+                "2yVjuQwpsvdsrywzsJJVs9Ueh4zayyo5DYJbBNc3DDpn"
+              ),
+            },
+            {
+              // message
+              isSigner: true,
+              isWritable: true,
+              pubkey: acct.publicKey,
+            },
+            {
+              // emitter
+              isSigner: true,
+              isWritable: false,
+              pubkey: program.provider.publicKey,
+            },
+            {
+              // sequence
+              isSigner: false,
+              isWritable: true,
+              pubkey: anchor.web3.PublicKey.findProgramAddressSync(
+                [
+                  Buffer.from("Sequence"),
+                  program.provider.publicKey.toBuffer(),
+                ],
+                CORE_BRIDGE_PROGRAM_ID
+              )[0],
+            },
+            {
+              // payer
+              isSigner: true,
+              isWritable: true,
+              pubkey: program.provider.publicKey,
+            },
+            {
+              // fee collector
+              isSigner: false,
+              isWritable: true,
+              pubkey: new anchor.web3.PublicKey(
+                "9bFNrXNb2WTx8fMHXCheaZqkLZ3YCCaiqTftHxeintHy"
+              ),
+            },
+            {
+              // clock
+              isSigner: false,
+              isWritable: false,
+              pubkey: new anchor.web3.PublicKey(
+                "SysvarC1ock11111111111111111111111111111111"
+              ),
+            },
+            {
+              // system program
+              isSigner: false,
+              isWritable: false,
+              pubkey: new anchor.web3.PublicKey(
+                "11111111111111111111111111111111"
+              ),
+            },
+          ],
+          programId: new anchor.web3.PublicKey(
+            "worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth"
+          ),
+          data,
+        })
+      );
+      const tx = await program.provider.sendAndConfirm(transaction, [acct]);
+      await logCostAndCompute("core", tx);
+    }
+    {
+      const tx = await postMessage("hello world");
+      await logCostAndCompute("shim", tx);
+    }
+  });
+});

Diff do ficheiro suprimidas por serem muito extensas
+ 12 - 0
svm/wormhole-core-shims/anchor/tests/wormhole-vaa-verification-comparison.ts


+ 68 - 0
svm/wormhole-core-shims/anchor/tests/wormhole-verify-vaa-shim.ts

@@ -0,0 +1,68 @@
+import { keccak256 } from "@certusone/wormhole-sdk/lib/cjs/utils/keccak";
+import { parseVaa } from "@certusone/wormhole-sdk/lib/cjs/vaa/wormhole";
+import * as anchor from "@coral-xyz/anchor";
+import { Program } from "@coral-xyz/anchor";
+import { WormholeVerifyVaaShim } from "../idls/wormhole_verify_vaa_shim";
+import { logCostAndCompute } from "./helpers";
+
+// VAA from https://wormholescan.io/#/tx/AEa98mf68bcwUmT8Yheidw4C4KUVSG9732SVg5kqnfmH?view=advanced
+const VAA =
+  "AQAAAAQNAL1qji7v9KnngyX0VxK+3fCMVscWTLoYX8L48NWquq2WGrcHd4H0wYc0KF4ZOWjLD2okXoBjGQIDJzx4qIrbSzQBAQq69h+neXGb58VfhZgraPVCxJmnTj8JIDq5jqi3Qav1e+IW51mIJlOhSAdCRbEyQLzf6Z3C19WJJqSyt/z1XF0AAvFgDHkseyMZTE5vQjflu4tc5OLPJe2VYCxTJT15LA02YPrWgOM6HhfUhXDhFoG5AI/s2ApjK8jaqi7LGJILAUMBA6cp4vfko8hYyRvogqQWsdk9e20g0O6s60h4ewweapXCQHerQpoJYdDxlCehN4fuYnuudEhW+6FaXLjwNJBdqsoABDg9qXjXB47nBVCZAGns2eosVqpjkyDaCfo/p1x8AEjBA80CyC1/QlbG9L4zlnnDIfZWylsf3keJqx28+fZNC5oABi6XegfozgE8JKqvZLvd7apDhrJ6Qv+fMiynaXASkafeVJOqgFOFbCMXdMKehD38JXvz3JrlnZ92E+I5xOJaDVgABzDSte4mxUMBMJB9UUgJBeAVsokFvK4DOfvh6G3CVqqDJplLwmjUqFB7fAgRfGcA8PWNStRc+YDZiG66YxPnptwACe84S31Kh9voz2xRk1THMpqHQ4fqE7DizXPNWz6Z6ebEXGcd7UP9PBXoNNvjkLWZJZOdbkZyZqztaIiAo4dgWUABCobiuQP92WjTxOZz0KhfWVJ3YBVfsXUwaVQH4/p6khX0HCEVHR9VHmjvrAAGDMdJGWW+zu8mFQc4gPU6m4PZ6swADO7voA5GWZZPiztz22pftwxKINGvOjCPlLpM1Y2+Vq6AQuez/mlUAmaL0NKgs+5VYcM1SGBz0TL3ABRhKQAhUEMADWmiMo0J1Qaj8gElb+9711ZjvAY663GIyG/E6EdPW+nPKJI9iZE180sLct+krHj0J7PlC9BjDiO2y149oCOJ6FgAEcaVkYK43EpN7XqxrdpanX6R6TaqECgZTjvtN3L6AP2ceQr8mJJraYq+qY8pTfFvPKEqmW9CBYvnA5gIMpX59WsAEjIL9Hdnx+zFY0qSPB1hB9AhqWeBP/QfJjqzqafsczaeCN/rWUf6iNBgXI050ywtEp8JQ36rCn8w6dRhUusn+MEAZ32XyAAAAAAAFczO6yk0j3G90i/+9DoqGcH1teF8XMpUEVKRIBgmcq3lAAAAAAAC/1wAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAC6Q7dAAAAAAAAAAAAAAAAAAoLhpkcYhizbB0Z1KLp6wzjYG60gAAgAAAAAAAAAAAAAAAInNTEvk5b/1WVF+JawF1smtAdicABAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA==";
+const CORE_BRIDGE_PROGRAM_ID = new anchor.web3.PublicKey(
+  "worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth"
+);
+const GUARDIAN_SET_SEED = "GuardianSet";
+
+describe("wormhole-verify-vaa-shim", () => {
+  // Configure the client to use the local cluster.
+  anchor.setProvider(anchor.AnchorProvider.env());
+
+  const program = anchor.workspace
+    .WormholeVerifyVaaShim as Program<WormholeVerifyVaaShim>;
+
+  it("Verifies a VAA!", async () => {
+    const signatureKeypair = anchor.web3.Keypair.generate();
+    const buf = Buffer.from(VAA, "base64");
+    const vaa = parseVaa(buf);
+    const tx = await program.methods
+      .postSignatures(
+        vaa.guardianSetIndex,
+        vaa.guardianSignatures.length,
+        vaa.guardianSignatures.map((s) => [s.index, ...s.signature])
+      )
+      .accounts({ guardianSignatures: signatureKeypair.publicKey })
+      .signers([signatureKeypair])
+      .rpc();
+    await logCostAndCompute("shim", tx);
+
+    // Convert guardian_set_index to big-endian bytes
+    const guardianSetIndex = vaa.guardianSetIndex;
+    const indexBuffer = Buffer.alloc(4); // guardian_set_index is a u32
+    indexBuffer.writeUInt32BE(guardianSetIndex);
+    const [guardianSet, guardianSetBump] =
+      anchor.web3.PublicKey.findProgramAddressSync(
+        [Buffer.from(GUARDIAN_SET_SEED), indexBuffer],
+        CORE_BRIDGE_PROGRAM_ID
+      );
+
+    const tx2 = await program.methods
+      .verifyHash(guardianSetBump, [...keccak256(vaa.hash)])
+      .accounts({
+        guardianSet,
+        guardianSignatures: signatureKeypair.publicKey,
+      })
+      .preInstructions([
+        anchor.web3.ComputeBudgetProgram.setComputeUnitLimit({
+          units: 420_000,
+        }),
+      ])
+      .postInstructions([
+        await program.methods
+          .closeSignatures()
+          .accounts({ guardianSignatures: signatureKeypair.publicKey })
+          .instruction(),
+      ])
+      .rpc();
+    await logCostAndCompute("shim", tx2);
+  });
+});

+ 11 - 0
svm/wormhole-core-shims/anchor/tsconfig.json

@@ -0,0 +1,11 @@
+{
+  "compilerOptions": {
+    "types": ["mocha", "chai"],
+    "typeRoots": ["./node_modules/@types"],
+    "lib": ["es2015"],
+    "module": "commonjs",
+    "target": "es6",
+    "esModuleInterop": true,
+    "resolveJsonModule": true
+  }
+}

+ 3951 - 0
svm/wormhole-core-shims/anchor/yarn.lock

@@ -0,0 +1,3951 @@
+# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY.
+# yarn lockfile v1
+
+
+"@apollo/client@^3.5.8":
+  version "3.12.4"
+  resolved "https://registry.yarnpkg.com/@apollo/client/-/client-3.12.4.tgz#d8b5753459dfb5642129247c02956fe5b5ad8f11"
+  integrity sha512-S/eC9jxEW9Jg1BjD6AZonE1fHxYuvC3gFHop8FRQkUdeK63MmBD5r0DOrN2WlJbwha1MSD6A97OwXwjaujEQpA==
+  dependencies:
+    "@graphql-typed-document-node/core" "^3.1.1"
+    "@wry/caches" "^1.0.0"
+    "@wry/equality" "^0.5.6"
+    "@wry/trie" "^0.5.0"
+    graphql-tag "^2.12.6"
+    hoist-non-react-statics "^3.3.2"
+    optimism "^0.18.0"
+    prop-types "^15.7.2"
+    rehackt "^0.1.0"
+    response-iterator "^0.2.6"
+    symbol-observable "^4.0.0"
+    ts-invariant "^0.10.3"
+    tslib "^2.3.0"
+    zen-observable-ts "^1.2.5"
+
+"@babel/runtime@7.26.0", "@babel/runtime@^7.25.0":
+  version "7.26.0"
+  resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.26.0.tgz#8600c2f595f277c60815256418b85356a65173c1"
+  integrity sha512-FDSOghenHTiToteC/QRlv2q3DhPZ/oOXTBoirfWNx1Cx3TMVcGWQtMMmQcSvb/JjpNeGzx8Pq/b4fKEJuWm1sw==
+  dependencies:
+    regenerator-runtime "^0.14.0"
+
+"@certusone/wormhole-sdk-proto-web@0.0.7":
+  version "0.0.7"
+  resolved "https://registry.yarnpkg.com/@certusone/wormhole-sdk-proto-web/-/wormhole-sdk-proto-web-0.0.7.tgz#3b4ff42227eaabbd8c89a781df1856b476b03764"
+  integrity sha512-GCe1/bcqMS0Mt+hsWp4SE4NLL59pWmK0lhQXO0oqAKl0G9AuuTdudySMDF/sLc7z5H2w34bSuSrIEKvPuuSC+w==
+  dependencies:
+    "@improbable-eng/grpc-web" "^0.15.0"
+    protobufjs "^7.0.0"
+    rxjs "^7.5.6"
+
+"@certusone/wormhole-sdk-wasm@^0.0.1":
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/@certusone/wormhole-sdk-wasm/-/wormhole-sdk-wasm-0.0.1.tgz#05f0c05acba3df5bd025eaa9c7a3d55e017a771f"
+  integrity sha512-LdIwLhOyr4pPs2jqYubqC7d4UkqYBX0EG/ppspQlW3qlVE0LZRMrH6oVzzLMyHtV0Rw7O9sIKzORW/T3mrJv2w==
+  dependencies:
+    "@types/long" "^4.0.2"
+    "@types/node" "^18.0.3"
+
+"@certusone/wormhole-sdk@^0.10.18":
+  version "0.10.18"
+  resolved "https://registry.yarnpkg.com/@certusone/wormhole-sdk/-/wormhole-sdk-0.10.18.tgz#a300381b838108d3fc31be2056de976e718029a9"
+  integrity sha512-VuN4AGB018ELkzTT/jN+yWgE6TWqXsHilxxCVWqGctzow2hKSFd8ADUhxhHigies436rS0vPvrgXi6m0J1+Ecw==
+  dependencies:
+    "@certusone/wormhole-sdk-proto-web" "0.0.7"
+    "@certusone/wormhole-sdk-wasm" "^0.0.1"
+    "@coral-xyz/borsh" "0.2.6"
+    "@mysten/sui.js" "0.32.2"
+    "@project-serum/anchor" "^0.25.0"
+    "@solana/spl-token" "^0.3.5"
+    "@solana/web3.js" "^1.66.2"
+    "@terra-money/terra.js" "3.1.9"
+    "@xpla/xpla.js" "^0.2.1"
+    algosdk "^2.4.0"
+    aptos "1.5.0"
+    axios "^0.24.0"
+    bech32 "^2.0.0"
+    binary-parser "^2.2.1"
+    bs58 "^4.0.1"
+    elliptic "^6.5.4"
+    js-base64 "^3.6.1"
+    near-api-js "^1.0.0"
+  optionalDependencies:
+    "@injectivelabs/networks" "1.10.12"
+    "@injectivelabs/sdk-ts" "1.10.72"
+    "@injectivelabs/utils" "1.10.12"
+
+"@classic-terra/terra.proto@^1.1.0":
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/@classic-terra/terra.proto/-/terra.proto-1.1.0.tgz#e314d89f59b49e79a04db25f66f658e5e5aa1890"
+  integrity sha512-bYhQG5LUaGF0KPRY9hYT/HEcd1QExZPQd6zLV/rQkCe/eDxfwFRLzZHpaaAdfWoAAZjsRWqJbUCqCg7gXBbJpw==
+  dependencies:
+    "@improbable-eng/grpc-web" "^0.14.1"
+    google-protobuf "^3.17.3"
+    long "^4.0.0"
+    protobufjs "~6.11.2"
+
+"@confio/ics23@^0.6.8":
+  version "0.6.8"
+  resolved "https://registry.yarnpkg.com/@confio/ics23/-/ics23-0.6.8.tgz#2a6b4f1f2b7b20a35d9a0745bb5a446e72930b3d"
+  integrity sha512-wB6uo+3A50m0sW/EWcU64xpV/8wShZ6bMTa7pF8eYsTrSkQA7oLUIJcs/wb8g4y2Oyq701BaGiO6n/ak5WXO1w==
+  dependencies:
+    "@noble/hashes" "^1.0.0"
+    protobufjs "^6.8.8"
+
+"@coral-xyz/anchor-errors@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@coral-xyz/anchor-errors/-/anchor-errors-0.30.1.tgz#bdfd3a353131345244546876eb4afc0e125bec30"
+  integrity sha512-9Mkradf5yS5xiLWrl9WrpjqOrAV+/W2RQHDlbnAZBivoGpOs1ECjoDCkVk4aRG8ZdiFiB8zQEVlxf+8fKkmSfQ==
+
+"@coral-xyz/anchor@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@coral-xyz/anchor/-/anchor-0.30.1.tgz#17f3e9134c28cd0ea83574c6bab4e410bcecec5d"
+  integrity sha512-gDXFoF5oHgpriXAaLpxyWBHdCs8Awgf/gLHIo6crv7Aqm937CNdY+x+6hoj7QR5vaJV7MxWSQ0NGFzL3kPbWEQ==
+  dependencies:
+    "@coral-xyz/anchor-errors" "^0.30.1"
+    "@coral-xyz/borsh" "^0.30.1"
+    "@noble/hashes" "^1.3.1"
+    "@solana/web3.js" "^1.68.0"
+    bn.js "^5.1.2"
+    bs58 "^4.0.1"
+    buffer-layout "^1.2.2"
+    camelcase "^6.3.0"
+    cross-fetch "^3.1.5"
+    crypto-hash "^1.3.0"
+    eventemitter3 "^4.0.7"
+    pako "^2.0.3"
+    snake-case "^3.0.4"
+    superstruct "^0.15.4"
+    toml "^3.0.0"
+
+"@coral-xyz/borsh@0.2.6":
+  version "0.2.6"
+  resolved "https://registry.yarnpkg.com/@coral-xyz/borsh/-/borsh-0.2.6.tgz#0f11b223bf2967574310705afd3c53ce26688ada"
+  integrity sha512-y6nmHw1bFcJib7sMHsQPpC8r47xhqDZVvhUdna7NUPzpSbOZG6f46N21+aXsQ2w/tG8Ggls488J/ZmwbgVmyjg==
+  dependencies:
+    bn.js "^5.1.2"
+    buffer-layout "^1.2.0"
+
+"@coral-xyz/borsh@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@coral-xyz/borsh/-/borsh-0.30.1.tgz#869d8833abe65685c72e9199b8688477a4f6b0e3"
+  integrity sha512-aaxswpPrCFKl8vZTbxLssA2RvwX2zmKLlRCIktJOwW+VpVwYtXRtlWiIP+c2pPRKneiTiWCN2GEMSH9j1zTlWQ==
+  dependencies:
+    bn.js "^5.1.2"
+    buffer-layout "^1.2.0"
+
+"@cosmjs/amino@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/amino/-/amino-0.30.1.tgz#7c18c14627361ba6c88e3495700ceea1f76baace"
+  integrity sha512-yNHnzmvAlkETDYIpeCTdVqgvrdt1qgkOXwuRVi8s27UKI5hfqyE9fJ/fuunXE6ZZPnKkjIecDznmuUOMrMvw4w==
+  dependencies:
+    "@cosmjs/crypto" "^0.30.1"
+    "@cosmjs/encoding" "^0.30.1"
+    "@cosmjs/math" "^0.30.1"
+    "@cosmjs/utils" "^0.30.1"
+
+"@cosmjs/crypto@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/crypto/-/crypto-0.30.1.tgz#21e94d5ca8f8ded16eee1389d2639cb5c43c3eb5"
+  integrity sha512-rAljUlake3MSXs9xAm87mu34GfBLN0h/1uPPV6jEwClWjNkAMotzjC0ab9MARy5FFAvYHL3lWb57bhkbt2GtzQ==
+  dependencies:
+    "@cosmjs/encoding" "^0.30.1"
+    "@cosmjs/math" "^0.30.1"
+    "@cosmjs/utils" "^0.30.1"
+    "@noble/hashes" "^1"
+    bn.js "^5.2.0"
+    elliptic "^6.5.4"
+    libsodium-wrappers "^0.7.6"
+
+"@cosmjs/encoding@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/encoding/-/encoding-0.30.1.tgz#b5c4e0ef7ceb1f2753688eb96400ed70f35c6058"
+  integrity sha512-rXmrTbgqwihORwJ3xYhIgQFfMSrwLu1s43RIK9I8EBudPx3KmnmyAKzMOVsRDo9edLFNuZ9GIvysUCwQfq3WlQ==
+  dependencies:
+    base64-js "^1.3.0"
+    bech32 "^1.1.4"
+    readonly-date "^1.0.0"
+
+"@cosmjs/json-rpc@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/json-rpc/-/json-rpc-0.30.1.tgz#16f21305fc167598c8a23a45549b85106b2372bc"
+  integrity sha512-pitfC/2YN9t+kXZCbNuyrZ6M8abnCC2n62m+JtU9vQUfaEtVsgy+1Fk4TRQ175+pIWSdBMFi2wT8FWVEE4RhxQ==
+  dependencies:
+    "@cosmjs/stream" "^0.30.1"
+    xstream "^11.14.0"
+
+"@cosmjs/math@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/math/-/math-0.30.1.tgz#8b816ef4de5d3afa66cb9fdfb5df2357a7845b8a"
+  integrity sha512-yaoeI23pin9ZiPHIisa6qqLngfnBR/25tSaWpkTm8Cy10MX70UF5oN4+/t1heLaM6SSmRrhk3psRkV4+7mH51Q==
+  dependencies:
+    bn.js "^5.2.0"
+
+"@cosmjs/proto-signing@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/proto-signing/-/proto-signing-0.30.1.tgz#f0dda372488df9cd2677150b89b3e9c72b3cb713"
+  integrity sha512-tXh8pPYXV4aiJVhTKHGyeZekjj+K9s2KKojMB93Gcob2DxUjfKapFYBMJSgfKPuWUPEmyr8Q9km2hplI38ILgQ==
+  dependencies:
+    "@cosmjs/amino" "^0.30.1"
+    "@cosmjs/crypto" "^0.30.1"
+    "@cosmjs/encoding" "^0.30.1"
+    "@cosmjs/math" "^0.30.1"
+    "@cosmjs/utils" "^0.30.1"
+    cosmjs-types "^0.7.1"
+    long "^4.0.0"
+
+"@cosmjs/socket@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/socket/-/socket-0.30.1.tgz#00b22f4b5e2ab01f4d82ccdb7b2e59536bfe5ce0"
+  integrity sha512-r6MpDL+9N+qOS/D5VaxnPaMJ3flwQ36G+vPvYJsXArj93BjgyFB7BwWwXCQDzZ+23cfChPUfhbINOenr8N2Kow==
+  dependencies:
+    "@cosmjs/stream" "^0.30.1"
+    isomorphic-ws "^4.0.1"
+    ws "^7"
+    xstream "^11.14.0"
+
+"@cosmjs/stargate@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/stargate/-/stargate-0.30.1.tgz#e1b22e1226cffc6e93914a410755f1f61057ba04"
+  integrity sha512-RdbYKZCGOH8gWebO7r6WvNnQMxHrNXInY/gPHPzMjbQF6UatA6fNM2G2tdgS5j5u7FTqlCI10stNXrknaNdzog==
+  dependencies:
+    "@confio/ics23" "^0.6.8"
+    "@cosmjs/amino" "^0.30.1"
+    "@cosmjs/encoding" "^0.30.1"
+    "@cosmjs/math" "^0.30.1"
+    "@cosmjs/proto-signing" "^0.30.1"
+    "@cosmjs/stream" "^0.30.1"
+    "@cosmjs/tendermint-rpc" "^0.30.1"
+    "@cosmjs/utils" "^0.30.1"
+    cosmjs-types "^0.7.1"
+    long "^4.0.0"
+    protobufjs "~6.11.3"
+    xstream "^11.14.0"
+
+"@cosmjs/stream@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/stream/-/stream-0.30.1.tgz#ba038a2aaf41343696b1e6e759d8e03a9516ec1a"
+  integrity sha512-Fg0pWz1zXQdoxQZpdHRMGvUH5RqS6tPv+j9Eh7Q953UjMlrwZVo0YFLC8OTf/HKVf10E4i0u6aM8D69Q6cNkgQ==
+  dependencies:
+    xstream "^11.14.0"
+
+"@cosmjs/tendermint-rpc@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/tendermint-rpc/-/tendermint-rpc-0.30.1.tgz#c16378892ba1ac63f72803fdf7567eab9d4f0aa0"
+  integrity sha512-Z3nCwhXSbPZJ++v85zHObeUggrEHVfm1u18ZRwXxFE9ZMl5mXTybnwYhczuYOl7KRskgwlB+rID0WYACxj4wdQ==
+  dependencies:
+    "@cosmjs/crypto" "^0.30.1"
+    "@cosmjs/encoding" "^0.30.1"
+    "@cosmjs/json-rpc" "^0.30.1"
+    "@cosmjs/math" "^0.30.1"
+    "@cosmjs/socket" "^0.30.1"
+    "@cosmjs/stream" "^0.30.1"
+    "@cosmjs/utils" "^0.30.1"
+    axios "^0.21.2"
+    readonly-date "^1.0.0"
+    xstream "^11.14.0"
+
+"@cosmjs/utils@^0.30.1":
+  version "0.30.1"
+  resolved "https://registry.yarnpkg.com/@cosmjs/utils/-/utils-0.30.1.tgz#6d92582341be3c2ec8d82090253cfa4b7f959edb"
+  integrity sha512-KvvX58MGMWh7xA+N+deCfunkA/ZNDvFLw4YbOmX3f/XBIkqrVY7qlotfy2aNb1kgp6h4B6Yc8YawJPDTfvWX7g==
+
+"@esbuild/aix-ppc64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.23.1.tgz#51299374de171dbd80bb7d838e1cfce9af36f353"
+  integrity sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==
+
+"@esbuild/android-arm64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.23.1.tgz#58565291a1fe548638adb9c584237449e5e14018"
+  integrity sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==
+
+"@esbuild/android-arm@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.23.1.tgz#5eb8c652d4c82a2421e3395b808e6d9c42c862ee"
+  integrity sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==
+
+"@esbuild/android-x64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.23.1.tgz#ae19d665d2f06f0f48a6ac9a224b3f672e65d517"
+  integrity sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==
+
+"@esbuild/darwin-arm64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.23.1.tgz#05b17f91a87e557b468a9c75e9d85ab10c121b16"
+  integrity sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==
+
+"@esbuild/darwin-x64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.23.1.tgz#c58353b982f4e04f0d022284b8ba2733f5ff0931"
+  integrity sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==
+
+"@esbuild/freebsd-arm64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.23.1.tgz#f9220dc65f80f03635e1ef96cfad5da1f446f3bc"
+  integrity sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==
+
+"@esbuild/freebsd-x64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.23.1.tgz#69bd8511fa013b59f0226d1609ac43f7ce489730"
+  integrity sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==
+
+"@esbuild/linux-arm64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.23.1.tgz#8050af6d51ddb388c75653ef9871f5ccd8f12383"
+  integrity sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==
+
+"@esbuild/linux-arm@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.23.1.tgz#ecaabd1c23b701070484990db9a82f382f99e771"
+  integrity sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==
+
+"@esbuild/linux-ia32@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.23.1.tgz#3ed2273214178109741c09bd0687098a0243b333"
+  integrity sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==
+
+"@esbuild/linux-loong64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.23.1.tgz#a0fdf440b5485c81b0fbb316b08933d217f5d3ac"
+  integrity sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==
+
+"@esbuild/linux-mips64el@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.23.1.tgz#e11a2806346db8375b18f5e104c5a9d4e81807f6"
+  integrity sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==
+
+"@esbuild/linux-ppc64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.23.1.tgz#06a2744c5eaf562b1a90937855b4d6cf7c75ec96"
+  integrity sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==
+
+"@esbuild/linux-riscv64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.23.1.tgz#65b46a2892fc0d1af4ba342af3fe0fa4a8fe08e7"
+  integrity sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==
+
+"@esbuild/linux-s390x@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.23.1.tgz#e71ea18c70c3f604e241d16e4e5ab193a9785d6f"
+  integrity sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==
+
+"@esbuild/linux-x64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.23.1.tgz#d47f97391e80690d4dfe811a2e7d6927ad9eed24"
+  integrity sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==
+
+"@esbuild/netbsd-x64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.23.1.tgz#44e743c9778d57a8ace4b72f3c6b839a3b74a653"
+  integrity sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==
+
+"@esbuild/openbsd-arm64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/openbsd-arm64/-/openbsd-arm64-0.23.1.tgz#05c5a1faf67b9881834758c69f3e51b7dee015d7"
+  integrity sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==
+
+"@esbuild/openbsd-x64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.23.1.tgz#2e58ae511bacf67d19f9f2dcd9e8c5a93f00c273"
+  integrity sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==
+
+"@esbuild/sunos-x64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.23.1.tgz#adb022b959d18d3389ac70769cef5a03d3abd403"
+  integrity sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==
+
+"@esbuild/win32-arm64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.23.1.tgz#84906f50c212b72ec360f48461d43202f4c8b9a2"
+  integrity sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==
+
+"@esbuild/win32-ia32@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.23.1.tgz#5e3eacc515820ff729e90d0cb463183128e82fac"
+  integrity sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==
+
+"@esbuild/win32-x64@0.23.1":
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.23.1.tgz#81fd50d11e2c32b2d6241470e3185b70c7b30699"
+  integrity sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==
+
+"@ethereumjs/common@^2.6.4":
+  version "2.6.5"
+  resolved "https://registry.yarnpkg.com/@ethereumjs/common/-/common-2.6.5.tgz#0a75a22a046272579d91919cb12d84f2756e8d30"
+  integrity sha512-lRyVQOeCDaIVtgfbowla32pzeDv2Obr8oR8Put5RdUBNRGr1VGPGQNGP6elWIpgK3YdpzqTOh4GyUGOureVeeA==
+  dependencies:
+    crc-32 "^1.2.0"
+    ethereumjs-util "^7.1.5"
+
+"@ethereumjs/tx@3.5.2":
+  version "3.5.2"
+  resolved "https://registry.yarnpkg.com/@ethereumjs/tx/-/tx-3.5.2.tgz#197b9b6299582ad84f9527ca961466fce2296c1c"
+  integrity sha512-gQDNJWKrSDGu2w7w0PzVXVBNMzb7wwdDOmOqczmhNjqFxFuIbhVJDwiGEnxFNC2/b8ifcZzY7MLcluizohRzNw==
+  dependencies:
+    "@ethereumjs/common" "^2.6.4"
+    ethereumjs-util "^7.1.5"
+
+"@ethersproject/abi@5.7.0", "@ethersproject/abi@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/abi/-/abi-5.7.0.tgz#b3f3e045bbbeed1af3947335c247ad625a44e449"
+  integrity sha512-351ktp42TiRcYB3H1OP8yajPeAQstMW/yCFokj/AthP9bLHzQFPlOrxOcwYEDkUAICmOHljvN4K39OMTMUa9RA==
+  dependencies:
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/hash" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/abstract-provider@5.7.0", "@ethersproject/abstract-provider@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/abstract-provider/-/abstract-provider-5.7.0.tgz#b0a8550f88b6bf9d51f90e4795d48294630cb9ef"
+  integrity sha512-R41c9UkchKCpAqStMYUpdunjo3pkEvZC3FAwZn5S5MGbXoMQOHIdHItezTETxAO5bevtMApSyEhn9+CHcDsWBw==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/networks" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/web" "^5.7.0"
+
+"@ethersproject/abstract-signer@5.7.0", "@ethersproject/abstract-signer@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/abstract-signer/-/abstract-signer-5.7.0.tgz#13f4f32117868452191a4649723cb086d2b596b2"
+  integrity sha512-a16V8bq1/Cz+TGCkE2OPMTOUDLS3grCpdjoJCYNnVBbdYEMSgKrU0+B90s8b6H+ByYTBZN7a3g76jdIJi7UfKQ==
+  dependencies:
+    "@ethersproject/abstract-provider" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+
+"@ethersproject/address@5.7.0", "@ethersproject/address@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/address/-/address-5.7.0.tgz#19b56c4d74a3b0a46bfdbb6cfcc0a153fc697f37"
+  integrity sha512-9wYhYt7aghVGo758POM5nqcOMaE168Q6aRLJZwUmiqSrAungkG74gSSeKEIR7ukixesdRZGPgVqme6vmxs1fkA==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/rlp" "^5.7.0"
+
+"@ethersproject/base64@5.7.0", "@ethersproject/base64@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/base64/-/base64-5.7.0.tgz#ac4ee92aa36c1628173e221d0d01f53692059e1c"
+  integrity sha512-Dr8tcHt2mEbsZr/mwTPIQAf3Ai0Bks/7gTw9dSqk1mQvhW3XvRlmDJr/4n+wg1JmCl16NZue17CDh8xb/vZ0sQ==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+
+"@ethersproject/basex@5.7.0", "@ethersproject/basex@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/basex/-/basex-5.7.0.tgz#97034dc7e8938a8ca943ab20f8a5e492ece4020b"
+  integrity sha512-ywlh43GwZLv2Voc2gQVTKBoVQ1mti3d8HK5aMxsfu/nRDnMmNqaSJ3r3n85HBByT8OpoY96SXM1FogC533T4zw==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+
+"@ethersproject/bignumber@5.7.0", "@ethersproject/bignumber@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/bignumber/-/bignumber-5.7.0.tgz#e2f03837f268ba655ffba03a57853e18a18dc9c2"
+  integrity sha512-n1CAdIHRWjSucQO3MC1zPSVgV/6dy/fjL9pMrPP9peL+QxEg9wOsVqwD4+818B6LUEtaXzVHQiuivzRoxPxUGw==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    bn.js "^5.2.1"
+
+"@ethersproject/bytes@5.7.0", "@ethersproject/bytes@^5.6.1", "@ethersproject/bytes@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/bytes/-/bytes-5.7.0.tgz#a00f6ea8d7e7534d6d87f47188af1148d71f155d"
+  integrity sha512-nsbxwgFXWh9NyYWo+U8atvmMsSdKJprTcICAkvbBffT75qDocbuggBU0SJiVK2MuTrp0q+xvLkTnGMPK1+uA9A==
+  dependencies:
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/constants@5.7.0", "@ethersproject/constants@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/constants/-/constants-5.7.0.tgz#df80a9705a7e08984161f09014ea012d1c75295e"
+  integrity sha512-DHI+y5dBNvkpYUMiRQyxRBYBefZkJfo70VUkUAsRjcPs47muV9evftfZ0PJVCXYbAiCgght0DtcF9srFQmIgWA==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+
+"@ethersproject/contracts@5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/contracts/-/contracts-5.7.0.tgz#c305e775abd07e48aa590e1a877ed5c316f8bd1e"
+  integrity sha512-5GJbzEU3X+d33CdfPhcyS+z8MzsTrBGk/sc+G+59+tPa9yFkl6HQ9D6L0QMgNTA9q8dT0XKxxkyp883XsQvbbg==
+  dependencies:
+    "@ethersproject/abi" "^5.7.0"
+    "@ethersproject/abstract-provider" "^5.7.0"
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+
+"@ethersproject/hash@5.7.0", "@ethersproject/hash@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/hash/-/hash-5.7.0.tgz#eb7aca84a588508369562e16e514b539ba5240a7"
+  integrity sha512-qX5WrQfnah1EFnO5zJv1v46a8HW0+E5xuBBDTwMFZLuVTx0tbU2kkx15NqdjxecrLGatQN9FGQKpb1FKdHCt+g==
+  dependencies:
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/base64" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/hdnode@5.7.0", "@ethersproject/hdnode@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/hdnode/-/hdnode-5.7.0.tgz#e627ddc6b466bc77aebf1a6b9e47405ca5aef9cf"
+  integrity sha512-OmyYo9EENBPPf4ERhR7oj6uAtUAhYGqOnIS+jE5pTXvdKBS99ikzq1E7Iv0ZQZ5V36Lqx1qZLeak0Ra16qpeOg==
+  dependencies:
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/basex" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/pbkdf2" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/sha2" "^5.7.0"
+    "@ethersproject/signing-key" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/wordlists" "^5.7.0"
+
+"@ethersproject/json-wallets@5.7.0", "@ethersproject/json-wallets@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/json-wallets/-/json-wallets-5.7.0.tgz#5e3355287b548c32b368d91014919ebebddd5360"
+  integrity sha512-8oee5Xgu6+RKgJTkvEMl2wDgSPSAQ9MB/3JYjFV9jlKvcYHUXZC+cQp0njgmxdHkYWn8s6/IqIZYm0YWCjO/0g==
+  dependencies:
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/hdnode" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/pbkdf2" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/random" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    aes-js "3.0.0"
+    scrypt-js "3.0.1"
+
+"@ethersproject/keccak256@5.7.0", "@ethersproject/keccak256@^5.6.1", "@ethersproject/keccak256@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/keccak256/-/keccak256-5.7.0.tgz#3186350c6e1cd6aba7940384ec7d6d9db01f335a"
+  integrity sha512-2UcPboeL/iW+pSg6vZ6ydF8tCnv3Iu/8tUmLLzWWGzxWKFFqOBQFLo6uLUv6BDrLgCDfN28RJ/wtByx+jZ4KBg==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    js-sha3 "0.8.0"
+
+"@ethersproject/logger@5.7.0", "@ethersproject/logger@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/logger/-/logger-5.7.0.tgz#6ce9ae168e74fecf287be17062b590852c311892"
+  integrity sha512-0odtFdXu/XHtjQXJYA3u9G0G8btm0ND5Cu8M7i5vhEcE8/HmF4Lbdqanwyv4uQTr2tx6b7fQRmgLrsnpQlmnig==
+
+"@ethersproject/networks@5.7.1", "@ethersproject/networks@^5.7.0":
+  version "5.7.1"
+  resolved "https://registry.yarnpkg.com/@ethersproject/networks/-/networks-5.7.1.tgz#118e1a981d757d45ccea6bb58d9fd3d9db14ead6"
+  integrity sha512-n/MufjFYv3yFcUyfhnXotyDlNdFb7onmkSy8aQERi2PjNcnWQ66xXxa3XlS8nCcA8aJKJjIIMNJTC7tu80GwpQ==
+  dependencies:
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/pbkdf2@5.7.0", "@ethersproject/pbkdf2@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/pbkdf2/-/pbkdf2-5.7.0.tgz#d2267d0a1f6e123f3771007338c47cccd83d3102"
+  integrity sha512-oR/dBRZR6GTyaofd86DehG72hY6NpAjhabkhxgr3X2FpJtJuodEl2auADWBZfhDHgVCbu3/H/Ocq2uC6dpNjjw==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/sha2" "^5.7.0"
+
+"@ethersproject/properties@5.7.0", "@ethersproject/properties@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/properties/-/properties-5.7.0.tgz#a6e12cb0439b878aaf470f1902a176033067ed30"
+  integrity sha512-J87jy8suntrAkIZtecpxEPxY//szqr1mlBaYlQ0r4RCaiD2hjheqF9s1LVE8vVuJCXisjIP+JgtK/Do54ej4Sw==
+  dependencies:
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/providers@5.7.2":
+  version "5.7.2"
+  resolved "https://registry.yarnpkg.com/@ethersproject/providers/-/providers-5.7.2.tgz#f8b1a4f275d7ce58cf0a2eec222269a08beb18cb"
+  integrity sha512-g34EWZ1WWAVgr4aptGlVBF8mhl3VWjv+8hoAnzStu8Ah22VHBsuGzP17eb6xDVRzw895G4W7vvx60lFFur/1Rg==
+  dependencies:
+    "@ethersproject/abstract-provider" "^5.7.0"
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/base64" "^5.7.0"
+    "@ethersproject/basex" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/hash" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/networks" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/random" "^5.7.0"
+    "@ethersproject/rlp" "^5.7.0"
+    "@ethersproject/sha2" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/web" "^5.7.0"
+    bech32 "1.1.4"
+    ws "7.4.6"
+
+"@ethersproject/random@5.7.0", "@ethersproject/random@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/random/-/random-5.7.0.tgz#af19dcbc2484aae078bb03656ec05df66253280c"
+  integrity sha512-19WjScqRA8IIeWclFme75VMXSBvi4e6InrUNuaR4s5pTF2qNhcGdCUwdxUVGtDDqC00sDLCO93jPQoDUH4HVmQ==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/rlp@5.7.0", "@ethersproject/rlp@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/rlp/-/rlp-5.7.0.tgz#de39e4d5918b9d74d46de93af80b7685a9c21304"
+  integrity sha512-rBxzX2vK8mVF7b0Tol44t5Tb8gomOHkj5guL+HhzQ1yBh/ydjGnpw6at+X6Iw0Kp3OzzzkcKp8N9r0W4kYSs9w==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/sha2@5.7.0", "@ethersproject/sha2@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/sha2/-/sha2-5.7.0.tgz#9a5f7a7824ef784f7f7680984e593a800480c9fb"
+  integrity sha512-gKlH42riwb3KYp0reLsFTokByAKoJdgFCwI+CCiX/k+Jm2mbNs6oOaCjYQSlI1+XBVejwH2KrmCbMAT/GnRDQw==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    hash.js "1.1.7"
+
+"@ethersproject/signing-key@5.7.0", "@ethersproject/signing-key@^5.6.2", "@ethersproject/signing-key@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/signing-key/-/signing-key-5.7.0.tgz#06b2df39411b00bc57c7c09b01d1e41cf1b16ab3"
+  integrity sha512-MZdy2nL3wO0u7gkB4nA/pEf8lu1TlFswPNmy8AiYkfKTdO6eXBJyUdmHO/ehm/htHw9K/qF8ujnTyUAD+Ry54Q==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    bn.js "^5.2.1"
+    elliptic "6.5.4"
+    hash.js "1.1.7"
+
+"@ethersproject/solidity@5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/solidity/-/solidity-5.7.0.tgz#5e9c911d8a2acce2a5ebb48a5e2e0af20b631cb8"
+  integrity sha512-HmabMd2Dt/raavyaGukF4XxizWKhKQ24DoLtdNbBmNKUOPqwjsKQSdV9GQtj9CBEea9DlzETlVER1gYeXXBGaA==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/sha2" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/strings@5.7.0", "@ethersproject/strings@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/strings/-/strings-5.7.0.tgz#54c9d2a7c57ae8f1205c88a9d3a56471e14d5ed2"
+  integrity sha512-/9nu+lj0YswRNSH0NXYqrh8775XNyEdUQAuf3f+SmOrnVewcJ5SBNAjF7lpgehKi4abvNNXyf+HX86czCdJ8Mg==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/transactions@5.7.0", "@ethersproject/transactions@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/transactions/-/transactions-5.7.0.tgz#91318fc24063e057885a6af13fdb703e1f993d3b"
+  integrity sha512-kmcNicCp1lp8qanMTC3RIikGgoJ80ztTyvtsFvCYpSCfkjhD0jZ2LOrnbcuxuToLIUYYf+4XwD1rP+B/erDIhQ==
+  dependencies:
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/rlp" "^5.7.0"
+    "@ethersproject/signing-key" "^5.7.0"
+
+"@ethersproject/units@5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/units/-/units-5.7.0.tgz#637b563d7e14f42deeee39245275d477aae1d8b1"
+  integrity sha512-pD3xLMy3SJu9kG5xDGI7+xhTEmGXlEqXU4OfNapmfnxLVY4EMSSRp7j1k7eezutBPH7RBN/7QPnwR7hzNlEFeg==
+  dependencies:
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/constants" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+
+"@ethersproject/wallet@5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/wallet/-/wallet-5.7.0.tgz#4e5d0790d96fe21d61d38fb40324e6c7ef350b2d"
+  integrity sha512-MhmXlJXEJFBFVKrDLB4ZdDzxcBxQ3rLyCkhNqVu3CDYvR97E+8r01UgrI+TI99Le+aYm/in/0vp86guJuM7FCA==
+  dependencies:
+    "@ethersproject/abstract-provider" "^5.7.0"
+    "@ethersproject/abstract-signer" "^5.7.0"
+    "@ethersproject/address" "^5.7.0"
+    "@ethersproject/bignumber" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/hash" "^5.7.0"
+    "@ethersproject/hdnode" "^5.7.0"
+    "@ethersproject/json-wallets" "^5.7.0"
+    "@ethersproject/keccak256" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/random" "^5.7.0"
+    "@ethersproject/signing-key" "^5.7.0"
+    "@ethersproject/transactions" "^5.7.0"
+    "@ethersproject/wordlists" "^5.7.0"
+
+"@ethersproject/web@5.7.1", "@ethersproject/web@^5.7.0":
+  version "5.7.1"
+  resolved "https://registry.yarnpkg.com/@ethersproject/web/-/web-5.7.1.tgz#de1f285b373149bee5928f4eb7bcb87ee5fbb4ae"
+  integrity sha512-Gueu8lSvyjBWL4cYsWsjh6MtMwM0+H4HvqFPZfB6dV8ctbP9zFAO73VG1cMWae0FLPCtz0peKPpZY8/ugJJX2w==
+  dependencies:
+    "@ethersproject/base64" "^5.7.0"
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@ethersproject/wordlists@5.7.0", "@ethersproject/wordlists@^5.7.0":
+  version "5.7.0"
+  resolved "https://registry.yarnpkg.com/@ethersproject/wordlists/-/wordlists-5.7.0.tgz#8fb2c07185d68c3e09eb3bfd6e779ba2774627f5"
+  integrity sha512-S2TFNJNfHWVHNE6cNDjbVlZ6MgE17MIxMbMg2zv3wn+3XSJGosL1m9ZVv3GXCf/2ymSsQ+hRI5IzoMJTG6aoVA==
+  dependencies:
+    "@ethersproject/bytes" "^5.7.0"
+    "@ethersproject/hash" "^5.7.0"
+    "@ethersproject/logger" "^5.7.0"
+    "@ethersproject/properties" "^5.7.0"
+    "@ethersproject/strings" "^5.7.0"
+
+"@graphql-typed-document-node/core@^3.1.1":
+  version "3.2.0"
+  resolved "https://registry.yarnpkg.com/@graphql-typed-document-node/core/-/core-3.2.0.tgz#5f3d96ec6b2354ad6d8a28bf216a1d97b5426861"
+  integrity sha512-mB9oAsNCm9aM3/SOv4YtBMqZbYj10R7dkq8byBqxGY/ncFwhf2oQzMV+LCRlWoDSEBJ3COiR1yeDvMtsoOsuFQ==
+
+"@improbable-eng/grpc-web-node-http-transport@0.15.0":
+  version "0.15.0"
+  resolved "https://registry.yarnpkg.com/@improbable-eng/grpc-web-node-http-transport/-/grpc-web-node-http-transport-0.15.0.tgz#5a064472ef43489cbd075a91fb831c2abeb09d68"
+  integrity sha512-HLgJfVolGGpjc9DWPhmMmXJx8YGzkek7jcCFO1YYkSOoO81MWRZentPOd/JiKiZuU08wtc4BG+WNuGzsQB5jZA==
+
+"@improbable-eng/grpc-web@^0.14.1":
+  version "0.14.1"
+  resolved "https://registry.yarnpkg.com/@improbable-eng/grpc-web/-/grpc-web-0.14.1.tgz#f4662f64dc89c0f956a94bb8a3b576556c74589c"
+  integrity sha512-XaIYuunepPxoiGVLLHmlnVminUGzBTnXr8Wv7khzmLWbNw4TCwJKX09GSMJlKhu/TRk6gms0ySFxewaETSBqgw==
+  dependencies:
+    browser-headers "^0.4.1"
+
+"@improbable-eng/grpc-web@^0.15.0":
+  version "0.15.0"
+  resolved "https://registry.yarnpkg.com/@improbable-eng/grpc-web/-/grpc-web-0.15.0.tgz#3e47e9fdd90381a74abd4b7d26e67422a2a04bef"
+  integrity sha512-ERft9/0/8CmYalqOVnJnpdDry28q+j+nAlFFARdjyxXDJ+Mhgv9+F600QC8BR9ygOfrXRlAk6CvST2j+JCpQPg==
+  dependencies:
+    browser-headers "^0.4.1"
+
+"@injectivelabs/core-proto-ts@^0.0.14":
+  version "0.0.14"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/core-proto-ts/-/core-proto-ts-0.0.14.tgz#88f3c3c5c88484906785cd93b2e2eb1d464ec5d4"
+  integrity sha512-NZWlgBzgVrXow9IknFQHvcYKX4QkUD25taRigoNYQK8PDn4+VXd9xM5WFUDRhzm2smTCguyl/+MghpEp4oTPWw==
+  dependencies:
+    "@injectivelabs/grpc-web" "^0.0.1"
+    google-protobuf "^3.14.0"
+    protobufjs "^7.0.0"
+    rxjs "^7.4.0"
+
+"@injectivelabs/exceptions@^1.10.12", "@injectivelabs/exceptions@^1.14.11", "@injectivelabs/exceptions@^1.14.33":
+  version "1.14.33"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/exceptions/-/exceptions-1.14.33.tgz#3bcc3e613aac13fce72b3557b4260820dde8ce3a"
+  integrity sha512-2c8YzLgwTOOsyc1WheqdM8jEfgGBhVrXN4cZ0jsikFVsLF619IDRn3hjIYqTeNERaEpeRPiuJGfZDu0DomwFrQ==
+  dependencies:
+    "@injectivelabs/grpc-web" "^0.0.1"
+    "@injectivelabs/ts-types" "^1.14.33"
+    http-status-codes "^2.2.0"
+    shx "^0.3.2"
+
+"@injectivelabs/grpc-web-node-http-transport@^0.0.2":
+  version "0.0.2"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/grpc-web-node-http-transport/-/grpc-web-node-http-transport-0.0.2.tgz#87c9bbd4db1f70cf18d6a55b54b2cf17d3cf30c0"
+  integrity sha512-rpyhXLiGY/UMs6v6YmgWHJHiO9l0AgDyVNv+jcutNVt4tQrmNvnpvz2wCAGOFtq5LuX/E9ChtTVpk3gWGqXcGA==
+
+"@injectivelabs/grpc-web-react-native-transport@^0.0.2":
+  version "0.0.2"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/grpc-web-react-native-transport/-/grpc-web-react-native-transport-0.0.2.tgz#07601b76bf1f165c7a9b97ee42d0d42b9e2b76fa"
+  integrity sha512-mk+aukQXnYNgPsPnu3KBi+FD0ZHQpazIlaBZ2jNZG7QAVmxTWtv3R66Zoq99Wx2dnE946NsZBYAoa0K5oSjnow==
+
+"@injectivelabs/grpc-web@^0.0.1":
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/grpc-web/-/grpc-web-0.0.1.tgz#24c028f6db50e589e30505efd2077110c8b492ba"
+  integrity sha512-Pu5YgaZp+OvR5UWfqbrPdHer3+gDf+b5fQoY+t2VZx1IAVHX8bzbN9EreYTvTYtFeDpYRWM8P7app2u4EX5wTw==
+  dependencies:
+    browser-headers "^0.4.1"
+
+"@injectivelabs/indexer-proto-ts@1.10.8-rc.4":
+  version "1.10.8-rc.4"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/indexer-proto-ts/-/indexer-proto-ts-1.10.8-rc.4.tgz#ab8424cd713bb5a69ab130b64bf0b3f9f9089946"
+  integrity sha512-IwbepTfsHHAv3Z36As6yH/+HIplOEpUu6SFHBCVgdSIaQ8GuvTib4HETiVnV4mjYqoyVgWs+zLSAfih46rdMJQ==
+  dependencies:
+    "@injectivelabs/grpc-web" "^0.0.1"
+    google-protobuf "^3.14.0"
+    protobufjs "^7.0.0"
+    rxjs "^7.4.0"
+
+"@injectivelabs/mito-proto-ts@1.0.9":
+  version "1.0.9"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/mito-proto-ts/-/mito-proto-ts-1.0.9.tgz#ad04165ad63f52a9f67082318e0620cccc4986f2"
+  integrity sha512-+TZMvJ4SHwcn6SFPdqaiQFZdNhjH7hyRFozY15nOTC2utdGij9jEsjz1NsyOejfYDA0s1z5Wm1SgrMYKaVpAmQ==
+  dependencies:
+    "@injectivelabs/grpc-web" "^0.0.1"
+    google-protobuf "^3.14.0"
+    protobufjs "^7.0.0"
+    rxjs "^7.4.0"
+
+"@injectivelabs/networks@1.10.12":
+  version "1.10.12"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/networks/-/networks-1.10.12.tgz#8c3fcf96a8930c28dcd44d779e1ef31e68ec7b0b"
+  integrity sha512-tTHyLls1Nik5QTs/S03qqG2y/ITvNwI8CJOQbMmmsr1CL2CdjJBtzRYn9Dyx2p8XgzRFf9hmlybpe20tq9O3SA==
+  dependencies:
+    "@injectivelabs/exceptions" "^1.10.12"
+    "@injectivelabs/ts-types" "^1.10.12"
+    "@injectivelabs/utils" "^1.10.12"
+    link-module-alias "^1.2.0"
+    shx "^0.3.2"
+
+"@injectivelabs/networks@^1.10.12", "@injectivelabs/networks@^1.14.11", "@injectivelabs/networks@^1.14.33":
+  version "1.14.33"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/networks/-/networks-1.14.33.tgz#340bef9948ace5abd7b9a2a69ef1d9bc6ee54c33"
+  integrity sha512-XDhAYwWYKdKBRfwO/MIfMyKjKRWz/AliMJG9yaM1C/cDlGHmA3EY7Au2Nf+PdkRhuxl2FzLV2Hp4uWeC0g8BYw==
+  dependencies:
+    "@injectivelabs/exceptions" "^1.14.33"
+    "@injectivelabs/ts-types" "^1.14.33"
+    "@injectivelabs/utils" "^1.14.33"
+    shx "^0.3.2"
+
+"@injectivelabs/sdk-ts@1.10.72":
+  version "1.10.72"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/sdk-ts/-/sdk-ts-1.10.72.tgz#07920960ef74aa9db96978a85d86042037836a51"
+  integrity sha512-A5mHNNBgO4fI1c/7CZ0bGfVXliy8laP+VaYZ++aWh1YyudoZw4CTCEmLetZRy7AUU3XcfbHa8sAImRi7db+v6Q==
+  dependencies:
+    "@apollo/client" "^3.5.8"
+    "@cosmjs/amino" "^0.30.1"
+    "@cosmjs/proto-signing" "^0.30.1"
+    "@cosmjs/stargate" "^0.30.1"
+    "@ethersproject/bytes" "^5.7.0"
+    "@injectivelabs/core-proto-ts" "^0.0.14"
+    "@injectivelabs/exceptions" "^1.10.12"
+    "@injectivelabs/grpc-web" "^0.0.1"
+    "@injectivelabs/grpc-web-node-http-transport" "^0.0.2"
+    "@injectivelabs/grpc-web-react-native-transport" "^0.0.2"
+    "@injectivelabs/indexer-proto-ts" "1.10.8-rc.4"
+    "@injectivelabs/mito-proto-ts" "1.0.9"
+    "@injectivelabs/networks" "^1.10.12"
+    "@injectivelabs/test-utils" "^1.10.12"
+    "@injectivelabs/token-metadata" "^1.10.42"
+    "@injectivelabs/ts-types" "^1.10.12"
+    "@injectivelabs/utils" "^1.10.12"
+    "@metamask/eth-sig-util" "^4.0.0"
+    axios "^0.27.2"
+    bech32 "^2.0.0"
+    bip39 "^3.0.4"
+    cosmjs-types "^0.7.1"
+    eth-crypto "^2.6.0"
+    ethereumjs-util "^7.1.4"
+    ethers "^5.7.2"
+    google-protobuf "^3.21.0"
+    graphql "^16.3.0"
+    http-status-codes "^2.2.0"
+    js-sha3 "^0.8.0"
+    jscrypto "^1.0.3"
+    keccak256 "^1.0.6"
+    link-module-alias "^1.2.0"
+    rxjs "^7.8.0"
+    secp256k1 "^4.0.3"
+    shx "^0.3.2"
+    snakecase-keys "^5.4.1"
+
+"@injectivelabs/test-utils@^1.10.12":
+  version "1.14.33"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/test-utils/-/test-utils-1.14.33.tgz#2c800231f3b6179c808d8fff63cb9b2e87d462bc"
+  integrity sha512-1SfIRsMnWcJAYNrrpY+ZUWmbD62lWWdIvD6c+FYmFKS14zU3yDIK9NXe9g1lTM/GdUVkVKQgGg2QAYZ5f2G/xA==
+  dependencies:
+    "@injectivelabs/exceptions" "^1.14.33"
+    "@injectivelabs/networks" "^1.14.33"
+    "@injectivelabs/ts-types" "^1.14.33"
+    "@injectivelabs/utils" "^1.14.33"
+    axios "^1.6.4"
+    bignumber.js "^9.0.1"
+    shx "^0.3.2"
+    snakecase-keys "^5.1.2"
+    store2 "^2.12.0"
+
+"@injectivelabs/token-metadata@^1.10.42":
+  version "1.14.11"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/token-metadata/-/token-metadata-1.14.11.tgz#296b6db3c9c9f6147273252cf41b803902c35414"
+  integrity sha512-WKJlvjKiTRHxpOeez4kYrzIwgWmpspD1IMxWclkTysAcwGltUfUsvUhu1cKuACleIjFFWmiZv/HoGRFdvEAZ8w==
+  dependencies:
+    "@injectivelabs/exceptions" "^1.14.11"
+    "@injectivelabs/networks" "^1.14.11"
+    "@injectivelabs/ts-types" "^1.14.11"
+    "@injectivelabs/utils" "^1.14.11"
+    "@types/lodash.values" "^4.3.6"
+    copyfiles "^2.4.1"
+    jsonschema "^1.4.0"
+    link-module-alias "^1.2.0"
+    lodash "^4.17.21"
+    lodash.values "^4.3.0"
+    shx "^0.3.2"
+
+"@injectivelabs/ts-types@^1.10.12", "@injectivelabs/ts-types@^1.14.11", "@injectivelabs/ts-types@^1.14.33":
+  version "1.14.33"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/ts-types/-/ts-types-1.14.33.tgz#c06da4757a07f2140d01a4f4b0087c6053e8e58f"
+  integrity sha512-sJZzMNJtZFFZoPKZ91G09bxrZqQ5aS9omoTjQWy+7OxfiRAakzhsarTueX47hm6oTaN0XeBgD3wkMukkWUaobw==
+  dependencies:
+    shx "^0.3.2"
+
+"@injectivelabs/utils@1.10.12":
+  version "1.10.12"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/utils/-/utils-1.10.12.tgz#063835b25c11182945881e82a2cbad50f6ca10b3"
+  integrity sha512-c8al79nxIJgV1cBAdW2TPDGldj/8gm5k0h5TIN/AJs8/AeIjpTwwVGfLY3QvPOpRsxuQ9CjBkTXrAcSL1wwkcw==
+  dependencies:
+    "@injectivelabs/exceptions" "^1.10.12"
+    "@injectivelabs/ts-types" "^1.10.12"
+    axios "^0.21.1"
+    bignumber.js "^9.0.1"
+    http-status-codes "^2.2.0"
+    link-module-alias "^1.2.0"
+    shx "^0.3.2"
+    snakecase-keys "^5.1.2"
+    store2 "^2.12.0"
+
+"@injectivelabs/utils@^1.10.12", "@injectivelabs/utils@^1.14.11", "@injectivelabs/utils@^1.14.33":
+  version "1.14.33"
+  resolved "https://registry.yarnpkg.com/@injectivelabs/utils/-/utils-1.14.33.tgz#9e6eef0e61f306ef5cd68ceed198ea98c47ada4e"
+  integrity sha512-zsezML4dTujF0xGLhcGmWBCghfJiy9MW+r6VqR8zJUlxnmnEdNpmsvBhBI6cmmov6Se4FL+yALAIFRvTm3txbg==
+  dependencies:
+    "@injectivelabs/exceptions" "^1.14.33"
+    "@injectivelabs/ts-types" "^1.14.33"
+    axios "^1.6.4"
+    bignumber.js "^9.0.1"
+    http-status-codes "^2.2.0"
+    shx "^0.3.2"
+    snakecase-keys "^5.1.2"
+    store2 "^2.12.0"
+
+"@metamask/eth-sig-util@^4.0.0":
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/@metamask/eth-sig-util/-/eth-sig-util-4.0.1.tgz#3ad61f6ea9ad73ba5b19db780d40d9aae5157088"
+  integrity sha512-tghyZKLHZjcdlDqCA3gNZmLeR0XvOE9U1qoQO9ohyAZT6Pya+H9vkBPcsyXytmYLNgVoin7CKCmweo/R43V+tQ==
+  dependencies:
+    ethereumjs-abi "^0.6.8"
+    ethereumjs-util "^6.2.1"
+    ethjs-util "^0.1.6"
+    tweetnacl "^1.0.3"
+    tweetnacl-util "^0.15.1"
+
+"@mysten/bcs@0.7.1":
+  version "0.7.1"
+  resolved "https://registry.yarnpkg.com/@mysten/bcs/-/bcs-0.7.1.tgz#e8db25718143e2a15688ea858470970b743f7248"
+  integrity sha512-wFPb8bkhwrbiStfZMV5rFM7J+umpke59/dNjDp+UYJKykNlW23LCk2ePyEUvGdb62HGJM1jyOJ8g4egE3OmdKA==
+  dependencies:
+    bs58 "^5.0.0"
+
+"@mysten/sui.js@0.32.2":
+  version "0.32.2"
+  resolved "https://registry.yarnpkg.com/@mysten/sui.js/-/sui.js-0.32.2.tgz#dff953a336c787465af181d8158f34a65700a74f"
+  integrity sha512-/Hm4xkGolJhqj8FvQr7QSHDTlxIvL52mtbOao9f75YjrBh7y1Uh9kbJSY7xiTF1NY9sv6p5hUVlYRJuM0Hvn9A==
+  dependencies:
+    "@mysten/bcs" "0.7.1"
+    "@noble/curves" "^1.0.0"
+    "@noble/hashes" "^1.3.0"
+    "@scure/bip32" "^1.3.0"
+    "@scure/bip39" "^1.2.0"
+    "@suchipi/femver" "^1.0.0"
+    jayson "^4.0.0"
+    rpc-websockets "^7.5.1"
+    superstruct "^1.0.3"
+    tweetnacl "^1.0.3"
+
+"@noble/curves@^1.0.0", "@noble/curves@~1.8.0":
+  version "1.8.0"
+  resolved "https://registry.yarnpkg.com/@noble/curves/-/curves-1.8.0.tgz#fe035a23959e6aeadf695851b51a87465b5ba8f7"
+  integrity sha512-j84kjAbzEnQHaSIhRPUmB3/eVXu2k3dKPl2LOrR8fSOIL+89U+7lV117EWHtq/GHM3ReGHM46iRBdZfpc4HRUQ==
+  dependencies:
+    "@noble/hashes" "1.7.0"
+
+"@noble/curves@^1.4.2":
+  version "1.7.0"
+  resolved "https://registry.yarnpkg.com/@noble/curves/-/curves-1.7.0.tgz#0512360622439256df892f21d25b388f52505e45"
+  integrity sha512-UTMhXK9SeDhFJVrHeUJ5uZlI6ajXg10O6Ddocf9S6GjbSBVZsJo88HzKwXznNfGpMTRDyJkqMjNDPYgf0qFWnw==
+  dependencies:
+    "@noble/hashes" "1.6.0"
+
+"@noble/hashes@1.1.3":
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.1.3.tgz#360afc77610e0a61f3417e497dcf36862e4f8111"
+  integrity sha512-CE0FCR57H2acVI5UOzIGSSIYxZ6v/HOhDR0Ro9VLyhnzLwx0o8W1mmgaqlEUx4049qJDlIBRztv5k+MM8vbO3A==
+
+"@noble/hashes@1.6.0":
+  version "1.6.0"
+  resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.6.0.tgz#d4bfb516ad6e7b5111c216a5cc7075f4cf19e6c5"
+  integrity sha512-YUULf0Uk4/mAA89w+k3+yUYh6NrEvxZa5T6SY3wlMvE2chHkxFUUIDI8/XW1QSC357iA5pSnqt7XEhvFOqmDyQ==
+
+"@noble/hashes@1.7.0", "@noble/hashes@^1", "@noble/hashes@^1.0.0", "@noble/hashes@^1.2.0", "@noble/hashes@^1.3.0", "@noble/hashes@~1.7.0":
+  version "1.7.0"
+  resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.7.0.tgz#5d9e33af2c7d04fee35de1519b80c958b2e35e39"
+  integrity sha512-HXydb0DgzTpDPwbVeDGCG1gIu7X6+AuU6Zl6av/E/KG8LMsvPntvq+w17CHRpKBmN6Ybdrt1eP3k4cj8DJa78w==
+
+"@noble/hashes@^1.3.1", "@noble/hashes@^1.4.0":
+  version "1.6.1"
+  resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.6.1.tgz#df6e5943edcea504bac61395926d6fd67869a0d5"
+  integrity sha512-pq5D8h10hHBjyqX+cfBm0i8JUXJ0UhczFc4r74zbuT9XgewFo2E3J1cOaGtdZynILNmQ685YWGzGE1Zv6io50w==
+
+"@noble/hashes@~1.1.1":
+  version "1.1.5"
+  resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.1.5.tgz#1a0377f3b9020efe2fae03290bd2a12140c95c11"
+  integrity sha512-LTMZiiLc+V4v1Yi16TD6aX2gmtKszNye0pQgbaLqkvhIqP7nVsSaJsWloGQjJfJ8offaoP5GtX3yY5swbcJxxQ==
+
+"@project-serum/anchor@^0.25.0":
+  version "0.25.0"
+  resolved "https://registry.yarnpkg.com/@project-serum/anchor/-/anchor-0.25.0.tgz#88ee4843336005cf5a64c80636ce626f0996f503"
+  integrity sha512-E6A5Y/ijqpfMJ5psJvbw0kVTzLZFUcOFgs6eSM2M2iWE1lVRF18T6hWZVNl6zqZsoz98jgnNHtVGJMs+ds9A7A==
+  dependencies:
+    "@project-serum/borsh" "^0.2.5"
+    "@solana/web3.js" "^1.36.0"
+    base64-js "^1.5.1"
+    bn.js "^5.1.2"
+    bs58 "^4.0.1"
+    buffer-layout "^1.2.2"
+    camelcase "^5.3.1"
+    cross-fetch "^3.1.5"
+    crypto-hash "^1.3.0"
+    eventemitter3 "^4.0.7"
+    js-sha256 "^0.9.0"
+    pako "^2.0.3"
+    snake-case "^3.0.4"
+    superstruct "^0.15.4"
+    toml "^3.0.0"
+
+"@project-serum/borsh@^0.2.5":
+  version "0.2.5"
+  resolved "https://registry.yarnpkg.com/@project-serum/borsh/-/borsh-0.2.5.tgz#6059287aa624ecebbfc0edd35e4c28ff987d8663"
+  integrity sha512-UmeUkUoKdQ7rhx6Leve1SssMR/Ghv8qrEiyywyxSWg7ooV7StdpPBhciiy5eB3T0qU1BXvdRNC8TdrkxK7WC5Q==
+  dependencies:
+    bn.js "^5.1.2"
+    buffer-layout "^1.2.0"
+
+"@protobufjs/aspromise@^1.1.1", "@protobufjs/aspromise@^1.1.2":
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/@protobufjs/aspromise/-/aspromise-1.1.2.tgz#9b8b0cc663d669a7d8f6f5d0893a14d348f30fbf"
+  integrity sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ==
+
+"@protobufjs/base64@^1.1.2":
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/@protobufjs/base64/-/base64-1.1.2.tgz#4c85730e59b9a1f1f349047dbf24296034bb2735"
+  integrity sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg==
+
+"@protobufjs/codegen@^2.0.4":
+  version "2.0.4"
+  resolved "https://registry.yarnpkg.com/@protobufjs/codegen/-/codegen-2.0.4.tgz#7ef37f0d010fb028ad1ad59722e506d9262815cb"
+  integrity sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg==
+
+"@protobufjs/eventemitter@^1.1.0":
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz#355cbc98bafad5978f9ed095f397621f1d066b70"
+  integrity sha512-j9ednRT81vYJ9OfVuXG6ERSTdEL1xVsNgqpkxMsbIabzSo3goCjDIveeGv5d03om39ML71RdmrGNjG5SReBP/Q==
+
+"@protobufjs/fetch@^1.1.0":
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/@protobufjs/fetch/-/fetch-1.1.0.tgz#ba99fb598614af65700c1619ff06d454b0d84c45"
+  integrity sha512-lljVXpqXebpsijW71PZaCYeIcE5on1w5DlQy5WH6GLbFryLUrBD4932W/E2BSpfRJWseIL4v/KPgBFxDOIdKpQ==
+  dependencies:
+    "@protobufjs/aspromise" "^1.1.1"
+    "@protobufjs/inquire" "^1.1.0"
+
+"@protobufjs/float@^1.0.2":
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/@protobufjs/float/-/float-1.0.2.tgz#5e9e1abdcb73fc0a7cb8b291df78c8cbd97b87d1"
+  integrity sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ==
+
+"@protobufjs/inquire@^1.1.0":
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/@protobufjs/inquire/-/inquire-1.1.0.tgz#ff200e3e7cf2429e2dcafc1140828e8cc638f089"
+  integrity sha512-kdSefcPdruJiFMVSbn801t4vFK7KB/5gd2fYvrxhuJYg8ILrmn9SKSX2tZdV6V+ksulWqS7aXjBcRXl3wHoD9Q==
+
+"@protobufjs/path@^1.1.2":
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/@protobufjs/path/-/path-1.1.2.tgz#6cc2b20c5c9ad6ad0dccfd21ca7673d8d7fbf68d"
+  integrity sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA==
+
+"@protobufjs/pool@^1.1.0":
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/@protobufjs/pool/-/pool-1.1.0.tgz#09fd15f2d6d3abfa9b65bc366506d6ad7846ff54"
+  integrity sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw==
+
+"@protobufjs/utf8@^1.1.0":
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/@protobufjs/utf8/-/utf8-1.1.0.tgz#a777360b5b39a1a2e5106f8e858f2fd2d060c570"
+  integrity sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw==
+
+"@scure/base@~1.1.0":
+  version "1.1.9"
+  resolved "https://registry.yarnpkg.com/@scure/base/-/base-1.1.9.tgz#e5e142fbbfe251091f9c5f1dd4c834ac04c3dbd1"
+  integrity sha512-8YKhl8GHiNI/pU2VMaofa2Tor7PJRAjwQLBBuilkJ9L5+13yVbC7JO/wS7piioAvPSwR3JKM1IJ/u4xQzbcXKg==
+
+"@scure/base@~1.2.1":
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/@scure/base/-/base-1.2.1.tgz#dd0b2a533063ca612c17aa9ad26424a2ff5aa865"
+  integrity sha512-DGmGtC8Tt63J5GfHgfl5CuAXh96VF/LD8K9Hr/Gv0J2lAoRGlPOMpqMpMbCTOoOJMZCk2Xt+DskdDyn6dEFdzQ==
+
+"@scure/bip32@^1.3.0":
+  version "1.6.1"
+  resolved "https://registry.yarnpkg.com/@scure/bip32/-/bip32-1.6.1.tgz#848eca1bc96f6b5ce6aa750798853fb142dace05"
+  integrity sha512-jSO+5Ud1E588Y+LFo8TaB8JVPNAZw/lGGao+1SepHDeTs2dFLurdNIAgUuDlwezqEjRjElkCJajVrtrZaBxvaQ==
+  dependencies:
+    "@noble/curves" "~1.8.0"
+    "@noble/hashes" "~1.7.0"
+    "@scure/base" "~1.2.1"
+
+"@scure/bip39@1.1.0":
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/@scure/bip39/-/bip39-1.1.0.tgz#92f11d095bae025f166bef3defcc5bf4945d419a"
+  integrity sha512-pwrPOS16VeTKg98dYXQyIjJEcWfz7/1YJIwxUEPFfQPtc86Ym/1sVgQ2RLoD43AazMk2l/unK4ITySSpW2+82w==
+  dependencies:
+    "@noble/hashes" "~1.1.1"
+    "@scure/base" "~1.1.0"
+
+"@scure/bip39@^1.2.0":
+  version "1.5.1"
+  resolved "https://registry.yarnpkg.com/@scure/bip39/-/bip39-1.5.1.tgz#a056868d672c7203a6035c808893742a79e151f6"
+  integrity sha512-GnlufVSP9UdAo/H2Patfv22VTtpNTyfi+I3qCKpvuB5l1KWzEYx+l2TNpBy9Ksh4xTs3Rn06tBlpWCi/1Vz8gw==
+  dependencies:
+    "@noble/hashes" "~1.7.0"
+    "@scure/base" "~1.2.1"
+
+"@solana/buffer-layout-utils@^0.2.0":
+  version "0.2.0"
+  resolved "https://registry.yarnpkg.com/@solana/buffer-layout-utils/-/buffer-layout-utils-0.2.0.tgz#b45a6cab3293a2eb7597cceb474f229889d875ca"
+  integrity sha512-szG4sxgJGktbuZYDg2FfNmkMi0DYQoVjN2h7ta1W1hPrwzarcFLBq9UpX1UjNXsNpT9dn+chgprtWGioUAr4/g==
+  dependencies:
+    "@solana/buffer-layout" "^4.0.0"
+    "@solana/web3.js" "^1.32.0"
+    bigint-buffer "^1.1.5"
+    bignumber.js "^9.0.1"
+
+"@solana/buffer-layout@^4.0.0", "@solana/buffer-layout@^4.0.1":
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/@solana/buffer-layout/-/buffer-layout-4.0.1.tgz#b996235eaec15b1e0b5092a8ed6028df77fa6c15"
+  integrity sha512-E1ImOIAD1tBZFRdjeM4/pzTiTApC0AOBGwyAMS4fwIodCWArzJ3DWdoh8cKxeFM2fElkxBh2Aqts1BPC373rHA==
+  dependencies:
+    buffer "~6.0.3"
+
+"@solana/codecs-core@2.0.0-rc.1":
+  version "2.0.0-rc.1"
+  resolved "https://registry.yarnpkg.com/@solana/codecs-core/-/codecs-core-2.0.0-rc.1.tgz#1a2d76b9c7b9e7b7aeb3bd78be81c2ba21e3ce22"
+  integrity sha512-bauxqMfSs8EHD0JKESaNmNuNvkvHSuN3bbWAF5RjOfDu2PugxHrvRebmYauvSumZ3cTfQ4HJJX6PG5rN852qyQ==
+  dependencies:
+    "@solana/errors" "2.0.0-rc.1"
+
+"@solana/codecs-data-structures@2.0.0-rc.1":
+  version "2.0.0-rc.1"
+  resolved "https://registry.yarnpkg.com/@solana/codecs-data-structures/-/codecs-data-structures-2.0.0-rc.1.tgz#d47b2363d99fb3d643f5677c97d64a812982b888"
+  integrity sha512-rinCv0RrAVJ9rE/rmaibWJQxMwC5lSaORSZuwjopSUE6T0nb/MVg6Z1siNCXhh/HFTOg0l8bNvZHgBcN/yvXog==
+  dependencies:
+    "@solana/codecs-core" "2.0.0-rc.1"
+    "@solana/codecs-numbers" "2.0.0-rc.1"
+    "@solana/errors" "2.0.0-rc.1"
+
+"@solana/codecs-numbers@2.0.0-rc.1":
+  version "2.0.0-rc.1"
+  resolved "https://registry.yarnpkg.com/@solana/codecs-numbers/-/codecs-numbers-2.0.0-rc.1.tgz#f34978ddf7ea4016af3aaed5f7577c1d9869a614"
+  integrity sha512-J5i5mOkvukXn8E3Z7sGIPxsThRCgSdgTWJDQeZvucQ9PT6Y3HiVXJ0pcWiOWAoQ3RX8e/f4I3IC+wE6pZiJzDQ==
+  dependencies:
+    "@solana/codecs-core" "2.0.0-rc.1"
+    "@solana/errors" "2.0.0-rc.1"
+
+"@solana/codecs-strings@2.0.0-rc.1":
+  version "2.0.0-rc.1"
+  resolved "https://registry.yarnpkg.com/@solana/codecs-strings/-/codecs-strings-2.0.0-rc.1.tgz#e1d9167075b8c5b0b60849f8add69c0f24307018"
+  integrity sha512-9/wPhw8TbGRTt6mHC4Zz1RqOnuPTqq1Nb4EyuvpZ39GW6O2t2Q7Q0XxiB3+BdoEjwA2XgPw6e2iRfvYgqty44g==
+  dependencies:
+    "@solana/codecs-core" "2.0.0-rc.1"
+    "@solana/codecs-numbers" "2.0.0-rc.1"
+    "@solana/errors" "2.0.0-rc.1"
+
+"@solana/codecs@2.0.0-rc.1":
+  version "2.0.0-rc.1"
+  resolved "https://registry.yarnpkg.com/@solana/codecs/-/codecs-2.0.0-rc.1.tgz#146dc5db58bd3c28e04b4c805e6096c2d2a0a875"
+  integrity sha512-qxoR7VybNJixV51L0G1RD2boZTcxmwUWnKCaJJExQ5qNKwbpSyDdWfFJfM5JhGyKe9DnPVOZB+JHWXnpbZBqrQ==
+  dependencies:
+    "@solana/codecs-core" "2.0.0-rc.1"
+    "@solana/codecs-data-structures" "2.0.0-rc.1"
+    "@solana/codecs-numbers" "2.0.0-rc.1"
+    "@solana/codecs-strings" "2.0.0-rc.1"
+    "@solana/options" "2.0.0-rc.1"
+
+"@solana/errors@2.0.0-rc.1":
+  version "2.0.0-rc.1"
+  resolved "https://registry.yarnpkg.com/@solana/errors/-/errors-2.0.0-rc.1.tgz#3882120886eab98a37a595b85f81558861b29d62"
+  integrity sha512-ejNvQ2oJ7+bcFAYWj225lyRkHnixuAeb7RQCixm+5mH4n1IA4Qya/9Bmfy5RAAHQzxK43clu3kZmL5eF9VGtYQ==
+  dependencies:
+    chalk "^5.3.0"
+    commander "^12.1.0"
+
+"@solana/options@2.0.0-rc.1":
+  version "2.0.0-rc.1"
+  resolved "https://registry.yarnpkg.com/@solana/options/-/options-2.0.0-rc.1.tgz#06924ba316dc85791fc46726a51403144a85fc4d"
+  integrity sha512-mLUcR9mZ3qfHlmMnREdIFPf9dpMc/Bl66tLSOOWxw4ml5xMT2ohFn7WGqoKcu/UHkT9CrC6+amEdqCNvUqI7AA==
+  dependencies:
+    "@solana/codecs-core" "2.0.0-rc.1"
+    "@solana/codecs-data-structures" "2.0.0-rc.1"
+    "@solana/codecs-numbers" "2.0.0-rc.1"
+    "@solana/codecs-strings" "2.0.0-rc.1"
+    "@solana/errors" "2.0.0-rc.1"
+
+"@solana/spl-token-metadata@^0.1.2":
+  version "0.1.6"
+  resolved "https://registry.yarnpkg.com/@solana/spl-token-metadata/-/spl-token-metadata-0.1.6.tgz#d240947aed6e7318d637238022a7b0981b32ae80"
+  integrity sha512-7sMt1rsm/zQOQcUWllQX9mD2O6KhSAtY1hFR2hfFwgqfFWzSY9E9GDvFVNYUI1F0iQKcm6HmePU9QbKRXTEBiA==
+  dependencies:
+    "@solana/codecs" "2.0.0-rc.1"
+
+"@solana/spl-token@^0.3.5":
+  version "0.3.11"
+  resolved "https://registry.yarnpkg.com/@solana/spl-token/-/spl-token-0.3.11.tgz#cdc10f9472b29b39c8983c92592cadd06627fb9a"
+  integrity sha512-bvohO3rIMSVL24Pb+I4EYTJ6cL82eFpInEXD/I8K8upOGjpqHsKUoAempR/RnUlI1qSFNyFlWJfu6MNUgfbCQQ==
+  dependencies:
+    "@solana/buffer-layout" "^4.0.0"
+    "@solana/buffer-layout-utils" "^0.2.0"
+    "@solana/spl-token-metadata" "^0.1.2"
+    buffer "^6.0.3"
+
+"@solana/web3.js@^1.32.0", "@solana/web3.js@^1.36.0", "@solana/web3.js@^1.66.2", "@solana/web3.js@^1.68.0":
+  version "1.98.0"
+  resolved "https://registry.yarnpkg.com/@solana/web3.js/-/web3.js-1.98.0.tgz#21ecfe8198c10831df6f0cfde7f68370d0405917"
+  integrity sha512-nz3Q5OeyGFpFCR+erX2f6JPt3sKhzhYcSycBCSPkWjzSVDh/Rr1FqTVMRe58FKO16/ivTUcuJjeS5MyBvpkbzA==
+  dependencies:
+    "@babel/runtime" "^7.25.0"
+    "@noble/curves" "^1.4.2"
+    "@noble/hashes" "^1.4.0"
+    "@solana/buffer-layout" "^4.0.1"
+    agentkeepalive "^4.5.0"
+    bigint-buffer "^1.1.5"
+    bn.js "^5.2.1"
+    borsh "^0.7.0"
+    bs58 "^4.0.1"
+    buffer "6.0.3"
+    fast-stable-stringify "^1.0.0"
+    jayson "^4.1.1"
+    node-fetch "^2.7.0"
+    rpc-websockets "^9.0.2"
+    superstruct "^2.0.2"
+
+"@suchipi/femver@^1.0.0":
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/@suchipi/femver/-/femver-1.0.0.tgz#4909dcc069695e07bd23a64c4bfe411d11d9692f"
+  integrity sha512-bprE8+K5V+DPX7q2e2K57ImqNBdfGHDIWaGI5xHxZoxbKOuQZn4wzPiUxOAHnsUr3w3xHrWXwN7gnG/iIuEMIg==
+
+"@swc/helpers@^0.5.11":
+  version "0.5.15"
+  resolved "https://registry.yarnpkg.com/@swc/helpers/-/helpers-0.5.15.tgz#79efab344c5819ecf83a43f3f9f811fc84b516d7"
+  integrity sha512-JQ5TuMi45Owi4/BIMAJBoSQoOJu12oOk/gADqlcUL9JEdHB8vyjUSsxqeNXnmXHjYKMi2WcYtezGEEhqUI/E2g==
+  dependencies:
+    tslib "^2.8.0"
+
+"@terra-money/legacy.proto@npm:@terra-money/terra.proto@^0.1.7":
+  version "0.1.7"
+  resolved "https://registry.yarnpkg.com/@terra-money/terra.proto/-/terra.proto-0.1.7.tgz#59c18f30da10d43200bab3ba8feb5b17e43a365f"
+  integrity sha512-NXD7f6pQCulvo6+mv6MAPzhOkUzRjgYVuHZE/apih+lVnPG5hDBU0rRYnOGGofwvKT5/jQoOENnFn/gioWWnyQ==
+  dependencies:
+    google-protobuf "^3.17.3"
+    long "^4.0.0"
+    protobufjs "~6.11.2"
+
+"@terra-money/terra.js@3.1.9":
+  version "3.1.9"
+  resolved "https://registry.yarnpkg.com/@terra-money/terra.js/-/terra.js-3.1.9.tgz#2d78f241902238ba0613a089c56c334eb1351c68"
+  integrity sha512-JulSvOHLM56fL7s+cIjIbZeWPBluq883X1soWxA4TG5rKkDythT/DHeLXr3jP5Ld/26VENPSg6lNvK7cEYKpiw==
+  dependencies:
+    "@classic-terra/terra.proto" "^1.1.0"
+    "@terra-money/terra.proto" "^2.1.0"
+    axios "^0.27.2"
+    bech32 "^2.0.0"
+    bip32 "^2.0.6"
+    bip39 "^3.0.3"
+    bufferutil "^4.0.3"
+    decimal.js "^10.2.1"
+    jscrypto "^1.0.1"
+    readable-stream "^3.6.0"
+    secp256k1 "^4.0.2"
+    tmp "^0.2.1"
+    utf-8-validate "^5.0.5"
+    ws "^7.5.9"
+
+"@terra-money/terra.proto@^2.1.0":
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/@terra-money/terra.proto/-/terra.proto-2.1.0.tgz#5a2ed85fc8146a346d6095adfc5d205b6fb6d387"
+  integrity sha512-rhaMslv3Rkr+QsTQEZs64FKA4QlfO0DfQHaR6yct/EovenMkibDEQ63dEL6yJA6LCaEQGYhyVB9JO9pTUA8ybw==
+  dependencies:
+    "@improbable-eng/grpc-web" "^0.14.1"
+    google-protobuf "^3.17.3"
+    long "^4.0.0"
+    protobufjs "~6.11.2"
+
+"@types/bn.js@5.1.6", "@types/bn.js@^5.1.0":
+  version "5.1.6"
+  resolved "https://registry.yarnpkg.com/@types/bn.js/-/bn.js-5.1.6.tgz#9ba818eec0c85e4d3c679518428afdf611d03203"
+  integrity sha512-Xh8vSwUeMKeYYrj3cX4lGQgFSF/N03r+tv4AiLl1SucqV+uTQpxRcnM8AkXKHwYP9ZPXOYXRr2KPXpVlIvqh9w==
+  dependencies:
+    "@types/node" "*"
+
+"@types/bn.js@^4.11.3":
+  version "4.11.6"
+  resolved "https://registry.yarnpkg.com/@types/bn.js/-/bn.js-4.11.6.tgz#c306c70d9358aaea33cd4eda092a742b9505967c"
+  integrity sha512-pqr857jrp2kPuO9uRjZ3PwnJTjoQy+fcdxvBTvHm6dkmEL9q+hDD/2j/0ELOBPtPnS8LjCX0gI9nbl8lVkadpg==
+  dependencies:
+    "@types/node" "*"
+
+"@types/chai@^4.3.0":
+  version "4.3.20"
+  resolved "https://registry.yarnpkg.com/@types/chai/-/chai-4.3.20.tgz#cb291577ed342ca92600430841a00329ba05cecc"
+  integrity sha512-/pC9HAB5I/xMlc5FP77qjCnI16ChlJfW0tGa0IUcFn38VJrTV6DeZ60NU5KZBtaOZqjdpwTWohz5HU1RrhiYxQ==
+
+"@types/connect@^3.4.33":
+  version "3.4.38"
+  resolved "https://registry.yarnpkg.com/@types/connect/-/connect-3.4.38.tgz#5ba7f3bc4fbbdeaff8dded952e5ff2cc53f8d858"
+  integrity sha512-K6uROf1LD88uDQqJCktA4yzL1YYAK6NgfsI0v/mTgyPKWsX1CnJ0XPSDhViejru1GcRkLWb8RlzFYJRqGUbaug==
+  dependencies:
+    "@types/node" "*"
+
+"@types/json5@^0.0.29":
+  version "0.0.29"
+  resolved "https://registry.yarnpkg.com/@types/json5/-/json5-0.0.29.tgz#ee28707ae94e11d2b827bcbe5270bcea7f3e71ee"
+  integrity sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==
+
+"@types/lodash.values@^4.3.6":
+  version "4.3.9"
+  resolved "https://registry.yarnpkg.com/@types/lodash.values/-/lodash.values-4.3.9.tgz#a30e7d7cb715b1a01726838844da1dfbca549c2e"
+  integrity sha512-IJ20OEfqNwm3k8ENwoM3q0yOs4UMpgtD4GqxB4lwBHToGthHWqhyh5DdSgQjioocz0QK2SSBkJfCq95ZTV8BTw==
+  dependencies:
+    "@types/lodash" "*"
+
+"@types/lodash@*":
+  version "4.17.14"
+  resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.17.14.tgz#bafc053533f4cdc5fcc9635af46a963c1f3deaff"
+  integrity sha512-jsxagdikDiDBeIRaPYtArcT8my4tN1og7MtMRquFT3XNA6axxyHDRUemqDz/taRDdOUn0GnGHRCuff4q48sW9A==
+
+"@types/long@^4.0.1", "@types/long@^4.0.2":
+  version "4.0.2"
+  resolved "https://registry.yarnpkg.com/@types/long/-/long-4.0.2.tgz#b74129719fc8d11c01868010082d483b7545591a"
+  integrity sha512-MqTGEo5bj5t157U6fA/BiDynNkn0YknVdh48CMPkTSpFTVmvao5UQmm7uEF6xBEo7qIMAlY/JSleYaE6VOdpaA==
+
+"@types/mocha@^9.0.0":
+  version "9.1.1"
+  resolved "https://registry.yarnpkg.com/@types/mocha/-/mocha-9.1.1.tgz#e7c4f1001eefa4b8afbd1eee27a237fee3bf29c4"
+  integrity sha512-Z61JK7DKDtdKTWwLeElSEBcWGRLY8g95ic5FoQqI9CMx0ns/Ghep3B4DfcEimiKMvtamNVULVNKEsiwV3aQmXw==
+
+"@types/node@*":
+  version "22.10.4"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-22.10.4.tgz#da36bebcc4b124f3d62bfde1cd1dafd7763949c1"
+  integrity sha512-99l6wv4HEzBQhvaU/UGoeBoCK61SCROQaCCGyQSgX2tEQ3rKkNZ2S7CEWnS/4s1LV+8ODdK21UeyR1fHP2mXug==
+  dependencies:
+    undici-types "~6.20.0"
+
+"@types/node@10.12.18":
+  version "10.12.18"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-10.12.18.tgz#1d3ca764718915584fcd9f6344621b7672665c67"
+  integrity sha512-fh+pAqt4xRzPfqA6eh3Z2y6fyZavRIumvjhaCL753+TVkGKGhpPeyrJG2JftD0T9q4GF00KjefsQ+PQNDdWQaQ==
+
+"@types/node@>=13.7.0":
+  version "22.10.5"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-22.10.5.tgz#95af89a3fb74a2bb41ef9927f206e6472026e48b"
+  integrity sha512-F8Q+SeGimwOo86fiovQh8qiXfFEh2/ocYv7tU5pJ3EXMSSxk1Joj5wefpFK2fHTf/N6HKGSxIDBT9f3gCxXPkQ==
+  dependencies:
+    undici-types "~6.20.0"
+
+"@types/node@^12.12.54":
+  version "12.20.55"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-12.20.55.tgz#c329cbd434c42164f846b909bd6f85b5537f6240"
+  integrity sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ==
+
+"@types/node@^18.0.3":
+  version "18.19.70"
+  resolved "https://registry.yarnpkg.com/@types/node/-/node-18.19.70.tgz#5a77508f5568d16fcd3b711c8102d7a430a04df7"
+  integrity sha512-RE+K0+KZoEpDUbGGctnGdkrLFwi1eYKTlIHNl2Um98mUkGsm1u2Ff6Ltd0e8DktTtC98uy7rSj+hO8t/QuLoVQ==
+  dependencies:
+    undici-types "~5.26.4"
+
+"@types/pbkdf2@^3.0.0":
+  version "3.1.2"
+  resolved "https://registry.yarnpkg.com/@types/pbkdf2/-/pbkdf2-3.1.2.tgz#2dc43808e9985a2c69ff02e2d2027bd4fe33e8dc"
+  integrity sha512-uRwJqmiXmh9++aSu1VNEn3iIxWOhd8AHXNSdlaLfdAAdSTY9jYVeGWnzejM3dvrkbqE3/hyQkQQ29IFATEGlew==
+  dependencies:
+    "@types/node" "*"
+
+"@types/secp256k1@^4.0.1":
+  version "4.0.6"
+  resolved "https://registry.yarnpkg.com/@types/secp256k1/-/secp256k1-4.0.6.tgz#d60ba2349a51c2cbc5e816dcd831a42029d376bf"
+  integrity sha512-hHxJU6PAEUn0TP4S/ZOzuTUvJWuZ6eIKeNKb5RBpODvSl6hp1Wrw4s7ATY50rklRCScUDpHzVA/DQdSjJ3UoYQ==
+  dependencies:
+    "@types/node" "*"
+
+"@types/uuid@^8.3.4":
+  version "8.3.4"
+  resolved "https://registry.yarnpkg.com/@types/uuid/-/uuid-8.3.4.tgz#bd86a43617df0594787d38b735f55c805becf1bc"
+  integrity sha512-c/I8ZRb51j+pYGAu5CrFMRxqZ2ke4y2grEBO5AUjgSkSk+qT2Ea+OdWElz/OiMf5MNpn2b17kuVBwZLQJXzihw==
+
+"@types/ws@^7.4.4":
+  version "7.4.7"
+  resolved "https://registry.yarnpkg.com/@types/ws/-/ws-7.4.7.tgz#f7c390a36f7a0679aa69de2d501319f4f8d9b702"
+  integrity sha512-JQbbmxZTZehdc2iszGKs5oC3NFnjeay7mtAWrdt7qNtAVK0g19muApzAy4bm9byz79xa2ZnO/BOBC2R8RC5Lww==
+  dependencies:
+    "@types/node" "*"
+
+"@types/ws@^8.2.2":
+  version "8.5.13"
+  resolved "https://registry.yarnpkg.com/@types/ws/-/ws-8.5.13.tgz#6414c280875e2691d0d1e080b05addbf5cb91e20"
+  integrity sha512-osM/gWBTPKgHV8XkTunnegTRIsvF6owmf5w+JtAfOw472dptdm0dlGv4xCt6GwQRcC2XVOvvRE/0bAoQcL2QkA==
+  dependencies:
+    "@types/node" "*"
+
+"@ungap/promise-all-settled@1.1.2":
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/@ungap/promise-all-settled/-/promise-all-settled-1.1.2.tgz#aa58042711d6e3275dd37dc597e5d31e8c290a44"
+  integrity sha512-sL/cEvJWAnClXw0wHk85/2L0G6Sj8UB0Ctc1TEMbKSsmpRosqhwj9gWgFRZSrBr2f9tiXISwNhCPmlfqUqyb9Q==
+
+"@wry/caches@^1.0.0":
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/@wry/caches/-/caches-1.0.1.tgz#8641fd3b6e09230b86ce8b93558d44cf1ece7e52"
+  integrity sha512-bXuaUNLVVkD20wcGBWRyo7j9N3TxePEWFZj2Y+r9OoUzfqmavM84+mFykRicNsBqatba5JLay1t48wxaXaWnlA==
+  dependencies:
+    tslib "^2.3.0"
+
+"@wry/context@^0.7.0":
+  version "0.7.4"
+  resolved "https://registry.yarnpkg.com/@wry/context/-/context-0.7.4.tgz#e32d750fa075955c4ab2cfb8c48095e1d42d5990"
+  integrity sha512-jmT7Sb4ZQWI5iyu3lobQxICu2nC/vbUhP0vIdd6tHC9PTfenmRmuIFqktc6GH9cgi+ZHnsLWPvfSvc4DrYmKiQ==
+  dependencies:
+    tslib "^2.3.0"
+
+"@wry/equality@^0.5.6":
+  version "0.5.7"
+  resolved "https://registry.yarnpkg.com/@wry/equality/-/equality-0.5.7.tgz#72ec1a73760943d439d56b7b1e9985aec5d497bb"
+  integrity sha512-BRFORjsTuQv5gxcXsuDXx6oGRhuVsEGwZy6LOzRRfgu+eSfxbhUQ9L9YtSEIuIjY/o7g3iWFjrc5eSY1GXP2Dw==
+  dependencies:
+    tslib "^2.3.0"
+
+"@wry/trie@^0.5.0":
+  version "0.5.0"
+  resolved "https://registry.yarnpkg.com/@wry/trie/-/trie-0.5.0.tgz#11e783f3a53f6e4cd1d42d2d1323f5bc3fa99c94"
+  integrity sha512-FNoYzHawTMk/6KMQoEG5O4PuioX19UbwdQKF44yw0nLfOypfQdjtfZzo/UIJWAJ23sNIFbD1Ug9lbaDGMwbqQA==
+  dependencies:
+    tslib "^2.3.0"
+
+"@xpla/xpla.js@^0.2.1":
+  version "0.2.3"
+  resolved "https://registry.yarnpkg.com/@xpla/xpla.js/-/xpla.js-0.2.3.tgz#c614d98e599fe03f9edc425f50f66481983e8dac"
+  integrity sha512-Tfk7hCGWXtwr08reY3Pi6dmzIqFbzri9jcyzJdfNmdo4cN0PMwpRJuZZcPmtxiIUnNef3AN1E/6nJUD5MKniuA==
+  dependencies:
+    "@ethersproject/bytes" "^5.6.1"
+    "@ethersproject/keccak256" "^5.6.1"
+    "@ethersproject/signing-key" "^5.6.2"
+    "@terra-money/legacy.proto" "npm:@terra-money/terra.proto@^0.1.7"
+    "@terra-money/terra.proto" "^2.1.0"
+    axios "^0.26.1"
+    bech32 "^2.0.0"
+    bip32 "^2.0.6"
+    bip39 "^3.0.3"
+    bufferutil "^4.0.3"
+    crypto-addr-codec "^0.1.7"
+    decimal.js "^10.2.1"
+    elliptic "^6.5.4"
+    ethereumjs-util "^7.1.5"
+    jscrypto "^1.0.1"
+    readable-stream "^3.6.0"
+    secp256k1 "^4.0.2"
+    tmp "^0.2.1"
+    utf-8-validate "^5.0.5"
+    ws "^7.5.8"
+
+JSONStream@^1.3.5:
+  version "1.3.5"
+  resolved "https://registry.yarnpkg.com/JSONStream/-/JSONStream-1.3.5.tgz#3208c1f08d3a4d99261ab64f92302bc15e111ca0"
+  integrity sha512-E+iruNOY8VV9s4JEbe1aNEm6MiszPRr/UfcHMz0TQh1BXSxHK+ASV1R6W4HpjBhSeS+54PIsAMCBmwD06LLsqQ==
+  dependencies:
+    jsonparse "^1.2.0"
+    through ">=2.2.7 <3"
+
+acorn@7.1.1:
+  version "7.1.1"
+  resolved "https://registry.yarnpkg.com/acorn/-/acorn-7.1.1.tgz#e35668de0b402f359de515c5482a1ab9f89a69bf"
+  integrity sha512-add7dgA5ppRPxCFJoAGfMDi7PIBXq1RtGo7BhbLaxwrXPOmw8gq48Y9ozT01hUKy9byMjlR20EJhu5zlkErEkg==
+
+aes-js@3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/aes-js/-/aes-js-3.0.0.tgz#e21df10ad6c2053295bcbb8dab40b09dbea87e4d"
+  integrity sha512-H7wUZRn8WpTq9jocdxQ2c8x2sKo9ZVmzfRE13GiNJXfp7NcKYEdvl3vspKjXox6RIG2VtaRe4JFvxG4rqp2Zuw==
+
+agentkeepalive@^4.5.0:
+  version "4.6.0"
+  resolved "https://registry.yarnpkg.com/agentkeepalive/-/agentkeepalive-4.6.0.tgz#35f73e94b3f40bf65f105219c623ad19c136ea6a"
+  integrity sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ==
+  dependencies:
+    humanize-ms "^1.2.1"
+
+algo-msgpack-with-bigint@^2.1.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/algo-msgpack-with-bigint/-/algo-msgpack-with-bigint-2.1.1.tgz#38bb717220525b3ff42232eefdcd9efb9ad405d6"
+  integrity sha512-F1tGh056XczEaEAqu7s+hlZUDWwOBT70Eq0lfMpBP2YguSQVyxRbprLq5rELXKQOyOaixTWYhMeMQMzP0U5FoQ==
+
+algosdk@^2.4.0:
+  version "2.9.0"
+  resolved "https://registry.yarnpkg.com/algosdk/-/algosdk-2.9.0.tgz#a93cc26675e5310667ce5e9667305ce48a1818f9"
+  integrity sha512-o0n0nLMbTX6SFQdMUk2/2sy50jmEmZk5OTPYSh2aAeP8DUPxrhjMPfwGsYNvaO+qk75MixC2eWpfA9vygCQ/Mg==
+  dependencies:
+    algo-msgpack-with-bigint "^2.1.1"
+    buffer "^6.0.3"
+    hi-base32 "^0.5.1"
+    js-sha256 "^0.9.0"
+    js-sha3 "^0.8.0"
+    js-sha512 "^0.8.0"
+    json-bigint "^1.0.0"
+    tweetnacl "^1.0.3"
+    vlq "^2.0.4"
+
+ansi-colors@4.1.1:
+  version "4.1.1"
+  resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-4.1.1.tgz#cbb9ae256bf750af1eab344f229aa27fe94ba348"
+  integrity sha512-JoX0apGbHaUJBNl6yF+p6JAFYZ666/hhCGKN5t9QFjbJQKUU/g8MNbFDbvfrgKXvI1QpZplPOnwIo99lX/AAmA==
+
+ansi-regex@^5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304"
+  integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ==
+
+ansi-styles@^3.2.1:
+  version "3.2.1"
+  resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d"
+  integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==
+  dependencies:
+    color-convert "^1.9.0"
+
+ansi-styles@^4.0.0, ansi-styles@^4.1.0:
+  version "4.3.0"
+  resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937"
+  integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg==
+  dependencies:
+    color-convert "^2.0.1"
+
+anymatch@~3.1.2:
+  version "3.1.3"
+  resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.3.tgz#790c58b19ba1720a84205b57c618d5ad8524973e"
+  integrity sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==
+  dependencies:
+    normalize-path "^3.0.0"
+    picomatch "^2.0.4"
+
+aptos@1.5.0:
+  version "1.5.0"
+  resolved "https://registry.yarnpkg.com/aptos/-/aptos-1.5.0.tgz#43263b13c878d2497d875d50aaee10355932c2e6"
+  integrity sha512-N7OuRtU7IYHkDkNx+4QS3g/QQGCp+36KzYn3oXPmT7Kttfuv+UKliQVdjy3cLmwd/DCQSh9ObTovwdxnHjUn0g==
+  dependencies:
+    "@noble/hashes" "1.1.3"
+    "@scure/bip39" "1.1.0"
+    axios "0.27.2"
+    form-data "4.0.0"
+    tweetnacl "1.0.3"
+
+argparse@^2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38"
+  integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q==
+
+arrify@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/arrify/-/arrify-1.0.1.tgz#898508da2226f380df904728456849c1501a4b0d"
+  integrity sha512-3CYzex9M9FGQjCGMGyi6/31c8GJbgb0qGyrx5HWxPd0aCwh4cB2YjMb2Xf9UuoogrMrlO9cTqnB5rI5GHZTcUA==
+
+assertion-error@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/assertion-error/-/assertion-error-1.1.0.tgz#e60b6b0e8f301bd97e5375215bda406c85118c0b"
+  integrity sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw==
+
+asynckit@^0.4.0:
+  version "0.4.0"
+  resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79"
+  integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==
+
+axios@0.27.2, axios@^0.27.2:
+  version "0.27.2"
+  resolved "https://registry.yarnpkg.com/axios/-/axios-0.27.2.tgz#207658cc8621606e586c85db4b41a750e756d972"
+  integrity sha512-t+yRIyySRTp/wua5xEr+z1q60QmLq8ABsS5O9Me1AsE5dfKqgnCFzwiCZZ/cGNd1lq4/7akDWMxdhVlucjmnOQ==
+  dependencies:
+    follow-redirects "^1.14.9"
+    form-data "^4.0.0"
+
+axios@^0.21.1, axios@^0.21.2:
+  version "0.21.4"
+  resolved "https://registry.yarnpkg.com/axios/-/axios-0.21.4.tgz#c67b90dc0568e5c1cf2b0b858c43ba28e2eda575"
+  integrity sha512-ut5vewkiu8jjGBdqpM44XxjuCjq9LAKeHVmoVfHVzy8eHgxxq8SbAVQNovDA8mVi05kP0Ea/n/UzcSHcTJQfNg==
+  dependencies:
+    follow-redirects "^1.14.0"
+
+axios@^0.24.0:
+  version "0.24.0"
+  resolved "https://registry.yarnpkg.com/axios/-/axios-0.24.0.tgz#804e6fa1e4b9c5288501dd9dff56a7a0940d20d6"
+  integrity sha512-Q6cWsys88HoPgAaFAVUb0WpPk0O8iTeisR9IMqy9G8AbO4NlpVknrnQS03zzF9PGAWgO3cgletO3VjV/P7VztA==
+  dependencies:
+    follow-redirects "^1.14.4"
+
+axios@^0.26.1:
+  version "0.26.1"
+  resolved "https://registry.yarnpkg.com/axios/-/axios-0.26.1.tgz#1ede41c51fcf51bbbd6fd43669caaa4f0495aaa9"
+  integrity sha512-fPwcX4EvnSHuInCMItEhAGnaSEXRBjtzh9fOtsE6E1G6p7vl7edEeZe11QHf18+6+9gR5PbKV/sGKNaD8YaMeA==
+  dependencies:
+    follow-redirects "^1.14.8"
+
+axios@^1.6.4:
+  version "1.7.9"
+  resolved "https://registry.yarnpkg.com/axios/-/axios-1.7.9.tgz#d7d071380c132a24accda1b2cfc1535b79ec650a"
+  integrity sha512-LhLcE7Hbiryz8oMDdDptSrWowmB4Bl6RCt6sIJKpRB4XtVf0iEgewX3au/pJqm+Py1kCASkb/FFKjxQaLtxJvw==
+  dependencies:
+    follow-redirects "^1.15.6"
+    form-data "^4.0.0"
+    proxy-from-env "^1.1.0"
+
+balanced-match@^1.0.0:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee"
+  integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==
+
+base-x@^3.0.2, base-x@^3.0.8:
+  version "3.0.10"
+  resolved "https://registry.yarnpkg.com/base-x/-/base-x-3.0.10.tgz#62de58653f8762b5d6f8d9fe30fa75f7b2585a75"
+  integrity sha512-7d0s06rR9rYaIWHkpfLIFICM/tkSVdoPC9qYAQRpxn9DdKNWNsKC0uk++akckyLq16Tx2WIinnZ6WRriAt6njQ==
+  dependencies:
+    safe-buffer "^5.0.1"
+
+base-x@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/base-x/-/base-x-4.0.0.tgz#d0e3b7753450c73f8ad2389b5c018a4af7b2224a"
+  integrity sha512-FuwxlW4H5kh37X/oW59pwTzzTKRzfrrQwhmyspRM7swOEZcHtDZSCt45U6oKgtuFE+WYPblePMVIPR4RZrh/hw==
+
+base64-js@^1.3.0, base64-js@^1.3.1, base64-js@^1.5.1:
+  version "1.5.1"
+  resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a"
+  integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==
+
+bech32@1.1.4, bech32@^1.1.4:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/bech32/-/bech32-1.1.4.tgz#e38c9f37bf179b8eb16ae3a772b40c356d4832e9"
+  integrity sha512-s0IrSOzLlbvX7yp4WBfPITzpAU8sqQcpsmwXDiKwrG4r491vwCO/XpejasRNl0piBMe/DvP4Tz0mIS/X1DPJBQ==
+
+bech32@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/bech32/-/bech32-2.0.0.tgz#078d3686535075c8c79709f054b1b226a133b355"
+  integrity sha512-LcknSilhIGatDAsY1ak2I8VtGaHNhgMSYVxFrGLXv+xLHytaKZKcaUJJUE7qmBr7h33o5YQwP55pMI0xmkpJwg==
+
+big-integer@1.6.36:
+  version "1.6.36"
+  resolved "https://registry.yarnpkg.com/big-integer/-/big-integer-1.6.36.tgz#78631076265d4ae3555c04f85e7d9d2f3a071a36"
+  integrity sha512-t70bfa7HYEA1D9idDbmuv7YbsbVkQ+Hp+8KFSul4aE5e/i1bjCNIRYJZlA8Q8p0r9T8cF/RVvwUgRA//FydEyg==
+
+bigint-buffer@^1.1.5:
+  version "1.1.5"
+  resolved "https://registry.yarnpkg.com/bigint-buffer/-/bigint-buffer-1.1.5.tgz#d038f31c8e4534c1f8d0015209bf34b4fa6dd442"
+  integrity sha512-trfYco6AoZ+rKhKnxA0hgX0HAbVP/s808/EuDSe2JDzUnCp/xAsli35Orvk67UrTEcwuxZqYZDmfA2RXJgxVvA==
+  dependencies:
+    bindings "^1.3.0"
+
+bignumber.js@^9.0.0, bignumber.js@^9.0.1:
+  version "9.1.2"
+  resolved "https://registry.yarnpkg.com/bignumber.js/-/bignumber.js-9.1.2.tgz#b7c4242259c008903b13707983b5f4bbd31eda0c"
+  integrity sha512-2/mKyZH9K85bzOEfhXDBFZTGd1CTs+5IHpeFQo9luiBG7hghdC851Pj2WAhb6E3R6b9tZj/XKhbg4fum+Kepug==
+
+binary-extensions@^2.0.0:
+  version "2.3.0"
+  resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.3.0.tgz#f6e14a97858d327252200242d4ccfe522c445522"
+  integrity sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==
+
+binary-parser@^2.2.1:
+  version "2.2.1"
+  resolved "https://registry.yarnpkg.com/binary-parser/-/binary-parser-2.2.1.tgz#4edc6da2dc56db73fa5ba450dfe6382ede8294ce"
+  integrity sha512-5ATpz/uPDgq5GgEDxTB4ouXCde7q2lqAQlSdBRQVl/AJnxmQmhIfyxJx+0MGu//D5rHQifkfGbWWlaysG0o9NA==
+
+bindings@^1.3.0, bindings@^1.5.0:
+  version "1.5.0"
+  resolved "https://registry.yarnpkg.com/bindings/-/bindings-1.5.0.tgz#10353c9e945334bc0511a6d90b38fbc7c9c504df"
+  integrity sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==
+  dependencies:
+    file-uri-to-path "1.0.0"
+
+bip32@^2.0.6:
+  version "2.0.6"
+  resolved "https://registry.yarnpkg.com/bip32/-/bip32-2.0.6.tgz#6a81d9f98c4cd57d05150c60d8f9e75121635134"
+  integrity sha512-HpV5OMLLGTjSVblmrtYRfFFKuQB+GArM0+XP8HGWfJ5vxYBqo+DesvJwOdC2WJ3bCkZShGf0QIfoIpeomVzVdA==
+  dependencies:
+    "@types/node" "10.12.18"
+    bs58check "^2.1.1"
+    create-hash "^1.2.0"
+    create-hmac "^1.1.7"
+    tiny-secp256k1 "^1.1.3"
+    typeforce "^1.11.5"
+    wif "^2.0.6"
+
+bip39@^3.0.3, bip39@^3.0.4:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/bip39/-/bip39-3.1.0.tgz#c55a418deaf48826a6ceb34ac55b3ee1577e18a3"
+  integrity sha512-c9kiwdk45Do5GL0vJMe7tS95VjCii65mYAH7DfWl3uW8AVzXKQVUm64i3hzVybBDMp9r7j9iNxR85+ul8MdN/A==
+  dependencies:
+    "@noble/hashes" "^1.2.0"
+
+bip66@^1.1.5:
+  version "1.1.5"
+  resolved "https://registry.yarnpkg.com/bip66/-/bip66-1.1.5.tgz#01fa8748785ca70955d5011217d1b3139969ca22"
+  integrity sha512-nemMHz95EmS38a26XbbdxIYj5csHd3RMP3H5bwQknX0WYHF01qhpufP42mLOwVICuH2JmhIhXiWs89MfUGL7Xw==
+  dependencies:
+    safe-buffer "^5.0.1"
+
+blakejs@^1.1.0:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/blakejs/-/blakejs-1.2.1.tgz#5057e4206eadb4a97f7c0b6e197a505042fc3814"
+  integrity sha512-QXUSXI3QVc/gJME0dBpXrag1kbzOqCjCX8/b54ntNyW6sjtoqxqRk3LTmXzaJoh71zMsDCjM+47jS7XiwN/+fQ==
+
+bn.js@5.2.1, bn.js@^5.1.2, bn.js@^5.2.0, bn.js@^5.2.1:
+  version "5.2.1"
+  resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.2.1.tgz#0bc527a6a0d18d0aa8d5b0538ce4a77dccfa7b70"
+  integrity sha512-eXRvHzWyYPBuB4NBy0cmYQjGitUrtqwbvlzP3G6VFnNRbsZQIxQ10PbKKHt8gZ/HW/D/747aDl+QkDqg3KQLMQ==
+
+bn.js@^4.11.0, bn.js@^4.11.8, bn.js@^4.11.9:
+  version "4.12.1"
+  resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-4.12.1.tgz#215741fe3c9dba2d7e12c001d0cfdbae43975ba7"
+  integrity sha512-k8TVBiPkPJT9uHLdOKfFpqcfprwBFOAAXXozRubr7R7PfIuKvQlzcI4M0pALeqXN09vdaMbUdUj+pass+uULAg==
+
+borsh@^0.7.0:
+  version "0.7.0"
+  resolved "https://registry.yarnpkg.com/borsh/-/borsh-0.7.0.tgz#6e9560d719d86d90dc589bca60ffc8a6c51fec2a"
+  integrity sha512-CLCsZGIBCFnPtkNnieW/a8wmreDmfUtjU2m9yHrzPXIlNbqVs0AQrSatSG6vdNYUqdc83tkQi2eHfF98ubzQLA==
+  dependencies:
+    bn.js "^5.2.0"
+    bs58 "^4.0.0"
+    text-encoding-utf-8 "^1.0.2"
+
+brace-expansion@^1.1.7:
+  version "1.1.11"
+  resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd"
+  integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA==
+  dependencies:
+    balanced-match "^1.0.0"
+    concat-map "0.0.1"
+
+braces@~3.0.2:
+  version "3.0.3"
+  resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.3.tgz#490332f40919452272d55a8480adc0c441358789"
+  integrity sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==
+  dependencies:
+    fill-range "^7.1.1"
+
+brorand@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/brorand/-/brorand-1.1.0.tgz#12c25efe40a45e3c323eb8675a0a0ce57b22371f"
+  integrity sha512-cKV8tMCEpQs4hK/ik71d6LrPOnpkpGBR0wzxqr68g2m/LB2GxVYQroAjMJZRVM1Y4BCjCKc3vAamxSzOY2RP+w==
+
+browser-headers@^0.4.1:
+  version "0.4.1"
+  resolved "https://registry.yarnpkg.com/browser-headers/-/browser-headers-0.4.1.tgz#4308a7ad3b240f4203dbb45acedb38dc2d65dd02"
+  integrity sha512-CA9hsySZVo9371qEHjHZtYxV2cFtVj5Wj/ZHi8ooEsrtm4vOnl9Y9HmyYWk9q+05d7K3rdoAE0j3MVEFVvtQtg==
+
+browser-stdout@1.3.1:
+  version "1.3.1"
+  resolved "https://registry.yarnpkg.com/browser-stdout/-/browser-stdout-1.3.1.tgz#baa559ee14ced73452229bad7326467c61fabd60"
+  integrity sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw==
+
+browserify-aes@^1.0.6, browserify-aes@^1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/browserify-aes/-/browserify-aes-1.2.0.tgz#326734642f403dabc3003209853bb70ad428ef48"
+  integrity sha512-+7CHXqGuspUn/Sl5aO7Ea0xWGAtETPXNSAjHo48JfLdPWcMng33Xe4znFvQweqc/uzk5zSOI3H52CYnjCfb5hA==
+  dependencies:
+    buffer-xor "^1.0.3"
+    cipher-base "^1.0.0"
+    create-hash "^1.1.0"
+    evp_bytestokey "^1.0.3"
+    inherits "^2.0.1"
+    safe-buffer "^5.0.1"
+
+bs58@^4.0.0, bs58@^4.0.1:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/bs58/-/bs58-4.0.1.tgz#be161e76c354f6f788ae4071f63f34e8c4f0a42a"
+  integrity sha512-Ok3Wdf5vOIlBrgCvTq96gBkJw+JUEzdBgyaza5HLtPm7yTHkjRy8+JzNyHF7BHa0bNWOQIp3m5YF0nnFcOIKLw==
+  dependencies:
+    base-x "^3.0.2"
+
+bs58@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/bs58/-/bs58-5.0.0.tgz#865575b4d13c09ea2a84622df6c8cbeb54ffc279"
+  integrity sha512-r+ihvQJvahgYT50JD05dyJNKlmmSlMoOGwn1lCcEzanPglg7TxYjioQUYehQ9mAR/+hOSd2jRc/Z2y5UxBymvQ==
+  dependencies:
+    base-x "^4.0.0"
+
+bs58check@<3.0.0, bs58check@^2.1.1, bs58check@^2.1.2:
+  version "2.1.2"
+  resolved "https://registry.yarnpkg.com/bs58check/-/bs58check-2.1.2.tgz#53b018291228d82a5aa08e7d796fdafda54aebfc"
+  integrity sha512-0TS1jicxdU09dwJMNZtVAfzPi6Q6QeN0pM1Fkzrjn+XYHvzMKPU3pHVpva+769iNVSfIYWf7LJ6WR+BuuMf8cA==
+  dependencies:
+    bs58 "^4.0.0"
+    create-hash "^1.1.0"
+    safe-buffer "^5.1.2"
+
+buffer-from@^1.0.0, buffer-from@^1.1.0:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.2.tgz#2b146a6fd72e80b4f55d255f35ed59a3a9a41bd5"
+  integrity sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ==
+
+buffer-layout@^1.2.0, buffer-layout@^1.2.2:
+  version "1.2.2"
+  resolved "https://registry.yarnpkg.com/buffer-layout/-/buffer-layout-1.2.2.tgz#b9814e7c7235783085f9ca4966a0cfff112259d5"
+  integrity sha512-kWSuLN694+KTk8SrYvCqwP2WcgQjoRCiF5b4QDvkkz8EmgD+aWAIceGFKMIAdmF/pH+vpgNV3d3kAKorcdAmWA==
+
+buffer-xor@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/buffer-xor/-/buffer-xor-1.0.3.tgz#26e61ed1422fb70dd42e6e36729ed51d855fe8d9"
+  integrity sha512-571s0T7nZWK6vB67HI5dyUF7wXiNcfaPPPTl6zYCNApANjIvYJTg7hlud/+cJpdAhS7dVzqMLmfhfHR3rAcOjQ==
+
+buffer@6.0.3, buffer@^6.0.3, buffer@~6.0.3:
+  version "6.0.3"
+  resolved "https://registry.yarnpkg.com/buffer/-/buffer-6.0.3.tgz#2ace578459cc8fbe2a70aaa8f52ee63b6a74c6c6"
+  integrity sha512-FTiCpNxtwiZZHEZbcbTIcZjERVICn9yq/pDFkTl95/AxzD1naBctN7YO68riM/gLSDY7sdrMby8hofADYuuqOA==
+  dependencies:
+    base64-js "^1.3.1"
+    ieee754 "^1.2.1"
+
+bufferutil@^4.0.1, bufferutil@^4.0.3:
+  version "4.0.9"
+  resolved "https://registry.yarnpkg.com/bufferutil/-/bufferutil-4.0.9.tgz#6e81739ad48a95cad45a279588e13e95e24a800a"
+  integrity sha512-WDtdLmJvAuNNPzByAYpRo2rF1Mmradw6gvWsQKf63476DDXmomT9zUiGypLcG4ibIM67vhAj8jJRdbmEws2Aqw==
+  dependencies:
+    node-gyp-build "^4.3.0"
+
+camelcase@^5.3.1:
+  version "5.3.1"
+  resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.3.1.tgz#e3c9b31569e106811df242f715725a1f4c494320"
+  integrity sha512-L28STB170nwWS63UjtlEOE3dldQApaJXZkOI1uMFfzf3rRuPegHaHesyee+YxQ+W6SvRDQV6UrdOdRiR153wJg==
+
+camelcase@^6.0.0, camelcase@^6.3.0:
+  version "6.3.0"
+  resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.3.0.tgz#5685b95eb209ac9c0c177467778c9c84df58ba9a"
+  integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA==
+
+capability@^0.2.5:
+  version "0.2.5"
+  resolved "https://registry.yarnpkg.com/capability/-/capability-0.2.5.tgz#51ad87353f1936ffd77f2f21c74633a4dea88801"
+  integrity sha512-rsJZYVCgXd08sPqwmaIqjAd5SUTfonV0z/gDJ8D6cN8wQphky1kkAYEqQ+hmDxTw7UihvBfjUVUSY+DBEe44jg==
+
+chai@^4.3.4:
+  version "4.5.0"
+  resolved "https://registry.yarnpkg.com/chai/-/chai-4.5.0.tgz#707e49923afdd9b13a8b0b47d33d732d13812fd8"
+  integrity sha512-RITGBfijLkBddZvnn8jdqoTypxvqbOLYQkGGxXzeFjVHvudaPw0HNFD9x928/eUwYWd2dPCugVqspGALTZZQKw==
+  dependencies:
+    assertion-error "^1.1.0"
+    check-error "^1.0.3"
+    deep-eql "^4.1.3"
+    get-func-name "^2.0.2"
+    loupe "^2.3.6"
+    pathval "^1.1.1"
+    type-detect "^4.1.0"
+
+chalk@^2.4.1:
+  version "2.4.2"
+  resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424"
+  integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==
+  dependencies:
+    ansi-styles "^3.2.1"
+    escape-string-regexp "^1.0.5"
+    supports-color "^5.3.0"
+
+chalk@^4.1.0:
+  version "4.1.2"
+  resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01"
+  integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA==
+  dependencies:
+    ansi-styles "^4.1.0"
+    supports-color "^7.1.0"
+
+chalk@^5.3.0:
+  version "5.4.1"
+  resolved "https://registry.yarnpkg.com/chalk/-/chalk-5.4.1.tgz#1b48bf0963ec158dce2aacf69c093ae2dd2092d8"
+  integrity sha512-zgVZuo2WcZgfUEmsn6eO3kINexW8RAE4maiQ8QNs8CtpPCSyMiYsULR3HQYkm3w8FIA3SberyMJMSldGsW+U3w==
+
+check-error@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/check-error/-/check-error-1.0.3.tgz#a6502e4312a7ee969f646e83bb3ddd56281bd694"
+  integrity sha512-iKEoDYaRmd1mxM90a2OEfWhjsjPpYPuQ+lMYsoxB126+t8fw7ySEO48nmDg5COTjxDI65/Y2OWpeEHk3ZOe8zg==
+  dependencies:
+    get-func-name "^2.0.2"
+
+chokidar@3.5.3:
+  version "3.5.3"
+  resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.5.3.tgz#1cf37c8707b932bd1af1ae22c0432e2acd1903bd"
+  integrity sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw==
+  dependencies:
+    anymatch "~3.1.2"
+    braces "~3.0.2"
+    glob-parent "~5.1.2"
+    is-binary-path "~2.1.0"
+    is-glob "~4.0.1"
+    normalize-path "~3.0.0"
+    readdirp "~3.6.0"
+  optionalDependencies:
+    fsevents "~2.3.2"
+
+cipher-base@^1.0.0, cipher-base@^1.0.1, cipher-base@^1.0.3:
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/cipher-base/-/cipher-base-1.0.6.tgz#8fe672437d01cd6c4561af5334e0cc50ff1955f7"
+  integrity sha512-3Ek9H3X6pj5TgenXYtNWdaBon1tgYCaebd+XPg0keyjEbEfkD4KkmAxkQ/i1vYvxdcT5nscLBfq9VJRmCBcFSw==
+  dependencies:
+    inherits "^2.0.4"
+    safe-buffer "^5.2.1"
+
+cliui@^7.0.2:
+  version "7.0.4"
+  resolved "https://registry.yarnpkg.com/cliui/-/cliui-7.0.4.tgz#a0265ee655476fc807aea9df3df8df7783808b4f"
+  integrity sha512-OcRE68cOsVMXp1Yvonl/fzkQOyjLSu/8bhPDfQt0e0/Eb283TKP20Fs2MqoPsr9SwA595rRCA+QMzYc9nBP+JQ==
+  dependencies:
+    string-width "^4.2.0"
+    strip-ansi "^6.0.0"
+    wrap-ansi "^7.0.0"
+
+color-convert@^1.9.0:
+  version "1.9.3"
+  resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8"
+  integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==
+  dependencies:
+    color-name "1.1.3"
+
+color-convert@^2.0.1:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3"
+  integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ==
+  dependencies:
+    color-name "~1.1.4"
+
+color-name@1.1.3:
+  version "1.1.3"
+  resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25"
+  integrity sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw==
+
+color-name@~1.1.4:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2"
+  integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA==
+
+combined-stream@^1.0.8:
+  version "1.0.8"
+  resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f"
+  integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==
+  dependencies:
+    delayed-stream "~1.0.0"
+
+commander@^12.1.0:
+  version "12.1.0"
+  resolved "https://registry.yarnpkg.com/commander/-/commander-12.1.0.tgz#01423b36f501259fdaac4d0e4d60c96c991585d3"
+  integrity sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA==
+
+commander@^2.20.3:
+  version "2.20.3"
+  resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.3.tgz#fd485e84c03eb4881c20722ba48035e8531aeb33"
+  integrity sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==
+
+concat-map@0.0.1:
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b"
+  integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg==
+
+copyfiles@^2.4.1:
+  version "2.4.1"
+  resolved "https://registry.yarnpkg.com/copyfiles/-/copyfiles-2.4.1.tgz#d2dcff60aaad1015f09d0b66e7f0f1c5cd3c5da5"
+  integrity sha512-fereAvAvxDrQDOXybk3Qu3dPbOoKoysFMWtkY3mv5BsL8//OSZVL5DCLYqgRfY5cWirgRzlC+WSrxp6Bo3eNZg==
+  dependencies:
+    glob "^7.0.5"
+    minimatch "^3.0.3"
+    mkdirp "^1.0.4"
+    noms "0.0.0"
+    through2 "^2.0.1"
+    untildify "^4.0.0"
+    yargs "^16.1.0"
+
+core-util-is@~1.0.0:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.3.tgz#a6042d3634c2b27e9328f837b965fac83808db85"
+  integrity sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==
+
+cosmjs-types@^0.7.1:
+  version "0.7.2"
+  resolved "https://registry.yarnpkg.com/cosmjs-types/-/cosmjs-types-0.7.2.tgz#a757371abd340949c5bd5d49c6f8379ae1ffd7e2"
+  integrity sha512-vf2uLyktjr/XVAgEq0DjMxeAWh1yYREe7AMHDKd7EiHVqxBPCaBS+qEEQUkXbR9ndnckqr1sUG8BQhazh4X5lA==
+  dependencies:
+    long "^4.0.0"
+    protobufjs "~6.11.2"
+
+crc-32@^1.2.0:
+  version "1.2.2"
+  resolved "https://registry.yarnpkg.com/crc-32/-/crc-32-1.2.2.tgz#3cad35a934b8bf71f25ca524b6da51fb7eace2ff"
+  integrity sha512-ROmzCKrTnOwybPcJApAA6WBWij23HVfGVNKqqrZpuyZOHqK2CwHSvpGuyt/UNNvaIjEd8X5IFGp4Mh+Ie1IHJQ==
+
+create-hash@^1.1.0, create-hash@^1.1.2, create-hash@^1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/create-hash/-/create-hash-1.2.0.tgz#889078af11a63756bcfb59bd221996be3a9ef196"
+  integrity sha512-z00bCGNHDG8mHAkP7CtT1qVu+bFQUPjYq/4Iv3C3kWjTFV10zIjfSoeqXo9Asws8gwSHDGj/hl2u4OGIjapeCg==
+  dependencies:
+    cipher-base "^1.0.1"
+    inherits "^2.0.1"
+    md5.js "^1.3.4"
+    ripemd160 "^2.0.1"
+    sha.js "^2.4.0"
+
+create-hmac@^1.1.4, create-hmac@^1.1.7:
+  version "1.1.7"
+  resolved "https://registry.yarnpkg.com/create-hmac/-/create-hmac-1.1.7.tgz#69170c78b3ab957147b2b8b04572e47ead2243ff"
+  integrity sha512-MJG9liiZ+ogc4TzUwuvbER1JRdgvUFSB5+VR/g5h82fGaIRWMWddtKBHi7/sVhfjQZ6SehlyhvQYrcYkaUIpLg==
+  dependencies:
+    cipher-base "^1.0.3"
+    create-hash "^1.1.0"
+    inherits "^2.0.1"
+    ripemd160 "^2.0.0"
+    safe-buffer "^5.0.1"
+    sha.js "^2.4.8"
+
+cross-fetch@^3.1.5:
+  version "3.2.0"
+  resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-3.2.0.tgz#34e9192f53bc757d6614304d9e5e6fb4edb782e3"
+  integrity sha512-Q+xVJLoGOeIMXZmbUK4HYk+69cQH6LudR0Vu/pRm2YlU/hDV9CiS0gKUMaWY5f2NeUH9C1nV3bsTlCo0FsTV1Q==
+  dependencies:
+    node-fetch "^2.7.0"
+
+crypto-addr-codec@^0.1.7:
+  version "0.1.8"
+  resolved "https://registry.yarnpkg.com/crypto-addr-codec/-/crypto-addr-codec-0.1.8.tgz#45c4b24e2ebce8e24a54536ee0ca25b65787b016"
+  integrity sha512-GqAK90iLLgP3FvhNmHbpT3wR6dEdaM8hZyZtLX29SPardh3OA13RFLHDR6sntGCgRWOfiHqW6sIyohpNqOtV/g==
+  dependencies:
+    base-x "^3.0.8"
+    big-integer "1.6.36"
+    blakejs "^1.1.0"
+    bs58 "^4.0.1"
+    ripemd160-min "0.0.6"
+    safe-buffer "^5.2.0"
+    sha3 "^2.1.1"
+
+crypto-hash@^1.3.0:
+  version "1.3.0"
+  resolved "https://registry.yarnpkg.com/crypto-hash/-/crypto-hash-1.3.0.tgz#b402cb08f4529e9f4f09346c3e275942f845e247"
+  integrity sha512-lyAZ0EMyjDkVvz8WOeVnuCPvKVBXcMv1l5SVqO1yC7PzTwrD/pPje/BIRbWhMoPe436U+Y2nD7f5bFx0kt+Sbg==
+
+debug@4.3.3:
+  version "4.3.3"
+  resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.3.tgz#04266e0b70a98d4462e6e288e38259213332b664"
+  integrity sha512-/zxw5+vh1Tfv+4Qn7a5nsbcJKPaSvCDhojn6FEl9vupwK2VCSDtEiEtqr8DFtzYFOdz63LBkxec7DYuc2jon6Q==
+  dependencies:
+    ms "2.1.2"
+
+decamelize@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-4.0.0.tgz#aa472d7bf660eb15f3494efd531cab7f2a709837"
+  integrity sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ==
+
+decimal.js@^10.2.1:
+  version "10.4.3"
+  resolved "https://registry.yarnpkg.com/decimal.js/-/decimal.js-10.4.3.tgz#1044092884d245d1b7f65725fa4ad4c6f781cc23"
+  integrity sha512-VBBaLc1MgL5XpzgIP7ny5Z6Nx3UrRkIViUkPUdtl9aya5amy3De1gsUUSB1g3+3sExYNjCAsAznmukyxCb1GRA==
+
+deep-eql@^4.1.3:
+  version "4.1.4"
+  resolved "https://registry.yarnpkg.com/deep-eql/-/deep-eql-4.1.4.tgz#d0d3912865911bb8fac5afb4e3acfa6a28dc72b7"
+  integrity sha512-SUwdGfqdKOwxCPeVYjwSyRpJ7Z+fhpwIAtmCUdZIWZ/YP5R9WAsyuSgpLVDi9bjWoN2LXHNss/dk3urXtdQxGg==
+  dependencies:
+    type-detect "^4.0.0"
+
+define-data-property@^1.0.1:
+  version "1.1.4"
+  resolved "https://registry.yarnpkg.com/define-data-property/-/define-data-property-1.1.4.tgz#894dc141bb7d3060ae4366f6a0107e68fbe48c5e"
+  integrity sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A==
+  dependencies:
+    es-define-property "^1.0.0"
+    es-errors "^1.3.0"
+    gopd "^1.0.1"
+
+define-properties@^1.2.1:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.2.1.tgz#10781cc616eb951a80a034bafcaa7377f6af2b6c"
+  integrity sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg==
+  dependencies:
+    define-data-property "^1.0.1"
+    has-property-descriptors "^1.0.0"
+    object-keys "^1.1.1"
+
+delay@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/delay/-/delay-5.0.0.tgz#137045ef1b96e5071060dd5be60bf9334436bd1d"
+  integrity sha512-ReEBKkIfe4ya47wlPYf/gu5ib6yUG0/Aez0JQZQz94kiWtRQvZIQbTiehsnwHvLSWJnQdhVeqYue7Id1dKr0qw==
+
+delayed-stream@~1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619"
+  integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==
+
+depd@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/depd/-/depd-2.0.0.tgz#b696163cc757560d09cf22cc8fad1571b79e76df"
+  integrity sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==
+
+depd@~1.1.2:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/depd/-/depd-1.1.2.tgz#9bcd52e14c097763e749b274c4346ed2e560b5a9"
+  integrity sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ==
+
+diff@5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/diff/-/diff-5.0.0.tgz#7ed6ad76d859d030787ec35855f5b1daf31d852b"
+  integrity sha512-/VTCrvm5Z0JGty/BWHljh+BAiw3IK+2j87NGMu8Nwc/f48WoDAC395uomO9ZD117ZOBaHmkX1oyLvkVM/aIT3w==
+
+diff@^3.1.0:
+  version "3.5.0"
+  resolved "https://registry.yarnpkg.com/diff/-/diff-3.5.0.tgz#800c0dd1e0a8bfbc95835c202ad220fe317e5a12"
+  integrity sha512-A46qtFgd+g7pDZinpnwiRJtxbC1hpgf0uzP3iG89scHk0AUC7A1TGxf5OiiOUv/JMZR8GOt8hL900hV0bOy5xA==
+
+dot-case@^3.0.4:
+  version "3.0.4"
+  resolved "https://registry.yarnpkg.com/dot-case/-/dot-case-3.0.4.tgz#9b2b670d00a431667a8a75ba29cd1b98809ce751"
+  integrity sha512-Kv5nKlh6yRrdrGvxeJ2e5y2eRUpkUosIW4A2AS38zwSz27zu7ufDwQPi5Jhs3XAlGNetl3bmnGhQsMtkKJnj3w==
+  dependencies:
+    no-case "^3.0.4"
+    tslib "^2.0.3"
+
+drbg.js@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/drbg.js/-/drbg.js-1.0.1.tgz#3e36b6c42b37043823cdbc332d58f31e2445480b"
+  integrity sha512-F4wZ06PvqxYLFEZKkFxTDcns9oFNk34hvmJSEwdzsxVQ8YI5YaxtACgQatkYgv2VI2CFkUd2Y+xosPQnHv809g==
+  dependencies:
+    browserify-aes "^1.0.6"
+    create-hash "^1.1.2"
+    create-hmac "^1.1.4"
+
+eccrypto@1.1.6:
+  version "1.1.6"
+  resolved "https://registry.yarnpkg.com/eccrypto/-/eccrypto-1.1.6.tgz#846bd1222323036f7a3515613704386399702bd3"
+  integrity sha512-d78ivVEzu7Tn0ZphUUaL43+jVPKTMPFGtmgtz1D0LrFn7cY3K8CdrvibuLz2AAkHBLKZtR8DMbB2ukRYFk987A==
+  dependencies:
+    acorn "7.1.1"
+    elliptic "6.5.4"
+    es6-promise "4.2.8"
+    nan "2.14.0"
+  optionalDependencies:
+    secp256k1 "3.7.1"
+
+elliptic@6.5.4:
+  version "6.5.4"
+  resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.5.4.tgz#da37cebd31e79a1367e941b592ed1fbebd58abbb"
+  integrity sha512-iLhC6ULemrljPZb+QutR5TQGB+pdW6KGD5RSegS+8sorOZT+rdQFbsQFJgvN3eRqNALqJer4oQ16YvJHlU8hzQ==
+  dependencies:
+    bn.js "^4.11.9"
+    brorand "^1.1.0"
+    hash.js "^1.0.0"
+    hmac-drbg "^1.0.1"
+    inherits "^2.0.4"
+    minimalistic-assert "^1.0.1"
+    minimalistic-crypto-utils "^1.0.1"
+
+elliptic@^6.4.0, elliptic@^6.4.1, elliptic@^6.5.2, elliptic@^6.5.4, elliptic@^6.5.7:
+  version "6.6.1"
+  resolved "https://registry.yarnpkg.com/elliptic/-/elliptic-6.6.1.tgz#3b8ffb02670bf69e382c7f65bf524c97c5405c06"
+  integrity sha512-RaddvvMatK2LJHqFJ+YA4WysVN5Ita9E35botqIYspQ4TkRAlCicdzKOjlyv/1Za5RyTNn7di//eEV0uTAfe3g==
+  dependencies:
+    bn.js "^4.11.9"
+    brorand "^1.1.0"
+    hash.js "^1.0.0"
+    hmac-drbg "^1.0.1"
+    inherits "^2.0.4"
+    minimalistic-assert "^1.0.1"
+    minimalistic-crypto-utils "^1.0.1"
+
+emoji-regex@^8.0.0:
+  version "8.0.0"
+  resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37"
+  integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A==
+
+error-polyfill@^0.1.3:
+  version "0.1.3"
+  resolved "https://registry.yarnpkg.com/error-polyfill/-/error-polyfill-0.1.3.tgz#df848b61ad8834f7a5db69a70b9913df86721d15"
+  integrity sha512-XHJk60ufE+TG/ydwp4lilOog549iiQF2OAPhkk9DdiYWMrltz5yhDz/xnKuenNwP7gy3dsibssO5QpVhkrSzzg==
+  dependencies:
+    capability "^0.2.5"
+    o3 "^1.0.3"
+    u3 "^0.1.1"
+
+es-define-property@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/es-define-property/-/es-define-property-1.0.1.tgz#983eb2f9a6724e9303f61addf011c72e09e0b0fa"
+  integrity sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==
+
+es-errors@^1.3.0:
+  version "1.3.0"
+  resolved "https://registry.yarnpkg.com/es-errors/-/es-errors-1.3.0.tgz#05f75a25dab98e4fb1dcd5e1472c0546d5057c8f"
+  integrity sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==
+
+es6-promise@4.2.8, es6-promise@^4.0.3:
+  version "4.2.8"
+  resolved "https://registry.yarnpkg.com/es6-promise/-/es6-promise-4.2.8.tgz#4eb21594c972bc40553d276e510539143db53e0a"
+  integrity sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==
+
+es6-promisify@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/es6-promisify/-/es6-promisify-5.0.0.tgz#5109d62f3e56ea967c4b63505aef08291c8a5203"
+  integrity sha512-C+d6UdsYDk0lMebHNR4S2NybQMMngAOnOwYBQjTOiv0MkoJMP0Myw2mgpDLBcpfCmRLxyFqYhS/CfOENq4SJhQ==
+  dependencies:
+    es6-promise "^4.0.3"
+
+esbuild@~0.23.0:
+  version "0.23.1"
+  resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.23.1.tgz#40fdc3f9265ec0beae6f59824ade1bd3d3d2dab8"
+  integrity sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==
+  optionalDependencies:
+    "@esbuild/aix-ppc64" "0.23.1"
+    "@esbuild/android-arm" "0.23.1"
+    "@esbuild/android-arm64" "0.23.1"
+    "@esbuild/android-x64" "0.23.1"
+    "@esbuild/darwin-arm64" "0.23.1"
+    "@esbuild/darwin-x64" "0.23.1"
+    "@esbuild/freebsd-arm64" "0.23.1"
+    "@esbuild/freebsd-x64" "0.23.1"
+    "@esbuild/linux-arm" "0.23.1"
+    "@esbuild/linux-arm64" "0.23.1"
+    "@esbuild/linux-ia32" "0.23.1"
+    "@esbuild/linux-loong64" "0.23.1"
+    "@esbuild/linux-mips64el" "0.23.1"
+    "@esbuild/linux-ppc64" "0.23.1"
+    "@esbuild/linux-riscv64" "0.23.1"
+    "@esbuild/linux-s390x" "0.23.1"
+    "@esbuild/linux-x64" "0.23.1"
+    "@esbuild/netbsd-x64" "0.23.1"
+    "@esbuild/openbsd-arm64" "0.23.1"
+    "@esbuild/openbsd-x64" "0.23.1"
+    "@esbuild/sunos-x64" "0.23.1"
+    "@esbuild/win32-arm64" "0.23.1"
+    "@esbuild/win32-ia32" "0.23.1"
+    "@esbuild/win32-x64" "0.23.1"
+
+escalade@^3.1.1:
+  version "3.2.0"
+  resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.2.0.tgz#011a3f69856ba189dffa7dc8fcce99d2a87903e5"
+  integrity sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==
+
+escape-string-regexp@4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34"
+  integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==
+
+escape-string-regexp@^1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4"
+  integrity sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg==
+
+eth-crypto@^2.6.0:
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/eth-crypto/-/eth-crypto-2.7.0.tgz#b859bf4f45660750fa144d74281b5560e362085d"
+  integrity sha512-MWbDl7OAoBAjkF2a7tklffAJv68uDI/MGPJKontt460nldJ8/2xT4cQacS8sGa6XJlon4ux1nAVzRoa4GxspOQ==
+  dependencies:
+    "@babel/runtime" "7.26.0"
+    "@ethereumjs/tx" "3.5.2"
+    "@types/bn.js" "5.1.6"
+    eccrypto "1.1.6"
+    ethereumjs-util "7.1.5"
+    ethers "5.7.2"
+    secp256k1 "5.0.1"
+
+ethereum-cryptography@^0.1.3:
+  version "0.1.3"
+  resolved "https://registry.yarnpkg.com/ethereum-cryptography/-/ethereum-cryptography-0.1.3.tgz#8d6143cfc3d74bf79bbd8edecdf29e4ae20dd191"
+  integrity sha512-w8/4x1SGGzc+tO97TASLja6SLd3fRIK2tLVcV2Gx4IB21hE19atll5Cq9o3d0ZmAYC/8aw0ipieTSiekAea4SQ==
+  dependencies:
+    "@types/pbkdf2" "^3.0.0"
+    "@types/secp256k1" "^4.0.1"
+    blakejs "^1.1.0"
+    browserify-aes "^1.2.0"
+    bs58check "^2.1.2"
+    create-hash "^1.2.0"
+    create-hmac "^1.1.7"
+    hash.js "^1.1.7"
+    keccak "^3.0.0"
+    pbkdf2 "^3.0.17"
+    randombytes "^2.1.0"
+    safe-buffer "^5.1.2"
+    scrypt-js "^3.0.0"
+    secp256k1 "^4.0.1"
+    setimmediate "^1.0.5"
+
+ethereumjs-abi@^0.6.8:
+  version "0.6.8"
+  resolved "https://registry.yarnpkg.com/ethereumjs-abi/-/ethereumjs-abi-0.6.8.tgz#71bc152db099f70e62f108b7cdfca1b362c6fcae"
+  integrity sha512-Tx0r/iXI6r+lRsdvkFDlut0N08jWMnKRZ6Gkq+Nmw75lZe4e6o3EkSnkaBP5NF6+m5PTGAr9JP43N3LyeoglsA==
+  dependencies:
+    bn.js "^4.11.8"
+    ethereumjs-util "^6.0.0"
+
+ethereumjs-util@7.1.5, ethereumjs-util@^7.1.4, ethereumjs-util@^7.1.5:
+  version "7.1.5"
+  resolved "https://registry.yarnpkg.com/ethereumjs-util/-/ethereumjs-util-7.1.5.tgz#9ecf04861e4fbbeed7465ece5f23317ad1129181"
+  integrity sha512-SDl5kKrQAudFBUe5OJM9Ac6WmMyYmXX/6sTmLZ3ffG2eY6ZIGBes3pEDxNN6V72WyOw4CPD5RomKdsa8DAAwLg==
+  dependencies:
+    "@types/bn.js" "^5.1.0"
+    bn.js "^5.1.2"
+    create-hash "^1.1.2"
+    ethereum-cryptography "^0.1.3"
+    rlp "^2.2.4"
+
+ethereumjs-util@^6.0.0, ethereumjs-util@^6.2.1:
+  version "6.2.1"
+  resolved "https://registry.yarnpkg.com/ethereumjs-util/-/ethereumjs-util-6.2.1.tgz#fcb4e4dd5ceacb9d2305426ab1a5cd93e3163b69"
+  integrity sha512-W2Ktez4L01Vexijrm5EB6w7dg4n/TgpoYU4avuT5T3Vmnw/eCRtiBrJfQYS/DCSvDIOLn2k57GcHdeBcgVxAqw==
+  dependencies:
+    "@types/bn.js" "^4.11.3"
+    bn.js "^4.11.0"
+    create-hash "^1.1.2"
+    elliptic "^6.5.2"
+    ethereum-cryptography "^0.1.3"
+    ethjs-util "0.1.6"
+    rlp "^2.2.3"
+
+ethers@5.7.2, ethers@^5.7.2:
+  version "5.7.2"
+  resolved "https://registry.yarnpkg.com/ethers/-/ethers-5.7.2.tgz#3a7deeabbb8c030d4126b24f84e525466145872e"
+  integrity sha512-wswUsmWo1aOK8rR7DIKiWSw9DbLWe6x98Jrn8wcTflTVvaXhAMaB5zGAXy0GYQEQp9iO1iSHWVyARQm11zUtyg==
+  dependencies:
+    "@ethersproject/abi" "5.7.0"
+    "@ethersproject/abstract-provider" "5.7.0"
+    "@ethersproject/abstract-signer" "5.7.0"
+    "@ethersproject/address" "5.7.0"
+    "@ethersproject/base64" "5.7.0"
+    "@ethersproject/basex" "5.7.0"
+    "@ethersproject/bignumber" "5.7.0"
+    "@ethersproject/bytes" "5.7.0"
+    "@ethersproject/constants" "5.7.0"
+    "@ethersproject/contracts" "5.7.0"
+    "@ethersproject/hash" "5.7.0"
+    "@ethersproject/hdnode" "5.7.0"
+    "@ethersproject/json-wallets" "5.7.0"
+    "@ethersproject/keccak256" "5.7.0"
+    "@ethersproject/logger" "5.7.0"
+    "@ethersproject/networks" "5.7.1"
+    "@ethersproject/pbkdf2" "5.7.0"
+    "@ethersproject/properties" "5.7.0"
+    "@ethersproject/providers" "5.7.2"
+    "@ethersproject/random" "5.7.0"
+    "@ethersproject/rlp" "5.7.0"
+    "@ethersproject/sha2" "5.7.0"
+    "@ethersproject/signing-key" "5.7.0"
+    "@ethersproject/solidity" "5.7.0"
+    "@ethersproject/strings" "5.7.0"
+    "@ethersproject/transactions" "5.7.0"
+    "@ethersproject/units" "5.7.0"
+    "@ethersproject/wallet" "5.7.0"
+    "@ethersproject/web" "5.7.1"
+    "@ethersproject/wordlists" "5.7.0"
+
+ethjs-util@0.1.6, ethjs-util@^0.1.6:
+  version "0.1.6"
+  resolved "https://registry.yarnpkg.com/ethjs-util/-/ethjs-util-0.1.6.tgz#f308b62f185f9fe6237132fb2a9818866a5cd536"
+  integrity sha512-CUnVOQq7gSpDHZVVrQW8ExxUETWrnrvXYvYz55wOU8Uj4VCgw56XC2B/fVqQN+f7gmrnRHSLVnFAwsCuNwji8w==
+  dependencies:
+    is-hex-prefixed "1.0.0"
+    strip-hex-prefix "1.0.0"
+
+eventemitter3@^4.0.7:
+  version "4.0.7"
+  resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.7.tgz#2de9b68f6528d5644ef5c59526a1b4a07306169f"
+  integrity sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==
+
+eventemitter3@^5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-5.0.1.tgz#53f5ffd0a492ac800721bb42c66b841de96423c4"
+  integrity sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA==
+
+evp_bytestokey@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/evp_bytestokey/-/evp_bytestokey-1.0.3.tgz#7fcbdb198dc71959432efe13842684e0525acb02"
+  integrity sha512-/f2Go4TognH/KvCISP7OUsHn85hT9nUkxxA9BEWxFn+Oj9o8ZNLm/40hdlgSLyuOimsrTKLUMEorQexp/aPQeA==
+  dependencies:
+    md5.js "^1.3.4"
+    safe-buffer "^5.1.1"
+
+eyes@^0.1.8:
+  version "0.1.8"
+  resolved "https://registry.yarnpkg.com/eyes/-/eyes-0.1.8.tgz#62cf120234c683785d902348a800ef3e0cc20bc0"
+  integrity sha512-GipyPsXO1anza0AOZdy69Im7hGFCNB7Y/NGjDlZGJ3GJJLtwNSb2vrzYrTYJRrRloVx7pl+bhUaTB8yiccPvFQ==
+
+fast-stable-stringify@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/fast-stable-stringify/-/fast-stable-stringify-1.0.0.tgz#5c5543462b22aeeefd36d05b34e51c78cb86d313"
+  integrity sha512-wpYMUmFu5f00Sm0cj2pfivpmawLZ0NKdviQ4w9zJeR8JVtOpOxHmLaJuj0vxvGqMJQWyP/COUkF75/57OKyRag==
+
+file-uri-to-path@1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz#553a7b8446ff6f684359c445f1e37a05dacc33dd"
+  integrity sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==
+
+fill-range@^7.1.1:
+  version "7.1.1"
+  resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.1.1.tgz#44265d3cac07e3ea7dc247516380643754a05292"
+  integrity sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==
+  dependencies:
+    to-regex-range "^5.0.1"
+
+find-up@5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc"
+  integrity sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng==
+  dependencies:
+    locate-path "^6.0.0"
+    path-exists "^4.0.0"
+
+flat@^5.0.2:
+  version "5.0.2"
+  resolved "https://registry.yarnpkg.com/flat/-/flat-5.0.2.tgz#8ca6fe332069ffa9d324c327198c598259ceb241"
+  integrity sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ==
+
+follow-redirects@^1.14.0, follow-redirects@^1.14.4, follow-redirects@^1.14.8, follow-redirects@^1.14.9, follow-redirects@^1.15.6:
+  version "1.15.9"
+  resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.9.tgz#a604fa10e443bf98ca94228d9eebcc2e8a2c8ee1"
+  integrity sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ==
+
+form-data@4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/form-data/-/form-data-4.0.0.tgz#93919daeaf361ee529584b9b31664dc12c9fa452"
+  integrity sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==
+  dependencies:
+    asynckit "^0.4.0"
+    combined-stream "^1.0.8"
+    mime-types "^2.1.12"
+
+form-data@^4.0.0:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/form-data/-/form-data-4.0.1.tgz#ba1076daaaa5bfd7e99c1a6cb02aa0a5cff90d48"
+  integrity sha512-tzN8e4TX8+kkxGPK8D5u0FNmjPUjw3lwC9lSLxxoB/+GtsJG91CO8bSWy73APlgAZzZbXEYZJuxjkHH2w+Ezhw==
+  dependencies:
+    asynckit "^0.4.0"
+    combined-stream "^1.0.8"
+    mime-types "^2.1.12"
+
+fs.realpath@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f"
+  integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw==
+
+fsevents@~2.3.2, fsevents@~2.3.3:
+  version "2.3.3"
+  resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6"
+  integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==
+
+function-bind@^1.1.2:
+  version "1.1.2"
+  resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.2.tgz#2c02d864d97f3ea6c8830c464cbd11ab6eab7a1c"
+  integrity sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==
+
+get-caller-file@^2.0.5:
+  version "2.0.5"
+  resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e"
+  integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg==
+
+get-func-name@^2.0.1, get-func-name@^2.0.2:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/get-func-name/-/get-func-name-2.0.2.tgz#0d7cf20cd13fda808669ffa88f4ffc7a3943fc41"
+  integrity sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ==
+
+get-tsconfig@^4.7.5:
+  version "4.10.0"
+  resolved "https://registry.yarnpkg.com/get-tsconfig/-/get-tsconfig-4.10.0.tgz#403a682b373a823612475a4c2928c7326fc0f6bb"
+  integrity sha512-kGzZ3LWWQcGIAmg6iWvXn0ei6WDtV26wzHRMwDSzmAbcXrTEXxHy6IehI6/4eT6VRKyMP1eF1VqwrVUmE/LR7A==
+  dependencies:
+    resolve-pkg-maps "^1.0.0"
+
+glob-parent@~5.1.2:
+  version "5.1.2"
+  resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4"
+  integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==
+  dependencies:
+    is-glob "^4.0.1"
+
+glob@7.2.0:
+  version "7.2.0"
+  resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.0.tgz#d15535af7732e02e948f4c41628bd910293f6023"
+  integrity sha512-lmLf6gtyrPq8tTjSmrO94wBeQbFR3HbLHbuyD69wuyQkImp2hWqMGB47OX65FBkPffO641IP9jWa1z4ivqG26Q==
+  dependencies:
+    fs.realpath "^1.0.0"
+    inflight "^1.0.4"
+    inherits "2"
+    minimatch "^3.0.4"
+    once "^1.3.0"
+    path-is-absolute "^1.0.0"
+
+glob@^7.0.0, glob@^7.0.5:
+  version "7.2.3"
+  resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b"
+  integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q==
+  dependencies:
+    fs.realpath "^1.0.0"
+    inflight "^1.0.4"
+    inherits "2"
+    minimatch "^3.1.1"
+    once "^1.3.0"
+    path-is-absolute "^1.0.0"
+
+globalthis@^1.0.1:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.4.tgz#7430ed3a975d97bfb59bcce41f5cabbafa651236"
+  integrity sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ==
+  dependencies:
+    define-properties "^1.2.1"
+    gopd "^1.0.1"
+
+google-protobuf@^3.14.0, google-protobuf@^3.17.3, google-protobuf@^3.21.0:
+  version "3.21.4"
+  resolved "https://registry.yarnpkg.com/google-protobuf/-/google-protobuf-3.21.4.tgz#2f933e8b6e5e9f8edde66b7be0024b68f77da6c9"
+  integrity sha512-MnG7N936zcKTco4Jd2PX2U96Kf9PxygAPKBug+74LHzmHXmceN16MmRcdgZv+DGef/S9YvQAfRsNCn4cjf9yyQ==
+
+gopd@^1.0.1:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.2.0.tgz#89f56b8217bdbc8802bd299df6d7f1081d7e51a1"
+  integrity sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==
+
+graphql-tag@^2.12.6:
+  version "2.12.6"
+  resolved "https://registry.yarnpkg.com/graphql-tag/-/graphql-tag-2.12.6.tgz#d441a569c1d2537ef10ca3d1633b48725329b5f1"
+  integrity sha512-FdSNcu2QQcWnM2VNvSCCDCVS5PpPqpzgFT8+GXzqJuoDd0CBncxCY278u4mhRO7tMgo2JjgJA5aZ+nWSQ/Z+xg==
+  dependencies:
+    tslib "^2.1.0"
+
+graphql@^16.3.0:
+  version "16.10.0"
+  resolved "https://registry.yarnpkg.com/graphql/-/graphql-16.10.0.tgz#24c01ae0af6b11ea87bf55694429198aaa8e220c"
+  integrity sha512-AjqGKbDGUFRKIRCP9tCKiIGHyriz2oHEbPIbEtcSLSs4YjReZOIPQQWek4+6hjw62H9QShXHyaGivGiYVLeYFQ==
+
+growl@1.10.5:
+  version "1.10.5"
+  resolved "https://registry.yarnpkg.com/growl/-/growl-1.10.5.tgz#f2735dc2283674fa67478b10181059355c369e5e"
+  integrity sha512-qBr4OuELkhPenW6goKVXiv47US3clb3/IbuWF9KNKEijAy9oeHxU9IgzjvJhHkUzhaj7rOUD7+YGWqUjLp5oSA==
+
+has-flag@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd"
+  integrity sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw==
+
+has-flag@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b"
+  integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==
+
+has-property-descriptors@^1.0.0:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz#963ed7d071dc7bf5f084c5bfbe0d1b6222586854"
+  integrity sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg==
+  dependencies:
+    es-define-property "^1.0.0"
+
+hash-base@^3.0.0:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/hash-base/-/hash-base-3.1.0.tgz#55c381d9e06e1d2997a883b4a3fddfe7f0d3af33"
+  integrity sha512-1nmYp/rhMDiE7AYkDw+lLwlAzz0AntGIe51F3RfFfEqyQ3feY2eI/NcwC6umIQVOASPMsWJLJScWKSSvzL9IVA==
+  dependencies:
+    inherits "^2.0.4"
+    readable-stream "^3.6.0"
+    safe-buffer "^5.2.0"
+
+hash.js@1.1.7, hash.js@^1.0.0, hash.js@^1.0.3, hash.js@^1.1.7:
+  version "1.1.7"
+  resolved "https://registry.yarnpkg.com/hash.js/-/hash.js-1.1.7.tgz#0babca538e8d4ee4a0f8988d68866537a003cf42"
+  integrity sha512-taOaskGt4z4SOANNseOviYDvjEJinIkRgmp7LbKP2YTTmVxWBl87s/uzK9r+44BclBSp2X7K1hqeNfz9JbBeXA==
+  dependencies:
+    inherits "^2.0.3"
+    minimalistic-assert "^1.0.1"
+
+hasown@^2.0.2:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.2.tgz#003eaf91be7adc372e84ec59dc37252cedb80003"
+  integrity sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==
+  dependencies:
+    function-bind "^1.1.2"
+
+he@1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/he/-/he-1.2.0.tgz#84ae65fa7eafb165fddb61566ae14baf05664f0f"
+  integrity sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw==
+
+hi-base32@^0.5.1:
+  version "0.5.1"
+  resolved "https://registry.yarnpkg.com/hi-base32/-/hi-base32-0.5.1.tgz#1279f2ddae2673219ea5870c2121d2a33132857e"
+  integrity sha512-EmBBpvdYh/4XxsnUybsPag6VikPYnN30td+vQk+GI3qpahVEG9+gTkG0aXVxTjBqQ5T6ijbWIu77O+C5WFWsnA==
+
+hmac-drbg@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/hmac-drbg/-/hmac-drbg-1.0.1.tgz#d2745701025a6c775a6c545793ed502fc0c649a1"
+  integrity sha512-Tti3gMqLdZfhOQY1Mzf/AanLiqh1WTiJgEj26ZuYQ9fbkLomzGchCws4FyrSd4VkpBfiNhaE1On+lOz894jvXg==
+  dependencies:
+    hash.js "^1.0.3"
+    minimalistic-assert "^1.0.0"
+    minimalistic-crypto-utils "^1.0.1"
+
+hoist-non-react-statics@^3.3.2:
+  version "3.3.2"
+  resolved "https://registry.yarnpkg.com/hoist-non-react-statics/-/hoist-non-react-statics-3.3.2.tgz#ece0acaf71d62c2969c2ec59feff42a4b1a85b45"
+  integrity sha512-/gGivxi8JPKWNm/W0jSmzcMPpfpPLc3dY/6GxhX2hQ9iGj3aDfklV4ET7NjKpSinLpJ5vafa9iiGIEZg10SfBw==
+  dependencies:
+    react-is "^16.7.0"
+
+http-errors@^1.7.2:
+  version "1.8.1"
+  resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-1.8.1.tgz#7c3f28577cbc8a207388455dbd62295ed07bd68c"
+  integrity sha512-Kpk9Sm7NmI+RHhnj6OIWDI1d6fIoFAtFt9RLaTMRlg/8w49juAStsrBgp0Dp4OdxdVbRIeKhtCUvoi/RuAhO4g==
+  dependencies:
+    depd "~1.1.2"
+    inherits "2.0.4"
+    setprototypeof "1.2.0"
+    statuses ">= 1.5.0 < 2"
+    toidentifier "1.0.1"
+
+http-status-codes@^2.2.0:
+  version "2.3.0"
+  resolved "https://registry.yarnpkg.com/http-status-codes/-/http-status-codes-2.3.0.tgz#987fefb28c69f92a43aecc77feec2866349a8bfc"
+  integrity sha512-RJ8XvFvpPM/Dmc5SV+dC4y5PCeOhT3x1Hq0NU3rjGeg5a/CqlhZ7uudknPwZFz4aeAXDcbAyaeP7GAo9lvngtA==
+
+humanize-ms@^1.2.1:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/humanize-ms/-/humanize-ms-1.2.1.tgz#c46e3159a293f6b896da29316d8b6fe8bb79bbed"
+  integrity sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ==
+  dependencies:
+    ms "^2.0.0"
+
+ieee754@^1.2.1:
+  version "1.2.1"
+  resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352"
+  integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==
+
+inflight@^1.0.4:
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9"
+  integrity sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==
+  dependencies:
+    once "^1.3.0"
+    wrappy "1"
+
+inherits@2, inherits@2.0.4, inherits@^2.0.1, inherits@^2.0.3, inherits@^2.0.4, inherits@~2.0.1, inherits@~2.0.3:
+  version "2.0.4"
+  resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c"
+  integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==
+
+interpret@^1.0.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/interpret/-/interpret-1.4.0.tgz#665ab8bc4da27a774a40584e812e3e0fa45b1a1e"
+  integrity sha512-agE4QfB2Lkp9uICn7BAqoscw4SZP9kTE2hxiFI3jBPmXJfdqiahTbUuKGsMoN2GtqL9AxhYioAcVvgsb1HvRbA==
+
+is-binary-path@~2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-2.1.0.tgz#ea1f7f3b80f064236e83470f86c09c254fb45b09"
+  integrity sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==
+  dependencies:
+    binary-extensions "^2.0.0"
+
+is-core-module@^2.16.0:
+  version "2.16.1"
+  resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.16.1.tgz#2a98801a849f43e2add644fbb6bc6229b19a4ef4"
+  integrity sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==
+  dependencies:
+    hasown "^2.0.2"
+
+is-extglob@^2.1.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2"
+  integrity sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==
+
+is-fullwidth-code-point@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d"
+  integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg==
+
+is-glob@^4.0.1, is-glob@~4.0.1:
+  version "4.0.3"
+  resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084"
+  integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==
+  dependencies:
+    is-extglob "^2.1.1"
+
+is-hex-prefixed@1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/is-hex-prefixed/-/is-hex-prefixed-1.0.0.tgz#7d8d37e6ad77e5d127148913c573e082d777f554"
+  integrity sha512-WvtOiug1VFrE9v1Cydwm+FnXd3+w9GaeVUss5W4v/SLy3UW00vP+6iNF2SdnfiBoLy4bTqVdkftNGTUeOFVsbA==
+
+is-number@^7.0.0:
+  version "7.0.0"
+  resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b"
+  integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==
+
+is-plain-obj@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-2.1.0.tgz#45e42e37fccf1f40da8e5f76ee21515840c09287"
+  integrity sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA==
+
+is-unicode-supported@^0.1.0:
+  version "0.1.0"
+  resolved "https://registry.yarnpkg.com/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz#3f26c76a809593b52bfa2ecb5710ed2779b522a7"
+  integrity sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw==
+
+isarray@0.0.1:
+  version "0.0.1"
+  resolved "https://registry.yarnpkg.com/isarray/-/isarray-0.0.1.tgz#8a18acfca9a8f4177e09abfc6038939b05d1eedf"
+  integrity sha512-D2S+3GLxWH+uhrNEcoh/fnmYeP8E8/zHl644d/jdA0g2uyXvy3sb0qxotE+ne0LtccHknQzWwZEzhak7oJ0COQ==
+
+isarray@~1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11"
+  integrity sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ==
+
+isexe@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10"
+  integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw==
+
+isomorphic-ws@^4.0.1:
+  version "4.0.1"
+  resolved "https://registry.yarnpkg.com/isomorphic-ws/-/isomorphic-ws-4.0.1.tgz#55fd4cd6c5e6491e76dc125938dd863f5cd4f2dc"
+  integrity sha512-BhBvN2MBpWTaSHdWRb/bwdZJ1WaehQ2L1KngkCkfLUGF0mAWAT1sQUQacEmQ0jXkFw/czDXPNQSL5u2/Krsz1w==
+
+jayson@^4.0.0, jayson@^4.1.1:
+  version "4.1.3"
+  resolved "https://registry.yarnpkg.com/jayson/-/jayson-4.1.3.tgz#db9be2e4287d9fef4fc05b5fe367abe792c2eee8"
+  integrity sha512-LtXh5aYZodBZ9Fc3j6f2w+MTNcnxteMOrb+QgIouguGOulWi0lieEkOUg+HkjjFs0DGoWDds6bi4E9hpNFLulQ==
+  dependencies:
+    "@types/connect" "^3.4.33"
+    "@types/node" "^12.12.54"
+    "@types/ws" "^7.4.4"
+    JSONStream "^1.3.5"
+    commander "^2.20.3"
+    delay "^5.0.0"
+    es6-promisify "^5.0.0"
+    eyes "^0.1.8"
+    isomorphic-ws "^4.0.1"
+    json-stringify-safe "^5.0.1"
+    uuid "^8.3.2"
+    ws "^7.5.10"
+
+js-base64@^3.6.1:
+  version "3.7.7"
+  resolved "https://registry.yarnpkg.com/js-base64/-/js-base64-3.7.7.tgz#e51b84bf78fbf5702b9541e2cb7bfcb893b43e79"
+  integrity sha512-7rCnleh0z2CkXhH67J8K1Ytz0b2Y+yxTPL+/KOJoa20hfnVQ/3/T6W/KflYI4bRHRagNeXeU2bkNGI3v1oS/lw==
+
+js-sha256@^0.9.0:
+  version "0.9.0"
+  resolved "https://registry.yarnpkg.com/js-sha256/-/js-sha256-0.9.0.tgz#0b89ac166583e91ef9123644bd3c5334ce9d0966"
+  integrity sha512-sga3MHh9sgQN2+pJ9VYZ+1LPwXOxuBJBA5nrR5/ofPfuiJBE2hnjsaN8se8JznOmGLN2p49Pe5U/ttafcs/apA==
+
+js-sha3@0.8.0, js-sha3@^0.8.0:
+  version "0.8.0"
+  resolved "https://registry.yarnpkg.com/js-sha3/-/js-sha3-0.8.0.tgz#b9b7a5da73afad7dedd0f8c463954cbde6818840"
+  integrity sha512-gF1cRrHhIzNfToc802P800N8PpXS+evLLXfsVpowqmAFR9uwbi89WvXg2QspOmXL8QL86J4T1EpFu+yUkwJY3Q==
+
+js-sha512@^0.8.0:
+  version "0.8.0"
+  resolved "https://registry.yarnpkg.com/js-sha512/-/js-sha512-0.8.0.tgz#dd22db8d02756faccf19f218e3ed61ec8249f7d4"
+  integrity sha512-PWsmefG6Jkodqt+ePTvBZCSMFgN7Clckjd0O7su3I0+BW2QWUTJNzjktHsztGLhncP2h8mcF9V9Y2Ha59pAViQ==
+
+"js-tokens@^3.0.0 || ^4.0.0":
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499"
+  integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==
+
+js-yaml@4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602"
+  integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==
+  dependencies:
+    argparse "^2.0.1"
+
+jscrypto@^1.0.1, jscrypto@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/jscrypto/-/jscrypto-1.0.3.tgz#598febca2a939d6f679c54f56e1fe364cef30cc9"
+  integrity sha512-lryZl0flhodv4SZHOqyb1bx5sKcJxj0VBo0Kzb4QMAg3L021IC9uGpl0RCZa+9KJwlRGSK2C80ITcwbe19OKLQ==
+
+json-bigint@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/json-bigint/-/json-bigint-1.0.0.tgz#ae547823ac0cad8398667f8cd9ef4730f5b01ff1"
+  integrity sha512-SiPv/8VpZuWbvLSMtTDU8hEfrZWg/mH/nV/b4o0CYbSxu1UIQPLdwKOCIyLQX+VIPO5vrLX3i8qtqFyhdPSUSQ==
+  dependencies:
+    bignumber.js "^9.0.0"
+
+json-stringify-safe@^5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb"
+  integrity sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA==
+
+json5@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.2.tgz#63d98d60f21b313b77c4d6da18bfa69d80e1d593"
+  integrity sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA==
+  dependencies:
+    minimist "^1.2.0"
+
+jsonparse@^1.2.0:
+  version "1.3.1"
+  resolved "https://registry.yarnpkg.com/jsonparse/-/jsonparse-1.3.1.tgz#3f4dae4a91fac315f71062f8521cc239f1366280"
+  integrity sha512-POQXvpdL69+CluYsillJ7SUhKvytYjW9vG/GKpnf+xP8UWgYEM/RaMzHHofbALDiKbbP1W8UEYmgGl39WkPZsg==
+
+jsonschema@^1.4.0:
+  version "1.5.0"
+  resolved "https://registry.yarnpkg.com/jsonschema/-/jsonschema-1.5.0.tgz#f6aceb1ab9123563dd901d05f81f9d4883d3b7d8"
+  integrity sha512-K+A9hhqbn0f3pJX17Q/7H6yQfD/5OXgdrR5UE12gMXCiN9D5Xq2o5mddV2QEcX/bjla99ASsAAQUyMCCRWAEhw==
+
+keccak256@^1.0.6:
+  version "1.0.6"
+  resolved "https://registry.yarnpkg.com/keccak256/-/keccak256-1.0.6.tgz#dd32fb771558fed51ce4e45a035ae7515573da58"
+  integrity sha512-8GLiM01PkdJVGUhR1e6M/AvWnSqYS0HaERI+K/QtStGDGlSTx2B1zTqZk4Zlqu5TxHJNTxWAdP9Y+WI50OApUw==
+  dependencies:
+    bn.js "^5.2.0"
+    buffer "^6.0.3"
+    keccak "^3.0.2"
+
+keccak@^3.0.0, keccak@^3.0.2:
+  version "3.0.4"
+  resolved "https://registry.yarnpkg.com/keccak/-/keccak-3.0.4.tgz#edc09b89e633c0549da444432ecf062ffadee86d"
+  integrity sha512-3vKuW0jV8J3XNTzvfyicFR5qvxrSAGl7KIhvgOu5cmWwM7tZRj3fMbj/pfIf4be7aznbc+prBWGjywox/g2Y6Q==
+  dependencies:
+    node-addon-api "^2.0.0"
+    node-gyp-build "^4.2.0"
+    readable-stream "^3.6.0"
+
+libsodium-wrappers@^0.7.6:
+  version "0.7.15"
+  resolved "https://registry.yarnpkg.com/libsodium-wrappers/-/libsodium-wrappers-0.7.15.tgz#53f13e483820272a3d55b23be2e34402ac988055"
+  integrity sha512-E4anqJQwcfiC6+Yrl01C1m8p99wEhLmJSs0VQqST66SbQXXBoaJY0pF4BNjRYa/sOQAxx6lXAaAFIlx+15tXJQ==
+  dependencies:
+    libsodium "^0.7.15"
+
+libsodium@^0.7.15:
+  version "0.7.15"
+  resolved "https://registry.yarnpkg.com/libsodium/-/libsodium-0.7.15.tgz#ac284e3dcb1c29ae9526c5581cdada6a072f6d20"
+  integrity sha512-sZwRknt/tUpE2AwzHq3jEyUU5uvIZHtSssktXq7owd++3CSgn8RGrv6UZJJBpP7+iBghBqe7Z06/2M31rI2NKw==
+
+link-module-alias@^1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/link-module-alias/-/link-module-alias-1.2.0.tgz#6a3b7b014cfe18b2759a1222fffce6a40fc120e4"
+  integrity sha512-ahPjXepbSVKbahTB6LxR//VHm8HPfI+QQygCH+E82spBY4HR5VPJTvlhKBc9F7muVxnS6C1rRfoPOXAbWO/fyw==
+  dependencies:
+    chalk "^2.4.1"
+
+locate-path@^6.0.0:
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286"
+  integrity sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw==
+  dependencies:
+    p-locate "^5.0.0"
+
+lodash.values@^4.3.0:
+  version "4.3.0"
+  resolved "https://registry.yarnpkg.com/lodash.values/-/lodash.values-4.3.0.tgz#a3a6c2b0ebecc5c2cba1c17e6e620fe81b53d347"
+  integrity sha512-r0RwvdCv8id9TUblb/O7rYPwVy6lerCbcawrfdo9iC/1t1wsNMJknO79WNBgwkH0hIeJ08jmvvESbFpNb4jH0Q==
+
+lodash@^4.17.21:
+  version "4.17.21"
+  resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c"
+  integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg==
+
+log-symbols@4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/log-symbols/-/log-symbols-4.1.0.tgz#3fbdbb95b4683ac9fc785111e792e558d4abd503"
+  integrity sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg==
+  dependencies:
+    chalk "^4.1.0"
+    is-unicode-supported "^0.1.0"
+
+long@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/long/-/long-4.0.0.tgz#9a7b71cfb7d361a194ea555241c92f7468d5bf28"
+  integrity sha512-XsP+KhQif4bjX1kbuSiySJFNAehNxgLb6hPRGJ9QsUr8ajHkuXGdrHmFUTUUXhDwVX2R5bY4JNZEwbUiMhV+MA==
+
+long@^5.0.0:
+  version "5.2.3"
+  resolved "https://registry.yarnpkg.com/long/-/long-5.2.3.tgz#a3ba97f3877cf1d778eccbcb048525ebb77499e1"
+  integrity sha512-lcHwpNoggQTObv5apGNCTdJrO69eHOZMi4BNC+rTLER8iHAqGrUVeLh/irVIM7zTw2bOXA8T6uNPeujwOLg/2Q==
+
+loose-envify@^1.4.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/loose-envify/-/loose-envify-1.4.0.tgz#71ee51fa7be4caec1a63839f7e682d8132d30caf"
+  integrity sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==
+  dependencies:
+    js-tokens "^3.0.0 || ^4.0.0"
+
+loupe@^2.3.6:
+  version "2.3.7"
+  resolved "https://registry.yarnpkg.com/loupe/-/loupe-2.3.7.tgz#6e69b7d4db7d3ab436328013d37d1c8c3540c697"
+  integrity sha512-zSMINGVYkdpYSOBmLi0D1Uo7JU9nVdQKrHxC8eYlV+9YKK9WePqAlL7lSlorG/U2Fw1w0hTBmaa/jrQ3UbPHtA==
+  dependencies:
+    get-func-name "^2.0.1"
+
+lower-case@^2.0.2:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/lower-case/-/lower-case-2.0.2.tgz#6fa237c63dbdc4a82ca0fd882e4722dc5e634e28"
+  integrity sha512-7fm3l3NAF9WfN6W3JOmf5drwpVqX78JtoGJ3A6W0a6ZnldM41w2fV5D490psKFTpMds8TJse/eHLFFsNHHjHgg==
+  dependencies:
+    tslib "^2.0.3"
+
+make-error@^1.1.1:
+  version "1.3.6"
+  resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2"
+  integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==
+
+map-obj@^4.1.0:
+  version "4.3.0"
+  resolved "https://registry.yarnpkg.com/map-obj/-/map-obj-4.3.0.tgz#9304f906e93faae70880da102a9f1df0ea8bb05a"
+  integrity sha512-hdN1wVrZbb29eBGiGjJbeP8JbKjq1urkHJ/LIP/NY48MZ1QVXUsQBV1G1zvYFHn1XE06cwjBsOI2K3Ulnj1YXQ==
+
+md5.js@^1.3.4:
+  version "1.3.5"
+  resolved "https://registry.yarnpkg.com/md5.js/-/md5.js-1.3.5.tgz#b5d07b8e3216e3e27cd728d72f70d1e6a342005f"
+  integrity sha512-xitP+WxNPcTTOgnTJcrhM0xvdPepipPSf3I8EIpGKeFLjt3PlJLIDG3u8EX53ZIubkb+5U2+3rELYpEhHhzdkg==
+  dependencies:
+    hash-base "^3.0.0"
+    inherits "^2.0.1"
+    safe-buffer "^5.1.2"
+
+mime-db@1.52.0:
+  version "1.52.0"
+  resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70"
+  integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==
+
+mime-types@^2.1.12:
+  version "2.1.35"
+  resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a"
+  integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==
+  dependencies:
+    mime-db "1.52.0"
+
+minimalistic-assert@^1.0.0, minimalistic-assert@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/minimalistic-assert/-/minimalistic-assert-1.0.1.tgz#2e194de044626d4a10e7f7fbc00ce73e83e4d5c7"
+  integrity sha512-UtJcAD4yEaGtjPezWuO9wC4nwUnVH/8/Im3yEHQP4b67cXlD/Qr9hdITCU1xDbSEXg2XKNaP8jsReV7vQd00/A==
+
+minimalistic-crypto-utils@^1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/minimalistic-crypto-utils/-/minimalistic-crypto-utils-1.0.1.tgz#f6c00c1c0b082246e5c4d99dfb8c7c083b2b582a"
+  integrity sha512-JIYlbt6g8i5jKfJ3xz7rF0LXmv2TkDxBLUkiBeZ7bAx4GnnNMr8xFpGnOxn6GhTEHx3SjRrZEoU+j04prX1ktg==
+
+minimatch@4.2.1:
+  version "4.2.1"
+  resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-4.2.1.tgz#40d9d511a46bdc4e563c22c3080cde9c0d8299b4"
+  integrity sha512-9Uq1ChtSZO+Mxa/CL1eGizn2vRn3MlLgzhT0Iz8zaY8NdvxvB0d5QdPFmCKf7JKA9Lerx5vRrnwO03jsSfGG9g==
+  dependencies:
+    brace-expansion "^1.1.7"
+
+minimatch@^3.0.3, minimatch@^3.0.4, minimatch@^3.1.1:
+  version "3.1.2"
+  resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b"
+  integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==
+  dependencies:
+    brace-expansion "^1.1.7"
+
+minimist@^1.2.0, minimist@^1.2.3, minimist@^1.2.6:
+  version "1.2.8"
+  resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c"
+  integrity sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==
+
+mkdirp@^0.5.1:
+  version "0.5.6"
+  resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-0.5.6.tgz#7def03d2432dcae4ba1d611445c48396062255f6"
+  integrity sha512-FP+p8RB8OWpF3YZBCrP5gtADmtXApB5AMLn+vdyA+PyxCjrCs00mjyUozssO33cwDeT3wNGdLxJ5M//YqtHAJw==
+  dependencies:
+    minimist "^1.2.6"
+
+mkdirp@^1.0.4:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-1.0.4.tgz#3eb5ed62622756d79a5f0e2a221dfebad75c2f7e"
+  integrity sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw==
+
+mocha@^9.0.3:
+  version "9.2.2"
+  resolved "https://registry.yarnpkg.com/mocha/-/mocha-9.2.2.tgz#d70db46bdb93ca57402c809333e5a84977a88fb9"
+  integrity sha512-L6XC3EdwT6YrIk0yXpavvLkn8h+EU+Y5UcCHKECyMbdUIxyMuZj4bX4U9e1nvnvUUvQVsV2VHQr5zLdcUkhW/g==
+  dependencies:
+    "@ungap/promise-all-settled" "1.1.2"
+    ansi-colors "4.1.1"
+    browser-stdout "1.3.1"
+    chokidar "3.5.3"
+    debug "4.3.3"
+    diff "5.0.0"
+    escape-string-regexp "4.0.0"
+    find-up "5.0.0"
+    glob "7.2.0"
+    growl "1.10.5"
+    he "1.2.0"
+    js-yaml "4.1.0"
+    log-symbols "4.1.0"
+    minimatch "4.2.1"
+    ms "2.1.3"
+    nanoid "3.3.1"
+    serialize-javascript "6.0.0"
+    strip-json-comments "3.1.1"
+    supports-color "8.1.1"
+    which "2.0.2"
+    workerpool "6.2.0"
+    yargs "16.2.0"
+    yargs-parser "20.2.4"
+    yargs-unparser "2.0.0"
+
+ms@2.1.2:
+  version "2.1.2"
+  resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009"
+  integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w==
+
+ms@2.1.3, ms@^2.0.0:
+  version "2.1.3"
+  resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2"
+  integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA==
+
+mustache@^4.0.0:
+  version "4.2.0"
+  resolved "https://registry.yarnpkg.com/mustache/-/mustache-4.2.0.tgz#e5892324d60a12ec9c2a73359edca52972bf6f64"
+  integrity sha512-71ippSywq5Yb7/tVYyGbkBggbU8H3u5Rz56fH60jGFgr8uHwxs+aSKeqmluIVzM0m0kB7xQjKS6qPfd0b2ZoqQ==
+
+nan@2.14.0:
+  version "2.14.0"
+  resolved "https://registry.yarnpkg.com/nan/-/nan-2.14.0.tgz#7818f722027b2459a86f0295d434d1fc2336c52c"
+  integrity sha512-INOFj37C7k3AfaNTtX8RhsTw7qRy7eLET14cROi9+5HAVbbHuIWUHEauBv5qT4Av2tWasiTY1Jw6puUNqRJXQg==
+
+nan@^2.13.2, nan@^2.14.0:
+  version "2.22.0"
+  resolved "https://registry.yarnpkg.com/nan/-/nan-2.22.0.tgz#31bc433fc33213c97bad36404bb68063de604de3"
+  integrity sha512-nbajikzWTMwsW+eSsNm3QwlOs7het9gGJU5dDZzRTQGk03vyBOauxgI4VakDzE0PtsGTmXPsXTbbjVhRwR5mpw==
+
+nanoid@3.3.1:
+  version "3.3.1"
+  resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.1.tgz#6347a18cac88af88f58af0b3594b723d5e99bb35"
+  integrity sha512-n6Vs/3KGyxPQd6uO0eH4Bv0ojGSUvuLlIHtC3Y0kEO23YRge8H9x1GCzLn28YX0H66pMkxuaeESFq4tKISKwdw==
+
+near-api-js@^1.0.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/near-api-js/-/near-api-js-1.1.0.tgz#907e807f052c1f043c6fbf28f61872de3c02235a"
+  integrity sha512-qYKv1mYsaDZc2uYndhS+ttDhR9+60qFc+ZjD6lWsAxr3ZskMjRwPffDGQZYhC7BRDQMe1HEbk6d5mf+TVm0Lqg==
+  dependencies:
+    bn.js "5.2.1"
+    borsh "^0.7.0"
+    bs58 "^4.0.0"
+    depd "^2.0.0"
+    error-polyfill "^0.1.3"
+    http-errors "^1.7.2"
+    js-sha256 "^0.9.0"
+    mustache "^4.0.0"
+    node-fetch "^2.6.1"
+    text-encoding-utf-8 "^1.0.2"
+    tweetnacl "^1.0.1"
+
+no-case@^3.0.4:
+  version "3.0.4"
+  resolved "https://registry.yarnpkg.com/no-case/-/no-case-3.0.4.tgz#d361fd5c9800f558551a8369fc0dcd4662b6124d"
+  integrity sha512-fgAN3jGAh+RoxUGZHTSOLJIqUc2wmoBwGR4tbpNAKmmovFoWq0OdRkb0VkldReO2a2iBT/OEulG9XSUc10r3zg==
+  dependencies:
+    lower-case "^2.0.2"
+    tslib "^2.0.3"
+
+node-addon-api@^2.0.0:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-2.0.2.tgz#432cfa82962ce494b132e9d72a15b29f71ff5d32"
+  integrity sha512-Ntyt4AIXyaLIuMHF6IOoTakB3K+RWxwtsHNRxllEoA6vPwP9o4866g6YWDLUdnucilZhmkxiHwHr11gAENw+QA==
+
+node-addon-api@^5.0.0:
+  version "5.1.0"
+  resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-5.1.0.tgz#49da1ca055e109a23d537e9de43c09cca21eb762"
+  integrity sha512-eh0GgfEkpnoWDq+VY8OyvYhFEzBk6jIYbRKdIlyTiAXIVJ8PyBaKb0rp7oDtoddbdoHWhq8wwr+XZ81F1rpNdA==
+
+node-fetch@^2.6.1, node-fetch@^2.7.0:
+  version "2.7.0"
+  resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.7.0.tgz#d0f0fa6e3e2dc1d27efcd8ad99d550bda94d187d"
+  integrity sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A==
+  dependencies:
+    whatwg-url "^5.0.0"
+
+node-gyp-build@^4.2.0, node-gyp-build@^4.3.0:
+  version "4.8.4"
+  resolved "https://registry.yarnpkg.com/node-gyp-build/-/node-gyp-build-4.8.4.tgz#8a70ee85464ae52327772a90d66c6077a900cfc8"
+  integrity sha512-LA4ZjwlnUblHVgq0oBF3Jl/6h/Nvs5fzBLwdEF4nuxnFdsfajde4WfxtJr3CaiH+F6ewcIB/q4jQ4UzPyid+CQ==
+
+noms@0.0.0:
+  version "0.0.0"
+  resolved "https://registry.yarnpkg.com/noms/-/noms-0.0.0.tgz#da8ebd9f3af9d6760919b27d9cdc8092a7332859"
+  integrity sha512-lNDU9VJaOPxUmXcLb+HQFeUgQQPtMI24Gt6hgfuMHRJgMRHMF/qZ4HJD3GDru4sSw9IQl2jPjAYnQrdIeLbwow==
+  dependencies:
+    inherits "^2.0.1"
+    readable-stream "~1.0.31"
+
+normalize-path@^3.0.0, normalize-path@~3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65"
+  integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==
+
+o3@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/o3/-/o3-1.0.3.tgz#192ce877a882dfa6751f0412a865fafb2da1dac0"
+  integrity sha512-f+4n+vC6s4ysy7YO7O2gslWZBUu8Qj2i2OUJOvjRxQva7jVjYjB29jrr9NCjmxZQR0gzrOcv1RnqoYOeMs5VRQ==
+  dependencies:
+    capability "^0.2.5"
+
+object-assign@^4.1.1:
+  version "4.1.1"
+  resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863"
+  integrity sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==
+
+object-keys@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e"
+  integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA==
+
+once@^1.3.0:
+  version "1.4.0"
+  resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1"
+  integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==
+  dependencies:
+    wrappy "1"
+
+optimism@^0.18.0:
+  version "0.18.1"
+  resolved "https://registry.yarnpkg.com/optimism/-/optimism-0.18.1.tgz#5cf16847921413dbb0ac809907370388b9c6335f"
+  integrity sha512-mLXNwWPa9dgFyDqkNi54sjDyNJ9/fTI6WGBLgnXku1vdKY/jovHfZT5r+aiVeFFLOz+foPNOm5YJ4mqgld2GBQ==
+  dependencies:
+    "@wry/caches" "^1.0.0"
+    "@wry/context" "^0.7.0"
+    "@wry/trie" "^0.5.0"
+    tslib "^2.3.0"
+
+p-limit@^3.0.2:
+  version "3.1.0"
+  resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b"
+  integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ==
+  dependencies:
+    yocto-queue "^0.1.0"
+
+p-locate@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834"
+  integrity sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw==
+  dependencies:
+    p-limit "^3.0.2"
+
+pako@^2.0.3:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/pako/-/pako-2.1.0.tgz#266cc37f98c7d883545d11335c00fbd4062c9a86"
+  integrity sha512-w+eufiZ1WuJYgPXbV/PO3NCMEc3xqylkKHzp8bxp1uW4qaSNQUkwmLLEc3kKsfz8lpV1F8Ht3U1Cm+9Srog2ug==
+
+path-exists@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3"
+  integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w==
+
+path-is-absolute@^1.0.0:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f"
+  integrity sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==
+
+path-parse@^1.0.7:
+  version "1.0.7"
+  resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.7.tgz#fbc114b60ca42b30d9daf5858e4bd68bbedb6735"
+  integrity sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==
+
+pathval@^1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/pathval/-/pathval-1.1.1.tgz#8534e77a77ce7ac5a2512ea21e0fdb8fcf6c3d8d"
+  integrity sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ==
+
+pbkdf2@^3.0.17:
+  version "3.1.2"
+  resolved "https://registry.yarnpkg.com/pbkdf2/-/pbkdf2-3.1.2.tgz#dd822aa0887580e52f1a039dc3eda108efae3075"
+  integrity sha512-iuh7L6jA7JEGu2WxDwtQP1ddOpaJNC4KlDEFfdQajSGgGPNi4OyDc2R7QnbY2bR9QjBVGwgvTdNJZoE7RaxUMA==
+  dependencies:
+    create-hash "^1.1.2"
+    create-hmac "^1.1.4"
+    ripemd160 "^2.0.1"
+    safe-buffer "^5.0.1"
+    sha.js "^2.4.8"
+
+picomatch@^2.0.4, picomatch@^2.2.1:
+  version "2.3.1"
+  resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42"
+  integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA==
+
+prettier@^2.6.2:
+  version "2.8.8"
+  resolved "https://registry.yarnpkg.com/prettier/-/prettier-2.8.8.tgz#e8c5d7e98a4305ffe3de2e1fc4aca1a71c28b1da"
+  integrity sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q==
+
+process-nextick-args@~2.0.0:
+  version "2.0.1"
+  resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2"
+  integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag==
+
+prop-types@^15.7.2:
+  version "15.8.1"
+  resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.8.1.tgz#67d87bf1a694f48435cf332c24af10214a3140b5"
+  integrity sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==
+  dependencies:
+    loose-envify "^1.4.0"
+    object-assign "^4.1.1"
+    react-is "^16.13.1"
+
+protobufjs@^6.8.8, protobufjs@~6.11.2, protobufjs@~6.11.3:
+  version "6.11.4"
+  resolved "https://registry.yarnpkg.com/protobufjs/-/protobufjs-6.11.4.tgz#29a412c38bf70d89e537b6d02d904a6f448173aa"
+  integrity sha512-5kQWPaJHi1WoCpjTGszzQ32PG2F4+wRY6BmAT4Vfw56Q2FZ4YZzK20xUYQH4YkfehY1e6QSICrJquM6xXZNcrw==
+  dependencies:
+    "@protobufjs/aspromise" "^1.1.2"
+    "@protobufjs/base64" "^1.1.2"
+    "@protobufjs/codegen" "^2.0.4"
+    "@protobufjs/eventemitter" "^1.1.0"
+    "@protobufjs/fetch" "^1.1.0"
+    "@protobufjs/float" "^1.0.2"
+    "@protobufjs/inquire" "^1.1.0"
+    "@protobufjs/path" "^1.1.2"
+    "@protobufjs/pool" "^1.1.0"
+    "@protobufjs/utf8" "^1.1.0"
+    "@types/long" "^4.0.1"
+    "@types/node" ">=13.7.0"
+    long "^4.0.0"
+
+protobufjs@^7.0.0:
+  version "7.4.0"
+  resolved "https://registry.yarnpkg.com/protobufjs/-/protobufjs-7.4.0.tgz#7efe324ce9b3b61c82aae5de810d287bc08a248a"
+  integrity sha512-mRUWCc3KUU4w1jU8sGxICXH/gNS94DvI1gxqDvBzhj1JpcsimQkYiOJfwsPUykUI5ZaspFbSgmBLER8IrQ3tqw==
+  dependencies:
+    "@protobufjs/aspromise" "^1.1.2"
+    "@protobufjs/base64" "^1.1.2"
+    "@protobufjs/codegen" "^2.0.4"
+    "@protobufjs/eventemitter" "^1.1.0"
+    "@protobufjs/fetch" "^1.1.0"
+    "@protobufjs/float" "^1.0.2"
+    "@protobufjs/inquire" "^1.1.0"
+    "@protobufjs/path" "^1.1.2"
+    "@protobufjs/pool" "^1.1.0"
+    "@protobufjs/utf8" "^1.1.0"
+    "@types/node" ">=13.7.0"
+    long "^5.0.0"
+
+proxy-from-env@^1.1.0:
+  version "1.1.0"
+  resolved "https://registry.yarnpkg.com/proxy-from-env/-/proxy-from-env-1.1.0.tgz#e102f16ca355424865755d2c9e8ea4f24d58c3e2"
+  integrity sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg==
+
+randombytes@^2.1.0:
+  version "2.1.0"
+  resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a"
+  integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==
+  dependencies:
+    safe-buffer "^5.1.0"
+
+react-is@^16.13.1, react-is@^16.7.0:
+  version "16.13.1"
+  resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4"
+  integrity sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==
+
+readable-stream@^3.6.0:
+  version "3.6.2"
+  resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.2.tgz#56a9b36ea965c00c5a93ef31eb111a0f11056967"
+  integrity sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==
+  dependencies:
+    inherits "^2.0.3"
+    string_decoder "^1.1.1"
+    util-deprecate "^1.0.1"
+
+readable-stream@~1.0.31:
+  version "1.0.34"
+  resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-1.0.34.tgz#125820e34bc842d2f2aaafafe4c2916ee32c157c"
+  integrity sha512-ok1qVCJuRkNmvebYikljxJA/UEsKwLl2nI1OmaqAu4/UE+h0wKCHok4XkL/gvi39OacXvw59RJUOFUkDib2rHg==
+  dependencies:
+    core-util-is "~1.0.0"
+    inherits "~2.0.1"
+    isarray "0.0.1"
+    string_decoder "~0.10.x"
+
+readable-stream@~2.3.6:
+  version "2.3.8"
+  resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.8.tgz#91125e8042bba1b9887f49345f6277027ce8be9b"
+  integrity sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA==
+  dependencies:
+    core-util-is "~1.0.0"
+    inherits "~2.0.3"
+    isarray "~1.0.0"
+    process-nextick-args "~2.0.0"
+    safe-buffer "~5.1.1"
+    string_decoder "~1.1.1"
+    util-deprecate "~1.0.1"
+
+readdirp@~3.6.0:
+  version "3.6.0"
+  resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.6.0.tgz#74a370bd857116e245b29cc97340cd431a02a6c7"
+  integrity sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==
+  dependencies:
+    picomatch "^2.2.1"
+
+readonly-date@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/readonly-date/-/readonly-date-1.0.0.tgz#5af785464d8c7d7c40b9d738cbde8c646f97dcd9"
+  integrity sha512-tMKIV7hlk0h4mO3JTmmVuIlJVXjKk3Sep9Bf5OH0O+758ruuVkUy2J9SttDLm91IEX/WHlXPSpxMGjPj4beMIQ==
+
+rechoir@^0.6.2:
+  version "0.6.2"
+  resolved "https://registry.yarnpkg.com/rechoir/-/rechoir-0.6.2.tgz#85204b54dba82d5742e28c96756ef43af50e3384"
+  integrity sha512-HFM8rkZ+i3zrV+4LQjwQ0W+ez98pApMGM3HUrN04j3CqzPOzl9nmP15Y8YXNm8QHGv/eacOVEjqhmWpkRV0NAw==
+  dependencies:
+    resolve "^1.1.6"
+
+regenerator-runtime@^0.14.0:
+  version "0.14.1"
+  resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz#356ade10263f685dda125100cd862c1db895327f"
+  integrity sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw==
+
+rehackt@^0.1.0:
+  version "0.1.0"
+  resolved "https://registry.yarnpkg.com/rehackt/-/rehackt-0.1.0.tgz#a7c5e289c87345f70da8728a7eb878e5d03c696b"
+  integrity sha512-7kRDOuLHB87D/JESKxQoRwv4DzbIdwkAGQ7p6QKGdVlY1IZheUnVhlk/4UZlNUVxdAXpyxikE3URsG067ybVzw==
+
+require-directory@^2.1.1:
+  version "2.1.1"
+  resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42"
+  integrity sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==
+
+resolve-pkg-maps@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/resolve-pkg-maps/-/resolve-pkg-maps-1.0.0.tgz#616b3dc2c57056b5588c31cdf4b3d64db133720f"
+  integrity sha512-seS2Tj26TBVOC2NIc2rOe2y2ZO7efxITtLZcGSOnHHNOQ7CkiUBfw0Iw2ck6xkIhPwLhKNLS8BO+hEpngQlqzw==
+
+resolve@^1.1.6:
+  version "1.22.10"
+  resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.10.tgz#b663e83ffb09bbf2386944736baae803029b8b39"
+  integrity sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w==
+  dependencies:
+    is-core-module "^2.16.0"
+    path-parse "^1.0.7"
+    supports-preserve-symlinks-flag "^1.0.0"
+
+response-iterator@^0.2.6:
+  version "0.2.11"
+  resolved "https://registry.yarnpkg.com/response-iterator/-/response-iterator-0.2.11.tgz#6d3e4f12d2e212f28bc742bc639a37939a4483ca"
+  integrity sha512-5tdhcAeGMSyM0/FoxAYjoOxQZ2tRR2H/S/t6kGRXu6iiWcGY5UnZgkVANbTwBVUSGqWu0ADctmoi6lOCIF8uKQ==
+
+ripemd160-min@0.0.6:
+  version "0.0.6"
+  resolved "https://registry.yarnpkg.com/ripemd160-min/-/ripemd160-min-0.0.6.tgz#a904b77658114474d02503e819dcc55853b67e62"
+  integrity sha512-+GcJgQivhs6S9qvLogusiTcS9kQUfgR75whKuy5jIhuiOfQuJ8fjqxV6EGD5duH1Y/FawFUMtMhyeq3Fbnib8A==
+
+ripemd160@^2.0.0, ripemd160@^2.0.1:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/ripemd160/-/ripemd160-2.0.2.tgz#a1c1a6f624751577ba5d07914cbc92850585890c"
+  integrity sha512-ii4iagi25WusVoiC4B4lq7pbXfAp3D9v5CwfkY33vffw2+pkDjY1D8GaN7spsxvCSx8dkPqOZCEZyfxcmJG2IA==
+  dependencies:
+    hash-base "^3.0.0"
+    inherits "^2.0.1"
+
+rlp@^2.2.3, rlp@^2.2.4:
+  version "2.2.7"
+  resolved "https://registry.yarnpkg.com/rlp/-/rlp-2.2.7.tgz#33f31c4afac81124ac4b283e2bd4d9720b30beaf"
+  integrity sha512-d5gdPmgQ0Z+AklL2NVXr/IoSjNZFfTVvQWzL/AM2AOcSzYP2xjlb0AC8YyCLc41MSNf6P6QVtjgPdmVtzb+4lQ==
+  dependencies:
+    bn.js "^5.2.0"
+
+rpc-websockets@^7.5.1:
+  version "7.11.2"
+  resolved "https://registry.yarnpkg.com/rpc-websockets/-/rpc-websockets-7.11.2.tgz#582910c425b9f2c860327481c1d1e0e431bf4a6d"
+  integrity sha512-pL9r5N6AVHlMN/vT98+fcO+5+/UcPLf/4tq+WUaid/PPUGS/ttJ3y8e9IqmaWKtShNAysMSjkczuEA49NuV7UQ==
+  dependencies:
+    eventemitter3 "^4.0.7"
+    uuid "^8.3.2"
+    ws "^8.5.0"
+  optionalDependencies:
+    bufferutil "^4.0.1"
+    utf-8-validate "^5.0.2"
+
+rpc-websockets@^9.0.2:
+  version "9.0.4"
+  resolved "https://registry.yarnpkg.com/rpc-websockets/-/rpc-websockets-9.0.4.tgz#9d8ee82533b5d1e13d9ded729e3e38d0d8fa083f"
+  integrity sha512-yWZWN0M+bivtoNLnaDbtny4XchdAIF5Q4g/ZsC5UC61Ckbp0QczwO8fg44rV3uYmY4WHd+EZQbn90W1d8ojzqQ==
+  dependencies:
+    "@swc/helpers" "^0.5.11"
+    "@types/uuid" "^8.3.4"
+    "@types/ws" "^8.2.2"
+    buffer "^6.0.3"
+    eventemitter3 "^5.0.1"
+    uuid "^8.3.2"
+    ws "^8.5.0"
+  optionalDependencies:
+    bufferutil "^4.0.1"
+    utf-8-validate "^5.0.2"
+
+rxjs@^7.4.0, rxjs@^7.5.6, rxjs@^7.8.0:
+  version "7.8.1"
+  resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-7.8.1.tgz#6f6f3d99ea8044291efd92e7c7fcf562c4057543"
+  integrity sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg==
+  dependencies:
+    tslib "^2.1.0"
+
+safe-buffer@^5.0.1, safe-buffer@^5.1.0, safe-buffer@^5.1.1, safe-buffer@^5.1.2, safe-buffer@^5.2.0, safe-buffer@^5.2.1, safe-buffer@~5.2.0:
+  version "5.2.1"
+  resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6"
+  integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==
+
+safe-buffer@~5.1.0, safe-buffer@~5.1.1:
+  version "5.1.2"
+  resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d"
+  integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==
+
+scrypt-js@3.0.1, scrypt-js@^3.0.0:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/scrypt-js/-/scrypt-js-3.0.1.tgz#d314a57c2aef69d1ad98a138a21fe9eafa9ee312"
+  integrity sha512-cdwTTnqPu0Hyvf5in5asVdZocVDTNRmR7XEcJuIzMjJeSHybHl7vpB66AzwTaIg6CLSbtjcxc8fqcySfnTkccA==
+
+secp256k1@3.7.1:
+  version "3.7.1"
+  resolved "https://registry.yarnpkg.com/secp256k1/-/secp256k1-3.7.1.tgz#12e473e0e9a7c2f2d4d4818e722ad0e14cc1e2f1"
+  integrity sha512-1cf8sbnRreXrQFdH6qsg2H71Xw91fCCS9Yp021GnUNJzWJS/py96fS4lHbnTnouLp08Xj6jBoBB6V78Tdbdu5g==
+  dependencies:
+    bindings "^1.5.0"
+    bip66 "^1.1.5"
+    bn.js "^4.11.8"
+    create-hash "^1.2.0"
+    drbg.js "^1.0.1"
+    elliptic "^6.4.1"
+    nan "^2.14.0"
+    safe-buffer "^5.1.2"
+
+secp256k1@5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/secp256k1/-/secp256k1-5.0.1.tgz#dc2c86187d48ff2da756f0f7e96417ee03c414b1"
+  integrity sha512-lDFs9AAIaWP9UCdtWrotXWWF9t8PWgQDcxqgAnpM9rMqxb3Oaq2J0thzPVSxBwdJgyQtkU/sYtFtbM1RSt/iYA==
+  dependencies:
+    elliptic "^6.5.7"
+    node-addon-api "^5.0.0"
+    node-gyp-build "^4.2.0"
+
+secp256k1@^4.0.1, secp256k1@^4.0.2, secp256k1@^4.0.3:
+  version "4.0.4"
+  resolved "https://registry.yarnpkg.com/secp256k1/-/secp256k1-4.0.4.tgz#58f0bfe1830fe777d9ca1ffc7574962a8189f8ab"
+  integrity sha512-6JfvwvjUOn8F/jUoBY2Q1v5WY5XS+rj8qSe0v8Y4ezH4InLgTEeOOPQsRll9OV429Pvo6BCHGavIyJfr3TAhsw==
+  dependencies:
+    elliptic "^6.5.7"
+    node-addon-api "^5.0.0"
+    node-gyp-build "^4.2.0"
+
+serialize-javascript@6.0.0:
+  version "6.0.0"
+  resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-6.0.0.tgz#efae5d88f45d7924141da8b5c3a7a7e663fefeb8"
+  integrity sha512-Qr3TosvguFt8ePWqsvRfrKyQXIiW+nGbYpy8XK24NQHE83caxWt+mIymTT19DGFbNWNLfEwsrkSmN64lVWB9ag==
+  dependencies:
+    randombytes "^2.1.0"
+
+setimmediate@^1.0.5:
+  version "1.0.5"
+  resolved "https://registry.yarnpkg.com/setimmediate/-/setimmediate-1.0.5.tgz#290cbb232e306942d7d7ea9b83732ab7856f8285"
+  integrity sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA==
+
+setprototypeof@1.2.0:
+  version "1.2.0"
+  resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.2.0.tgz#66c9a24a73f9fc28cbe66b09fed3d33dcaf1b424"
+  integrity sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw==
+
+sha.js@^2.4.0, sha.js@^2.4.8:
+  version "2.4.11"
+  resolved "https://registry.yarnpkg.com/sha.js/-/sha.js-2.4.11.tgz#37a5cf0b81ecbc6943de109ba2960d1b26584ae7"
+  integrity sha512-QMEp5B7cftE7APOjk5Y6xgrbWu+WkLVQwk8JNjZ8nKRciZaByEW6MubieAiToS7+dwvrjGhH8jRXz3MVd0AYqQ==
+  dependencies:
+    inherits "^2.0.1"
+    safe-buffer "^5.0.1"
+
+sha3@^2.1.1:
+  version "2.1.4"
+  resolved "https://registry.yarnpkg.com/sha3/-/sha3-2.1.4.tgz#000fac0fe7c2feac1f48a25e7a31b52a6492cc8f"
+  integrity sha512-S8cNxbyb0UGUM2VhRD4Poe5N58gJnJsLJ5vC7FYWGUmGhcsj4++WaIOBFVDxlG0W3To6xBuiRh+i0Qp2oNCOtg==
+  dependencies:
+    buffer "6.0.3"
+
+shelljs@^0.8.5:
+  version "0.8.5"
+  resolved "https://registry.yarnpkg.com/shelljs/-/shelljs-0.8.5.tgz#de055408d8361bed66c669d2f000538ced8ee20c"
+  integrity sha512-TiwcRcrkhHvbrZbnRcFYMLl30Dfov3HKqzp5tO5b4pt6G/SezKcYhmDg15zXVBswHmctSAQKznqNW2LO5tTDow==
+  dependencies:
+    glob "^7.0.0"
+    interpret "^1.0.0"
+    rechoir "^0.6.2"
+
+shx@^0.3.2:
+  version "0.3.4"
+  resolved "https://registry.yarnpkg.com/shx/-/shx-0.3.4.tgz#74289230b4b663979167f94e1935901406e40f02"
+  integrity sha512-N6A9MLVqjxZYcVn8hLmtneQWIJtp8IKzMP4eMnx+nqkvXoqinUPCbUFLp2UcWTEIUONhlk0ewxr/jaVGlc+J+g==
+  dependencies:
+    minimist "^1.2.3"
+    shelljs "^0.8.5"
+
+snake-case@^3.0.4:
+  version "3.0.4"
+  resolved "https://registry.yarnpkg.com/snake-case/-/snake-case-3.0.4.tgz#4f2bbd568e9935abdfd593f34c691dadb49c452c"
+  integrity sha512-LAOh4z89bGQvl9pFfNF8V146i7o7/CqFPbqzYgP+yYzDIDeS9HaNFtXABamRW+AQzEVODcvE79ljJ+8a9YSdMg==
+  dependencies:
+    dot-case "^3.0.4"
+    tslib "^2.0.3"
+
+snakecase-keys@^5.1.2, snakecase-keys@^5.4.1:
+  version "5.5.0"
+  resolved "https://registry.yarnpkg.com/snakecase-keys/-/snakecase-keys-5.5.0.tgz#fbf9ca4a47fcc37c7496d00c6d9e455da9750403"
+  integrity sha512-r3kRtnoPu3FxGJ3fny6PKNnU3pteb29o6qAa0ugzhSseKNWRkw1dw8nIjXMyyKaU9vQxxVIE62Mb3bKbdrgpiw==
+  dependencies:
+    map-obj "^4.1.0"
+    snake-case "^3.0.4"
+    type-fest "^3.12.0"
+
+source-map-support@^0.5.6:
+  version "0.5.21"
+  resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.21.tgz#04fe7c7f9e1ed2d662233c28cb2b35b9f63f6e4f"
+  integrity sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w==
+  dependencies:
+    buffer-from "^1.0.0"
+    source-map "^0.6.0"
+
+source-map@^0.6.0:
+  version "0.6.1"
+  resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263"
+  integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==
+
+"statuses@>= 1.5.0 < 2":
+  version "1.5.0"
+  resolved "https://registry.yarnpkg.com/statuses/-/statuses-1.5.0.tgz#161c7dac177659fd9811f43771fa99381478628c"
+  integrity sha512-OpZ3zP+jT1PI7I8nemJX4AKmAX070ZkYPVWV/AaKTJl+tXCTGyVdC1a4SL8RUQYEwk/f34ZX8UTykN68FwrqAA==
+
+store2@^2.12.0:
+  version "2.14.4"
+  resolved "https://registry.yarnpkg.com/store2/-/store2-2.14.4.tgz#81b313abaddade4dcd7570c5cc0e3264a8f7a242"
+  integrity sha512-srTItn1GOvyvOycgxjAnPA63FZNwy0PTyUBFMHRM+hVFltAeoh0LmNBz9SZqUS9mMqGk8rfyWyXn3GH5ReJ8Zw==
+
+string-width@^4.1.0, string-width@^4.2.0:
+  version "4.2.3"
+  resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010"
+  integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g==
+  dependencies:
+    emoji-regex "^8.0.0"
+    is-fullwidth-code-point "^3.0.0"
+    strip-ansi "^6.0.1"
+
+string_decoder@^1.1.1:
+  version "1.3.0"
+  resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e"
+  integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==
+  dependencies:
+    safe-buffer "~5.2.0"
+
+string_decoder@~0.10.x:
+  version "0.10.31"
+  resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-0.10.31.tgz#62e203bc41766c6c28c9fc84301dab1c5310fa94"
+  integrity sha512-ev2QzSzWPYmy9GuqfIVildA4OdcGLeFZQrq5ys6RtiuF+RQQiZWr8TZNyAcuVXyQRYfEO+MsoB/1BuQVhOJuoQ==
+
+string_decoder@~1.1.1:
+  version "1.1.1"
+  resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8"
+  integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg==
+  dependencies:
+    safe-buffer "~5.1.0"
+
+strip-ansi@^6.0.0, strip-ansi@^6.0.1:
+  version "6.0.1"
+  resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9"
+  integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A==
+  dependencies:
+    ansi-regex "^5.0.1"
+
+strip-bom@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3"
+  integrity sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA==
+
+strip-hex-prefix@1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/strip-hex-prefix/-/strip-hex-prefix-1.0.0.tgz#0c5f155fef1151373377de9dbb588da05500e36f"
+  integrity sha512-q8d4ue7JGEiVcypji1bALTos+0pWtyGlivAWyPuTkHzuTCJqrK9sWxYQZUq6Nq3cuyv3bm734IhHvHtGGURU6A==
+  dependencies:
+    is-hex-prefixed "1.0.0"
+
+strip-json-comments@3.1.1:
+  version "3.1.1"
+  resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006"
+  integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig==
+
+superstruct@^0.15.4:
+  version "0.15.5"
+  resolved "https://registry.yarnpkg.com/superstruct/-/superstruct-0.15.5.tgz#0f0a8d3ce31313f0d84c6096cd4fa1bfdedc9dab"
+  integrity sha512-4AOeU+P5UuE/4nOUkmcQdW5y7i9ndt1cQd/3iUe+LTz3RxESf/W/5lg4B74HbDMMv8PHnPnGCQFH45kBcrQYoQ==
+
+superstruct@^1.0.3:
+  version "1.0.4"
+  resolved "https://registry.yarnpkg.com/superstruct/-/superstruct-1.0.4.tgz#0adb99a7578bd2f1c526220da6571b2d485d91ca"
+  integrity sha512-7JpaAoX2NGyoFlI9NBh66BQXGONc+uE+MRS5i2iOBKuS4e+ccgMDjATgZldkah+33DakBxDHiss9kvUcGAO8UQ==
+
+superstruct@^2.0.2:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/superstruct/-/superstruct-2.0.2.tgz#3f6d32fbdc11c357deff127d591a39b996300c54"
+  integrity sha512-uV+TFRZdXsqXTL2pRvujROjdZQ4RAlBUS5BTh9IGm+jTqQntYThciG/qu57Gs69yjnVUSqdxF9YLmSnpupBW9A==
+
+supports-color@8.1.1:
+  version "8.1.1"
+  resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-8.1.1.tgz#cd6fc17e28500cff56c1b86c0a7fd4a54a73005c"
+  integrity sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q==
+  dependencies:
+    has-flag "^4.0.0"
+
+supports-color@^5.3.0:
+  version "5.5.0"
+  resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz#e2e69a44ac8772f78a1ec0b35b689df6530efc8f"
+  integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==
+  dependencies:
+    has-flag "^3.0.0"
+
+supports-color@^7.1.0:
+  version "7.2.0"
+  resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da"
+  integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==
+  dependencies:
+    has-flag "^4.0.0"
+
+supports-preserve-symlinks-flag@^1.0.0:
+  version "1.0.0"
+  resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09"
+  integrity sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==
+
+symbol-observable@^2.0.3:
+  version "2.0.3"
+  resolved "https://registry.yarnpkg.com/symbol-observable/-/symbol-observable-2.0.3.tgz#5b521d3d07a43c351055fa43b8355b62d33fd16a"
+  integrity sha512-sQV7phh2WCYAn81oAkakC5qjq2Ml0g8ozqz03wOGnx9dDlG1de6yrF+0RAzSJD8fPUow3PTSMf2SAbOGxb93BA==
+
+symbol-observable@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/symbol-observable/-/symbol-observable-4.0.0.tgz#5b425f192279e87f2f9b937ac8540d1984b39205"
+  integrity sha512-b19dMThMV4HVFynSAM1++gBHAbk2Tc/osgLIBZMKsyqh34jb2e8Os7T6ZW/Bt3pJFdBTd2JwAnAAEQV7rSNvcQ==
+
+text-encoding-utf-8@^1.0.2:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/text-encoding-utf-8/-/text-encoding-utf-8-1.0.2.tgz#585b62197b0ae437e3c7b5d0af27ac1021e10d13"
+  integrity sha512-8bw4MY9WjdsD2aMtO0OzOCY3pXGYNx2d2FfHRVUKkiCPDWjKuOlhLVASS+pD7VkLTVjW268LYJHwsnPFlBpbAg==
+
+through2@^2.0.1:
+  version "2.0.5"
+  resolved "https://registry.yarnpkg.com/through2/-/through2-2.0.5.tgz#01c1e39eb31d07cb7d03a96a70823260b23132cd"
+  integrity sha512-/mrRod8xqpA+IHSLyGCQ2s8SPHiCDEeQJSep1jqLYeEUClOFG2Qsh+4FU6G9VeqpZnGW/Su8LQGc4YKni5rYSQ==
+  dependencies:
+    readable-stream "~2.3.6"
+    xtend "~4.0.1"
+
+"through@>=2.2.7 <3":
+  version "2.3.8"
+  resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5"
+  integrity sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg==
+
+tiny-secp256k1@^1.1.3:
+  version "1.1.7"
+  resolved "https://registry.yarnpkg.com/tiny-secp256k1/-/tiny-secp256k1-1.1.7.tgz#0c1b6b9d2d93404f9093dc7e287b0aa834480573"
+  integrity sha512-eb+F6NabSnjbLwNoC+2o5ItbmP1kg7HliWue71JgLegQt6A5mTN8YbvTLCazdlg6e5SV6A+r8OGvZYskdlmhqQ==
+  dependencies:
+    bindings "^1.3.0"
+    bn.js "^4.11.8"
+    create-hmac "^1.1.7"
+    elliptic "^6.4.0"
+    nan "^2.13.2"
+
+tmp@^0.2.1:
+  version "0.2.3"
+  resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.2.3.tgz#eb783cc22bc1e8bebd0671476d46ea4eb32a79ae"
+  integrity sha512-nZD7m9iCPC5g0pYmcaxogYKggSfLsdxl8of3Q/oIbqCqLLIO9IAF0GWjX1z9NZRHPiXv8Wex4yDCaZsgEw0Y8w==
+
+to-regex-range@^5.0.1:
+  version "5.0.1"
+  resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4"
+  integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==
+  dependencies:
+    is-number "^7.0.0"
+
+toidentifier@1.0.1:
+  version "1.0.1"
+  resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.1.tgz#3be34321a88a820ed1bd80dfaa33e479fbb8dd35"
+  integrity sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==
+
+toml@^3.0.0:
+  version "3.0.0"
+  resolved "https://registry.yarnpkg.com/toml/-/toml-3.0.0.tgz#342160f1af1904ec9d204d03a5d61222d762c5ee"
+  integrity sha512-y/mWCZinnvxjTKYhJ+pYxwD0mRLVvOtdS2Awbgxln6iEnt4rk0yBxeSBHkGJcPucRiG0e55mwWp+g/05rsrd6w==
+
+tr46@~0.0.3:
+  version "0.0.3"
+  resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a"
+  integrity sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw==
+
+ts-invariant@^0.10.3:
+  version "0.10.3"
+  resolved "https://registry.yarnpkg.com/ts-invariant/-/ts-invariant-0.10.3.tgz#3e048ff96e91459ffca01304dbc7f61c1f642f6c"
+  integrity sha512-uivwYcQaxAucv1CzRp2n/QdYPo4ILf9VXgH19zEIjFx2EJufV16P0JtJVpYHy89DItG6Kwj2oIUjrcK5au+4tQ==
+  dependencies:
+    tslib "^2.1.0"
+
+ts-mocha@^10.0.0:
+  version "10.0.0"
+  resolved "https://registry.yarnpkg.com/ts-mocha/-/ts-mocha-10.0.0.tgz#41a8d099ac90dbbc64b06976c5025ffaebc53cb9"
+  integrity sha512-VRfgDO+iiuJFlNB18tzOfypJ21xn2xbuZyDvJvqpTbWgkAgD17ONGr8t+Tl8rcBtOBdjXp5e/Rk+d39f7XBHRw==
+  dependencies:
+    ts-node "7.0.1"
+  optionalDependencies:
+    tsconfig-paths "^3.5.0"
+
+ts-node@7.0.1:
+  version "7.0.1"
+  resolved "https://registry.yarnpkg.com/ts-node/-/ts-node-7.0.1.tgz#9562dc2d1e6d248d24bc55f773e3f614337d9baf"
+  integrity sha512-BVwVbPJRspzNh2yfslyT1PSbl5uIk03EZlb493RKHN4qej/D06n1cEhjlOJG69oFsE7OT8XjpTUcYf6pKTLMhw==
+  dependencies:
+    arrify "^1.0.0"
+    buffer-from "^1.1.0"
+    diff "^3.1.0"
+    make-error "^1.1.1"
+    minimist "^1.2.0"
+    mkdirp "^0.5.1"
+    source-map-support "^0.5.6"
+    yn "^2.0.0"
+
+tsconfig-paths@^3.5.0:
+  version "3.15.0"
+  resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz#5299ec605e55b1abb23ec939ef15edaf483070d4"
+  integrity sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg==
+  dependencies:
+    "@types/json5" "^0.0.29"
+    json5 "^1.0.2"
+    minimist "^1.2.6"
+    strip-bom "^3.0.0"
+
+tslib@^2.0.3, tslib@^2.1.0, tslib@^2.3.0, tslib@^2.8.0:
+  version "2.8.1"
+  resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.8.1.tgz#612efe4ed235d567e8aba5f2a5fab70280ade83f"
+  integrity sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==
+
+tsx@^4.19.2:
+  version "4.19.2"
+  resolved "https://registry.yarnpkg.com/tsx/-/tsx-4.19.2.tgz#2d7814783440e0ae42354d0417d9c2989a2ae92c"
+  integrity sha512-pOUl6Vo2LUq/bSa8S5q7b91cgNSjctn9ugq/+Mvow99qW6x/UZYwzxy/3NmqoT66eHYfCVvFvACC58UBPFf28g==
+  dependencies:
+    esbuild "~0.23.0"
+    get-tsconfig "^4.7.5"
+  optionalDependencies:
+    fsevents "~2.3.3"
+
+tweetnacl-util@^0.15.1:
+  version "0.15.1"
+  resolved "https://registry.yarnpkg.com/tweetnacl-util/-/tweetnacl-util-0.15.1.tgz#b80fcdb5c97bcc508be18c44a4be50f022eea00b"
+  integrity sha512-RKJBIj8lySrShN4w6i/BonWp2Z/uxwC3h4y7xsRrpP59ZboCd0GpEVsOnMDYLMmKBpYhb5TgHzZXy7wTfYFBRw==
+
+tweetnacl@1.0.3, tweetnacl@^1.0.1, tweetnacl@^1.0.3:
+  version "1.0.3"
+  resolved "https://registry.yarnpkg.com/tweetnacl/-/tweetnacl-1.0.3.tgz#ac0af71680458d8a6378d0d0d050ab1407d35596"
+  integrity sha512-6rt+RN7aOi1nGMyC4Xa5DdYiukl2UWCbcJft7YhxReBGQD7OAM8Pbxw6YMo4r2diNEA8FEmu32YOn9rhaiE5yw==
+
+type-detect@^4.0.0, type-detect@^4.1.0:
+  version "4.1.0"
+  resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-4.1.0.tgz#deb2453e8f08dcae7ae98c626b13dddb0155906c"
+  integrity sha512-Acylog8/luQ8L7il+geoSxhEkazvkslg7PSNKOX59mbB9cOveP5aq9h74Y7YU8yDpJwetzQQrfIwtf4Wp4LKcw==
+
+type-fest@^3.12.0:
+  version "3.13.1"
+  resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-3.13.1.tgz#bb744c1f0678bea7543a2d1ec24e83e68e8c8706"
+  integrity sha512-tLq3bSNx+xSpwvAJnzrK0Ep5CLNWjvFTOp71URMaAEWBfRb9nnJiBoUe0tF8bI4ZFO3omgBR6NvnbzVUT3Ly4g==
+
+typeforce@^1.11.5:
+  version "1.18.0"
+  resolved "https://registry.yarnpkg.com/typeforce/-/typeforce-1.18.0.tgz#d7416a2c5845e085034d70fcc5b6cc4a90edbfdc"
+  integrity sha512-7uc1O8h1M1g0rArakJdf0uLRSSgFcYexrVoKo+bzJd32gd4gDy2L/Z+8/FjPnU9ydY3pEnVPtr9FyscYY60K1g==
+
+typescript@^4.3.5:
+  version "4.9.5"
+  resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.9.5.tgz#095979f9bcc0d09da324d58d03ce8f8374cbe65a"
+  integrity sha512-1FXk9E2Hm+QzZQ7z+McJiHL4NW1F2EzMu9Nq9i3zAaGqibafqYwCVU6WyWAuyQRRzOlxou8xZSyXLEN8oKj24g==
+
+u3@^0.1.1:
+  version "0.1.1"
+  resolved "https://registry.yarnpkg.com/u3/-/u3-0.1.1.tgz#5f52044f42ee76cd8de33148829e14528494b73b"
+  integrity sha512-+J5D5ir763y+Am/QY6hXNRlwljIeRMZMGs0cT6qqZVVzzT3X3nFPXVyPOFRMOR4kupB0T8JnCdpWdp6Q/iXn3w==
+
+undici-types@~5.26.4:
+  version "5.26.5"
+  resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-5.26.5.tgz#bcd539893d00b56e964fd2657a4866b221a65617"
+  integrity sha512-JlCMO+ehdEIKqlFxk6IfVoAUVmgz7cU7zD/h9XZ0qzeosSHmUJVOzSQvvYSYWXkFXC+IfLKSIffhv0sVZup6pA==
+
+undici-types@~6.20.0:
+  version "6.20.0"
+  resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.20.0.tgz#8171bf22c1f588d1554d55bf204bc624af388433"
+  integrity sha512-Ny6QZ2Nju20vw1SRHe3d9jVu6gJ+4e3+MMpqu7pqE5HT6WsTSlce++GQmK5UXS8mzV8DSYHrQH+Xrf2jVcuKNg==
+
+untildify@^4.0.0:
+  version "4.0.0"
+  resolved "https://registry.yarnpkg.com/untildify/-/untildify-4.0.0.tgz#2bc947b953652487e4600949fb091e3ae8cd919b"
+  integrity sha512-KK8xQ1mkzZeg9inewmFVDNkg3l5LUhoq9kN6iWYB/CC9YMG8HA+c1Q8HwDe6dEX7kErrEVNVBO3fWsVq5iDgtw==
+
+utf-8-validate@^5.0.2, utf-8-validate@^5.0.5:
+  version "5.0.10"
+  resolved "https://registry.yarnpkg.com/utf-8-validate/-/utf-8-validate-5.0.10.tgz#d7d10ea39318171ca982718b6b96a8d2442571a2"
+  integrity sha512-Z6czzLq4u8fPOyx7TU6X3dvUZVvoJmxSQ+IcrlmagKhilxlhZgxPK6C5Jqbkw1IDUmFTM+cz9QDnnLTwDz/2gQ==
+  dependencies:
+    node-gyp-build "^4.3.0"
+
+util-deprecate@^1.0.1, util-deprecate@~1.0.1:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf"
+  integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==
+
+uuid@^8.3.2:
+  version "8.3.2"
+  resolved "https://registry.yarnpkg.com/uuid/-/uuid-8.3.2.tgz#80d5b5ced271bb9af6c445f21a1a04c606cefbe2"
+  integrity sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg==
+
+vlq@^2.0.4:
+  version "2.0.4"
+  resolved "https://registry.yarnpkg.com/vlq/-/vlq-2.0.4.tgz#6057b85729245b9829e3cc7755f95b228d4fe041"
+  integrity sha512-aodjPa2wPQFkra1G8CzJBTHXhgk3EVSwxSWXNPr1fgdFLUb8kvLV1iEb6rFgasIsjP82HWI6dsb5Io26DDnasA==
+
+webidl-conversions@^3.0.0:
+  version "3.0.1"
+  resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871"
+  integrity sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ==
+
+whatwg-url@^5.0.0:
+  version "5.0.0"
+  resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-5.0.0.tgz#966454e8765462e37644d3626f6742ce8b70965d"
+  integrity sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw==
+  dependencies:
+    tr46 "~0.0.3"
+    webidl-conversions "^3.0.0"
+
+which@2.0.2:
+  version "2.0.2"
+  resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1"
+  integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA==
+  dependencies:
+    isexe "^2.0.0"
+
+wif@^2.0.6:
+  version "2.0.6"
+  resolved "https://registry.yarnpkg.com/wif/-/wif-2.0.6.tgz#08d3f52056c66679299726fade0d432ae74b4704"
+  integrity sha512-HIanZn1zmduSF+BQhkE+YXIbEiH0xPr1012QbFEGB0xsKqJii0/SqJjyn8dFv6y36kOznMgMB+LGcbZTJ1xACQ==
+  dependencies:
+    bs58check "<3.0.0"
+
+workerpool@6.2.0:
+  version "6.2.0"
+  resolved "https://registry.yarnpkg.com/workerpool/-/workerpool-6.2.0.tgz#827d93c9ba23ee2019c3ffaff5c27fccea289e8b"
+  integrity sha512-Rsk5qQHJ9eowMH28Jwhe8HEbmdYDX4lwoMWshiCXugjtHqMD9ZbiqSDLxcsfdqsETPzVUtX5s1Z5kStiIM6l4A==
+
+wrap-ansi@^7.0.0:
+  version "7.0.0"
+  resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43"
+  integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q==
+  dependencies:
+    ansi-styles "^4.0.0"
+    string-width "^4.1.0"
+    strip-ansi "^6.0.0"
+
+wrappy@1:
+  version "1.0.2"
+  resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f"
+  integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==
+
+ws@7.4.6:
+  version "7.4.6"
+  resolved "https://registry.yarnpkg.com/ws/-/ws-7.4.6.tgz#5654ca8ecdeee47c33a9a4bf6d28e2be2980377c"
+  integrity sha512-YmhHDO4MzaDLB+M9ym/mDA5z0naX8j7SIlT8f8z+I0VtzsRbekxEutHSme7NPS2qE8StCYQNUnfWdXta/Yu85A==
+
+ws@^7, ws@^7.5.10, ws@^7.5.8, ws@^7.5.9:
+  version "7.5.10"
+  resolved "https://registry.yarnpkg.com/ws/-/ws-7.5.10.tgz#58b5c20dc281633f6c19113f39b349bd8bd558d9"
+  integrity sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ==
+
+ws@^8.5.0:
+  version "8.18.0"
+  resolved "https://registry.yarnpkg.com/ws/-/ws-8.18.0.tgz#0d7505a6eafe2b0e712d232b42279f53bc289bbc"
+  integrity sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw==
+
+xstream@^11.14.0:
+  version "11.14.0"
+  resolved "https://registry.yarnpkg.com/xstream/-/xstream-11.14.0.tgz#2c071d26b18310523b6877e86b4e54df068a9ae5"
+  integrity sha512-1bLb+kKKtKPbgTK6i/BaoAn03g47PpFstlbe1BA+y3pNS/LfvcaghS5BFf9+EE1J+KwSQsEpfJvFN5GqFtiNmw==
+  dependencies:
+    globalthis "^1.0.1"
+    symbol-observable "^2.0.3"
+
+xtend@~4.0.1:
+  version "4.0.2"
+  resolved "https://registry.yarnpkg.com/xtend/-/xtend-4.0.2.tgz#bb72779f5fa465186b1f438f674fa347fdb5db54"
+  integrity sha512-LKYU1iAXJXUgAXn9URjiu+MWhyUXHsvfp7mcuYm9dSUKK0/CjtrUwFAxD82/mCWbtLsGjFIad0wIsod4zrTAEQ==
+
+y18n@^5.0.5:
+  version "5.0.8"
+  resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.8.tgz#7f4934d0f7ca8c56f95314939ddcd2dd91ce1d55"
+  integrity sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA==
+
+yargs-parser@20.2.4:
+  version "20.2.4"
+  resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.4.tgz#b42890f14566796f85ae8e3a25290d205f154a54"
+  integrity sha512-WOkpgNhPTlE73h4VFAFsOnomJVaovO8VqLDzy5saChRBFQFBoMYirowyW+Q9HB4HFF4Z7VZTiG3iSzJJA29yRA==
+
+yargs-parser@^20.2.2:
+  version "20.2.9"
+  resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-20.2.9.tgz#2eb7dc3b0289718fc295f362753845c41a0c94ee"
+  integrity sha512-y11nGElTIV+CT3Zv9t7VKl+Q3hTQoT9a1Qzezhhl6Rp21gJ/IVTW7Z3y9EWXhuUBC2Shnf+DX0antecpAwSP8w==
+
+yargs-unparser@2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/yargs-unparser/-/yargs-unparser-2.0.0.tgz#f131f9226911ae5d9ad38c432fe809366c2325eb"
+  integrity sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA==
+  dependencies:
+    camelcase "^6.0.0"
+    decamelize "^4.0.0"
+    flat "^5.0.2"
+    is-plain-obj "^2.1.0"
+
+yargs@16.2.0, yargs@^16.1.0:
+  version "16.2.0"
+  resolved "https://registry.yarnpkg.com/yargs/-/yargs-16.2.0.tgz#1c82bf0f6b6a66eafce7ef30e376f49a12477f66"
+  integrity sha512-D1mvvtDG0L5ft/jGWkLpG1+m0eQxOfaBvTNELraWj22wSVUMWxZUvYgJYcKh6jGGIkJFhH4IZPQhR4TKpc8mBw==
+  dependencies:
+    cliui "^7.0.2"
+    escalade "^3.1.1"
+    get-caller-file "^2.0.5"
+    require-directory "^2.1.1"
+    string-width "^4.2.0"
+    y18n "^5.0.5"
+    yargs-parser "^20.2.2"
+
+yn@^2.0.0:
+  version "2.0.0"
+  resolved "https://registry.yarnpkg.com/yn/-/yn-2.0.0.tgz#e5adabc8acf408f6385fc76495684c88e6af689a"
+  integrity sha512-uTv8J/wiWTgUTg+9vLTi//leUl5vDQS6uii/emeTb2ssY7vl6QWf2fFbIIGjnhjvbdKlU0ed7QPgY1htTC86jQ==
+
+yocto-queue@^0.1.0:
+  version "0.1.0"
+  resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b"
+  integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q==
+
+zen-observable-ts@^1.2.5:
+  version "1.2.5"
+  resolved "https://registry.yarnpkg.com/zen-observable-ts/-/zen-observable-ts-1.2.5.tgz#6c6d9ea3d3a842812c6e9519209365a122ba8b58"
+  integrity sha512-QZWQekv6iB72Naeake9hS1KxHlotfRpe+WGNbNx5/ta+R3DNjVO2bswf63gXlWDcs+EMd7XY8HfVQyP1X6T4Zg==
+  dependencies:
+    zen-observable "0.8.15"
+
+zen-observable@0.8.15:
+  version "0.8.15"
+  resolved "https://registry.yarnpkg.com/zen-observable/-/zen-observable-0.8.15.tgz#96415c512d8e3ffd920afd3889604e30b9eaac15"
+  integrity sha512-PQ2PC7R9rslx84ndNBZB/Dkv8V8fZEpk83RLgXtYd0fwUgEjseMn1Dgajh2x6S8QbZAFa9p2qVCEuYZNgve0dQ==

+ 32 - 0
svm/wormhole-core-shims/crates/definitions/Cargo.toml

@@ -0,0 +1,32 @@
+[package]
+name = "wormhole-svm-definitions"
+description = "Pubkeys and Account Definitions for Wormhole on SVM"
+readme = "README.md"
+
+edition.workspace = true
+license.workspace = true
+repository.workspace = true
+rust-version.workspace = true
+version.workspace = true
+
+# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
+
+[features]
+default = []
+borsh = ["dep:borsh"]
+
+### Network types
+testnet = []
+localnet = []
+
+### Specific networks
+solana = []
+
+[dependencies]
+borsh = { optional = true, workspace = true }
+cfg-if.workspace = true
+sha2-const-stable.workspace = true
+solana-program.workspace = true
+
+[dev-dependencies]
+base64.workspace = true

+ 34 - 0
svm/wormhole-core-shims/crates/definitions/README.md

@@ -0,0 +1,34 @@
+# Wormhole SVM Definitions
+
+Definitions relating to Wormhole SVM programs. These definitions include finding
+PDA addresses (and corresponding bump seeds), various consts (like program IDs),
+and other things that define these program accounts and data.
+
+## Cargo Features
+
+There are features that define network types and specific SVM networks.
+
+### Network Types
+
+The default network type is mainnet. There is no feature that defines mainnet.
+But if one of the following features are defined, program IDs and account
+addresses will not use the ones defined for mainnet.
+
+- `localnet`: Wormhole's Tilt devnet. Programs like the Wormhole Core Bridge and
+  its associated PDAs have addresses specific to this local development network.
+- `testnet`: Public devnet or testnet depending on the specific SVM network. For
+  Solana specifically, this feature corresponds to the public Solana devnet.
+
+### Specific Networks
+
+There are no default network features. These feature labels also exist as
+submodules in this crate. By defining a particular SVM network feature, the
+definitions found in this submodule are simply exported into the crate root.
+
+- `solana`
+
+### Other Features
+
+- `borsh`: Accounts and events relating to Wormhole SVM programs that follow
+  Borsh serialization. This feature also supports deserializing data with
+  discriminators.

+ 54 - 0
svm/wormhole-core-shims/crates/definitions/src/borsh.rs

@@ -0,0 +1,54 @@
+use borsh::{io, BorshDeserialize, BorshSerialize};
+use solana_program::pubkey::Pubkey;
+
+use super::DataDiscriminator;
+
+/// Wormhole Post Message Shim program message event. This message is encoded
+/// as instruction data when the Shim program calls itself via CPI.
+#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, BorshDeserialize, BorshSerialize)]
+pub struct MessageEvent {
+    pub emitter: Pubkey,
+    pub sequence: u64,
+    pub submission_time: u32,
+}
+
+impl DataDiscriminator for MessageEvent {
+    const DISCRIMINATOR: &'static [u8] = &super::MESSAGE_EVENT_DISCRIMINATOR;
+}
+
+#[derive(Debug, Clone, PartialEq, Eq, BorshDeserialize, BorshSerialize)]
+pub struct GuardianSignatures {
+    /// Payer of this guardian signatures account. Only this account may extend
+    /// signatures. When the close signatures instruction is invoked, rent will
+    /// be returned to this account.
+    pub refund_recipient: Pubkey,
+
+    /// Guardian set index that these signatures correspond to. This index will
+    /// be checked against the Wormhole Core Bridge program's guardian set when
+    /// the verify hash instruction is invoked.
+    ///
+    /// NOTE: Encoding the guardian set as big-endian matches the derivation
+    /// used by the Wormhole Core Bridge program.
+    pub guardian_set_index_be: [u8; 4],
+
+    /// Guardian signatures loaded via the post signatures instruction.
+    pub guardian_signatures: Vec<[u8; 66]>,
+}
+
+impl DataDiscriminator for GuardianSignatures {
+    const DISCRIMINATOR: &'static [u8] = &super::GUARDIAN_SIGNATURES_DISCRIMINATOR;
+}
+
+/// Deserializes Borsh-serialized data with a discriminator.
+pub fn deserialize_with_discriminator<T: BorshDeserialize + DataDiscriminator>(
+    data: &[u8],
+) -> io::Result<T> {
+    if data.len() < T::DISCRIMINATOR.len() || &data[..T::DISCRIMINATOR.len()] != T::DISCRIMINATOR {
+        return Err(io::Error::new(
+            io::ErrorKind::InvalidData,
+            "invalid discriminator",
+        ));
+    }
+
+    T::deserialize(&mut &data[T::DISCRIMINATOR.len()..])
+}

+ 170 - 0
svm/wormhole-core-shims/crates/definitions/src/lib.rs

@@ -0,0 +1,170 @@
+#![deny(dead_code, unused_imports, unused_mut, unused_variables)]
+#![doc = include_str!("../README.md")]
+
+#[cfg(feature = "borsh")]
+pub mod borsh;
+pub mod solana;
+pub mod zero_copy;
+
+// NOTE: Expand this conditional as Wormhole supports more SVM networks.
+cfg_if::cfg_if! {
+    if #[cfg(feature = "solana")] {
+        pub use solana::*;
+    }
+}
+
+pub use solana_program::keccak::{Hash, HASH_BYTES};
+
+use solana_program::pubkey::Pubkey;
+
+/// Wormhole Core Bridge program's bridge config account seed.
+pub const CORE_BRIDGE_CONFIG_SEED: &[u8] = b"Bridge";
+
+/// Wormhole Core Bridge program's fee collector account seed.
+pub const FEE_COLLECTOR_SEED: &[u8] = b"fee_collector";
+
+/// Wormhole Core Bridge program's emitter sequence account seed.
+pub const EMITTER_SEQUENCE_SEED: &[u8] = b"Sequence";
+
+/// Wormhole Core Bridge program's guardian set account seed.
+pub const GUARDIAN_SET_SEED: &[u8] = b"GuardianSet";
+
+/// Anchor event CPI's authority seed.
+pub const EVENT_AUTHORITY_SEED: &[u8] = b"__event_authority";
+
+/// Wormhole Post Message Shim's message event instruction data discriminator.
+pub const MESSAGE_EVENT_DISCRIMINATOR: [u8; 8] = make_anchor_discriminator(b"event:MessageEvent");
+
+/// Wormhole Verify VAA Shim's guardian signatures account data discriminator.
+pub const GUARDIAN_SIGNATURES_DISCRIMINATOR: [u8; 8] =
+    make_anchor_discriminator(b"account:GuardianSignatures");
+
+pub const GUARDIAN_SIGNATURE_LENGTH: usize = 66;
+pub const GUARDIAN_PUBKEY_LENGTH: usize = 20;
+
+pub const ANCHOR_EVENT_CPI_SELECTOR: [u8; 8] = u64::to_be_bytes(0xe445a52e51cb9a1d);
+
+/// Convenience method to compute the Keccak-256 digest of Keccak-256 hashed
+/// input data. For some messages (like query responses), there is a prefix
+/// prepended to this hash before producing the digest. Otherwise the hash is
+/// simply hashed again. This digest along with a guardian's signature is used
+/// to recover this guardian's pubkey.
+///
+/// NOTE: In an SVM program, prefer this method or
+/// [solana_program::keccak::hashv]. Both of these methods use the
+/// `sol_keccak256` syscall under the hood in SVM runtime, which uses minimal
+/// compute units.
+///
+/// # Examples
+///
+/// A v1 VAA digest can be computed as follows:
+/// ```rust
+/// use wormhole_svm_definitions::compute_keccak_digest;
+///
+/// // `vec_body` is the encoded body of the VAA.
+/// # let vaa_body = vec![];
+/// let digest = compute_keccak_digest(
+///     solana_program::keccak::hash(&vaa_body),
+///     None, // there is no prefix for V1 messages
+/// );
+/// ```
+///
+/// A QueryResponse digest can be computed as follows:
+/// ```rust
+/// # mod wormhole_query_sdk {
+/// #    pub const MESSAGE_PREFIX: &'static [u8] = b"ruh roh";
+/// # }
+/// use wormhole_query_sdk::MESSAGE_PREFIX;
+/// use wormhole_svm_definitions::compute_keccak_digest;
+///
+/// # let query_response_bytes = vec![];
+/// let digest = compute_keccak_digest(
+///     solana_program::keccak::hash(&query_response_bytes),
+///     Some(MESSAGE_PREFIX)
+/// );
+#[inline]
+pub fn compute_keccak_digest(hashed_data: Hash, prefix: Option<&[u8]>) -> Hash {
+    match prefix {
+        Some(prefix) => solana_program::keccak::hashv(&[prefix, &hashed_data.0]),
+        None => solana_program::keccak::hashv(&[&hashed_data.0]),
+    }
+}
+
+/// Derive the Wormhole Core Bridge program's bridge config account address and
+/// bump.
+pub fn find_core_bridge_config_address(wormhole_program_id: &Pubkey) -> (Pubkey, u8) {
+    Pubkey::find_program_address(&[CORE_BRIDGE_CONFIG_SEED], wormhole_program_id)
+}
+
+/// Derive the Wormhole Core Bridge program's fee collector account address and
+/// bump.
+pub fn find_fee_collector_address(wormhole_program_id: &Pubkey) -> (Pubkey, u8) {
+    Pubkey::find_program_address(&[FEE_COLLECTOR_SEED], wormhole_program_id)
+}
+
+/// Derive the Wormhole Core Bridge program's emitter sequence account address
+/// and bump.
+pub fn find_emitter_sequence_address(
+    emitter: &Pubkey,
+    wormhole_program_id: &Pubkey,
+) -> (Pubkey, u8) {
+    Pubkey::find_program_address(
+        &[EMITTER_SEQUENCE_SEED, emitter.as_ref()],
+        wormhole_program_id,
+    )
+}
+
+/// Derive the Wormhole Core Bridge program's guardian set address and bump.
+pub fn find_guardian_set_address(
+    index_be_bytes: [u8; 4],
+    wormhole_program_id: &Pubkey,
+) -> (Pubkey, u8) {
+    Pubkey::find_program_address(&[GUARDIAN_SET_SEED, &index_be_bytes], wormhole_program_id)
+}
+
+/// Derive the Wormhole Post Message Shim program's message account address and
+/// bump.
+pub fn find_shim_message_address(
+    emitter: &Pubkey,
+    post_message_program_id: &Pubkey,
+) -> (Pubkey, u8) {
+    Pubkey::find_program_address(&[emitter.as_ref()], post_message_program_id)
+}
+
+/// Derive the Anchor event CPI's authority address and bump.
+pub fn find_event_authority_address(program_id: &Pubkey) -> (Pubkey, u8) {
+    Pubkey::find_program_address(&[EVENT_AUTHORITY_SEED], program_id)
+}
+
+/// Generate an 8-byte Anchor discriminator from the input data.
+pub const fn make_anchor_discriminator(input: &[u8]) -> [u8; 8] {
+    let digest = sha2_const_stable::Sha256::new().update(input).finalize();
+    let mut trimmed = [0; 8];
+    let mut i = 0;
+
+    loop {
+        if i >= 8 {
+            break;
+        }
+        trimmed[i] = digest[i];
+        i += 1;
+    }
+
+    trimmed
+}
+
+/// Trait to encode and decode the SVM finality of a message.
+pub trait EncodeFinality: Sized + Copy {
+    /// Encode SVM finality into a byte.
+    fn encode(&self) -> u8;
+
+    /// Decode SVM finality from a byte.
+    fn decode(data: u8) -> Option<Self>;
+}
+
+/// Trait that defines an arbitrary discriminator for deserializing data. This
+/// discriminator acts as a prefix to identify which kind of data is encoded.
+/// For Anchor accounts and events, this discriminator is 8 bytes long.
+pub trait DataDiscriminator {
+    const DISCRIMINATOR: &'static [u8];
+}

+ 97 - 0
svm/wormhole-core-shims/crates/definitions/src/solana.rs

@@ -0,0 +1,97 @@
+/// Wormhole Chain ID identifying Solana's network. This ID is shared between Solana mainnet and
+/// devnet.
+pub const CHAIN_ID: u16 = 1;
+
+use solana_program::{pubkey, pubkey::Pubkey};
+
+pub const POST_MESSAGE_SHIM_PROGRAM_ID: Pubkey =
+    pubkey!("EtZMZM22ViKMo4r5y4Anovs3wKQ2owUmDpjygnMMcdEX");
+pub const POST_MESSAGE_SHIM_EVENT_AUTHORITY: Pubkey =
+    pubkey!("HQS31aApX3DDkuXgSpV9XyDUNtFgQ31pUn5BNWHG2PSp");
+pub const POST_MESSAGE_SHIM_EVENT_AUTHORITY_BUMP: u8 = 255;
+
+pub const VERIFY_VAA_SHIM_PROGRAM_ID: Pubkey =
+    pubkey!("EFaNWErqAtVWufdNb7yofSHHfWFos843DFpu4JBw24at");
+
+cfg_if::cfg_if! {
+    if #[cfg(feature = "testnet")] {
+        /// Core Bridge program ID on Solana devnet.
+        pub const CORE_BRIDGE_PROGRAM_ID: Pubkey = pubkey!("3u8hJUVTA4jH1wYAyUur7FFZVQ8H635K3tSHHF4ssjQ5");
+        pub const CORE_BRIDGE_FEE_COLLECTOR: Pubkey = pubkey!("7s3a1ycs16d6SNDumaRtjcoyMaTDZPavzgsmS3uUZYWX");
+        pub const CORE_BRIDGE_CONFIG: Pubkey = pubkey!("6bi4JGDoRwUs9TYBuvoA7dUVyikTJDrJsJU1ew6KVLiu");
+    } else if #[cfg(feature = "localnet")] {
+        /// Core Bridge program ID on Wormhole's Tilt (dev) network.
+        pub const CORE_BRIDGE_PROGRAM_ID: Pubkey = pubkey!("Bridge1p5gheXUvJ6jGWGeCsgPKgnE3YgdGKRVCMY9o");
+        pub const CORE_BRIDGE_FEE_COLLECTOR: Pubkey = pubkey!("GXBsgBD3LDn3vkRZF6TfY5RqgajVZ4W5bMAdiAaaUARs");
+        pub const CORE_BRIDGE_CONFIG: Pubkey = pubkey!("FKoMTctsC7vJbEqyRiiPskPnuQx2tX1kurmvWByq5uZP");
+    }
+    // Default to mainnet.
+    else {
+        /// Core Bridge program ID on Solana mainnet.
+        pub const CORE_BRIDGE_PROGRAM_ID: Pubkey = pubkey!("worm2ZoG2kUd4vFXhvjh93UUH596ayRfgQ2MgjNMTth");
+        pub const CORE_BRIDGE_FEE_COLLECTOR: Pubkey = pubkey!("9bFNrXNb2WTx8fMHXCheaZqkLZ3YCCaiqTftHxeintHy");
+        pub const CORE_BRIDGE_CONFIG: Pubkey = pubkey!("2yVjuQwpsvdsrywzsJJVs9Ueh4zayyo5DYJbBNc3DDpn");
+    }
+}
+
+/// Finality of the message (which is when the Wormhole guardians will attest to
+/// this message's observation).
+///
+/// On Solana, there are only two commitment levels that the Wormhole guardians
+/// recognize.
+#[cfg_attr(
+    feature = "borsh",
+    derive(borsh::BorshDeserialize, borsh::BorshSerialize)
+)]
+#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
+#[repr(u8)]
+pub enum Finality {
+    /// Equivalent to observing after one slot.
+    Confirmed,
+
+    /// Equivalent to observing after 32 slots.
+    Finalized,
+}
+
+impl super::EncodeFinality for Finality {
+    fn encode(&self) -> u8 {
+        *self as u8
+    }
+
+    fn decode(data: u8) -> Option<Self> {
+        match data {
+            0 => Some(Self::Confirmed),
+            1 => Some(Self::Finalized),
+            _ => None,
+        }
+    }
+}
+
+#[cfg(test)]
+mod tests {
+    use super::*;
+
+    #[test]
+    fn test_core_bridge_fee_collector() {
+        let (expected, _) = crate::find_fee_collector_address(&CORE_BRIDGE_PROGRAM_ID);
+        assert_eq!(CORE_BRIDGE_FEE_COLLECTOR, expected);
+    }
+
+    #[test]
+    fn test_core_bridge_config() {
+        let (expected, _) = crate::find_core_bridge_config_address(&CORE_BRIDGE_PROGRAM_ID);
+        assert_eq!(CORE_BRIDGE_CONFIG, expected);
+    }
+
+    #[test]
+    fn test_post_message_shim_event_authority() {
+        let expected = crate::find_event_authority_address(&POST_MESSAGE_SHIM_PROGRAM_ID);
+        assert_eq!(
+            (
+                POST_MESSAGE_SHIM_EVENT_AUTHORITY,
+                POST_MESSAGE_SHIM_EVENT_AUTHORITY_BUMP
+            ),
+            expected
+        );
+    }
+}

+ 398 - 0
svm/wormhole-core-shims/crates/definitions/src/zero_copy/guardian_set.rs

@@ -0,0 +1,398 @@
+use crate::GUARDIAN_PUBKEY_LENGTH;
+
+/// Guardian set account owned by the Wormhole Core Bridge program.
+///
+/// NOTE: This is a zero-copy struct only meant to read in account data. There
+/// is no verification of whether this account is owned by the Wormhole Verify
+/// VAA program. You must ensure that the account is owned by this program.
+///
+/// Because this account does not have a discriminator, you must also ensure
+/// that the account key matches the PDA's expected key with
+/// [find_guardian_set_address].
+///
+/// [find_guardian_set_address]: crate::find_guardian_set_address
+pub struct GuardianSet<'data>(&'data [u8]);
+
+impl<'data> GuardianSet<'data> {
+    pub const MINIMUM_SIZE: usize = {
+        4 // guardian set index
+        + 4 // keys length
+        + 4 // creation time
+        + 4 // expiration time
+    };
+
+    /// Attempts to read a guardian set account from the given data. This method
+    /// will return `None` if the data is not a valid guardian set account.
+    pub fn new(data: &'data [u8]) -> Option<Self> {
+        if data.len() < Self::MINIMUM_SIZE {
+            return None;
+        }
+
+        let account = Self(data);
+        let total_len = (account.keys_len() as usize)
+            .checked_mul(GUARDIAN_PUBKEY_LENGTH)?
+            .checked_add(Self::MINIMUM_SIZE)?;
+
+        if data.len() < total_len {
+            return None;
+        }
+
+        Some(account)
+    }
+
+    /// Guardian set index that these keys correspond to.
+    #[inline]
+    pub fn guardian_set_index(&self) -> u32 {
+        u32::from_le_bytes(self.guardian_set_index_slice().try_into().unwrap())
+    }
+
+    /// Number of guardian public keys in this set.
+    #[inline]
+    pub fn keys_len(&self) -> u32 {
+        u32::from_le_bytes(self.keys_len_slice().try_into().unwrap())
+    }
+
+    /// Guardian public key at the given index. This method will return `None`
+    /// if the index is out of bounds.
+    #[inline]
+    pub fn key(&self, index: usize) -> Option<[u8; GUARDIAN_PUBKEY_LENGTH]> {
+        self.key_slice(index)?.try_into().ok()
+    }
+
+    /// When the guardian set was created.
+    #[inline]
+    pub fn creation_time(&self) -> u32 {
+        u32::from_le_bytes(self.creation_time_slice().try_into().unwrap())
+    }
+
+    /// When the guardian set will expire. A value of 0 means that the guardian
+    /// set is the current one.
+    #[inline]
+    pub fn expiration_time(&self) -> u32 {
+        u32::from_le_bytes(self.expiration_time_slice().try_into().unwrap())
+    }
+
+    #[inline(always)]
+    pub fn guardian_set_index_slice(&self) -> &'data [u8] {
+        &self.0[..4]
+    }
+
+    #[inline(always)]
+    pub fn keys_len_slice(&self) -> &'data [u8] {
+        &self.0[4..8]
+    }
+
+    #[inline(always)]
+    pub fn key_slice(&self, index: usize) -> Option<&'data [u8]> {
+        if index >= self.keys_len() as usize {
+            return None;
+        }
+
+        let start_idx = 8 + index * GUARDIAN_PUBKEY_LENGTH;
+        let end_idx = 28 + index * GUARDIAN_PUBKEY_LENGTH;
+
+        Some(&self.0[start_idx..end_idx])
+    }
+
+    #[inline(always)]
+    pub fn creation_time_slice(&self) -> &'data [u8] {
+        let end_idx = self.keys_end_index();
+        &self.0[end_idx..(end_idx + 4)]
+    }
+
+    #[inline(always)]
+    pub fn expiration_time_slice(&self) -> &'data [u8] {
+        let end_idx = self.keys_end_index();
+        &self.0[(end_idx + 4)..(end_idx + 8)]
+    }
+
+    /// Checks if the guardian set is active at the given timestamp. A guardian
+    /// set is active if its expiration time is 0 or if the given timestamp is
+    /// less than or equal to the expiration time.
+    ///
+    /// NOTE: This method also handles a special case for the initial guardian
+    /// set on mainnet. The initial guardian set was never expired.
+    #[inline]
+    pub fn is_active(&self, timestamp: u32) -> bool {
+        const GUARDIAN_SET_ZERO_INDEX: [u8; 4] = [0, 0, 0, 0];
+        const GUARDIAN_SET_ZERO_CREATION_TIME: [u8; 4] = u32::to_le_bytes(1_628_099_186);
+
+        // Note: This is a fix for Wormhole on mainnet.  The initial guardian set was never expired
+        // so we block it here.
+        if self.guardian_set_index_slice() == GUARDIAN_SET_ZERO_INDEX
+            && self.creation_time_slice() == GUARDIAN_SET_ZERO_CREATION_TIME
+        {
+            false
+        } else {
+            let expiration_time = self.expiration_time();
+            expiration_time == 0 || timestamp <= expiration_time
+        }
+    }
+
+    /// Returns the number of guardian public keys required to form a quorum.
+    #[inline]
+    pub fn quorum(&self) -> u32 {
+        (self.keys_len() * 2) / 3 + 1
+    }
+
+    #[inline(always)]
+    fn keys_end_index(&self) -> usize {
+        8 + (self.keys_len() as usize) * GUARDIAN_PUBKEY_LENGTH
+    }
+}
+
+#[cfg(test)]
+mod test {
+    use base64::{prelude::BASE64_STANDARD, Engine};
+
+    use super::*;
+
+    /// Mainnet guardian set 4: AFEXK4A1BU7BZfi8niAmker98LH9EARB544wKGPXwMyy.
+    #[test]
+    fn test_guardian_set_active() {
+        let encoded_data = "BAAAABMAAABYk7WnbD9zlkVkiIW9zMBs1wo80/9suVJYm96GLCXvQ5ITL7nUpCFXEU3oRgGTvfOi/PgfhqCXZfR2L9EQegCGsy16CXeSaiBRMdhzHTnL64yCsv2C+u0nEdWa8PJJnRbnJvayEbOXVsBCRBvm2GULabVOvnFeI0NUzltNNI+3S5WOiWbi7D29SVinzRXnyvB8Tj3I58Rp+SyM2I+4AFogdKO/kTlT1pUmDYi8GqJaTu42PvAACsAHZyezX76i2sKP7lzLD+p2jq9FztE2udniSQNGSuiJ9cinI/wU+TEkt8c4hDy7iehkyGLDjN3Mz5XSzDek3ANqjSMrSPYs3UcxQS9IkNp5j2iWozMfZLSMEtHVf9nL5wgRcaob4dNsr+OGeRD5nAnjR4mcGcOBkrbnOHzNdoJ3wX2rG3pQJ8CzzxeOIa0ud64GcRVJz7sfnHqdgJboXhSH81UV0CqSdTUEqNdUcbn0nttvvryJj0A+R3PpX+sV6Ayamcg0jXiZHmYAAAAA";
+
+        let data = BASE64_STANDARD.decode(encoded_data).unwrap();
+
+        let guardian_set = GuardianSet::new(&data).unwrap();
+        assert_eq!(guardian_set.guardian_set_index(), 4);
+        assert_eq!(guardian_set.keys_len(), 19);
+        assert_eq!(guardian_set.creation_time(), 1_713_281_400);
+        assert_eq!(guardian_set.expiration_time(), 0);
+        assert!(guardian_set.is_active(0));
+        assert!(guardian_set.is_active(u32::MAX));
+
+        let expected_keys = [
+            [
+                0x58, 0x93, 0xb5, 0xa7, 0x6c, 0x3f, 0x73, 0x96, 0x45, 0x64, 0x88, 0x85, 0xbd, 0xcc,
+                0xc0, 0x6c, 0xd7, 0x0a, 0x3c, 0xd3,
+            ],
+            [
+                0xff, 0x6c, 0xb9, 0x52, 0x58, 0x9b, 0xde, 0x86, 0x2c, 0x25, 0xef, 0x43, 0x92, 0x13,
+                0x2f, 0xb9, 0xd4, 0xa4, 0x21, 0x57,
+            ],
+            [
+                0x11, 0x4d, 0xe8, 0x46, 0x01, 0x93, 0xbd, 0xf3, 0xa2, 0xfc, 0xf8, 0x1f, 0x86, 0xa0,
+                0x97, 0x65, 0xf4, 0x76, 0x2f, 0xd1,
+            ],
+            [
+                0x10, 0x7a, 0x00, 0x86, 0xb3, 0x2d, 0x7a, 0x09, 0x77, 0x92, 0x6a, 0x20, 0x51, 0x31,
+                0xd8, 0x73, 0x1d, 0x39, 0xcb, 0xeb,
+            ],
+            [
+                0x8c, 0x82, 0xb2, 0xfd, 0x82, 0xfa, 0xed, 0x27, 0x11, 0xd5, 0x9a, 0xf0, 0xf2, 0x49,
+                0x9d, 0x16, 0xe7, 0x26, 0xf6, 0xb2,
+            ],
+            [
+                0x11, 0xb3, 0x97, 0x56, 0xc0, 0x42, 0x44, 0x1b, 0xe6, 0xd8, 0x65, 0x0b, 0x69, 0xb5,
+                0x4e, 0xbe, 0x71, 0x5e, 0x23, 0x43,
+            ],
+            [
+                0x54, 0xce, 0x5b, 0x4d, 0x34, 0x8f, 0xb7, 0x4b, 0x95, 0x8e, 0x89, 0x66, 0xe2, 0xec,
+                0x3d, 0xbd, 0x49, 0x58, 0xa7, 0xcd,
+            ],
+            [
+                0x15, 0xe7, 0xca, 0xf0, 0x7c, 0x4e, 0x3d, 0xc8, 0xe7, 0xc4, 0x69, 0xf9, 0x2c, 0x8c,
+                0xd8, 0x8f, 0xb8, 0x00, 0x5a, 0x20,
+            ],
+            [
+                0x74, 0xa3, 0xbf, 0x91, 0x39, 0x53, 0xd6, 0x95, 0x26, 0x0d, 0x88, 0xbc, 0x1a, 0xa2,
+                0x5a, 0x4e, 0xee, 0x36, 0x3e, 0xf0,
+            ],
+            [
+                0x00, 0x0a, 0xc0, 0x07, 0x67, 0x27, 0xb3, 0x5f, 0xbe, 0xa2, 0xda, 0xc2, 0x8f, 0xee,
+                0x5c, 0xcb, 0x0f, 0xea, 0x76, 0x8e,
+            ],
+            [
+                0xaf, 0x45, 0xce, 0xd1, 0x36, 0xb9, 0xd9, 0xe2, 0x49, 0x03, 0x46, 0x4a, 0xe8, 0x89,
+                0xf5, 0xc8, 0xa7, 0x23, 0xfc, 0x14,
+            ],
+            [
+                0xf9, 0x31, 0x24, 0xb7, 0xc7, 0x38, 0x84, 0x3c, 0xbb, 0x89, 0xe8, 0x64, 0xc8, 0x62,
+                0xc3, 0x8c, 0xdd, 0xcc, 0xcf, 0x95,
+            ],
+            [
+                0xd2, 0xcc, 0x37, 0xa4, 0xdc, 0x03, 0x6a, 0x8d, 0x23, 0x2b, 0x48, 0xf6, 0x2c, 0xdd,
+                0x47, 0x31, 0x41, 0x2f, 0x48, 0x90,
+            ],
+            [
+                0xda, 0x79, 0x8f, 0x68, 0x96, 0xa3, 0x33, 0x1f, 0x64, 0xb4, 0x8c, 0x12, 0xd1, 0xd5,
+                0x7f, 0xd9, 0xcb, 0xe7, 0x08, 0x11,
+            ],
+            [
+                0x71, 0xaa, 0x1b, 0xe1, 0xd3, 0x6c, 0xaf, 0xe3, 0x86, 0x79, 0x10, 0xf9, 0x9c, 0x09,
+                0xe3, 0x47, 0x89, 0x9c, 0x19, 0xc3,
+            ],
+            [
+                0x81, 0x92, 0xb6, 0xe7, 0x38, 0x7c, 0xcd, 0x76, 0x82, 0x77, 0xc1, 0x7d, 0xab, 0x1b,
+                0x7a, 0x50, 0x27, 0xc0, 0xb3, 0xcf,
+            ],
+            [
+                0x17, 0x8e, 0x21, 0xad, 0x2e, 0x77, 0xae, 0x06, 0x71, 0x15, 0x49, 0xcf, 0xbb, 0x1f,
+                0x9c, 0x7a, 0x9d, 0x80, 0x96, 0xe8,
+            ],
+            [
+                0x5e, 0x14, 0x87, 0xf3, 0x55, 0x15, 0xd0, 0x2a, 0x92, 0x75, 0x35, 0x04, 0xa8, 0xd7,
+                0x54, 0x71, 0xb9, 0xf4, 0x9e, 0xdb,
+            ],
+            [
+                0x6f, 0xbe, 0xbc, 0x89, 0x8f, 0x40, 0x3e, 0x47, 0x73, 0xe9, 0x5f, 0xeb, 0x15, 0xe8,
+                0x0c, 0x9a, 0x99, 0xc8, 0x34, 0x8d,
+            ],
+        ];
+
+        for (i, expected_key) in expected_keys.iter().enumerate() {
+            let key = guardian_set.key(i).unwrap();
+            assert_eq!(&key, expected_key, "key[{}]", i);
+        }
+
+        // Add a byte at the end and we should expect the same output.
+        let mut data = BASE64_STANDARD.decode(encoded_data).unwrap();
+        data.push(0);
+
+        let guardian_set = GuardianSet::new(&data).unwrap();
+        assert_eq!(guardian_set.guardian_set_index(), 4);
+        assert_eq!(guardian_set.keys_len(), 19);
+        assert_eq!(guardian_set.creation_time(), 1_713_281_400);
+        assert_eq!(guardian_set.expiration_time(), 0);
+        assert!(guardian_set.is_active(0));
+        assert!(guardian_set.is_active(u32::MAX));
+
+        for (i, expected_key) in expected_keys.iter().enumerate() {
+            let key = guardian_set.key(i).unwrap();
+            assert_eq!(&key, expected_key, "key[{}]", i);
+        }
+
+        // Remove two bytes from this data and we should expect None.
+        data.pop();
+        data.pop();
+
+        assert!(GuardianSet::new(&data).is_none());
+    }
+
+    /// Mainnet guardian set 0: XMDS7qfSAgYsonPpKoAjcGhX9VFjXdGkiHjEDkTidf8H2P.
+    #[test]
+    fn test_guardian_set_zero() {
+        let encoded_data = "AAAAAAEAAABYzDrlwJeyE848gZeeG5+VcHRqpXLSCmEAAAAA";
+
+        let data = BASE64_STANDARD.decode(encoded_data).unwrap();
+
+        let guardian_set = GuardianSet::new(&data).unwrap();
+        assert_eq!(guardian_set.guardian_set_index(), 0);
+        assert_eq!(guardian_set.keys_len(), 1);
+        assert_eq!(guardian_set.creation_time(), 1_628_099_186);
+        assert_eq!(guardian_set.expiration_time(), 0);
+        assert!(!guardian_set.is_active(0));
+        assert!(!guardian_set.is_active(u32::MAX));
+
+        let expected_keys = [[
+            0x58, 0xcc, 0x3a, 0xe5, 0xc0, 0x97, 0xb2, 0x13, 0xce, 0x3c, 0x81, 0x97, 0x9e, 0x1b,
+            0x9f, 0x95, 0x70, 0x74, 0x6a, 0xa5,
+        ]];
+
+        for (i, expected_key) in expected_keys.iter().enumerate() {
+            let key = guardian_set.key(i).unwrap();
+            assert_eq!(&key, expected_key, "key[{}]", i);
+        }
+    }
+
+    /// Mainnet guardian set 3: 6d3w8mGjJauf6gCAg7WfLezbaPmUHYGuoNutnfYF1RYM.
+    #[test]
+    fn test_guardian_set_expired() {
+        let encoded_data = "AwAAABMAAABYzDrlwJeyE848gZeeG5+VcHRqpf9suVJYm96GLCXvQ5ITL7nUpCFXEU3oRgGTvfOi/PgfhqCXZfR2L9EQegCGsy16CXeSaiBRMdhzHTnL64yCsv2C+u0nEdWa8PJJnRbnJvayEbOXVsBCRBvm2GULabVOvnFeI0NUzltNNI+3S5WOiWbi7D29SVinzRXnyvB8Tj3I58Rp+SyM2I+4AFogdKO/kTlT1pUmDYi8GqJaTu42PvAACsAHZyezX76i2sKP7lzLD+p2jq9FztE2udniSQNGSuiJ9cinI/wU+TEkt8c4hDy7iehkyGLDjN3Mz5XSzDek3ANqjSMrSPYs3UcxQS9IkNp5j2iWozMfZLSMEtHVf9nL5wgRcaob4dNsr+OGeRD5nAnjR4mcGcOBkrbnOHzNdoJ3wX2rG3pQJ8CzzxeOIa0ud64GcRVJz7sfnHqdgJboXhSH81UV0CqSdTUEqNdUcbn0nttvvryJj0A+R3PpX+sV6Ayamcg0jUA8xWP46h9m";
+
+        let data = BASE64_STANDARD.decode(encoded_data).unwrap();
+
+        let guardian_set = GuardianSet::new(&data).unwrap();
+        assert_eq!(guardian_set.guardian_set_index(), 3);
+        assert_eq!(guardian_set.keys_len(), 19);
+        assert_eq!(guardian_set.creation_time(), 1_673_870_400);
+        assert_eq!(guardian_set.expiration_time(), 1_713_367_800);
+        assert!(guardian_set.is_active(1_713_367_800));
+        assert!(!guardian_set.is_active(1_713_367_801));
+
+        let expected_keys = [
+            [
+                0x58, 0xcc, 0x3a, 0xe5, 0xc0, 0x97, 0xb2, 0x13, 0xce, 0x3c, 0x81, 0x97, 0x9e, 0x1b,
+                0x9f, 0x95, 0x70, 0x74, 0x6a, 0xa5,
+            ],
+            [
+                0xff, 0x6c, 0xb9, 0x52, 0x58, 0x9b, 0xde, 0x86, 0x2c, 0x25, 0xef, 0x43, 0x92, 0x13,
+                0x2f, 0xb9, 0xd4, 0xa4, 0x21, 0x57,
+            ],
+            [
+                0x11, 0x4d, 0xe8, 0x46, 0x01, 0x93, 0xbd, 0xf3, 0xa2, 0xfc, 0xf8, 0x1f, 0x86, 0xa0,
+                0x97, 0x65, 0xf4, 0x76, 0x2f, 0xd1,
+            ],
+            [
+                0x10, 0x7a, 0x00, 0x86, 0xb3, 0x2d, 0x7a, 0x09, 0x77, 0x92, 0x6a, 0x20, 0x51, 0x31,
+                0xd8, 0x73, 0x1d, 0x39, 0xcb, 0xeb,
+            ],
+            [
+                0x8c, 0x82, 0xb2, 0xfd, 0x82, 0xfa, 0xed, 0x27, 0x11, 0xd5, 0x9a, 0xf0, 0xf2, 0x49,
+                0x9d, 0x16, 0xe7, 0x26, 0xf6, 0xb2,
+            ],
+            [
+                0x11, 0xb3, 0x97, 0x56, 0xc0, 0x42, 0x44, 0x1b, 0xe6, 0xd8, 0x65, 0x0b, 0x69, 0xb5,
+                0x4e, 0xbe, 0x71, 0x5e, 0x23, 0x43,
+            ],
+            [
+                0x54, 0xce, 0x5b, 0x4d, 0x34, 0x8f, 0xb7, 0x4b, 0x95, 0x8e, 0x89, 0x66, 0xe2, 0xec,
+                0x3d, 0xbd, 0x49, 0x58, 0xa7, 0xcd,
+            ],
+            [
+                0x15, 0xe7, 0xca, 0xf0, 0x7c, 0x4e, 0x3d, 0xc8, 0xe7, 0xc4, 0x69, 0xf9, 0x2c, 0x8c,
+                0xd8, 0x8f, 0xb8, 0x00, 0x5a, 0x20,
+            ],
+            [
+                0x74, 0xa3, 0xbf, 0x91, 0x39, 0x53, 0xd6, 0x95, 0x26, 0x0d, 0x88, 0xbc, 0x1a, 0xa2,
+                0x5a, 0x4e, 0xee, 0x36, 0x3e, 0xf0,
+            ],
+            [
+                0x00, 0x0a, 0xc0, 0x07, 0x67, 0x27, 0xb3, 0x5f, 0xbe, 0xa2, 0xda, 0xc2, 0x8f, 0xee,
+                0x5c, 0xcb, 0x0f, 0xea, 0x76, 0x8e,
+            ],
+            [
+                0xaf, 0x45, 0xce, 0xd1, 0x36, 0xb9, 0xd9, 0xe2, 0x49, 0x03, 0x46, 0x4a, 0xe8, 0x89,
+                0xf5, 0xc8, 0xa7, 0x23, 0xfc, 0x14,
+            ],
+            [
+                0xf9, 0x31, 0x24, 0xb7, 0xc7, 0x38, 0x84, 0x3c, 0xbb, 0x89, 0xe8, 0x64, 0xc8, 0x62,
+                0xc3, 0x8c, 0xdd, 0xcc, 0xcf, 0x95,
+            ],
+            [
+                0xd2, 0xcc, 0x37, 0xa4, 0xdc, 0x03, 0x6a, 0x8d, 0x23, 0x2b, 0x48, 0xf6, 0x2c, 0xdd,
+                0x47, 0x31, 0x41, 0x2f, 0x48, 0x90,
+            ],
+            [
+                0xda, 0x79, 0x8f, 0x68, 0x96, 0xa3, 0x33, 0x1f, 0x64, 0xb4, 0x8c, 0x12, 0xd1, 0xd5,
+                0x7f, 0xd9, 0xcb, 0xe7, 0x08, 0x11,
+            ],
+            [
+                0x71, 0xaa, 0x1b, 0xe1, 0xd3, 0x6c, 0xaf, 0xe3, 0x86, 0x79, 0x10, 0xf9, 0x9c, 0x09,
+                0xe3, 0x47, 0x89, 0x9c, 0x19, 0xc3,
+            ],
+            [
+                0x81, 0x92, 0xb6, 0xe7, 0x38, 0x7c, 0xcd, 0x76, 0x82, 0x77, 0xc1, 0x7d, 0xab, 0x1b,
+                0x7a, 0x50, 0x27, 0xc0, 0xb3, 0xcf,
+            ],
+            [
+                0x17, 0x8e, 0x21, 0xad, 0x2e, 0x77, 0xae, 0x06, 0x71, 0x15, 0x49, 0xcf, 0xbb, 0x1f,
+                0x9c, 0x7a, 0x9d, 0x80, 0x96, 0xe8,
+            ],
+            [
+                0x5e, 0x14, 0x87, 0xf3, 0x55, 0x15, 0xd0, 0x2a, 0x92, 0x75, 0x35, 0x04, 0xa8, 0xd7,
+                0x54, 0x71, 0xb9, 0xf4, 0x9e, 0xdb,
+            ],
+            [
+                0x6f, 0xbe, 0xbc, 0x89, 0x8f, 0x40, 0x3e, 0x47, 0x73, 0xe9, 0x5f, 0xeb, 0x15, 0xe8,
+                0x0c, 0x9a, 0x99, 0xc8, 0x34, 0x8d,
+            ],
+        ];
+
+        for (i, expected_key) in expected_keys.iter().enumerate() {
+            let key = guardian_set.key(i).unwrap();
+            assert_eq!(&key, expected_key, "key[{}]", i);
+        }
+    }
+}

+ 106 - 0
svm/wormhole-core-shims/crates/definitions/src/zero_copy/guardian_signatures.rs

@@ -0,0 +1,106 @@
+use solana_program::pubkey::Pubkey;
+
+use crate::{GUARDIAN_SIGNATURES_DISCRIMINATOR, GUARDIAN_SIGNATURE_LENGTH};
+
+/// Guardian signatures account owned by the Wormhole Verify VAA program.
+///
+/// NOTE: This is a zero-copy struct only meant to read in account data. There
+/// is no verification of whether this account is owned by the Wormhole Verify
+/// VAA program. You must ensure that the account is owned by this program.
+pub struct GuardianSignatures<'data>(&'data [u8]);
+
+impl<'data> GuardianSignatures<'data> {
+    pub const DISCRIMINATOR: [u8; 8] = GUARDIAN_SIGNATURES_DISCRIMINATOR;
+
+    pub const MINIMUM_SIZE: usize = {
+        8 // discriminator
+        + 32 // refund recipient
+        + 4 // guardian set index
+        + 4 // guardian signatures length
+    };
+
+    /// Attempts to read a guardian signatures account from the given data. This
+    /// method will return `None` if the data is not a valid guardian signatures
+    /// account.
+    pub fn new(data: &'data [u8]) -> Option<Self> {
+        if data.len() < Self::MINIMUM_SIZE || data[..8] != Self::DISCRIMINATOR {
+            return None;
+        }
+
+        let account = Self(data);
+        let total_len = (account.guardian_signatures_len() as usize)
+            .checked_mul(GUARDIAN_SIGNATURE_LENGTH)?
+            .checked_add(Self::MINIMUM_SIZE)?;
+
+        if data.len() < total_len {
+            return None;
+        }
+
+        Some(account)
+    }
+
+    /// Payer of this guardian signatures account. Only this account may extend
+    /// signatures. When the close signatures instruction is invoked, rent will
+    /// be returned to this account.
+    #[inline]
+    pub fn refund_recipient(&self) -> Pubkey {
+        Pubkey::new_from_array(self.refund_recipient_slice().try_into().unwrap())
+    }
+
+    /// Guardian set index that these signatures correspond to. This index will
+    /// be checked against the Wormhole Core Bridge program's guardian set when
+    /// the verify hash instruction is invoked.
+    #[inline]
+    pub fn guardian_set_index(&self) -> u32 {
+        u32::from_be_bytes(self.guardian_set_index_be())
+    }
+
+    /// Guardian set index that these signatures correspond to. This index will
+    /// be checked against the Wormhole Core Bridge program's guardian set when
+    /// the verify hash instruction is invoked.
+    ///
+    /// NOTE: Encoding the guardian set as big-endian matches the derivation
+    /// used by the Wormhole Core Bridge program.
+    #[inline]
+    pub fn guardian_set_index_be(&self) -> [u8; 4] {
+        self.guardian_index_be_slice().try_into().unwrap()
+    }
+
+    /// Number of guardian signatures in this account.
+    #[inline]
+    pub fn guardian_signatures_len(&self) -> u32 {
+        u32::from_le_bytes(self.guardian_signatures_len_slice().try_into().unwrap())
+    }
+
+    /// Guardian signature at the given index. This method will return `None` if
+    /// the index is out of bounds.
+    #[inline]
+    pub fn guardian_signature(&self, index: usize) -> Option<[u8; GUARDIAN_SIGNATURE_LENGTH]> {
+        let signature = self.guardian_signature_slice(index)?;
+        Some(signature.try_into().unwrap())
+    }
+
+    #[inline(always)]
+    pub fn refund_recipient_slice(&self) -> &'data [u8] {
+        &self.0[8..40]
+    }
+
+    #[inline(always)]
+    pub fn guardian_index_be_slice(&self) -> &'data [u8] {
+        &self.0[40..44]
+    }
+
+    #[inline(always)]
+    pub fn guardian_signatures_len_slice(&self) -> &'data [u8] {
+        &self.0[44..48]
+    }
+
+    #[inline(always)]
+    pub fn guardian_signature_slice(&self, index: usize) -> Option<&'data [u8]> {
+        let start = index
+            .checked_mul(GUARDIAN_SIGNATURE_LENGTH)?
+            .checked_add(Self::MINIMUM_SIZE)?;
+        let end = start.checked_add(GUARDIAN_SIGNATURE_LENGTH)?;
+        self.0.get(start..end)
+    }
+}

+ 5 - 0
svm/wormhole-core-shims/crates/definitions/src/zero_copy/mod.rs

@@ -0,0 +1,5 @@
+mod guardian_set;
+mod guardian_signatures;
+
+pub use guardian_set::*;
+pub use guardian_signatures::*;

+ 18 - 0
svm/wormhole-core-shims/crates/shim/Cargo.toml

@@ -0,0 +1,18 @@
+[package]
+name = "wormhole-svm-shim"
+description = "Wormhole SVM Shim Programs API"
+
+edition.workspace = true
+license.workspace = true
+repository.workspace = true
+rust-version.workspace = true
+version.workspace = true
+
+# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
+
+[features]
+default = []
+
+[dependencies]
+solana-program.workspace = true
+wormhole-svm-definitions.workspace = true

+ 4 - 0
svm/wormhole-core-shims/crates/shim/README.md

@@ -0,0 +1,4 @@
+# Wormhole SVM Shim
+
+Instruction builders for Wormhole Post Message and Wormhole Verify VAA Shim
+programs.

+ 5 - 0
svm/wormhole-core-shims/crates/shim/src/lib.rs

@@ -0,0 +1,5 @@
+#![deny(dead_code, unused_imports, unused_mut, unused_variables)]
+#![doc = include_str!("../README.md")]
+
+pub mod post_message;
+pub mod verify_vaa;

+ 281 - 0
svm/wormhole-core-shims/crates/shim/src/post_message.rs

@@ -0,0 +1,281 @@
+//! Wormhole Post Message Shim program instruction and related types.
+
+use solana_program::{
+    instruction::{AccountMeta, Instruction},
+    pubkey::Pubkey,
+};
+use wormhole_svm_definitions::{make_anchor_discriminator, EncodeFinality};
+
+#[derive(Debug, Clone, PartialEq, Eq)]
+pub enum PostMessageShimInstruction<'ix, F: EncodeFinality> {
+    PostMessage(PostMessageData<'ix, F>),
+}
+
+impl<'ix, F: EncodeFinality> PostMessageShimInstruction<'ix, F> {
+    pub const POST_MESSAGE_SELECTOR: [u8; 8] = make_anchor_discriminator(b"global:post_message");
+
+    #[inline]
+    pub fn to_vec(&self) -> Vec<u8> {
+        match self {
+            Self::PostMessage(data) => {
+                let payload_len = data.payload.len();
+
+                let mut out = Vec::with_capacity({
+                    8 // selector
+                    + 4 // nonce
+                    + 4 // payload length
+                    + payload_len // payload
+                    + 1 // finality
+                });
+                out.extend_from_slice(&Self::POST_MESSAGE_SELECTOR);
+                out.extend_from_slice(&data.nonce.to_le_bytes());
+                out.push(data.finality.encode());
+                out.extend_from_slice(&(payload_len as u32).to_le_bytes());
+                out.extend_from_slice(data.payload);
+
+                out
+            }
+        }
+    }
+
+    #[inline]
+    pub fn deserialize(data: &'ix [u8]) -> Option<Self> {
+        if data.len() < 8 {
+            return None;
+        }
+
+        match data[..8].try_into().unwrap() {
+            Self::POST_MESSAGE_SELECTOR => {
+                PostMessageData::deserialize(&data[8..]).map(Self::PostMessage)
+            }
+            _ => None,
+        }
+    }
+}
+
+#[derive(Debug, Clone, PartialEq, Eq)]
+pub struct PostMessageAccounts<'ix> {
+    /// Emitter of the Wormhole Core Bridge message. Wormhole Core Bridge
+    /// program's post message instruction requires this account to be a signer.
+    pub emitter: &'ix Pubkey,
+
+    /// Payer will pay the rent for the Wormhole Core Bridge emitter sequence
+    /// and message on the first post message call. Subsequent calls will not
+    /// require more lamports for rent.
+    pub payer: &'ix Pubkey,
+
+    /// Wormhole Core Bridge program.
+    pub wormhole_program_id: &'ix Pubkey,
+
+    /// Program-derived accounts required for the post message instruction.
+    ///
+    /// NOTE: If `None` is used for any of these accounts,
+    /// [PostMessage::instruction] will find these addresses. For on-chain
+    /// applications, it is recommended to provide these accounts to save on
+    /// CU costs (1,500 CU per bump iteration per account).
+    pub derived: PostMessageDerivedAccounts<'ix>,
+}
+
+#[derive(Debug, Default, Clone, PartialEq, Eq)]
+pub struct PostMessageDerivedAccounts<'ix> {
+    /// Wormhole Core Bridge config. The Wormhole Core Bridge program's post
+    /// message instruction requires this account to be mutable.
+    pub core_bridge_config: Option<&'ix Pubkey>,
+
+    /// Wormhole Message. The Wormhole Core Bridge program's post message
+    /// instruction requires this account to be a mutable signer.
+    ///
+    /// This program uses a PDA per emitter. Messages are already bottle-necked
+    /// by emitter sequence and the Wormhole Core Bridge program enforces that
+    /// emitter must be identical for reused accounts. While this could be
+    /// managed by the integrator, it seems more effective to have this Shim
+    /// program manage these accounts.
+    pub message: Option<&'ix Pubkey>,
+
+    /// Emitter's sequence account. Wormhole Core Bridge program's post message
+    /// instruction requires this account to be mutable.
+    pub sequence: Option<&'ix Pubkey>,
+
+    /// Wormhole Core Bridge fee collector. Wormhole Core Bridge program's post
+    /// message instruction requires this account to be mutable.
+    pub fee_collector: Option<&'ix Pubkey>,
+
+    /// Wormhole Post Message Shim program's self-CPI authority.
+    pub event_authority: Option<&'ix Pubkey>,
+}
+
+#[derive(Debug, Clone, Copy, PartialEq, Eq)]
+pub struct PostMessageData<'ix, F: EncodeFinality> {
+    /// Arbitrary message identifier specified by the message sender.
+    nonce: u32,
+
+    /// Finality of the message (which is when the Wormhole guardians will
+    /// attest to this message's observation).
+    finality: F,
+
+    /// Message payload.
+    payload: &'ix [u8],
+}
+
+impl<'ix, F: EncodeFinality> PostMessageData<'ix, F> {
+    pub const MINIMUM_SIZE: usize = {
+        4 // nonce
+        + 1 // finality
+        + 4 // payload length
+    };
+
+    #[inline]
+    pub fn nonce(&self) -> u32 {
+        self.nonce
+    }
+
+    #[inline]
+    pub fn finality(&self) -> F {
+        self.finality
+    }
+
+    #[inline]
+    pub fn payload(&self) -> &'ix [u8] {
+        self.payload
+    }
+
+    /// Construct new post message data. This method returns `None` if the
+    /// payload length exceeds the maximum allowed size of [u32::MAX].
+    pub fn new(nonce: u32, finality: F, payload: &'ix [u8]) -> Option<Self> {
+        if payload.len() > u32::MAX as usize {
+            None
+        } else {
+            Some(Self {
+                nonce,
+                finality,
+                payload,
+            })
+        }
+    }
+
+    #[inline(always)]
+    fn deserialize(data: &'ix [u8]) -> Option<Self> {
+        if data.len() < Self::MINIMUM_SIZE {
+            return None;
+        }
+
+        let nonce = u32::from_le_bytes(data[..4].try_into().unwrap());
+
+        // NOTE: There may be different finality requirements among SVM
+        // networks. This logic will have to change if that is the case.
+        let finality = EncodeFinality::decode(data[4])?;
+
+        let payload_len = u32::from_le_bytes(data[5..9].try_into().unwrap()) as usize;
+
+        // This operation is unlikely to overflow (beware 32-bit architectures).
+        // But it does not cost much to be paranoid.
+        let total_len = Self::MINIMUM_SIZE.checked_add(payload_len)?;
+
+        if data.len() < total_len {
+            return None;
+        }
+
+        let payload = &data[9..(9 + payload_len)];
+
+        // NOTE: We do not care about trailing bytes.
+
+        Some(Self {
+            nonce,
+            payload,
+            finality,
+        })
+    }
+}
+
+/// This instruction is intended to be a significantly cheaper alternative to
+/// the post message instruction on Wormhole Core Bridge program. It achieves
+/// this by reusing the message account (per emitter) via the post message
+/// unreliable instruction and emitting data via self-CPI (Anchor event)
+/// for the guardian to observe. This instruction data contains information
+/// previously found only in the resulting message account.
+///
+/// Because this instruction passes through the emitter and calls the post
+/// message unreliable instruction on the Wormhole Core Bridge, it can be used
+/// without disruption.
+///
+/// NOTE: In the initial message publication for a new emitter, this will
+/// require one additional CPI call depth when compared to using the Wormhole
+/// Core Bridge directly. If this initial call depth is an issue, emit an empty
+/// message on initialization (or migration) in order to instantiate the
+/// message account. Posting a message will result in a VAA from your emitter,
+/// so be careful to avoid any issues that may result from this first message.
+///
+/// Call depth of direct case:
+/// 1. post message (Wormhole Post Message Shim)
+/// 2. multiple CPI
+///     - post message unreliable (Wormhole Core Bridge)
+///     - Anchor event of `MesssageEvent` (Wormhole Post Message Shim)
+///
+/// Call depth of integrator case:
+/// 1. integrator instruction
+/// 2. CPI post message (Wormhole Post Message Shim)
+/// 3. multiple CPI
+///    - post message unreliable (Wormhole Core Bridge)
+///    - Anchor event of `MesssageEvent` (Wormhole Post Message Shim)
+#[derive(Debug, Clone, PartialEq, Eq)]
+pub struct PostMessage<'ix, F: EncodeFinality> {
+    pub program_id: &'ix Pubkey,
+    pub accounts: PostMessageAccounts<'ix>,
+    pub data: PostMessageData<'ix, F>,
+}
+
+impl<F: EncodeFinality> PostMessage<'_, F> {
+    /// Generate SVM instruction.
+    #[inline]
+    pub fn instruction(&self) -> Instruction {
+        let program_id = self.program_id;
+
+        let PostMessageAccounts {
+            emitter,
+            payer,
+            wormhole_program_id,
+            derived:
+                PostMessageDerivedAccounts {
+                    core_bridge_config,
+                    message,
+                    sequence,
+                    fee_collector,
+                    event_authority,
+                },
+        } = self.accounts;
+
+        let core_bridge_config = core_bridge_config.copied().unwrap_or_else(|| {
+            wormhole_svm_definitions::find_core_bridge_config_address(wormhole_program_id).0
+        });
+        let message = message.copied().unwrap_or_else(|| {
+            wormhole_svm_definitions::find_shim_message_address(emitter, program_id).0
+        });
+        let sequence = sequence.copied().unwrap_or_else(|| {
+            wormhole_svm_definitions::find_emitter_sequence_address(emitter, wormhole_program_id).0
+        });
+        let fee_collector = fee_collector.copied().unwrap_or_else(|| {
+            wormhole_svm_definitions::find_fee_collector_address(wormhole_program_id).0
+        });
+        let event_authority = event_authority.copied().unwrap_or_else(|| {
+            wormhole_svm_definitions::find_event_authority_address(program_id).0
+        });
+
+        Instruction {
+            program_id: *program_id,
+            accounts: vec![
+                AccountMeta::new(core_bridge_config, false),
+                AccountMeta::new(message, false),
+                AccountMeta::new_readonly(*emitter, true),
+                AccountMeta::new(sequence, false),
+                AccountMeta::new(*payer, true),
+                AccountMeta::new(fee_collector, false),
+                AccountMeta::new_readonly(solana_program::sysvar::clock::id(), false),
+                AccountMeta::new_readonly(solana_program::system_program::id(), false),
+                AccountMeta::new_readonly(*wormhole_program_id, false),
+                AccountMeta::new_readonly(event_authority, false),
+                AccountMeta::new_readonly(*program_id, false),
+            ],
+            data: PostMessageShimInstruction::PostMessage(self.data).to_vec(),
+        }
+    }
+}

+ 41 - 0
svm/wormhole-core-shims/crates/shim/src/verify_vaa/close_signatures.rs

@@ -0,0 +1,41 @@
+use solana_program::{
+    instruction::{AccountMeta, Instruction},
+    pubkey::Pubkey,
+};
+
+use super::VerifyVaaShimInstruction;
+
+/// Accounts for the close signatures instruction.
+#[derive(Debug, Clone, PartialEq, Eq)]
+pub struct CloseSignaturesAccounts<'ix> {
+    /// Guardian signatures account created using [PostSignatures]. This account
+    /// will be closed and the rent refunded to the original payer.
+    ///
+    /// [PostSignatures]: super::PostSignatures
+    pub guardian_signatures: &'ix Pubkey,
+
+    /// Original payer of the guardian signatures account. This account will
+    /// receive the rent refund when the guardian signatures account is closed.
+    pub refund_recipient: &'ix Pubkey,
+}
+
+/// Allows the initial payer to close the signature account, reclaiming the rent
+/// taken by the post signatures instruction.
+pub struct CloseSignatures<'ix> {
+    pub program_id: &'ix Pubkey,
+    pub accounts: CloseSignaturesAccounts<'ix>,
+}
+
+impl CloseSignatures<'_> {
+    /// Generate SVM instruction.
+    pub fn instruction(&self) -> Instruction {
+        Instruction {
+            program_id: *self.program_id,
+            accounts: vec![
+                AccountMeta::new(*self.accounts.guardian_signatures, false),
+                AccountMeta::new(*self.accounts.refund_recipient, true),
+            ],
+            data: VerifyVaaShimInstruction::CloseSignatures.to_vec(),
+        }
+    }
+}

+ 87 - 0
svm/wormhole-core-shims/crates/shim/src/verify_vaa/mod.rs

@@ -0,0 +1,87 @@
+//! Wormhole Verify VAA Shim program instructions and related types.
+
+mod close_signatures;
+mod post_signatures;
+mod verify_hash;
+
+pub use close_signatures::*;
+pub use post_signatures::*;
+pub use verify_hash::*;
+
+use wormhole_svm_definitions::{make_anchor_discriminator, Hash, GUARDIAN_SIGNATURE_LENGTH};
+
+/// Instructions for the Verify VAA Shim program.
+pub enum VerifyVaaShimInstruction<'ix, const CONTIGUOUS: bool> {
+    PostSignatures(PostSignaturesData<'ix, CONTIGUOUS>),
+    VerifyHash(VerifyHashData),
+    CloseSignatures,
+}
+
+impl<const CONTIGUOUS: bool> VerifyVaaShimInstruction<'_, CONTIGUOUS> {
+    pub const CLOSE_SIGNATURES_SELECTOR: [u8; 8] =
+        make_anchor_discriminator(b"global:close_signatures");
+    pub const POST_SIGNATURES_SELECTOR: [u8; 8] =
+        make_anchor_discriminator(b"global:post_signatures");
+    pub const VERIFY_HASH_SELECTOR: [u8; 8] = make_anchor_discriminator(b"global:verify_hash");
+}
+
+impl VerifyVaaShimInstruction<'_, false> {
+    /// Serialize the instruction data.
+    #[inline]
+    pub fn to_vec(&self) -> Vec<u8> {
+        match self {
+            Self::PostSignatures(data) => {
+                let guardian_signatures_len = data.guardian_signatures.len();
+                let encoded_signatures_len = guardian_signatures_len * GUARDIAN_SIGNATURE_LENGTH;
+
+                let mut out = Vec::with_capacity({
+                    8 // selector
+                    + PostSignaturesData::<false>::MINIMUM_SIZE
+                    + encoded_signatures_len // guardian_signatures
+                });
+                out.extend_from_slice(&Self::POST_SIGNATURES_SELECTOR);
+                out.extend_from_slice(&data.guardian_set_index.to_le_bytes());
+                out.push(data.total_signatures);
+                out.extend_from_slice(&(guardian_signatures_len as u32).to_le_bytes());
+                for signature in data.guardian_signatures.iter() {
+                    out.extend_from_slice(signature);
+                }
+
+                out
+            }
+            Self::VerifyHash(data) => {
+                let mut out = Vec::with_capacity({
+                    8 // selector
+                    + VerifyHashData::SIZE
+                });
+                out.extend_from_slice(&Self::VERIFY_HASH_SELECTOR);
+                out.push(data.guardian_set_bump);
+                out.extend_from_slice(&data.digest.0);
+
+                out
+            }
+            Self::CloseSignatures => Self::CLOSE_SIGNATURES_SELECTOR.to_vec(),
+        }
+    }
+}
+
+impl<'ix> VerifyVaaShimInstruction<'ix, true> {
+    /// Deserialize the instruction data.
+    #[inline]
+    pub fn deserialize(data: &'ix [u8]) -> Option<Self> {
+        if data.len() < 8 {
+            return None;
+        }
+
+        match data[..8].try_into().unwrap() {
+            Self::POST_SIGNATURES_SELECTOR => {
+                PostSignaturesData::deserialize(&data[8..]).map(Self::PostSignatures)
+            }
+            Self::VERIFY_HASH_SELECTOR => {
+                VerifyHashData::deserialize(&data[8..]).map(Self::VerifyHash)
+            }
+            Self::CLOSE_SIGNATURES_SELECTOR => Some(Self::CloseSignatures),
+            _ => None,
+        }
+    }
+}

+ 166 - 0
svm/wormhole-core-shims/crates/shim/src/verify_vaa/post_signatures.rs

@@ -0,0 +1,166 @@
+use solana_program::{
+    instruction::{AccountMeta, Instruction},
+    pubkey::Pubkey,
+};
+use wormhole_svm_definitions::GUARDIAN_SIGNATURE_LENGTH;
+
+use super::VerifyVaaShimInstruction;
+
+/// Accounts for the post signatures instruction.
+#[derive(Debug, Clone, PartialEq, Eq)]
+pub struct PostSignaturesAccounts<'ix> {
+    pub payer: &'ix Pubkey,
+
+    pub guardian_signatures: &'ix Pubkey,
+}
+
+/// Instruction data for the post signatures instruction.
+///
+/// Being contiguous is a feature that allows for the guardian signatures to be
+/// stored in a single slice, which is more efficient for the SVM runtime.
+/// When instruction data is initialized via [PostSignaturesData::new], the
+/// data is not guaranteed to be contiguous.
+#[derive(Debug, Clone, Copy, PartialEq, Eq)]
+pub struct PostSignaturesData<'ix, const CONTIGUOUS: bool> {
+    /// Argument to encode the guardian set index associated with the guardian
+    /// signatures.
+    pub(super) guardian_set_index: u32,
+
+    /// Total expected number of signatures, which determines the total size of
+    /// the guardian signatures account.
+    pub(super) total_signatures: u8,
+
+    /// Guardian signatures to load into the guardian signatures account at this
+    /// call.
+    pub(super) guardian_signatures: &'ix [[u8; GUARDIAN_SIGNATURE_LENGTH]],
+}
+
+impl<'ix, const CONTIGUOUS: bool> PostSignaturesData<'ix, CONTIGUOUS> {
+    pub const MINIMUM_SIZE: usize = {
+        4 // guardian_set_index
+        + 1 // total_signatures
+        + 4 // guardian_signatures length
+    };
+
+    #[inline]
+    pub fn guardian_set_index(&self) -> u32 {
+        self.guardian_set_index
+    }
+
+    #[inline]
+    pub fn total_signatures(&self) -> u8 {
+        self.total_signatures
+    }
+
+    #[inline]
+    pub fn guardian_signatures(&self) -> &'ix [[u8; GUARDIAN_SIGNATURE_LENGTH]] {
+        self.guardian_signatures
+    }
+}
+
+impl<'ix> PostSignaturesData<'ix, false> {
+    pub fn new(
+        guardian_set_index: u32,
+        total_signatures: u8,
+        guardian_signatures: &'ix [[u8; GUARDIAN_SIGNATURE_LENGTH]],
+    ) -> Self {
+        Self {
+            guardian_set_index,
+            total_signatures,
+            guardian_signatures,
+        }
+    }
+}
+
+impl<'ix> PostSignaturesData<'ix, true> {
+    #[inline(always)]
+    pub(super) fn deserialize(data: &'ix [u8]) -> Option<Self> {
+        if data.len() < Self::MINIMUM_SIZE {
+            return None;
+        }
+
+        let guardian_set_index = u32::from_le_bytes(data[..4].try_into().unwrap());
+        let total_signatures = data[4];
+        let guardian_signatures_len = u32::from_le_bytes(data[5..9].try_into().unwrap()) as usize;
+
+        let encoded_signatures_len =
+            guardian_signatures_len.checked_mul(GUARDIAN_SIGNATURE_LENGTH)?;
+        let total_len = Self::MINIMUM_SIZE.checked_add(encoded_signatures_len)?;
+
+        if data.len() < total_len {
+            return None;
+        }
+
+        let guardian_signatures = &data[9..total_len];
+
+        // Safety: Guardian signatures are contiguous and its length is a
+        // multiple of SIGNATURE_LENGTH.
+        let guardian_signatures = unsafe {
+            core::slice::from_raw_parts(
+                guardian_signatures.as_ptr() as *const [u8; GUARDIAN_SIGNATURE_LENGTH],
+                guardian_signatures_len,
+            )
+        };
+
+        // NOTE: We do not care about trailing bytes.
+
+        Some(Self {
+            guardian_set_index,
+            total_signatures,
+            guardian_signatures,
+        })
+    }
+
+    #[inline]
+    pub fn guardian_signatures_slice(&self) -> &'ix [u8] {
+        // Safety: Guardian signatures are contiguous and its length is a
+        // multiple of SIGNATURE_LENGTH.
+        unsafe {
+            core::slice::from_raw_parts(
+                self.guardian_signatures.as_ptr() as *const u8,
+                self.guardian_signatures.len() * GUARDIAN_SIGNATURE_LENGTH,
+            )
+        }
+    }
+}
+
+/// Creates or appends to a guardian signatures account for subsequent use by
+/// the verify hash instruction.
+///
+/// This instruction is necessary due to the Wormhole VAA body, which has an
+/// arbitrary size, and 13 guardian signatures (a quorum of the current 19
+/// mainnet guardians, 66 bytes each) alongside the required accounts is likely
+/// larger than the transaction size limit on Solana (1232 bytes).
+///
+/// This instruction will also allow for the verification of other messages
+/// which guardians sign, such as query results.
+///
+/// This instruction allows for the initial payer to append additional
+/// signatures to the account by calling the instruction again. Subsequent
+/// calls may be necessary if a quorum of signatures from the current guardian
+/// set grows larger than can fit into a single transaction.
+///
+/// The guardian signatures account can be closed by the initial payer via the
+/// close signatures instruction, which will refund this payer.
+#[derive(Debug, Clone, PartialEq, Eq)]
+pub struct PostSignatures<'ix> {
+    pub program_id: &'ix Pubkey,
+    pub accounts: PostSignaturesAccounts<'ix>,
+    pub data: PostSignaturesData<'ix, false>,
+}
+
+impl PostSignatures<'_> {
+    /// Generate SVM instruction.
+    #[inline]
+    pub fn instruction(&self) -> Instruction {
+        Instruction {
+            program_id: *self.program_id,
+            accounts: vec![
+                AccountMeta::new(*self.accounts.payer, true),
+                AccountMeta::new(*self.accounts.guardian_signatures, true),
+                AccountMeta::new_readonly(solana_program::system_program::ID, false),
+            ],
+            data: VerifyVaaShimInstruction::PostSignatures(self.data).to_vec(),
+        }
+    }
+}

+ 120 - 0
svm/wormhole-core-shims/crates/shim/src/verify_vaa/verify_hash.rs

@@ -0,0 +1,120 @@
+use solana_program::{
+    instruction::{AccountMeta, Instruction},
+    pubkey::Pubkey,
+};
+
+use super::{Hash, VerifyVaaShimInstruction};
+
+/// Accounts for the verify hash instruction.
+#[derive(Debug, Clone, PartialEq, Eq)]
+pub struct VerifyHashAccounts<'ix> {
+    /// Guardian set is used to verify the recovered public keys from the
+    /// signatures found in the guardian signatures account and
+    /// [VerifyHashData::digest].
+    pub guardian_set: &'ix Pubkey,
+
+    /// Guardian signatures account created using [PostSignatures].
+    ///
+    /// [PostSignatures]: super::PostSignatures
+    pub guardian_signatures: &'ix Pubkey,
+}
+
+/// Instruction data for the verify hash instruction.
+#[derive(Debug, Clone, Copy, PartialEq, Eq)]
+pub struct VerifyHashData {
+    pub(super) guardian_set_bump: u8,
+    pub(super) digest: Hash,
+}
+
+impl VerifyHashData {
+    pub const SIZE: usize = {
+        1 // guardian set bump
+        + 32 // digest
+    };
+
+    pub fn new(guardian_set_bump: u8, digest: Hash) -> Self {
+        Self {
+            guardian_set_bump,
+            digest,
+        }
+    }
+
+    #[inline]
+    pub fn guardian_set_bump(&self) -> u8 {
+        self.guardian_set_bump
+    }
+
+    #[inline]
+    pub fn digest(&self) -> Hash {
+        self.digest
+    }
+
+    #[inline(always)]
+    pub(super) fn deserialize(data: &[u8]) -> Option<Self> {
+        if data.len() < Self::SIZE {
+            return None;
+        }
+
+        Some(Self {
+            guardian_set_bump: data[0],
+            digest: Hash(data[1..33].try_into().unwrap()),
+        })
+    }
+}
+
+/// This instruction is intended to be invoked via CPI call. It verifies a
+/// digest against a guardian signatures account and a Wormhole Core Bridge
+/// guardian set account.
+///
+/// Prior to this call (and likely in a separate transaction), call the post
+/// signatures instruction to create the guardian signatures account.
+///
+/// Immediately after this verify call, call the close signatures instruction
+/// to reclaim the rent paid to create the guardian signatures account.
+///
+/// A v1 VAA digest can be computed as follows:
+/// ```rust
+/// use wormhole_svm_definitions::compute_keccak_digest;
+///
+/// // `vec_body` is the encoded body of the VAA.
+/// # let vaa_body = vec![];
+/// let digest = compute_keccak_digest(
+///     solana_program::keccak::hash(&vaa_body),
+///     None, // there is no prefix for V1 messages
+/// );
+/// ```
+///
+/// A QueryResponse digest can be computed as follows:
+/// ```rust
+/// # mod wormhole_query_sdk {
+/// #    pub const MESSAGE_PREFIX: &'static [u8] = b"ruh roh";
+/// # }
+/// use wormhole_query_sdk::MESSAGE_PREFIX;
+/// use wormhole_svm_definitions::compute_keccak_digest;
+///
+/// # let query_response_bytes = vec![];
+/// let digest = compute_keccak_digest(
+///     solana_program::keccak::hash(&query_response_bytes),
+///     Some(MESSAGE_PREFIX)
+/// );
+/// ```
+pub struct VerifyHash<'ix> {
+    pub program_id: &'ix Pubkey,
+    pub accounts: VerifyHashAccounts<'ix>,
+    pub data: VerifyHashData,
+}
+
+impl VerifyHash<'_> {
+    /// Generate SVM instruction.
+    #[inline]
+    pub fn instruction(&self) -> Instruction {
+        Instruction {
+            program_id: *self.program_id,
+            accounts: vec![
+                AccountMeta::new_readonly(*self.accounts.guardian_set, false),
+                AccountMeta::new_readonly(*self.accounts.guardian_signatures, false),
+            ],
+            data: VerifyVaaShimInstruction::VerifyHash(self.data).to_vec(),
+        }
+    }
+}

+ 33 - 0
svm/wormhole-core-shims/programs/post-message/Cargo.toml

@@ -0,0 +1,33 @@
+[package]
+name = "wormhole-post-message-shim"
+description = "Created with Anchor"
+publish = false
+
+edition.workspace = true
+license.workspace = true
+repository.workspace = true
+rust-version.workspace = true
+version.workspace = true
+
+[lib]
+crate-type = ["cdylib", "lib"]
+
+[features]
+default = []
+no-entrypoint = []
+
+### Network types
+testnet = ["wormhole-svm-definitions/testnet"]
+localnet = ["wormhole-svm-definitions/localnet"]
+
+### Specific networks
+solana = ["wormhole-svm-definitions/solana"]
+
+[dependencies]
+solana-program.workspace = true
+wormhole-svm-definitions.workspace = true
+wormhole-svm-shim.workspace = true
+
+[dev-dependencies]
+solana-program-test.workspace = true
+solana-sdk.workspace = true

+ 274 - 0
svm/wormhole-core-shims/programs/post-message/README.md

@@ -0,0 +1,274 @@
+# Post Message Shim
+
+## Objective
+
+This program aims to reduce the cost of Wormhole Core Bridge program message
+emission on SVM networks. It provides a new emission mechanism without making
+changes to the existing Wormhole Core Bridge program.
+
+## Background
+
+The existing [post message instructions] has the following limitations:
+
+- The post message instruction creates a new message account for every call.
+  These accounts cannot be reclaimed and take up state forever, even after
+  the VAA has been produced and consumed. The post message unreliable
+  instruction allows reusing a message account, but according to the comment in
+  the Wormhole Core Bridge program implementation, there is “no way” to recover
+  the message after it has been overwritten. It also requires that the payload
+  size matches the existing message account since the feature was written before
+  account resizing.
+- The post message instructions take mutable references to the bridge config
+  and fee_collector accounts. These writable accounts limit the ability to
+  parallelize calls across multiple integrators. The same could be said of the
+  emitter sequence account for a single integrator, but at least that is part of
+  the specification for the Wormhole Core Bridge.
+
+The emitter address on SVM networks is a signer for the post message
+instructions. Typically this address is a PDA of the integrating program.
+
+Though it is not guaranteed to be a unique identifier, many components of the
+Wormhole ecosystem use `chain/emitter/sequence` as an id. Furthermore,
+the [message digest] should be unique between two different messages. As such,
+the digest is recommended for replay protection. Therefore, it is important that
+a new emission mechanism not reuse sequence numbers for the same emitter.
+
+## Goals
+
+- Provide a new mechanism for publishing messages which does not require a new
+  account for every message in order to be “reliable”.
+- Provide a migration recommendation to avoid emitter sequence number reuse.
+  Ideally, migration should be as easy as possible for integrators.
+- Detail how this could be integrated into Modular Messaging.
+- Handle a message fee. If this is not required, this design could be revised to
+  further reduce costs and complexity as well as increase parallelization
+  potential.
+
+## Non-Goals
+
+- Prevent emissions on the existing Wormhole Core Bridge program. This design
+  must treat the existing core bridge as non-upgradeable.
+- Message verification. Integrators can use separate contracts or methods for
+  emission and verification.
+
+## Overview
+
+1. A new SVM program which takes the same arguments as the post message
+  unreliable instruction while providing the additional information necessary
+  for the guardian to make them reliable. It will act as a shim, augmenting the
+  existing Wormhole Core Bridge program.
+2. A modification to the guardian watcher which detects calls that are made via
+  this new program and observes the message solely via the instruction data
+  (using Anchor event CPI).
+3. New Guardian recommendations for Solana RPC instruction data availability.
+
+## Detailed Design
+
+### Solana Program Technical Details
+
+For ease of integration, this program will follow the [Anchor] program
+interface, which exposes one CPI instruction (post message), which:
+
+- MUST take at least the same accounts and instruction data as the Wormhole Core
+  Bridge program's post message unreliable instruction.
+- MUST read the sequence number from the sequence account and emit it in an
+  [Anchor CPI event], along with `Clock::unix_timestamp as u32` for convenience.
+- MUST call the Wormhole Core Bridge program's post message unreliable
+  instruction, replacing message's payload with an empty vector. This way,
+  payloads of different sizes can be supported and the most limited data can be
+  written to the account. This also keeps tracking the sequence number against
+  the emitter and ensures the appropriate fee is paid.
+
+With this mechanism, the guardian will have all of the information it needs to
+re-build the message from only the instruction data.
+
+1. Shim program’s post message instruction:
+  - `nonce`
+  - `payload`
+  - `consistency_level`
+2. Shim program’s CPI event
+  - `emitter`
+  - `timestamp`
+  - `sequence`
+
+### Guardian Changes
+
+The guardian code requires two modifications to make this successful:
+
+1. Look for the Wormhole Post Message Shim program's post message instruction
+   and read the corresponding message data from the instruction instead of the
+   message account. Ignore the corresponding unreliable message account from the
+   Wormhole Core Bridge (or see below). The guardian also MUST NOT re-observe
+   unreliable messages from these accounts as this could lead to a duplicate VAA
+   for the same sequence number, but with an empty payload (since that is what
+   is written to the message account).
+2. Support re-observations by transaction ID. See
+   https://github.com/wormhole-foundation/wormhole/pull/4101.
+
+Optionally, ignore unreliable messages with a zero length payload (which MAY be
+paired with a particular hard-coded nonce passed by the shim). Ignoring
+zero-length unreliable messages avoids any mechanism of accidentally picking up
+these messages via the account and avoids some added code complexity in skipping
+the corresponding instruction.
+
+### Guardian RPC Changes
+
+The Guardian’s SVM RPCs will likely need to support fetching transaction
+instruction information farther back in time than they currently support (about
+30 minutes or so from initial testing). For example,
+https://docs.anza.xyz/implemented-proposals/rpc-transaction-history/
+
+For reference, guardians must already set [the following flags].
+
+The Wormhole Foundation should work with Guardians and the community to
+establish an acceptable and recommended SLA.
+
+> 🚧 TODO: investigate the flags / process necessary to store the required data
+for a longer period of time. Perhaps it is
+`--enable-extended-tx-metadata-storage` or `--limit-ledger-size`?
+
+</aside>
+
+### Protocol Integration
+
+The monitor will need to change to support reobservation via transaction for
+these messages.
+
+Given that the existing SVM RPC configuration for most of the guardians is
+expected to be limited to approximately 30 minutes of transaction history, it
+would be prudent to have a separate re-observation pipeline for SVM emissions
+utilizing the shim.
+
+Wormholescan may want to indicate the difference in emission somehow.
+
+Modular Messaging - an adapter can use this program for message emission and
+its own mechanism (via loading the Wormhole Core Bridge program accounts) for
+verification.
+
+### API and Database Schema
+
+N/A
+
+## Caveats
+
+This approach would not increase the potential parallelization of Wormhole Core
+Bridge program message emission because it still relies on the shared writable
+fee collector account.
+
+For Solana specifically, it is critically important to note that this network
+has a [maximum call depth of 4] and the very first call with a given emitter
+will result in 1 additional CPI call stack depth than the equivalent
+Wormhole Core Bridge program's post message instruction call. However,
+subsequent calls will be equivalent. **For this reason, if an integrating program
+is _only_ called via CPI from another program, it is recommended that they emit
+an empty message upon initialization / migration so that they will not encounter
+an edge case here.**
+
+```jsx
+// Direct core integration, `post_message`
+Integrator Program -> Core Bridge -> System Program (init account)
+// Initial shim call, per emitter (+1)
+Integrator Program -> Shim -> Core Bridge -> System Program (init account)
+                           -> Shim
+// Subsequent shim calls, per emitter (+0)
+Integrator Program -> Shim -> Core Bridge
+                           -> Shim
+```
+
+# Alternatives Considered
+
+- Make account reuse for the Wormhole Core Bridge program's post message
+unreliable instruction reliable via guardian changes. Unfortunately, the only
+artifact of the sequence number at the time of the emission is the log message,
+which can be truncated. And the content of the message account can be
+overridden. There is currently a [guardian optimization] which relies on the log
+messages, but they should not be relied upon for critical information.
+- Avoid calling the existing Wormhole Core Bridge program and track sequence
+numbers independently. This path would be more reasonable if the message fee was
+not required, but would require special handling for migrations to avoid
+potentially duplicate message digests. Some possible solutions include:
+  - A new, unique emitter could be derived by this contract, which would avoid
+    conflicts but would make it more difficult to migrate. Existing protocols
+    would need to update their emitter and potential support two emitters
+    simultaneously during the migration.
+  - This contract could reuse the emitter but use new enum values for the
+    consistency levels. Like other alternatives (such as setting the first bit
+    of the sequence number to 1), this solution is not very intuitive, scalable
+    or repeatable.
+  - Document this issue and warn integrators against re-using an emitter.
+
+## Security Considerations
+
+As mentioned at various points in this document, it is critically important
+that a given message MUST NEVER result in two different VAAs. Incidents which
+may result in duplicate message emission for unique messages (such as reusing
+sequence numbers) should also be avoided, as it could lead to an unredeemable
+VAA for an integrator (even if they were using the recommended message digest
+approach).
+
+After initial testing is complete and proper configuration and functionality of
+the Wormhole Post Message Shim program is confirmed, the upgrade authority
+should be discarded in order to improve trust assumptions and avoid possible
+compromise of the upgrade authority key.
+
+## Test Plan
+
+The Wormhole Post Message Shim program may be developed in a standalone repo and
+can leverage Tilt akin to NTT. That way it can test emission e2e with a running
+guardian. Alternatively, if desired, it could exist in a new directory in the
+monorepo (the advantage of which would be to ensure that co-developed guardian
+code continues to work).
+
+Gas cost analysis should be performed for initial and subsequent message
+emissions with and without the shim and of various lengths.
+
+## Performance Impact
+
+This should greatly reduce the cost of emitting messages on Solana for
+integrators who are willing to accept the risk of not having a permanent account
+for each message, as it will save the current rent exemption cost per message.
+
+It does, however, increase the compute units required by introducing another
+program and two instructions (one via event CPI) into the mix. Some of this may
+be slightly reduced by meticulously optimizing the Wormhole Post Message Shim
+program instruction. Initial testing results below compare the existing
+Wormhole Core Bridge program's post message instruction call with one from this
+Shim program, with and without extra Anchor address enforcement that is not
+necessary but helps IDL generation.
+
+NOTE: The compute unit calculations below are dependent on the number of bump
+iterations using a specific emitter key. Because the Wormhole Post Message Shim
+program's message PDA and Wormhole Core Bridge program's emitter sequence
+accounts are derived using the emitter key, CU usage can vary. This CU report
+can may also change in the future with any optimizations introduced to the
+program.
+
+```jsx
+core: lamports 1638720 SOL 0.00163872, $0.3564871488, CU 26827
+shim: lamports 5120    SOL 0.00000512, $0.0011138048, CU 23057
+```
+
+## Rollout
+
+This will require a guardian release and deployment of the Wormhole Post Message
+Shim program. If desired, upgrade authority of the shim can be maintained for a
+limited time while testing.
+
+### Acceptance Criteria
+
+Test that emitting a message via Wormhole Core Bridge program's post message
+instructions along with the Wormhole Post Message Shim program's post message
+instruction all result in the appropriate VAAs.
+
+### Rollback
+
+Disable the guardian detection of shim instructions. Enabling could be gated
+behind a feature flag if desired.
+
+[Anchor]: https://www.anchor-lang.com/
+[Anchor CPI event]: https://www.anchor-lang.com/docs/features/events#emit_cpi
+[guardian optimization]: https://github.com/wormhole-foundation/wormhole/blob/1dbe8459b96e182932d0dd5ae4b6bbce6f48cb09/node/pkg/watchers/solana/client.go#L563
+[maximum call depth of 4]: https://solana.com/docs/core/cpi#key-points
+[message digest]: https://github.com/wormhole-foundation/wormhole/blob/main/whitepapers/0001_generic_message_passing.md#detailed-design
+[post message instructions]: https://github.com/wormhole-foundation/wormhole/blob/main/solana/bridge/program/src/api/post_message.rs
+[the following flags]: https://github.com/wormhole-foundation/wormhole/blob/main/docs/operations.md#solana-node-requirements

+ 272 - 0
svm/wormhole-core-shims/programs/post-message/src/lib.rs

@@ -0,0 +1,272 @@
+#![deny(dead_code, unused_imports, unused_mut, unused_variables)]
+
+use solana_program::{
+    account_info::AccountInfo,
+    entrypoint::ProgramResult,
+    instruction::{AccountMeta, Instruction},
+    msg,
+    program_error::ProgramError,
+    pubkey::Pubkey,
+};
+use wormhole_svm_definitions::{
+    find_shim_message_address, Finality, ANCHOR_EVENT_CPI_SELECTOR, CORE_BRIDGE_PROGRAM_ID,
+    EVENT_AUTHORITY_SEED, MESSAGE_EVENT_DISCRIMINATOR, POST_MESSAGE_SHIM_EVENT_AUTHORITY,
+    POST_MESSAGE_SHIM_EVENT_AUTHORITY_BUMP, POST_MESSAGE_SHIM_PROGRAM_ID as ID,
+};
+use wormhole_svm_shim::post_message::PostMessageShimInstruction;
+
+solana_program::entrypoint_no_alloc!(process_instruction);
+
+#[inline]
+fn process_instruction(
+    program_id: &Pubkey,
+    accounts: &[AccountInfo],
+    instruction_data: &[u8],
+) -> ProgramResult {
+    // Verify the program ID is what we expect.
+    if program_id != &ID {
+        return Err(ProgramError::IncorrectProgramId);
+    }
+
+    // Determine whether instruction data is post message.
+    match PostMessageShimInstruction::<Finality>::deserialize(instruction_data) {
+        Some(PostMessageShimInstruction::PostMessage(_)) => process_post_message(accounts),
+        None => process_privileged_invoke(accounts),
+    }
+}
+
+#[inline(always)]
+fn process_post_message(accounts: &[AccountInfo]) -> ProgramResult {
+    // This instruction requires 11 accounts. If there are more remaining, we
+    // won't do anything with them. We perform this check upfront so we can
+    // index into the accounts slice.
+    if accounts.len() < 11 {
+        return Err(ProgramError::NotEnoughAccountKeys);
+    }
+
+    // Verify accounts.
+
+    // Wormhole Core Bridge config. Wormhole Core Bridge program's post message
+    // instruction requires this account be mutable.
+    let bridge_info = &accounts[0];
+
+    // Wormhole Message. Wormhole Core Bridge program's post message
+    // instruction requires this account to be a mutable signer.
+    let message_info = &accounts[1];
+
+    // Emitter of the Wormhole Core Bridge message. Wormhole Core Bridge
+    // program's post message instruction requires this account to be a signer.
+    let emitter_info = &accounts[2];
+    let emitter_key = emitter_info.key;
+
+    // The Wormhole Post Message Shim program uses a PDA per emitter because
+    // these messages are already bottle-necked by sequence and the Wormhole
+    // Core Bridge program enforces that the emitter must be identical for
+    // reused accounts.
+    //
+    // While this could be managed by the integrator, it seems more effective
+    // to have the Wormhole Post Message Shim program manage these accounts.
+    let (expected_message_key, message_bump) = find_shim_message_address(emitter_key, &ID);
+    if message_info.key != &expected_message_key {
+        msg!("Message (account #2) seeds constraint violated");
+        msg!("Actual:");
+        msg!("{}", message_info.key);
+        msg!("Expected:");
+        msg!("{}", expected_message_key);
+        return Err(ProgramError::InvalidSeeds);
+    }
+
+    // Emitter's sequence account. Wormhole Core Bridge program's post message
+    // instruction requires this account to be mutable.
+    let sequence_info = &accounts[3];
+
+    // Payer will pay the rent for the Wormhole Core Bridge emitter sequence
+    // and message on the first post message call. Subsequent calls will not
+    // require more lamports for rent.
+    let payer_info = &accounts[4];
+
+    // Wormhole Core Bridge fee collector. Wormhole Core Bridge program's post
+    // message instruction requires this account to be mutable.
+    let fee_collector_info = &accounts[5];
+
+    // Clock sysvar, which will be used after the message is posted.
+    let clock_info = &accounts[6];
+
+    // System program. Wormhole Core Bridge program's post message instruction
+    // requires the System program to create the message account and emitter
+    // sequence account if they have not been created yet.
+    let system_program_info = &accounts[7];
+
+    // Wormhole Core Bridge program.
+    let wormhole_program_info = &accounts[8];
+
+    // We only want to use the Wormhole Core Bridge program to post messages.
+    if wormhole_program_info.key != &CORE_BRIDGE_PROGRAM_ID {
+        msg!("Wormhole program (account #9) address constraint violated");
+        return Err(ProgramError::InvalidAccountData);
+    }
+
+    // Wormhole Post Message Shim program's self-CPI authority. The emit message
+    // instruction will verify that this account is a signer and the pubkey is
+    // correct.
+    let event_authority_info = &accounts[9];
+
+    // NOTE: We are not checking account at index == 10 because the self CPI
+    // will fail if this program executable is not present.
+
+    // Perform two CPIs:
+    // 1. Post the message.
+    // 2. Emit the event (with self CPI)
+    //
+    // The max length of instruction data is 60 bytes between the two
+    // instructions, so we will reuse the same allocated memory for both.
+    const MAX_CPI_DATA_LEN: usize = 60;
+    const POST_MESSAGE_UNRELIABLE_DATA_LEN: usize = 10;
+
+    // The post message unreliable instruction needs more accounts than the self
+    // CPI call (which only needs one). We will initialize the instruction with
+    // the Wormhole Core Bridge program's ID and the post message accounts.
+    let mut cpi_ix = Instruction {
+        program_id: *wormhole_program_info.key,
+        accounts: vec![
+            AccountMeta::new(*bridge_info.key, false),
+            AccountMeta::new(expected_message_key, true),
+            AccountMeta::new_readonly(*emitter_key, true),
+            AccountMeta::new(*sequence_info.key, false),
+            AccountMeta::new(*payer_info.key, true),
+            AccountMeta::new(*fee_collector_info.key, false),
+            AccountMeta::new_readonly(*clock_info.key, false),
+            AccountMeta::new_readonly(*system_program_info.key, false),
+        ],
+        data: Vec::with_capacity(MAX_CPI_DATA_LEN),
+    };
+
+    let emitter_key_bytes = emitter_key.to_bytes();
+
+    // First post message.
+    {
+        let cpi_data = &mut cpi_ix.data;
+
+        // Safety: Because the capacity is > 10, it is safe to write to the
+        // first 10 elements and set the vector's length to 10.
+        //
+        // Encoding the post message instruction data is:
+        // 1. 1 byte for the selector.
+        // 2. 4 bytes for the nonce.
+        // 3. 4 bytes for the payload length (zero in this case).
+        // 4. 1 byte for the consistency level (finality).
+        //
+        // Nonce and consistency level will not be read from the Wormhole Core
+        // Bridge message account (only from this program's instruction data),
+        // so we might as well set them to zero.
+        unsafe {
+            core::ptr::write_bytes(cpi_data.as_mut_ptr(), 0, POST_MESSAGE_UNRELIABLE_DATA_LEN);
+            cpi_data.set_len(POST_MESSAGE_UNRELIABLE_DATA_LEN);
+        }
+
+        // We only need to encode the selector for post message unreliable.
+        cpi_data[0] = 8;
+
+        // There is no account data being borrowed at this point that the CPI'ed
+        // program will be using, so this is safe.
+        solana_program::program::invoke_signed_unchecked(
+            &cpi_ix,
+            accounts,
+            &[&[&emitter_key_bytes, &[message_bump]]],
+        )?;
+    }
+
+    // Finally perform self CPI.
+    {
+        // Parse the sequence from the account and emit the event reading the
+        // account after avoids having to handle when the account doesn't exist.
+        let sequence_info_data = sequence_info.data.borrow();
+        let sequence = u64::from_le_bytes(sequence_info_data[..8].try_into().unwrap()) - 1;
+
+        // NOTE: If the Wormhole Core Bridge ever changes to use Clock::get(),
+        // this clock account info may not exist anymore (so we will have to
+        // perform Clock::get() here instead).
+        let clock_info_data = clock_info.data.borrow();
+        let encoded_unix_timestamp = &clock_info_data[32..40];
+
+        // Casting to u32 matches the Wormhole Core Bridge's timestamp. This
+        // operation will panic quite far in the future.
+        //
+        // Clock encodes unix timestamp as little-endian i64, so the first 4
+        // bytes will be the timestamp encoded as u32. We expect the last 4
+        // bytes to be zero.
+        assert!(encoded_unix_timestamp[4..] == [0, 0, 0, 0]);
+
+        let cpi_accounts = &mut cpi_ix.accounts;
+
+        // Safety: Setting the length reduces the previous length from the last
+        // CPI call.
+        unsafe {
+            cpi_accounts.set_len(1);
+        }
+        cpi_accounts[0] = AccountMeta::new_readonly(*event_authority_info.key, true);
+
+        // "Emit" the MessageEvent. Its schema is:
+        //
+        // struct MessageEvent {
+        //     emitter: Pubkey,
+        //     sequence: u64,
+        //     submission_time: u32,
+        // }
+        //
+        // The invoke below emulates the Anchor event CPI pattern.
+        let cpi_data = &mut cpi_ix.data;
+
+        // Safety: The capacity of this vector is 60. This data will be
+        // overwritten for the next CPI call.
+        unsafe {
+            core::ptr::write_bytes(
+                cpi_data
+                    .as_mut_ptr()
+                    .offset(POST_MESSAGE_UNRELIABLE_DATA_LEN as isize),
+                0,
+                MAX_CPI_DATA_LEN - POST_MESSAGE_UNRELIABLE_DATA_LEN,
+            );
+            cpi_data.set_len(MAX_CPI_DATA_LEN);
+        }
+        cpi_data[..8].copy_from_slice(&ANCHOR_EVENT_CPI_SELECTOR);
+        cpi_data[8..16].copy_from_slice(&MESSAGE_EVENT_DISCRIMINATOR);
+        cpi_data[16..48].copy_from_slice(&emitter_key_bytes);
+        cpi_data[48..56].copy_from_slice(&sequence.to_le_bytes());
+        cpi_data[56..60].copy_from_slice(&encoded_unix_timestamp[..4]);
+
+        cpi_ix.program_id = ID;
+
+        // There is no account data being borrowed at this point that the CPI'ed
+        // program will be using, so this is safe.
+        solana_program::program::invoke_signed_unchecked(
+            &cpi_ix,
+            accounts,
+            &[&[
+                EVENT_AUTHORITY_SEED,
+                &[POST_MESSAGE_SHIM_EVENT_AUTHORITY_BUMP],
+            ]],
+        )?;
+    }
+
+    Ok(())
+}
+
+// Event CPI.
+
+#[inline(always)]
+fn process_privileged_invoke(accounts: &[AccountInfo]) -> ProgramResult {
+    // We want to ensure that this program's authority (which is the Anchor
+    // event CPI PDA) is the one invoking this instruction. If anything else
+    // invokes this instruction, we will pretend that the instruction does not
+    // exist by reverting with `InvalidInstructionData`.
+
+    if accounts.is_empty()
+        || !accounts[0].is_signer
+        || accounts[0].key != &POST_MESSAGE_SHIM_EVENT_AUTHORITY
+    {
+        return Err(ProgramError::InvalidInstructionData);
+    }
+
+    Ok(())
+}

+ 303 - 0
svm/wormhole-core-shims/programs/post-message/tests/integration_test.rs

@@ -0,0 +1,303 @@
+use solana_program_test::{tokio, BanksClient, ProgramTest};
+use solana_sdk::{
+    compute_budget::ComputeBudgetInstruction,
+    hash::Hash,
+    message::{v0::Message, VersionedMessage},
+    pubkey::Pubkey,
+    signature::Keypair,
+    signer::Signer,
+    transaction::VersionedTransaction,
+};
+use wormhole_svm_definitions::{
+    CORE_BRIDGE_CONFIG, CORE_BRIDGE_FEE_COLLECTOR, CORE_BRIDGE_PROGRAM_ID,
+    POST_MESSAGE_SHIM_PROGRAM_ID,
+};
+use wormhole_svm_shim::post_message;
+
+#[tokio::test]
+async fn test_post_message_no_emitter_sequence() {
+    let (banks_client, payer_signer, recent_blockhash) = start_test().await;
+
+    let emitter_signer = Keypair::new();
+    let (transaction, bump_costs) = set_up_post_message_transaction(
+        b"All your base are belong to us",
+        &payer_signer,
+        &emitter_signer,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_ok());
+
+    let details = out.simulation_details.unwrap();
+    let logs = details.logs;
+
+    let is_core_bridge_cpi_log = |line: &String| {
+        line.contains(format!("Program {} invoke [2]", CORE_BRIDGE_PROGRAM_ID).as_str())
+    };
+
+    // CPI to Core Bridge.
+    assert_eq!(
+        logs.iter()
+            .filter(|line| {
+                line.contains(format!("Program {} invoke [2]", CORE_BRIDGE_PROGRAM_ID).as_str())
+            })
+            .count(),
+        1
+    );
+    assert_eq!(
+        logs.iter()
+            .filter(|line| { line.contains("Program log: Sequence: 0") })
+            .count(),
+        1
+    );
+
+    let core_bridge_log_index = logs.iter().position(is_core_bridge_cpi_log).unwrap();
+
+    // Self CPI.
+    assert_eq!(
+        logs.iter()
+            .skip(core_bridge_log_index)
+            .filter(|line| {
+                line.contains(
+                    format!("Program {} invoke [2]", POST_MESSAGE_SHIM_PROGRAM_ID).as_str(),
+                )
+            })
+            .count(),
+        1
+    );
+
+    // Wormhole Core Bridge re-derives the sequence account when it needs to be
+    // created (cool). So we need to subtract the sequence bump cost twice for
+    // the first message.
+    assert_eq!(
+        details.units_consumed - bump_costs.message - 2 * bump_costs.sequence,
+        // 53_418
+        46_146
+    );
+}
+
+#[tokio::test]
+async fn test_cannot_post_message_invalid_message() {
+    let (banks_client, payer_signer, recent_blockhash) = start_test().await;
+
+    let emitter_signer = Keypair::new();
+    let (transaction, _) = set_up_post_message_transaction(
+        b"All your base are belong to us",
+        &payer_signer,
+        &emitter_signer,
+        recent_blockhash,
+        Some(AdditionalTestInputs {
+            invalid_message: Some(Pubkey::new_unique()),
+            ..Default::default()
+        }),
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    let details = out.simulation_details.unwrap();
+    dbg!(&details);
+    assert!(out.result.unwrap().is_err());
+    //assert!(details.logs.contains(&"Program log: AnchorError caused by account: message. Error Code: ConstraintSeeds. Error Number: 2006. Error Message: A seeds constraint was violated.".to_string()));
+    assert!(details
+        .logs
+        .contains(&"Program log: Message (account #2) seeds constraint violated".to_string()));
+}
+
+#[tokio::test]
+async fn test_cannot_post_message_invalid_core_bridge_program() {
+    let (banks_client, payer_signer, recent_blockhash) = start_test().await;
+
+    let emitter_signer = Keypair::new();
+    let (transaction, _) = set_up_post_message_transaction(
+        b"All your base are belong to us",
+        &payer_signer,
+        &emitter_signer,
+        recent_blockhash,
+        Some(AdditionalTestInputs {
+            invalid_core_bridge_program: Some(Pubkey::new_unique()),
+            ..Default::default()
+        }),
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    let details = out.simulation_details.unwrap();
+    dbg!(&details);
+    assert!(out.result.unwrap().is_err());
+    //assert!(details.logs.contains(&"Program log: AnchorError caused by account: wormhole_program. Error Code: ConstraintAddress. Error Number: 2012. Error Message: An address constraint was violated.".to_string()));
+    assert!(details.logs.contains(
+        &"Program log: Wormhole program (account #9) address constraint violated".to_string()
+    ));
+}
+
+#[tokio::test]
+async fn test_post_message() {
+    let (banks_client, payer_signer, recent_blockhash) = start_test().await;
+
+    let first_message = b"All your base";
+    let emitter_signer = Keypair::new();
+    let (transaction, _) = set_up_post_message_transaction(
+        first_message,
+        &payer_signer,
+        &emitter_signer,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    // Send one to create the emitter sequence account.
+    banks_client.process_transaction(transaction).await.unwrap();
+
+    let subsequent_message = b"are belong to us";
+    assert_ne!(&first_message[..], &subsequent_message[..]);
+
+    let recent_blockhash = banks_client.get_latest_blockhash().await.unwrap();
+    let (transaction, bump_costs) = set_up_post_message_transaction(
+        subsequent_message,
+        &payer_signer,
+        &emitter_signer,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_ok());
+    assert_eq!(
+        out.simulation_details.unwrap().units_consumed - bump_costs.message - bump_costs.sequence,
+        // 30_901
+        23_627
+    );
+}
+
+async fn start_test() -> (BanksClient, Keypair, Hash) {
+    let mut program_test = ProgramTest::new(
+        "wormhole_post_message_shim",
+        POST_MESSAGE_SHIM_PROGRAM_ID,
+        None,
+    );
+    program_test.add_program("core_bridge", CORE_BRIDGE_PROGRAM_ID, None);
+    program_test.add_account_with_base64_data(
+        CORE_BRIDGE_CONFIG,
+        1_057_920,
+        CORE_BRIDGE_PROGRAM_ID,
+        "BAAAAAQYDQ0AAAAAgFEBAGQAAAAAAAAA",
+    );
+    program_test.add_account_with_base64_data(
+        CORE_BRIDGE_FEE_COLLECTOR,
+        2_350_640_070,
+        CORE_BRIDGE_PROGRAM_ID,
+        "",
+    );
+    program_test.prefer_bpf(true);
+
+    program_test.start().await
+}
+
+#[derive(Debug, Default)]
+struct AdditionalTestInputs {
+    invalid_message: Option<Pubkey>,
+    invalid_core_bridge_program: Option<Pubkey>,
+}
+
+fn set_up_post_message_transaction(
+    payload: &[u8],
+    payer_signer: &Keypair,
+    emitter_signer: &Keypair,
+    recent_blockhash: Hash,
+    additional_inputs: Option<AdditionalTestInputs>,
+) -> (VersionedTransaction, BumpCosts) {
+    let emitter = emitter_signer.pubkey();
+    let payer = payer_signer.pubkey();
+
+    let AdditionalTestInputs {
+        invalid_message,
+        invalid_core_bridge_program,
+    } = additional_inputs.unwrap_or_default();
+
+    // Use an invalid message if provided.
+    let (message, message_bump) = invalid_message.map(|key| (key, 0)).unwrap_or(
+        wormhole_svm_definitions::find_shim_message_address(
+            &emitter,
+            &POST_MESSAGE_SHIM_PROGRAM_ID,
+        ),
+    );
+    // Use an invalid core bridge program if provided.
+    let core_bridge_program = invalid_core_bridge_program.unwrap_or(CORE_BRIDGE_PROGRAM_ID);
+
+    let (sequence, sequence_bump) =
+        wormhole_svm_definitions::find_emitter_sequence_address(&emitter, &core_bridge_program);
+
+    let transfer_fee_ix =
+        solana_sdk::system_instruction::transfer(&payer, &CORE_BRIDGE_FEE_COLLECTOR, 100);
+    let post_message_ix = post_message::PostMessage {
+        program_id: &POST_MESSAGE_SHIM_PROGRAM_ID,
+        accounts: post_message::PostMessageAccounts {
+            emitter: &emitter,
+            payer: &payer,
+            wormhole_program_id: &core_bridge_program,
+            derived: post_message::PostMessageDerivedAccounts {
+                message: Some(&message),
+                sequence: Some(&sequence),
+                ..Default::default()
+            },
+        },
+        data: post_message::PostMessageData::new(
+            420,
+            wormhole_svm_definitions::Finality::Finalized,
+            payload,
+        )
+        .unwrap(),
+    }
+    .instruction();
+
+    // Adding compute budget instructions to ensure all instructions fit into
+    // one transaction.
+    //
+    // NOTE: Invoking the compute budget costs in total 300 CU.
+    let message = Message::try_compile(
+        &payer,
+        &[
+            transfer_fee_ix,
+            post_message_ix,
+            ComputeBudgetInstruction::set_compute_unit_price(420),
+            ComputeBudgetInstruction::set_compute_unit_limit(100_000),
+        ],
+        &[],
+        recent_blockhash,
+    )
+    .unwrap();
+
+    let transaction = VersionedTransaction::try_new(
+        VersionedMessage::V0(message),
+        &[payer_signer, emitter_signer],
+    )
+    .unwrap();
+
+    (
+        transaction,
+        BumpCosts {
+            message: bump_cu_cost(message_bump),
+            sequence: bump_cu_cost(sequence_bump),
+        },
+    )
+}
+
+struct BumpCosts {
+    message: u64,
+    sequence: u64,
+}
+
+fn bump_cu_cost(bump: u8) -> u64 {
+    1_500 * (255 - u64::from(bump))
+}

+ 36 - 0
svm/wormhole-core-shims/programs/verify-vaa/Cargo.toml

@@ -0,0 +1,36 @@
+[package]
+name = "wormhole-verify-vaa-shim"
+description = "Created with Anchor"
+publish = false
+
+edition.workspace = true
+license.workspace = true
+repository.workspace = true
+rust-version.workspace = true
+version.workspace = true
+
+[lib]
+crate-type = ["cdylib", "lib"]
+
+[features]
+default = []
+no-entrypoint = []
+
+### Network types
+testnet = ["wormhole-svm-definitions/testnet"]
+localnet = ["wormhole-svm-definitions/localnet"]
+
+### Specific networks
+solana = ["wormhole-svm-definitions/solana"]
+
+[dependencies]
+solana-program.workspace = true
+wormhole-svm-definitions.workspace = true
+wormhole-svm-shim.workspace = true
+
+[dev-dependencies]
+base64.workspace = true
+borsh.workspace = true
+solana-program-test.workspace = true
+solana-sdk.workspace = true
+wormhole-svm-definitions = { workspace = true, features = ["borsh"]}

+ 261 - 0
svm/wormhole-core-shims/programs/verify-vaa/README.md

@@ -0,0 +1,261 @@
+# Verify VAA Shim
+
+## Objective
+
+Reduce the cost of Wormhole Core Bridge program message verification on SVM
+networks. Provide a new verification mechanism without making changes to the
+existing Wormhole Core Bridge program.
+
+## Background
+
+The current way most integrators on SVM networks verify VAAs is via the
+following process:
+
+1. Call the Secp256k1 Native program and the Wormhole Core Bridge program's
+   [verify signatures instruction] in order to verify and store all of the
+   signatures from the VAA into a signature set account. These pair of
+   instructions are called multiple times because the signature data for 13
+   signatures is too long to fit into a single transaction.
+   _There used to be a compute unit restriction as well, but that is no longer
+   an issue as a higher limit can now be requested._
+2. Call the Wormhole Core Bridge program's [post VAA instruction] to validate
+   the signature set account against the VAA and post the VAA data into an
+   account.
+3. Call an instruction on your program which takes a posted VAA account and
+   assumes its validity.
+
+Notably, neither the signature set nor is the posted VAA accounts are closed.
+And there exists no mechanism to close them either. This means that posters
+permanently lose the lamports dedicated to the rent exemption for these
+accounts.
+
+However, some integrations, such as [Solana World ID] use a different approach:
+
+1. Call a [post signatures instruction] on your program (potentially multiple
+   times) to store the signature data on-chain in a guardian signatures account.
+2. Call an instruction on your program which takes a Wormhole Core Bridge
+   program's [guardian set] account and your program's [guardian signatures]
+   account, [performing the required checks and verification]. If the
+   verification is successful, it performs whatever action is needed and closes
+   the signatures account, refunding the lamports.
+
+<!-- cspell:disable -->
+
+This approach does not require leaving any on-chain artifacts and is more cost
+effective. Part of the reason this is now possible in less instructions is due
+to the `secp256k1_recover` SVM syscall and ability to increase the compute unit
+budget.
+
+<!-- cspell:enable -->
+
+> 💡 This will also provide integrators the flexibility to post the VAA payload
+separately as well if needed for payloads that won’t fit into a transaction.
+
+## Goals
+
+- Recommend a documented mechanism for verifying VAAs which does not require
+  permanently storing accounts.
+- Provide a reusable approach to perform the bulk of the work.
+- Provide a cost savings estimation compared to the Wormhole Core Bridge
+  program's verify signatures and post VAA approach.
+
+## Non-Goals
+
+- Provide an on-chain program which generates an artifact of verification akin
+  to the Wormhole Core Bridge program's post VAA instruction.
+
+## Overview
+
+Save end-user costs on SVM network VAA verification by providing a shim, which
+integrators can CPI call to perform verification. This shim will allow the
+client to clean up any artifacts generated during the verification process,
+ultimately resulting in a lower cost verification.
+
+## Detailed Design
+
+The shim will have three instructions which can be leveraged over two
+transactions, resulting in less transactions and rent costs than the Wormhole
+Core Bridge program's approach.
+
+This method explicitly only operates on the signatures, Wormhole Core Bridge
+program's guardian set and digest, making it compatible with both v1 VAAs and
+Queries while maintaining a static instruction size for verification.
+
+As a bonus, this also allows for verification of VAA bodies that are larger what
+can fit in a single instruction, which was not possible with the Wormhole Core
+Bridge program. Integrators can pass the VAA body via instruction data or
+account, as suits their needs. How they choose to do so is outside the scope of
+this design.
+
+### Post Signatures Technical Details
+
+This instruction creates or appends to a guardian signatures account account for
+subsequent use by the [verify hash instruction](#verify-hash-technical-details).
+This step is necessary because the Wormhole VAA body, which has an arbitrary
+size, and 13 guardian signatures (a quorum of the current 19 mainnet guardians,
+66 bytes each) alongside the required accounts is likely larger than the
+transaction size limit on Solana (1232 bytes). This will also allow for the
+verification of other messages which guardians sign, such as query results.
+
+This instruction allows for the initial payer to append additional signatures to
+the account by calling the instruction again. Subsequent calls may be necessary
+if a quorum of signatures from the current guardian set grows larger than can
+fit into a single transaction.
+
+The guardian signatures account can be closed by the initial payer via
+the [close signatures instruction](#close-signatures-technical-details), which
+will refund the initial payer.
+
+### Verify Hash Technical Details
+
+<!-- cspell:disable -->
+
+This instruction is intended to be invoked via CPI call. It verifies a digest
+against a GuardianSignatures account and a Wormhole Core Bridge program's
+guardian set. Prior to this call, and likely in a separate transaction,
+the post signatures instruction must be called to create the account.
+Immediately after calling the verify hash instruction, the close signatures
+instruction should be called to reclaim the lamports.
+
+
+A v1 VAA digest can be computed as follows:
+
+```rust
+let message_hash = &solana_program::keccak::hashv(&[&vaa_body]).to_bytes();
+let digest = keccak::hash(message_hash.as_slice()).to_bytes();
+```
+
+A QueryResponse digest can be computed as follows:
+
+```rust
+use wormhole_query_sdk::MESSAGE_PREFIX;
+let message_hash = [
+  MESSAGE_PREFIX,
+  &solana_program::keccak::hashv(&[&bytes]).to_bytes(),
+]
+.concat();
+let digest = keccak::hash(message_hash.as_slice()).to_bytes();
+```
+
+<!-- cspell:enable -->
+
+### Close Signatures Technical Details
+
+Allows the initial payer to close the guardian signature account, reclaiming the
+rent taken by the post signatures instruction to create this account.
+
+### Protocol Integration
+
+There are no changes required to the protocol.
+
+### API / Database Schema
+
+N/A
+
+## Caveats
+
+This shim will need to include the following patch made to the Wormhole Core
+Bridge program when calculating the expiry for guardian sets.
+
+```rs
+pub fn is_active(&self, timestamp: u32) -> bool {
+    // Note: This is a fix for Wormhole on mainnet. The initial guardian set was
+    // never expired so we block it here.
+    if self.index == 0 && self.creation_time == 1_628_099_186 {
+        false
+    } else {
+        self.expiration_time == 0 || self.expiration_time >= timestamp
+    }
+}
+```
+
+Unlike the Wormhole Core Bridge program, this shim program will not need to
+perform signature set deny-listing, as it does the verification directly through
+`secp256k1_recover`.
+
+Since it is planned to be non-upgradeable, any similar mitigation strategies
+will not be possible. For example, the only way to expire a guardian set will be
+for the Wormhole Core Bridge program to properly expire it.
+
+## Alternatives Considered
+
+- Integrators perform verification logic themselves. This design does not
+  prohibit integrators from doing this. Having each integrator implement
+  verification in their own program is a significant amount of code duplication
+  to recommend. The purpose here is to provide an easy mechanism for integrators
+  to verify VAAs.
+- Post VAA data to an account. For some integrators, the VAA body may fit into
+  their instruction data, so avoiding the account would save cost and added
+  complexity. For integrators with larger payloads, they can first post the body
+  into an account and process it in a subsequent instruction. They may have
+  other efficiencies to gain in how they do that, such as automatically closing
+  the account after verifying the VAA and performing the desired action. It
+  seems best to leave this up to the integrator.
+- Provide an artifact of verification, akin to the Wormhole Core Bridge
+  program's post VAA instruction. While this has the advantage of using only 1
+  account instead of 3, it has the disadvantage of being another temporary
+  account to clean up _and_ does not enforce that the VAA is _currently_ valid.
+  Rather that it was at some time in the past. Since the advent of
+  [Address Lookup Tables], using more accounts is less of an issue than it once
+  was, and ensuring that the VAA is actually valid when you consume it is a
+  security gain.
+
+## Security Considerations
+
+See [Caveats](#caveats).
+
+After initial testing is complete and proper configuration and functionality of
+the shim is confirmed, the upgrade authority should be discarded in order to
+improve trust assumptions and avoid possible compromise of the key.
+
+## Test Plan
+
+Gas cost analysis should be performed for verifying VAAs with and without the shim.
+
+## Performance Impact
+
+This should greatly reduce the cost of verifying messages on Solana, as it will
+save the current rent exemption cost per signature set and posted VAA account.
+
+It does, however, increase the compute units required by using
+`secp256k1_recover` instead of the native instruction.
+
+```jsx
+core (1/4): lamports  1346520, SOL  0.00134652,  $ 0.2929219608,  CU 37058
+core (2/4): lamports    40000, SOL  0.00004,     $ 0.0087016,     CU 25016
+core (3/4): lamports  2482760, SOL  0.00248276,  $ 0.5400996104,  CU 82333
+core (4/4): lamports     4992, SOL  0.000004992, $ 0.00108595968, CU  2302
+total:                3874272,      0.003874272, $ 0.84280913088,   146709
+
+shim (1/2): lamports  7206592, SOL  0.007206592, $1.56772202368,  CU   3037
+shim (2/2): lamports -7191616  SOL -0.007191616, $-1.56446414464, CU 334846
+total:                  15040,      0.000015040, $ 0.00325787904,    337883
+
+savings:              3859232,      0.003859232, $ 0.83955125184,   -191174
+```
+
+That is ~190k CU for ~.3% of the cost.
+
+## Rollout
+
+This only requires deployment of the shim program. If desired, upgrade authority
+of the shim can be maintained for a limited time while testing.
+
+### Acceptance Criteria
+
+Test that verifying a valid VAA digest and signatures via the shim’s verify hash
+instruction succeeds. Test that an expired guardian set, incorrect digest, bad
+signatures, etc, fail.
+
+## Rollback
+
+N/A
+
+[Address Lookup Tables]: https://solana.com/docs/advanced/lookup-tables
+[guardian set]: https://github.com/wormholelabs-xyz/solana-world-id-program/blob/68f1740b2b9bad9d86bd933001a3716a2a993930/programs/solana-world-id-program/src/instructions/update_root_with_query.rs#L70
+[guardian signatures]: https://github.com/wormholelabs-xyz/solana-world-id-program/blob/68f1740b2b9bad9d86bd933001a3716a2a993930/programs/solana-world-id-program/src/instructions/update_root_with_query.rs#L74
+[performing the required checks and verification]: https://github.com/wormholelabs-xyz/solana-world-id-program/blob/68f1740b2b9bad9d86bd933001a3716a2a993930/programs/solana-world-id-program/src/instructions/update_root_with_query.rs#L113
+[post signatures instruction]: https://github.com/wormholelabs-xyz/solana-world-id-program/blob/68f1740b2b9bad9d86bd933001a3716a2a993930/programs/solana-world-id-program/src/instructions/post_signatures.rs#L46
+[post VAA instruction]: https://github.com/wormhole-foundation/wormhole/blob/db1ee86bfbaf16383c46150a44b1437a522514da/solana/bridge/program/src/api/post_vaa.rs#L102
+[Solana World ID]: https://github.com/wormholelabs-xyz/solana-world-id-program/tree/main
+[verify signatures instruction]: https://github.com/wormhole-foundation/wormhole/blob/db1ee86bfbaf16383c46150a44b1437a522514da/solana/bridge/program/src/api/verify_signature.rs#L67

+ 497 - 0
svm/wormhole-core-shims/programs/verify-vaa/src/lib.rs

@@ -0,0 +1,497 @@
+#![deny(dead_code, unused_imports, unused_mut, unused_variables)]
+
+use solana_program::{
+    account_info::AccountInfo,
+    clock::Clock,
+    entrypoint::ProgramResult,
+    instruction::{AccountMeta, Instruction},
+    msg,
+    program::invoke_signed_unchecked,
+    program_error::ProgramError,
+    pubkey::Pubkey,
+    rent::Rent,
+    system_instruction,
+    sysvar::Sysvar,
+};
+use wormhole_svm_definitions::{
+    zero_copy::{GuardianSet, GuardianSignatures},
+    CORE_BRIDGE_PROGRAM_ID, GUARDIAN_SET_SEED, GUARDIAN_SIGNATURE_LENGTH,
+    VERIFY_VAA_SHIM_PROGRAM_ID as ID,
+};
+use wormhole_svm_shim::verify_vaa::{PostSignaturesData, VerifyHashData, VerifyVaaShimInstruction};
+
+solana_program::entrypoint_no_alloc!(process_instruction);
+
+#[inline]
+fn process_instruction(
+    program_id: &Pubkey,
+    accounts: &[AccountInfo],
+    instruction_data: &[u8],
+) -> ProgramResult {
+    // Verify the program ID is what we expect.
+    if program_id != &ID {
+        return Err(ProgramError::IncorrectProgramId);
+    }
+
+    match VerifyVaaShimInstruction::deserialize(instruction_data) {
+        Some(VerifyVaaShimInstruction::PostSignatures(data)) => {
+            process_post_signatures(accounts, data)
+        }
+        Some(VerifyVaaShimInstruction::VerifyHash(data)) => process_verify_hash(accounts, data),
+        Some(VerifyVaaShimInstruction::CloseSignatures) => process_close_signatures(accounts),
+        _ => Err(ProgramError::InvalidInstructionData),
+    }
+}
+
+#[inline(always)]
+fn process_post_signatures(
+    accounts: &[AccountInfo],
+    data: PostSignaturesData<true>,
+) -> ProgramResult {
+    if accounts.len() < 3 {
+        return Err(ProgramError::NotEnoughAccountKeys);
+    }
+
+    // Verify accounts.
+
+    // Payer will pay the rent for the guardian signatures account if it does
+    // not exist. CPI calls to System program will fail if the account is not
+    // writable and is not a signer.
+    let payer_info = &accounts[0];
+
+    // Guardian signatures account will store the guardian signatures. This
+    // account will be created if it does not already exist. For subsequent
+    // calls to this instruction, this account does not need to be a signer.
+    let guardian_signatures_info = &accounts[1];
+
+    // NOTE: We are not checking account at index == 2 because the System
+    // program CPI will fail if the System program account is not present.
+
+    let guardian_signatures_slice = data.guardian_signatures_slice();
+    if guardian_signatures_slice.is_empty() {
+        msg!("Guardian signatures must not be empty");
+        return Err(ProgramError::InvalidArgument);
+    }
+
+    let total_signatures = data.total_signatures();
+    let mut guardian_signatures_len = data.guardian_signatures().len() as u32;
+
+    let start_data_index = if guardian_signatures_info.owner != &ID {
+        if guardian_signatures_info.key == payer_info.key {
+            msg!("Guardian signatures (account #2) cannot be initialized as payer (account #1)");
+            return Err(ProgramError::InvalidAccountData);
+        }
+
+        // There is no point in creating this account if there are no
+        // signatures to write.
+        if total_signatures == 0 {
+            msg!("Total signatures must not be 0");
+            return Err(ProgramError::InvalidArgument);
+        }
+
+        // TODO: Should we revert if total signatures exceeds 154? If it does, the
+        // System program's allocate attempt will fail. But it might be nicer to
+        // have an explicit error for this case.
+        create_account_reliably(
+            payer_info.key,
+            guardian_signatures_info.key,
+            guardian_signatures_info.lamports(),
+            compute_guardian_signatures_data_len(total_signatures),
+            accounts,
+        )?;
+
+        // Now write some data.
+        let mut account_data = guardian_signatures_info.data.borrow_mut();
+        account_data[..8].copy_from_slice(&GuardianSignatures::DISCRIMINATOR);
+        account_data[8..40].copy_from_slice(&payer_info.key.to_bytes());
+        account_data[40..44].copy_from_slice(&data.guardian_set_index().to_be_bytes());
+
+        GuardianSignatures::MINIMUM_SIZE
+    } else {
+        // Because the payer encoded in the guardian signatures is the only
+        // authority to write to this account, we must check that it is a
+        // signer.
+        if !payer_info.is_signer {
+            msg!("Payer (account #1) must be a signer");
+            return Err(ProgramError::MissingRequiredSignature);
+        }
+
+        let account_data = guardian_signatures_info.data.borrow();
+        let guardian_signatures = GuardianSignatures::new(&account_data).ok_or_else(|| {
+            msg!("Guardian signatures (account #2) failed to deserialize");
+            ProgramError::InvalidAccountData
+        })?;
+
+        // This check may not be necessary. But we ensure that the total
+        // signatures is consistent with the account data (even though it is
+        // only used when creating the guardian signatures account).
+        if compute_guardian_signatures_data_len(total_signatures) != account_data.len() {
+            let expected_total_signatures =
+                (account_data.len() - GuardianSignatures::MINIMUM_SIZE) / GUARDIAN_SIGNATURE_LENGTH;
+            msg!("Total signatures mismatch");
+            msg!("Actual:");
+            msg!("{}", total_signatures);
+            msg!("Expected:");
+            msg!("{}", expected_total_signatures);
+            return Err(ProgramError::InvalidArgument);
+        }
+
+        // Only the original payer (refund recipient) can write to this account
+        // with subsequent calls to this instruction.
+        if payer_info.key.as_ref() != guardian_signatures.refund_recipient_slice() {
+            msg!("Payer (account #1) must match refund recipient");
+            msg!("Actual:");
+            msg!("{}", payer_info.key);
+            msg!("Expected:");
+            msg!("{}", guardian_signatures.refund_recipient());
+            return Err(ProgramError::InvalidAccountData);
+        }
+
+        // Same as the total signatures check above, this check may not be
+        // necessary. But we ensure that the guardian set index is consistent
+        // with the account data (even though it is only used when creating the
+        // guardian signatures account).
+        if data.guardian_set_index() != guardian_signatures.guardian_set_index() {
+            msg!("Guardian set index mismatch");
+            msg!("Actual:");
+            msg!("{}", data.guardian_set_index());
+            msg!("Expected:");
+            msg!("{}", guardian_signatures.guardian_set_index());
+            return Err(ProgramError::InvalidArgument);
+        }
+
+        let current_guardian_signatures_len = guardian_signatures.guardian_signatures_len();
+
+        // Update the length, which will be written back to the account. Simply
+        // adding is safe because the number of signatures will never
+        // exceed 154 (which is much smaller than u32::MAX).
+        guardian_signatures_len += current_guardian_signatures_len;
+
+        GuardianSignatures::MINIMUM_SIZE
+            + (current_guardian_signatures_len as usize) * GUARDIAN_SIGNATURE_LENGTH
+    };
+
+    let mut account_data = guardian_signatures_info.data.borrow_mut();
+
+    // This operation is safe because this sum will never reach usize::MAX.
+    let end_data_index = start_data_index + guardian_signatures_slice.len();
+
+    // Check if the account data is not large enough to hold the signatures.
+    if end_data_index > account_data.len() {
+        msg!("Too many input guardian signatures");
+        return Err(ProgramError::AccountDataTooSmall);
+    }
+
+    // Write signatures.
+    account_data[start_data_index..end_data_index].copy_from_slice(guardian_signatures_slice);
+
+    // Update length.
+    account_data[44..48].copy_from_slice(&guardian_signatures_len.to_le_bytes());
+
+    Ok(())
+}
+
+#[inline(always)]
+fn process_verify_hash(accounts: &[AccountInfo], data: VerifyHashData) -> ProgramResult {
+    if accounts.len() < 2 {
+        return Err(ProgramError::NotEnoughAccountKeys);
+    }
+
+    // Verify accounts.
+
+    // Guardian set account warehouses the guardians' ETH public keys. These
+    // keys are used to verify the public keys recovered from verifying the
+    // signatures in the guardian signatures account with the provided digest.
+    let guardian_set_info = &accounts[0];
+
+    // Guardian signatures account. This account was created in the post
+    // signatures instruction. Its signatures will be validated against the
+    // guardian pubkeys found in the guardian set account.
+    let guardian_signatures_info = &accounts[1];
+    if guardian_signatures_info.owner != &ID {
+        msg!("Guardian signatures (account #2) must be owned by this program");
+        return Err(ProgramError::InvalidAccountData);
+    }
+
+    let guardian_signatures_info_data = guardian_signatures_info.data.borrow();
+    let guardian_signatures =
+        GuardianSignatures::new(&guardian_signatures_info_data).ok_or_else(|| {
+            msg!("Guardian signatures (account #2) failed to deserialize");
+            ProgramError::InvalidAccountData
+        })?;
+
+    // With the guardian signatures account's encoded guardian set index and
+    // provided bump, create the guardian set account's address.
+    let expected_address = Pubkey::create_program_address(
+        &[
+            GUARDIAN_SET_SEED,
+            guardian_signatures.guardian_index_be_slice(),
+            &[data.guardian_set_bump()],
+        ],
+        &CORE_BRIDGE_PROGRAM_ID,
+    )
+    .map_err(|err| {
+        msg!("Guardian set (account #1) address creation failed");
+        err
+    })?;
+    if guardian_set_info.key != &expected_address {
+        msg!("Guardian set (account #1) seeds constraint violated");
+        msg!("Actual:");
+        msg!("{}", guardian_set_info.key);
+        msg!("Expected:");
+        msg!("{}", expected_address);
+        return Err(ProgramError::InvalidAccountData);
+    }
+
+    let guardian_set_info_data = guardian_set_info.data.borrow();
+
+    // Unwrapping here is safe because we verified the guardian set's address.
+    let guardian_set = GuardianSet::new(&guardian_set_info_data).unwrap();
+
+    // This operation will panic quite far in the future.
+    let timestamp = u32::try_from(Clock::get().unwrap().unix_timestamp).unwrap();
+    if !guardian_set.is_active(timestamp) {
+        msg!("Guardian set (account #1) is expired");
+        return Err(ProgramError::InvalidAccountData);
+    }
+
+    let guardian_signatures_len = guardian_signatures.guardian_signatures_len();
+
+    // Number of signatures must meet quorum, which is at least two thirds of
+    // the guardian set's size.
+    if guardian_signatures_len < guardian_set.quorum() {
+        msg!("Guardian signatures (account #2) fails to meet quorum");
+        return Err(ProgramError::InvalidAccountData);
+    }
+
+    let digest = data.digest().0;
+    let mut last_guardian_index = None;
+
+    for i in 0..(guardian_signatures_len as usize) {
+        // Signature is encoded as:
+        // - 1 byte: guardian index
+        // - 64 bytes: r and s concatenated
+        // - 1 byte: recovery ID
+        let signature = guardian_signatures.guardian_signature_slice(i).unwrap();
+        let index = signature[0] as usize;
+        if let Some(last_index) = last_guardian_index {
+            if index <= last_index {
+                msg!("Guardian signatures (account #2) has non-increasing guardian index");
+                return Err(ProgramError::InvalidAccountData);
+            }
+        }
+
+        let guardian_pubkey = guardian_set.key_slice(index).ok_or_else(|| {
+            msg!("Guardian signatures (account #2) guardian index out of range");
+            ProgramError::InvalidAccountData
+        })?;
+
+        // Keccak-256 hash the recovered public key and compare it with the
+        // guardian set's public key.
+        let recovered_pubkey = solana_program::secp256k1_recover::secp256k1_recover(
+            &digest,
+            signature[65],
+            &signature[1..65],
+        )
+        .map(|pubkey| solana_program::keccak::hashv(&[&pubkey.0]).0)
+        .map_err(|_| {
+            msg!("Guardian signature index {} is invalid", i);
+            ProgramError::InvalidAccountData
+        })?;
+
+        if &recovered_pubkey[12..] != guardian_pubkey {
+            msg!(
+                "Guardian signature index {} does not recover guardian {} pubkey",
+                i,
+                index
+            );
+            return Err(ProgramError::InvalidAccountData);
+        }
+
+        last_guardian_index.replace(index);
+    }
+
+    Ok(())
+}
+
+#[inline(always)]
+fn process_close_signatures(accounts: &[AccountInfo]) -> ProgramResult {
+    if accounts.len() < 2 {
+        return Err(ProgramError::NotEnoughAccountKeys);
+    }
+
+    // Verify accounts.
+
+    // Guardian signatures account will be closed by the end of this
+    // instruction. Rent will be moved to the refund recipient.
+    let guardian_signatures_info = &accounts[0];
+
+    // Recipient of guardian signatures account's lamports. This pubkey must
+    // match the refund recipient stored in the guardian signatures account.
+    let refund_recipient_info = &accounts[1];
+
+    // Because the refund recipient encoded in the guardian signatures is the
+    // only authority to close this account, we must check that it is a signer.
+    if !refund_recipient_info.is_signer {
+        msg!("Refund recipient (account #2) must be a signer");
+        return Err(ProgramError::MissingRequiredSignature);
+    }
+
+    {
+        let account_data = guardian_signatures_info.data.borrow();
+        let guardian_signatures =
+            GuardianSignatures::new(&account_data).ok_or(ProgramError::InvalidAccountData)?;
+
+        if refund_recipient_info.key.as_ref() != guardian_signatures.refund_recipient_slice() {
+            msg!("Refund recipient (account #2) mismatch");
+            msg!("Actual:");
+            msg!("{}", refund_recipient_info.key);
+            msg!("Expected:");
+            msg!("{}", guardian_signatures.refund_recipient());
+            return Err(ProgramError::InvalidAccountData);
+        }
+    }
+
+    let mut guardian_signatures_lamports = guardian_signatures_info.lamports.borrow_mut();
+    **refund_recipient_info.lamports.borrow_mut() += **guardian_signatures_lamports;
+    **guardian_signatures_lamports = 0;
+
+    // The refund recipient is the only authority that can close this account.
+    // So even if lamports were sent to the guardian signatures account to keep
+    // it open, the refund recipient will still get the lamports he is owed.
+    // Therefore, we do not need to make sure that the account's size is zero
+    // and that its owner is ssigned to the System program.
+    //
+    // When the guardian signatures account's lamports are zero, SVM runtime
+    // will ensure that this account will be closed.
+
+    Ok(())
+}
+
+#[inline(always)]
+fn create_account_reliably(
+    payer_key: &Pubkey,
+    account_key: &Pubkey,
+    current_lamports: u64,
+    data_len: usize,
+    accounts: &[AccountInfo],
+) -> ProgramResult {
+    let lamports = Rent::get().unwrap().minimum_balance(data_len);
+
+    if current_lamports == 0 {
+        let ix = system_instruction::create_account(
+            payer_key,
+            account_key,
+            lamports,
+            data_len as u64,
+            &ID,
+        );
+
+        invoke_signed_unchecked(&ix, accounts, &[])?;
+    } else {
+        const MAX_CPI_DATA_LEN: usize = 36;
+        const TRANSFER_ALLOCATE_DATA_LEN: usize = 12;
+        const SYSTEM_PROGRAM_SELECTOR_LEN: usize = 4;
+
+        // Perform up to three CPIs:
+        // 1. Transfer lamports from payer to account (may not be necessary).
+        // 2. Allocate data to the account.
+        // 3. Assign the account owner to this program.
+        //
+        // The max length of instruction data is 36 bytes among the three
+        // instructions, so we will reuse the same allocated memory for all.
+        let mut cpi_ix = Instruction {
+            program_id: solana_program::system_program::ID,
+            accounts: vec![
+                AccountMeta::new(*payer_key, true),
+                AccountMeta::new(*account_key, true),
+            ],
+            data: Vec::with_capacity(MAX_CPI_DATA_LEN),
+        };
+
+        // Safety: Because capacity is > 12, it is safe to set this length and
+        // to set the first 4 elements to zero, which covers the System program
+        // instruction selectors.
+        //
+        // The transfer and allocate instructions are 12 bytes long:
+        // - 4 bytes for the discriminator
+        // - 8 bytes for the lamports (transfer) or data length (allocate)
+        //
+        // The last 8 bytes will be copied to the data slice.
+        unsafe {
+            let cpi_data = &mut cpi_ix.data;
+
+            core::ptr::write_bytes(cpi_data.as_mut_ptr(), 0, TRANSFER_ALLOCATE_DATA_LEN);
+            cpi_data.set_len(TRANSFER_ALLOCATE_DATA_LEN);
+        }
+
+        // We will have to transfer the remaining lamports needed to cover rent
+        // for the account.
+        let lamport_diff = lamports.saturating_sub(current_lamports);
+
+        // Only invoke transfer if there are lamports required.
+        if lamport_diff != 0 {
+            let cpi_data = &mut cpi_ix.data;
+
+            cpi_data[0] = 2; // transfer selector
+            cpi_data[SYSTEM_PROGRAM_SELECTOR_LEN..TRANSFER_ALLOCATE_DATA_LEN]
+                .copy_from_slice(&lamport_diff.to_le_bytes());
+
+            invoke_signed_unchecked(&cpi_ix, accounts, &[])?;
+        }
+
+        let cpi_accounts = &mut cpi_ix.accounts;
+
+        // Safety: Setting the length reduces the previous length from the last
+        // CPI call.
+        //
+        // Both allocate and assign instructions require one account (the
+        // account being created).
+        unsafe {
+            cpi_accounts.set_len(1);
+        }
+
+        // Because the payer and account are writable signers, we can simply
+        // overwrite the pubkey of the first account.
+        cpi_accounts[0].pubkey = *account_key;
+
+        {
+            let cpi_data = &mut cpi_ix.data;
+
+            cpi_data[0] = 8; // allocate selector
+            cpi_data[SYSTEM_PROGRAM_SELECTOR_LEN..TRANSFER_ALLOCATE_DATA_LEN]
+                .copy_from_slice(&(data_len as u64).to_le_bytes());
+
+            invoke_signed_unchecked(&cpi_ix, accounts, &[])?;
+        }
+
+        {
+            let cpi_data = &mut cpi_ix.data;
+
+            // Safety: The capacity of this vector is 36. This data will be
+            // overwritten for the next CPI call.
+            unsafe {
+                core::ptr::write_bytes(
+                    cpi_data
+                        .as_mut_ptr()
+                        .offset(TRANSFER_ALLOCATE_DATA_LEN as isize),
+                    0,
+                    MAX_CPI_DATA_LEN - TRANSFER_ALLOCATE_DATA_LEN,
+                );
+                cpi_data.set_len(MAX_CPI_DATA_LEN);
+            }
+
+            cpi_data[0] = 1; // assign selector
+            cpi_data[SYSTEM_PROGRAM_SELECTOR_LEN..MAX_CPI_DATA_LEN].copy_from_slice(&ID.to_bytes());
+
+            invoke_signed_unchecked(&cpi_ix, accounts, &[])?;
+        }
+    }
+
+    Ok(())
+}
+
+#[inline(always)]
+fn compute_guardian_signatures_data_len(total_signatures: u8) -> usize {
+    (total_signatures as usize) * GUARDIAN_SIGNATURE_LENGTH + GuardianSignatures::MINIMUM_SIZE
+}

+ 75 - 0
svm/wormhole-core-shims/programs/verify-vaa/tests/common/close_signatures.rs

@@ -0,0 +1,75 @@
+use solana_sdk::{
+    compute_budget::ComputeBudgetInstruction,
+    hash::Hash,
+    message::{v0::Message, VersionedMessage},
+    pubkey::Pubkey,
+    signature::Keypair,
+    signer::Signer,
+    transaction::VersionedTransaction,
+};
+use wormhole_svm_definitions::VERIFY_VAA_SHIM_PROGRAM_ID;
+use wormhole_svm_shim::verify_vaa;
+
+#[derive(Debug, Default)]
+pub struct AdditionalTestInputs {
+    pub refund_recipient_is_signer: Option<bool>,
+}
+
+pub fn set_up_transaction(
+    tx_payer_signer: &Keypair,
+    refund_recipient_signer: &Keypair,
+    guardian_signatures: &Pubkey,
+    recent_blockhash: Hash,
+    additional_inputs: Option<AdditionalTestInputs>,
+) -> VersionedTransaction {
+    let tx_payer = tx_payer_signer.pubkey();
+    let refund_recipient = refund_recipient_signer.pubkey();
+
+    let mut close_signatures_ix = verify_vaa::CloseSignatures {
+        program_id: &VERIFY_VAA_SHIM_PROGRAM_ID,
+        accounts: verify_vaa::CloseSignaturesAccounts {
+            guardian_signatures,
+            refund_recipient: &refund_recipient,
+        },
+    }
+    .instruction();
+
+    let AdditionalTestInputs {
+        refund_recipient_is_signer,
+    } = additional_inputs.unwrap_or_default();
+
+    let needs_additional_signer = match refund_recipient_is_signer {
+        Some(is_signer) => {
+            close_signatures_ix.accounts[1].is_signer = is_signer;
+            is_signer
+        }
+        None => tx_payer != refund_recipient,
+    };
+
+    // Adding compute budget instructions to ensure all instructions fit into
+    // one transaction.
+    //
+    // NOTE: Invoking the compute budget costs in total 300 CU.
+    let message = Message::try_compile(
+        &tx_payer,
+        &[
+            close_signatures_ix,
+            ComputeBudgetInstruction::set_compute_unit_price(69),
+            // NOTE: CU limit is higher than needed to resolve errors in test.
+            ComputeBudgetInstruction::set_compute_unit_limit(25_000),
+        ],
+        &[],
+        recent_blockhash,
+    )
+    .unwrap();
+
+    if needs_additional_signer {
+        VersionedTransaction::try_new(
+            VersionedMessage::V0(message),
+            &[tx_payer_signer, refund_recipient_signer],
+        )
+        .unwrap()
+    } else {
+        VersionedTransaction::try_new(VersionedMessage::V0(message), &[tx_payer_signer]).unwrap()
+    }
+}

+ 178 - 0
svm/wormhole-core-shims/programs/verify-vaa/tests/common/mod.rs

@@ -0,0 +1,178 @@
+pub mod close_signatures;
+pub mod post_signatures;
+pub mod verify_hash;
+
+use base64::{prelude::BASE64_STANDARD, Engine};
+use solana_program_test::{BanksClient, ProgramTest};
+use solana_sdk::{
+    hash::Hash, pubkey::Pubkey, signature::Keypair, signer::Signer, transaction::Transaction,
+};
+use wormhole_svm_definitions::{
+    borsh::GuardianSignatures, find_guardian_set_address, CORE_BRIDGE_CONFIG,
+    CORE_BRIDGE_PROGRAM_ID, GUARDIAN_SIGNATURE_LENGTH, VERIFY_VAA_SHIM_PROGRAM_ID,
+};
+
+pub async fn start_test(vaa: &str) -> (BanksClient, Keypair, Hash, DecodedVaa) {
+    let mut program_test =
+        ProgramTest::new("wormhole_verify_vaa_shim", VERIFY_VAA_SHIM_PROGRAM_ID, None);
+    // program_test.add_program("core_bridge", CORE_BRIDGE_PROGRAM_ID, None);
+    program_test.add_account_with_base64_data(
+        CORE_BRIDGE_CONFIG,
+        1_057_920,
+        CORE_BRIDGE_PROGRAM_ID,
+        "BAAAAAQYDQ0AAAAAgFEBAGQAAAAAAAAA",
+    );
+    // Guardian set 4 (active).
+    program_test.add_account_with_base64_data(
+        find_guardian_set_address(u32::to_be_bytes(4), &CORE_BRIDGE_PROGRAM_ID).0,
+        3_647_040,
+        CORE_BRIDGE_PROGRAM_ID,
+        "BAAAABMAAABYk7WnbD9zlkVkiIW9zMBs1wo80/9suVJYm96GLCXvQ5ITL7nUpCFXEU3oRgGTvfOi/PgfhqCXZfR2L9EQegCGsy16CXeSaiBRMdhzHTnL64yCsv2C+u0nEdWa8PJJnRbnJvayEbOXVsBCRBvm2GULabVOvnFeI0NUzltNNI+3S5WOiWbi7D29SVinzRXnyvB8Tj3I58Rp+SyM2I+4AFogdKO/kTlT1pUmDYi8GqJaTu42PvAACsAHZyezX76i2sKP7lzLD+p2jq9FztE2udniSQNGSuiJ9cinI/wU+TEkt8c4hDy7iehkyGLDjN3Mz5XSzDek3ANqjSMrSPYs3UcxQS9IkNp5j2iWozMfZLSMEtHVf9nL5wgRcaob4dNsr+OGeRD5nAnjR4mcGcOBkrbnOHzNdoJ3wX2rG3pQJ8CzzxeOIa0ud64GcRVJz7sfnHqdgJboXhSH81UV0CqSdTUEqNdUcbn0nttvvryJj0A+R3PpX+sV6Ayamcg0jXiZHmYAAAAA",
+    );
+    // Guardian set 3 (expired).
+    program_test.add_account_with_base64_data(
+        find_guardian_set_address(u32::to_be_bytes(3), &CORE_BRIDGE_PROGRAM_ID).0,
+        3_647_040,
+        CORE_BRIDGE_PROGRAM_ID,
+        "AwAAABMAAABYzDrlwJeyE848gZeeG5+VcHRqpf9suVJYm96GLCXvQ5ITL7nUpCFXEU3oRgGTvfOi/PgfhqCXZfR2L9EQegCGsy16CXeSaiBRMdhzHTnL64yCsv2C+u0nEdWa8PJJnRbnJvayEbOXVsBCRBvm2GULabVOvnFeI0NUzltNNI+3S5WOiWbi7D29SVinzRXnyvB8Tj3I58Rp+SyM2I+4AFogdKO/kTlT1pUmDYi8GqJaTu42PvAACsAHZyezX76i2sKP7lzLD+p2jq9FztE2udniSQNGSuiJ9cinI/wU+TEkt8c4hDy7iehkyGLDjN3Mz5XSzDek3ANqjSMrSPYs3UcxQS9IkNp5j2iWozMfZLSMEtHVf9nL5wgRcaob4dNsr+OGeRD5nAnjR4mcGcOBkrbnOHzNdoJ3wX2rG3pQJ8CzzxeOIa0ud64GcRVJz7sfnHqdgJboXhSH81UV0CqSdTUEqNdUcbn0nttvvryJj0A+R3PpX+sV6Ayamcg0jUA8xWP46h9m",
+    );
+    program_test.prefer_bpf(true);
+
+    let (banks_client, payer, recent_blockhash) = program_test.start().await;
+    (banks_client, payer, recent_blockhash, vaa.into())
+}
+
+pub struct DecodedVaa {
+    pub guardian_set_index: u32,
+    pub total_signatures: u8,
+    pub guardian_signatures: Vec<[u8; GUARDIAN_SIGNATURE_LENGTH]>,
+    pub body: Vec<u8>,
+}
+
+impl From<&str> for DecodedVaa {
+    fn from(vaa: &str) -> Self {
+        let mut buf = BASE64_STANDARD.decode(vaa).unwrap();
+        let guardian_set_index = u32::from_be_bytes(buf[1..5].try_into().unwrap());
+        let total_signatures = buf[5];
+
+        let body = buf
+            .drain((6 + total_signatures as usize * GUARDIAN_SIGNATURE_LENGTH)..)
+            .collect();
+
+        let mut guardian_signatures = Vec::with_capacity(total_signatures as usize);
+
+        for i in 0..usize::from(total_signatures) {
+            let offset = 6 + i * 66;
+            let mut signature = [0; GUARDIAN_SIGNATURE_LENGTH];
+            signature.copy_from_slice(&buf[offset..offset + GUARDIAN_SIGNATURE_LENGTH]);
+            guardian_signatures.push(signature);
+        }
+
+        Self {
+            guardian_set_index,
+            total_signatures,
+            guardian_signatures,
+            body,
+        }
+    }
+}
+
+pub async fn transfer_lamports(
+    banks_client: &BanksClient,
+    recent_blockhash: Hash,
+    payer: &Keypair,
+    recipient: &Pubkey,
+    lamports: u64,
+) -> Hash {
+    let transfer_ix =
+        solana_sdk::system_instruction::transfer(&payer.pubkey(), recipient, lamports);
+
+    banks_client
+        .process_transaction(Transaction::new_signed_with_payer(
+            &[transfer_ix],
+            Some(&payer.pubkey()),
+            &[payer],
+            recent_blockhash,
+        ))
+        .await
+        .unwrap();
+
+    banks_client.get_latest_blockhash().await.unwrap()
+}
+
+pub fn generate_expected_guardian_signatures_info(
+    payer: &Pubkey,
+    total_signatures: u8,
+    guardian_set_index: u32,
+    guardian_signatures: Vec<[u8; GUARDIAN_SIGNATURE_LENGTH]>,
+) -> (
+    usize, // expected length
+    GuardianSignatures,
+) {
+    let expected_length = {
+        8 // discriminator
+        + 32 // refund recipient
+        + 4 // guardian set index
+        + 4 // guardian signatures length
+        + (total_signatures as usize) * GUARDIAN_SIGNATURE_LENGTH
+    };
+
+    let guardian_signatures = GuardianSignatures {
+        refund_recipient: *payer,
+        guardian_set_index_be: guardian_set_index.to_be_bytes(),
+        guardian_signatures,
+    };
+
+    (expected_length, guardian_signatures)
+}
+
+pub fn bump_cu_cost(bump: u8) -> u64 {
+    1_500 * (255 - u64::from(bump))
+}
+
+pub async fn send_post_signatures_transaction(
+    banks_client: &BanksClient,
+    tx_payer_signer: &Keypair,
+    guardian_set_index: u32,
+    total_signatures: u8,
+    guardian_signatures: &[[u8; GUARDIAN_SIGNATURE_LENGTH]],
+    recent_blockhash: Hash,
+    guardian_signatures_signer: Option<&Keypair>,
+) -> (
+    Pubkey,  // guardian signatures
+    Keypair, // refund recipient signer
+    Hash,
+) {
+    let refund_recipient_signer = Keypair::new();
+
+    let recent_blockhash = transfer_lamports(
+        banks_client,
+        recent_blockhash,
+        tx_payer_signer,
+        &refund_recipient_signer.pubkey(),
+        2_000_000_000,
+    )
+    .await;
+
+    let arbitrary_signer = Keypair::new();
+    let guardian_signatures_signer = guardian_signatures_signer.unwrap_or(&arbitrary_signer);
+
+    let transaction = post_signatures::set_up_transaction(
+        tx_payer_signer,
+        guardian_set_index,
+        total_signatures,
+        guardian_signatures,
+        &refund_recipient_signer,
+        guardian_signatures_signer,
+        recent_blockhash,
+        None,
+    );
+
+    banks_client.process_transaction(transaction).await.unwrap();
+
+    (
+        guardian_signatures_signer.pubkey(),
+        refund_recipient_signer,
+        banks_client.get_latest_blockhash().await.unwrap(),
+    )
+}

+ 85 - 0
svm/wormhole-core-shims/programs/verify-vaa/tests/common/post_signatures.rs

@@ -0,0 +1,85 @@
+use solana_sdk::{
+    compute_budget::ComputeBudgetInstruction,
+    hash::Hash,
+    message::{v0::Message, VersionedMessage},
+    signature::Keypair,
+    signer::Signer,
+    transaction::VersionedTransaction,
+};
+use wormhole_svm_definitions::{GUARDIAN_SIGNATURE_LENGTH, VERIFY_VAA_SHIM_PROGRAM_ID};
+use wormhole_svm_shim::verify_vaa;
+
+#[derive(Debug, Default)]
+pub struct AdditionalTestInputs {
+    pub payer_is_signer: Option<bool>,
+}
+
+#[allow(clippy::too_many_arguments)]
+pub fn set_up_transaction(
+    tx_payer_signer: &Keypair,
+    guardian_set_index: u32,
+    total_signatures: u8,
+    guardian_signatures: &[[u8; GUARDIAN_SIGNATURE_LENGTH]],
+    payer_signer: &Keypair,
+    guardian_signatures_signer: &Keypair,
+    recent_blockhash: Hash,
+    additional_inputs: Option<AdditionalTestInputs>,
+) -> VersionedTransaction {
+    let tx_payer = tx_payer_signer.pubkey();
+    let payer = payer_signer.pubkey();
+
+    let mut post_signatures_ix = verify_vaa::PostSignatures {
+        program_id: &VERIFY_VAA_SHIM_PROGRAM_ID,
+        accounts: verify_vaa::PostSignaturesAccounts {
+            payer: &payer,
+            guardian_signatures: &guardian_signatures_signer.pubkey(),
+        },
+        data: verify_vaa::PostSignaturesData::new(
+            guardian_set_index,
+            total_signatures,
+            guardian_signatures,
+        ),
+    }
+    .instruction();
+
+    let AdditionalTestInputs { payer_is_signer } = additional_inputs.unwrap_or_default();
+
+    let needs_additional_signer = match payer_is_signer {
+        Some(is_signer) => {
+            post_signatures_ix.accounts[0].is_signer = is_signer;
+            is_signer
+        }
+        None => tx_payer != payer,
+    };
+
+    // Adding compute budget instructions to ensure all instructions fit into
+    // one transaction.
+    //
+    // NOTE: Invoking the compute budget costs in total 300 CU.
+    let message = Message::try_compile(
+        &tx_payer,
+        &[
+            post_signatures_ix,
+            ComputeBudgetInstruction::set_compute_unit_price(69),
+            // NOTE: CU limit is higher than needed to resolve errors in test.
+            ComputeBudgetInstruction::set_compute_unit_limit(25_000),
+        ],
+        &[],
+        recent_blockhash,
+    )
+    .unwrap();
+
+    if needs_additional_signer {
+        VersionedTransaction::try_new(
+            VersionedMessage::V0(message),
+            &[tx_payer_signer, payer_signer, guardian_signatures_signer],
+        )
+        .unwrap()
+    } else {
+        VersionedTransaction::try_new(
+            VersionedMessage::V0(message),
+            &[tx_payer_signer, guardian_signatures_signer],
+        )
+        .unwrap()
+    }
+}

+ 78 - 0
svm/wormhole-core-shims/programs/verify-vaa/tests/common/verify_hash.rs

@@ -0,0 +1,78 @@
+use solana_sdk::{
+    compute_budget::ComputeBudgetInstruction,
+    hash::Hash,
+    keccak,
+    message::{v0::Message, VersionedMessage},
+    pubkey::Pubkey,
+    signature::Keypair,
+    signer::Signer,
+    transaction::VersionedTransaction,
+};
+use wormhole_svm_definitions::{CORE_BRIDGE_PROGRAM_ID, VERIFY_VAA_SHIM_PROGRAM_ID};
+use wormhole_svm_shim::verify_vaa::{self, VerifyHashData};
+
+use super::bump_cu_cost;
+
+#[derive(Debug, Default)]
+pub struct AdditionalTestInputs {
+    pub invalid_guardian_set: Option<Pubkey>,
+}
+
+pub struct BumpCosts {
+    pub guardian_set: u64,
+}
+
+pub fn set_up_transaction(
+    tx_payer_signer: &Keypair,
+    guardian_set_index: u32,
+    guardian_signatures: &Pubkey,
+    digest: keccak::Hash,
+    recent_blockhash: Hash,
+    additional_inputs: Option<AdditionalTestInputs>,
+) -> (VersionedTransaction, BumpCosts) {
+    let tx_payer = tx_payer_signer.pubkey();
+
+    let AdditionalTestInputs {
+        invalid_guardian_set,
+    } = additional_inputs.unwrap_or_default();
+
+    let (guardian_set, guardian_set_bump) = invalid_guardian_set.map(|key| (key, 0)).unwrap_or(
+        wormhole_svm_definitions::find_guardian_set_address(
+            guardian_set_index.to_be_bytes(),
+            &CORE_BRIDGE_PROGRAM_ID,
+        ),
+    );
+
+    let verify_hash_ix = verify_vaa::VerifyHash {
+        program_id: &VERIFY_VAA_SHIM_PROGRAM_ID,
+        accounts: verify_vaa::VerifyHashAccounts {
+            guardian_set: &guardian_set,
+            guardian_signatures,
+        },
+        data: VerifyHashData::new(guardian_set_bump, digest),
+    }
+    .instruction();
+
+    // Adding compute budget instructions to ensure all instructions fit into
+    // one transaction.
+    //
+    // NOTE: Invoking the compute budget costs in total 300 CU.
+    let message = Message::try_compile(
+        &tx_payer,
+        &[
+            verify_hash_ix,
+            ComputeBudgetInstruction::set_compute_unit_price(69),
+            ComputeBudgetInstruction::set_compute_unit_limit(340_000),
+        ],
+        &[],
+        recent_blockhash,
+    )
+    .unwrap();
+
+    (
+        VersionedTransaction::try_new(VersionedMessage::V0(message), &[tx_payer_signer]).unwrap(),
+        BumpCosts {
+            guardian_set: bump_cu_cost(guardian_set_bump),
+        },
+    )
+}

+ 1153 - 0
svm/wormhole-core-shims/programs/verify-vaa/tests/integration_test.rs

@@ -0,0 +1,1153 @@
+use solana_program_test::tokio;
+use solana_sdk::{
+    compute_budget::ComputeBudgetInstruction,
+    message::{v0::Message, VersionedMessage},
+    pubkey::Pubkey,
+    signature::Keypair,
+    signer::Signer,
+    transaction::VersionedTransaction,
+};
+use wormhole_svm_definitions::{
+    borsh::{deserialize_with_discriminator, GuardianSignatures},
+    compute_keccak_digest, VERIFY_VAA_SHIM_PROGRAM_ID,
+};
+use wormhole_svm_shim::verify_vaa;
+
+mod common;
+
+const VAA: &str = "AQAAAAQNAL1qji7v9KnngyX0VxK+3fCMVscWTLoYX8L48NWquq2WGrcHd4H0wYc0KF4ZOWjLD2okXoBjGQIDJzx4qIrbSzQBAQq69h+neXGb58VfhZgraPVCxJmnTj8JIDq5jqi3Qav1e+IW51mIJlOhSAdCRbEyQLzf6Z3C19WJJqSyt/z1XF0AAvFgDHkseyMZTE5vQjflu4tc5OLPJe2VYCxTJT15LA02YPrWgOM6HhfUhXDhFoG5AI/s2ApjK8jaqi7LGJILAUMBA6cp4vfko8hYyRvogqQWsdk9e20g0O6s60h4ewweapXCQHerQpoJYdDxlCehN4fuYnuudEhW+6FaXLjwNJBdqsoABDg9qXjXB47nBVCZAGns2eosVqpjkyDaCfo/p1x8AEjBA80CyC1/QlbG9L4zlnnDIfZWylsf3keJqx28+fZNC5oABi6XegfozgE8JKqvZLvd7apDhrJ6Qv+fMiynaXASkafeVJOqgFOFbCMXdMKehD38JXvz3JrlnZ92E+I5xOJaDVgABzDSte4mxUMBMJB9UUgJBeAVsokFvK4DOfvh6G3CVqqDJplLwmjUqFB7fAgRfGcA8PWNStRc+YDZiG66YxPnptwACe84S31Kh9voz2xRk1THMpqHQ4fqE7DizXPNWz6Z6ebEXGcd7UP9PBXoNNvjkLWZJZOdbkZyZqztaIiAo4dgWUABCobiuQP92WjTxOZz0KhfWVJ3YBVfsXUwaVQH4/p6khX0HCEVHR9VHmjvrAAGDMdJGWW+zu8mFQc4gPU6m4PZ6swADO7voA5GWZZPiztz22pftwxKINGvOjCPlLpM1Y2+Vq6AQuez/mlUAmaL0NKgs+5VYcM1SGBz0TL3ABRhKQAhUEMADWmiMo0J1Qaj8gElb+9711ZjvAY663GIyG/E6EdPW+nPKJI9iZE180sLct+krHj0J7PlC9BjDiO2y149oCOJ6FgAEcaVkYK43EpN7XqxrdpanX6R6TaqECgZTjvtN3L6AP2ceQr8mJJraYq+qY8pTfFvPKEqmW9CBYvnA5gIMpX59WsAEjIL9Hdnx+zFY0qSPB1hB9AhqWeBP/QfJjqzqafsczaeCN/rWUf6iNBgXI050ywtEp8JQ36rCn8w6dRhUusn+MEAZ32XyAAAAAAAFczO6yk0j3G90i/+9DoqGcH1teF8XMpUEVKRIBgmcq3lAAAAAAAC/1wAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAC6Q7dAAAAAAAAAAAAAAAAAAoLhpkcYhizbB0Z1KLp6wzjYG60gAAgAAAAAAAAAAAAAAAInNTEvk5b/1WVF+JawF1smtAdicABAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA==";
+
+// Post signatures.
+
+#[tokio::test]
+async fn test_post_signatures_13_at_once() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let guardian_signatures_signer = Keypair::new();
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &decoded_vaa.guardian_signatures,
+        &payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction.clone())
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_ok());
+    assert_eq!(
+        out.simulation_details.unwrap().units_consumed,
+        // 13_355
+        3_343
+    );
+
+    banks_client.process_transaction(transaction).await.unwrap();
+
+    // Check guardian signatures account after processing the transaction.
+    let guardian_signatures_info = banks_client
+        .get_account(guardian_signatures_signer.pubkey())
+        .await
+        .unwrap()
+        .unwrap();
+
+    let account_data = &guardian_signatures_info.data;
+    let (expected_length, expected_guardian_signatures_data) =
+        common::generate_expected_guardian_signatures_info(
+            &payer_signer.pubkey(),
+            decoded_vaa.total_signatures,
+            decoded_vaa.guardian_set_index,
+            decoded_vaa.guardian_signatures,
+        );
+    assert_eq!(account_data.len(), expected_length);
+    assert_eq!(
+        deserialize_with_discriminator::<GuardianSignatures>(&account_data[..]).unwrap(),
+        expected_guardian_signatures_data
+    );
+}
+
+#[tokio::test]
+async fn test_post_signatures_lamports_already_in_guardian_signatures() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let guardian_signatures_signer = Keypair::new();
+
+    // Send lamports to the guardian signatures account.
+    let recent_blockhash = common::transfer_lamports(
+        &banks_client,
+        recent_blockhash,
+        &payer_signer,
+        &guardian_signatures_signer.pubkey(),
+        6_960 * 128,
+    )
+    .await;
+
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &decoded_vaa.guardian_signatures,
+        &payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction.clone())
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_ok());
+    assert_eq!(
+        out.simulation_details.unwrap().units_consumed,
+        // 17_267
+        5_654
+    );
+
+    banks_client.process_transaction(transaction).await.unwrap();
+
+    // Check guardian signatures account after processing the transaction.
+    let guardian_signatures_info = banks_client
+        .get_account(guardian_signatures_signer.pubkey())
+        .await
+        .unwrap()
+        .unwrap();
+
+    let account_data = &guardian_signatures_info.data;
+    let (expected_length, expected_guardian_signatures_data) =
+        common::generate_expected_guardian_signatures_info(
+            &payer_signer.pubkey(),
+            decoded_vaa.total_signatures,
+            decoded_vaa.guardian_set_index,
+            decoded_vaa.guardian_signatures,
+        );
+    assert_eq!(account_data.len(), expected_length);
+    assert_eq!(
+        deserialize_with_discriminator::<GuardianSignatures>(&account_data[..]).unwrap(),
+        expected_guardian_signatures_data
+    );
+}
+
+#[tokio::test]
+async fn test_post_signatures_separate_transactions() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    // Split up signatures
+    let guardian_signatures_1 = &decoded_vaa.guardian_signatures[..12];
+    let guardian_signatures_2 = &decoded_vaa.guardian_signatures[12..];
+    assert_eq!(guardian_signatures_2.len(), 1);
+
+    let guardian_signatures_signer = Keypair::new();
+
+    // First transaction.
+
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        guardian_signatures_1,
+        &payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction.clone())
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_ok());
+    assert_eq!(
+        out.simulation_details.unwrap().units_consumed,
+        // 12_828
+        3_343
+    );
+
+    banks_client.process_transaction(transaction).await.unwrap();
+
+    let guardian_signatures_info_data = banks_client
+        .get_account(guardian_signatures_signer.pubkey())
+        .await
+        .unwrap()
+        .unwrap()
+        .data;
+
+    let mut expected_guardian_signatures = guardian_signatures_1.to_vec();
+
+    // Check guardian signatures account after processing the transaction.
+    let (expected_length, expected_guardian_signatures_data) =
+        common::generate_expected_guardian_signatures_info(
+            &payer_signer.pubkey(),
+            decoded_vaa.total_signatures,
+            decoded_vaa.guardian_set_index,
+            expected_guardian_signatures.clone(),
+        );
+    assert_eq!(guardian_signatures_info_data.len(), expected_length);
+    assert_eq!(
+        deserialize_with_discriminator::<GuardianSignatures>(&guardian_signatures_info_data[..])
+            .unwrap(),
+        expected_guardian_signatures_data
+    );
+
+    // Second transaction.
+    let recent_blockhash = banks_client.get_latest_blockhash().await.unwrap();
+
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        guardian_signatures_2,
+        &payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction.clone())
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_ok());
+    assert_eq!(
+        out.simulation_details.unwrap().units_consumed,
+        // 7_628
+        1_007
+    );
+
+    banks_client.process_transaction(transaction).await.unwrap();
+
+    let guardian_signatures_info_data = banks_client
+        .get_account(guardian_signatures_signer.pubkey())
+        .await
+        .unwrap()
+        .unwrap()
+        .data;
+
+    expected_guardian_signatures.extend_from_slice(guardian_signatures_2);
+
+    // Check guardian signatures account after processing the transaction.
+    let (expected_length, expected_guardian_signatures_data) =
+        common::generate_expected_guardian_signatures_info(
+            &payer_signer.pubkey(),
+            decoded_vaa.total_signatures,
+            decoded_vaa.guardian_set_index,
+            expected_guardian_signatures.clone(),
+        );
+    assert_eq!(guardian_signatures_info_data.len(), expected_length);
+    assert_eq!(
+        deserialize_with_discriminator::<GuardianSignatures>(&guardian_signatures_info_data[..])
+            .unwrap(),
+        expected_guardian_signatures_data
+    );
+}
+
+#[tokio::test]
+async fn test_cannot_post_signatures_refund_recipient_mismatch() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    // Split up signatures
+    let guardian_signatures_1 = &decoded_vaa.guardian_signatures[..12];
+    let guardian_signatures_2 = &decoded_vaa.guardian_signatures[12..];
+    assert_eq!(guardian_signatures_2.len(), 1);
+
+    let guardian_signatures_signer = Keypair::new();
+
+    // First transaction.
+
+    let (_, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        guardian_signatures_1,
+        recent_blockhash,
+        Some(&guardian_signatures_signer),
+    )
+    .await;
+
+    let another_payer_signer = Keypair::new();
+
+    // Send some lamports to the another payer.
+    let recent_blockhash = common::transfer_lamports(
+        &banks_client,
+        recent_blockhash,
+        &payer_signer,
+        &another_payer_signer.pubkey(),
+        2_000_000_000,
+    )
+    .await;
+
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        guardian_signatures_2,
+        &another_payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError thrown in programs/verify-vaa/src/instructions/post_signatures.rs:42. Error Code: WriteAuthorityMismatch. Error Number: 6001. Error Message: WriteAuthorityMismatch.";
+    let err_msg = "Program log: Payer (account #1) must match refund recipient";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_post_signatures_total_signatures_mismatch() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    // Split up signatures
+    let guardian_signatures_1 = &decoded_vaa.guardian_signatures[..12];
+    let guardian_signatures_2 = &decoded_vaa.guardian_signatures[12..];
+    assert_eq!(guardian_signatures_2.len(), 1);
+
+    let guardian_signatures_signer = Keypair::new();
+
+    // First transaction.
+
+    let (_, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        guardian_signatures_1,
+        recent_blockhash,
+        Some(&guardian_signatures_signer),
+    )
+    .await;
+
+    // Second transaction.
+
+    let another_payer_signer = Keypair::new();
+
+    // Send some lamports to the another payer.
+    let recent_blockhash = common::transfer_lamports(
+        &banks_client,
+        recent_blockhash,
+        &payer_signer,
+        &another_payer_signer.pubkey(),
+        2_000_000_000,
+    )
+    .await;
+
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures - 1,
+        guardian_signatures_2,
+        &payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    let err_msg = "Program log: Total signatures mismatch";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_post_signatures_zero_signatures() {
+    let (banks_client, payer_signer, recent_blockhash, _) = common::start_test(VAA).await;
+
+    let guardian_signatures_signer = Keypair::new();
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        0,   // guardian set index
+        1,   // total signatures
+        &[], // guardian signatures
+        &payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError thrown in programs/verify-vaa/src/instructions/post_signatures.rs:24. Error Code: EmptyGuardianSignatures. Error Number: 6000. Error Message: EmptyGuardianSignatures.";
+    let err_msg = "Program log: Guardian signatures must not be empty";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()));
+}
+
+#[tokio::test]
+async fn test_cannot_post_signatures_total_signatures_too_large() {
+    let (banks_client, payer_signer, recent_blockhash, _) = common::start_test(VAA).await;
+
+    let guardian_signatures_signer = Keypair::new();
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        0,          // guardian set index
+        155,        // total signatures
+        &[[0; 66]], // guardian signatures
+        &payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    let err_msg = "Account data size realloc limited to 10240 in inner instructions";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_post_signatures_payer_is_guardian_signatures() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let guardian_signatures_signer = Keypair::new();
+
+    // First transfer lamports to guardian signatures signer.
+    let recent_blockhash = common::transfer_lamports(
+        &banks_client,
+        recent_blockhash,
+        &payer_signer,
+        &guardian_signatures_signer.pubkey(),
+        2_000_000_000,
+    )
+    .await;
+
+    let guardian_signatures = guardian_signatures_signer.pubkey();
+
+    let post_signatures_ix = verify_vaa::PostSignatures {
+        program_id: &VERIFY_VAA_SHIM_PROGRAM_ID,
+        accounts: verify_vaa::PostSignaturesAccounts {
+            payer: &guardian_signatures,
+            guardian_signatures: &guardian_signatures,
+        },
+        data: verify_vaa::PostSignaturesData::new(
+            decoded_vaa.guardian_set_index,
+            decoded_vaa.total_signatures,
+            &decoded_vaa.guardian_signatures,
+        ),
+    }
+    .instruction();
+
+    // Adding compute budget instructions to ensure all instructions fit into
+    // one transaction.
+    //
+    // NOTE: Invoking the compute budget costs in total 300 CU.
+    let message = Message::try_compile(
+        &guardian_signatures,
+        &[
+            post_signatures_ix,
+            ComputeBudgetInstruction::set_compute_unit_price(69),
+            ComputeBudgetInstruction::set_compute_unit_limit(8_000),
+        ],
+        &[],
+        recent_blockhash,
+    )
+    .unwrap();
+
+    let transaction = VersionedTransaction::try_new(
+        VersionedMessage::V0(message),
+        &[&guardian_signatures_signer],
+    )
+    .unwrap();
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError thrown in programs/verify-vaa/src/instructions/post_signatures.rs:4. Error Code: TryingToInitPayerAsProgramAccount. Error Number: 4101. Error Message: You cannot/should not initialize the payer account as a program account.";
+    let err_msg =
+        "Program log: Guardian signatures (account #2) cannot be initialized as payer (account #1)";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_post_signatures_more_signatures_than_total() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let guardian_signatures_signer = Keypair::new();
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        1,
+        decoded_vaa.total_signatures - 1,
+        &decoded_vaa.guardian_signatures,
+        &payer_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError caused by account: guardian_signatures. Error Code: AccountDidNotSerialize. Error Number: 3004. Error Message: Failed to serialize the account.";
+    let err_msg = "Program log: Too many input guardian signatures";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_post_signatures_refund_recipient_is_not_signer() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    // Split up signatures
+    let guardian_signatures_1 = &decoded_vaa.guardian_signatures[..12];
+    let guardian_signatures_2 = &decoded_vaa.guardian_signatures[12..];
+    assert_eq!(guardian_signatures_2.len(), 1);
+
+    let guardian_signatures_signer = Keypair::new();
+
+    // First transaction.
+
+    let (_, refund_recipient_signer, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        guardian_signatures_1,
+        recent_blockhash,
+        Some(&guardian_signatures_signer),
+    )
+    .await;
+
+    let transaction = common::post_signatures::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        guardian_signatures_2,
+        &refund_recipient_signer,
+        &guardian_signatures_signer,
+        recent_blockhash,
+        Some(common::post_signatures::AdditionalTestInputs {
+            payer_is_signer: Some(false),
+        }),
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    let err_msg = "Program log: Payer (account #1) must be a signer";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+// Close signatures.
+
+#[tokio::test]
+async fn test_close_signatures() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let (guardian_signatures, refund_recipient_signer, recent_blockhash) =
+        common::send_post_signatures_transaction(
+            &banks_client,
+            &payer_signer,
+            decoded_vaa.guardian_set_index,
+            decoded_vaa.total_signatures,
+            &decoded_vaa.guardian_signatures,
+            recent_blockhash,
+            None, // guardian_signatures_signer
+        )
+        .await;
+
+    let transaction = common::close_signatures::set_up_transaction(
+        &payer_signer,
+        &refund_recipient_signer,
+        &guardian_signatures,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction.clone())
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_ok());
+    assert_eq!(
+        out.simulation_details.unwrap().units_consumed,
+        // 5_165
+        715
+    );
+
+    banks_client.process_transaction(transaction).await.unwrap();
+
+    // Check that the guardian signatures account does not exist anymore.
+    //
+    // NOTE: Because there are only two accounts in this instruction, we can
+    // assume the refund recipient received all of the lamports from the
+    // guardian signatures account.
+    assert!(banks_client
+        .get_account(guardian_signatures)
+        .await
+        .unwrap()
+        .is_none());
+}
+
+#[tokio::test]
+async fn test_cannot_close_signatures_refund_recipient_mismatch() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &decoded_vaa.guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let another_refund_recipient_signer = Keypair::new();
+
+    // Send some lamports to the another payer.
+    let recent_blockhash = common::transfer_lamports(
+        &banks_client,
+        recent_blockhash,
+        &payer_signer,
+        &another_refund_recipient_signer.pubkey(),
+        2_000_000_000,
+    )
+    .await;
+
+    let transaction = common::close_signatures::set_up_transaction(
+        &payer_signer,
+        &another_refund_recipient_signer,
+        &guardian_signatures,
+        recent_blockhash,
+        None, // additional inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError caused by account: guardian_signatures. Error Code: ConstraintHasOne. Error Number: 2001. Error Message: A has one constraint was violated.";
+    let err_msg = "Program log: Refund recipient (account #2) mismatch";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()));
+}
+
+#[tokio::test]
+async fn test_cannot_close_signatures_refund_recipient_is_not_signer() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let (guardian_signatures, refund_recipient_signer, recent_blockhash) =
+        common::send_post_signatures_transaction(
+            &banks_client,
+            &payer_signer,
+            decoded_vaa.guardian_set_index,
+            decoded_vaa.total_signatures,
+            &decoded_vaa.guardian_signatures,
+            recent_blockhash,
+            None, // guardian_signatures_signer
+        )
+        .await;
+
+    let transaction = common::close_signatures::set_up_transaction(
+        &payer_signer,
+        &refund_recipient_signer,
+        &guardian_signatures,
+        recent_blockhash,
+        Some(common::close_signatures::AdditionalTestInputs {
+            refund_recipient_is_signer: Some(false),
+        }),
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction.clone())
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    let err_msg = "Program log: Refund recipient (account #2) must be a signer";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+// Verify hash.
+
+#[tokio::test]
+async fn test_verify_hash() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &decoded_vaa.guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+    let digest = compute_keccak_digest(message_hash, None);
+
+    let (transaction, bump_costs) = common::verify_hash::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        &guardian_signatures,
+        digest,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_ok());
+    assert_eq!(
+        out.simulation_details.unwrap().units_consumed - bump_costs.guardian_set,
+        // 342_276
+        334_558
+    );
+}
+
+#[tokio::test]
+async fn test_cannot_verify_hash_invalid_guardian_set() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &decoded_vaa.guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+    let digest = compute_keccak_digest(message_hash, None);
+
+    let (transaction, _) = common::verify_hash::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        &guardian_signatures,
+        digest,
+        recent_blockhash,
+        Some(common::verify_hash::AdditionalTestInputs {
+            invalid_guardian_set: Some(Pubkey::new_unique()),
+        }),
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError caused by account: guardian_set. Error Code: AccountNotInitialized. Error Number: 3012. Error Message: The program expected this account to be already initialized.";
+    let err_msg = "Program log: Guardian set (account #1) seeds constraint violated";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_verify_hash_invalid_guardian_set_index() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &decoded_vaa.guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+    let digest = compute_keccak_digest(message_hash, None);
+
+    let (transaction, _) = common::verify_hash::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index - 1,
+        &guardian_signatures,
+        digest,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError caused by account: guardian_set. Error Code: ConstraintSeeds. Error Number: 2006. Error Message: A seeds constraint was violated.";
+    let err_msg = "Program log: Guardian set (account #1) address creation failed";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_verify_hash_expired_guardian_set() {
+    const VAA_EXPIRED_SET: &str = "AQAAAAMNAnTlXVVK+fjEIYfgQggRENn0/f++V7dCtNCHnrSj05X8ctnm0x1Fzn8hODqvC44/eWTUso+tUPQpHjCgEP0g1xMBA/S7K8O34D/AkEkYLrQbgKFDq6W3uDycJ90B75GmLaniGrmPxDBX1gog6ISTqrDIB9OBL1e3fVGqaMOTCrrPS6gABFrsiVNLrIsU2hPJHWF0c/CT2+DWS+TAq05lfSVvVExfLjv6WejfY3PdN+dLbCKE0JpD14BiNbcYeRXGJxPE76sBBs8nYy7KLd5McKmEGvhN2zBdlHcNByoJf8ZK1WXifnUeWhHnnWDr05wWigqw657ZjRytNVzgFA+MruATReXsxM0BB+NIIlTBvvqF0WZ5FNpT53nzmcpypZYDIU6041CcY2FqU5ntQdEsjZDiu1Q4W0Sjjcfg3xfApMURec+5Q3IP0isBCTjCgO4J/+tgfxazGf8WC2kMRn21Dh1E1u6QG52wul7wIhCQTejShVbWG1U4f8y5mk3wA/X4qHVK3tizoozeJHsACjGD+X3upTjNNQZFfqMxYrnDxRLK6UCGSTb++1AlWHwNQGBfbPf/upVZBO1qIHAiEGQllkyTg4aEoinp38bSN9oADDcEbPbb9R+g1VTWOg8VS8ucHEX4ojahNghH/n6r9tOKfMwfprBtnasT5y1NjSDO7uvt9WTDMTgUCtDdtluz4ToBDQd4w/uKG0pQziUsWMZZeeNTgrdTP1LBJ/6eTWMmMV7QGHep0XwCsEhXOmIVgrDcny6+g8GOpS7bV5Y9d5WQBpYAD2xbXI9zaXN/mRqsk6dF87dzpYHqf4lGPXkv7sKACUZyI71qwyTfjua0XAuNpsUPWLT1pXEa5iIBpgzR8A81M04AEHOKvkaYNNmg6WXsq8noXD4iA3Q8ibC7r4mOkPsOhqPmYX1SzR4g3jLRSM/Ck4BTGo7hVXKv37zMcrAddEErqiIBESYgyM+I7XUdNNWGZ4lWxSnc5WF65DiHH1U8nRsZ5RiPJFic1xp8Zg/5uil3sRUKcXti6M3coO4N9x4W++PxV2MBEqUxI7EX5evuk6uHyoh8VVYYvVAo1XWt8Yx8sDlDqNLOWJaxpGzq0WbB8EUPpRlDzG8YgZVyXl49ZEFj/vMxgW0BZhTIZwAAAAAAAgAAAAAAAAAAAAAAAO4MzphZqQ+m4rdRggW+9MeZtxbXAAAAAAAAAAABAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAlQL5AASAAAAAAAAAAAAAAAAtTNofvd0WQkzaMQ+lfjfHCtaH3oAAAAAAAAAAAAAAACVPJV2dXAP9gZNOrN+ppSrFdP9wgAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAC0W8AD2fGpZ2cDa08dvhobF0SoEEHFqAoL7PN5HTYzDEM0lCf6DptD8OMI950PhoAt0aoIpOjLblmX6I3DGC27gAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAB";
+
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) =
+        common::start_test(VAA_EXPIRED_SET).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+    assert_eq!(decoded_vaa.guardian_set_index, 3);
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &decoded_vaa.guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+    let digest = compute_keccak_digest(message_hash, None);
+
+    let (transaction, _) = common::verify_hash::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        &guardian_signatures,
+        digest,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError thrown in programs/verify-vaa/src/instructions/verify_hash.rs:44. Error Code: GuardianSetExpired. Error Number: 6002. Error Message: GuardianSetExpired.";
+    let err_msg = "Program log: Guardian set (account #1) is expired";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_verify_hash_no_quorum() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let mut insufficient_guardian_signatures = decoded_vaa.guardian_signatures.clone();
+    insufficient_guardian_signatures.pop();
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &insufficient_guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+    let digest = compute_keccak_digest(message_hash, None);
+
+    let (transaction, _) = common::verify_hash::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        &guardian_signatures,
+        digest,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError thrown in programs/verify-vaa/src/instructions/verify_hash.rs:56. Error Code: NoQuorum. Error Number: 6003. Error Message: NoQuorum.";
+    let err_msg = "Program log: Guardian signatures (account #2) fails to meet quorum";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_verify_hash_non_increasing_guardian_index() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let mut non_increasing_guardian_signatures = decoded_vaa.guardian_signatures.clone();
+    non_increasing_guardian_signatures.rotate_right(1);
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &non_increasing_guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+    let digest = compute_keccak_digest(message_hash, None);
+
+    let (transaction, _) = common::verify_hash::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        &guardian_signatures,
+        digest,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError thrown in programs/verify-vaa/src/instructions/verify_hash.rs:69. Error Code: InvalidGuardianIndexNonIncreasing. Error Number: 6005. Error Message: InvalidGuardianIndexNonIncreasing.";
+    let err_msg = "Program log: Guardian signatures (account #2) has non-increasing guardian index";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_verify_hash_guardian_index_out_of_range() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let mut out_of_range_guardian_signatures = decoded_vaa.guardian_signatures.clone();
+    out_of_range_guardian_signatures[0][0] = 19;
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &out_of_range_guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+    let digest = compute_keccak_digest(message_hash, None);
+
+    let (transaction, _) = common::verify_hash::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        &guardian_signatures,
+        digest,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError thrown in programs/verify-vaa/src/instructions/verify_hash.rs:78. Error Code: InvalidGuardianIndexOutOfRange. Error Number: 6006. Error Message: InvalidGuardianIndexOutOfRange.";
+    let err_msg = "Program log: Guardian signatures (account #2) guardian index out of range";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}
+
+#[tokio::test]
+async fn test_cannot_verify_hash_invalid_signature() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    for i in 0..13 {
+        let mut out_of_range_guardian_signatures = decoded_vaa.guardian_signatures.clone();
+        out_of_range_guardian_signatures[i][65] = 255;
+
+        let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+            &banks_client,
+            &payer_signer,
+            decoded_vaa.guardian_set_index,
+            decoded_vaa.total_signatures,
+            &out_of_range_guardian_signatures,
+            recent_blockhash,
+            None, // guardian_signatures_signer
+        )
+        .await;
+
+        let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+        let digest = compute_keccak_digest(message_hash, None);
+
+        let (transaction, _) = common::verify_hash::set_up_transaction(
+            &payer_signer,
+            decoded_vaa.guardian_set_index,
+            &guardian_signatures,
+            digest,
+            recent_blockhash,
+            None, // additional_inputs
+        );
+
+        let out = banks_client
+            .simulate_transaction(transaction)
+            .await
+            .unwrap();
+        assert!(
+            out.result.unwrap().is_err(),
+            "Unexpected success at index {}",
+            i
+        );
+
+        // let err_msg = "Program log: AnchorError occurred. Error Code: InvalidSignature. Error Number: 6004. Error Message: InvalidSignature.";
+        let err_msg = format!("Program log: Guardian signature index {} is invalid", i);
+        assert!(
+            out.simulation_details.unwrap().logs.contains(&err_msg),
+            "Unexpected error message at index {}",
+            i
+        );
+    }
+}
+
+#[tokio::test]
+async fn test_cannot_verify_hash_invalid_guardian_recovery() {
+    let (banks_client, payer_signer, recent_blockhash, decoded_vaa) = common::start_test(VAA).await;
+    assert_eq!(decoded_vaa.total_signatures, 13);
+
+    let mut out_of_range_guardian_signatures = decoded_vaa.guardian_signatures.clone();
+
+    let mismatched_signature: [u8; 65] = out_of_range_guardian_signatures[11][1..]
+        .try_into()
+        .unwrap();
+    out_of_range_guardian_signatures[12][1..].copy_from_slice(&mismatched_signature);
+
+    let (guardian_signatures, _, recent_blockhash) = common::send_post_signatures_transaction(
+        &banks_client,
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        decoded_vaa.total_signatures,
+        &out_of_range_guardian_signatures,
+        recent_blockhash,
+        None, // guardian_signatures_signer
+    )
+    .await;
+
+    let message_hash = solana_sdk::keccak::hash(&decoded_vaa.body);
+    let digest = compute_keccak_digest(message_hash, None);
+
+    let (transaction, _) = common::verify_hash::set_up_transaction(
+        &payer_signer,
+        decoded_vaa.guardian_set_index,
+        &guardian_signatures,
+        digest,
+        recent_blockhash,
+        None, // additional_inputs
+    );
+
+    let out = banks_client
+        .simulate_transaction(transaction)
+        .await
+        .unwrap();
+    assert!(out.result.unwrap().is_err());
+
+    // let err_msg = "Program log: AnchorError thrown in programs/verify-vaa/src/instructions/verify_hash.rs:127. Error Code: InvalidGuardianKeyRecovery. Error Number: 6007. Error Message: InvalidGuardianKeyRecovery.";
+    let err_msg = "Program log: Guardian signature index 12 does not recover guardian 18 pubkey";
+    assert!(out
+        .simulation_details
+        .unwrap()
+        .logs
+        .contains(&err_msg.to_string()))
+}

+ 3 - 0
svm/wormhole-core-shims/rust-toolchain.toml

@@ -0,0 +1,3 @@
+[toolchain]
+channel = "1.83.0"
+components = [ "clippy", "rustc", "rustfmt" ]

Alguns ficheiros não foram mostrados porque muitos ficheiros mudaram neste diff