From 1c1c0817026b409bcc1f539da5d1dd927bc59d0d Mon Sep 17 00:00:00 2001 From: bishalbikram Date: Thu, 19 Dec 2024 08:39:53 +0545 Subject: [PATCH] feat: solana intent contract --- contracts/solana/.gitignore | 7 + contracts/solana/.prettierignore | 7 + contracts/solana/Anchor.toml | 18 + contracts/solana/Cargo.lock | 2700 +++++++++++++++++ contracts/solana/Cargo.toml | 16 + contracts/solana/package.json | 23 + contracts/solana/programs/intent/Cargo.toml | 27 + contracts/solana/programs/intent/Xargo.toml | 2 + .../solana/programs/intent/src/connection.rs | 33 + .../solana/programs/intent/src/constants.rs | 5 + .../solana/programs/intent/src/constatns.rs | 5 + contracts/solana/programs/intent/src/error.rs | 88 + contracts/solana/programs/intent/src/event.rs | 66 + .../solana/programs/intent/src/helpers.rs | 87 + .../intent/src/instructions/cancel.rs | 137 + .../intent/src/instructions/config.rs | 142 + .../programs/intent/src/instructions/fill.rs | 322 ++ .../programs/intent/src/instructions/mod.rs | 13 + .../intent/src/instructions/query_accounts.rs | 376 +++ .../intent/src/instructions/recv_message.rs | 146 + .../programs/intent/src/instructions/swap.rs | 134 + contracts/solana/programs/intent/src/lib.rs | 137 + contracts/solana/programs/intent/src/state.rs | 133 + .../solana/programs/intent/src/types/misc.rs | 89 + .../solana/programs/intent/src/types/mod.rs | 12 + .../programs/intent/src/types/order_cancel.rs | 62 + .../programs/intent/src/types/order_fill.rs | 104 + .../intent/src/types/order_message.rs | 120 + .../programs/intent/src/types/swap_order.rs | 230 ++ contracts/solana/scripts/setup.ts | 224 ++ contracts/solana/scripts/types/index.ts | 4 + .../solana/scripts/types/order-cancel.ts | 18 + contracts/solana/scripts/types/order-fill.ts | 22 + .../solana/scripts/types/order_message.ts | 25 + contracts/solana/scripts/types/swap-order.ts | 79 + contracts/solana/scripts/utils/index.ts | 42 + contracts/solana/scripts/utils/transaction.ts | 128 + contracts/solana/tests/begin.ts | 201 ++ contracts/solana/tests/cancel.ts | 167 + contracts/solana/tests/fill.ts | 575 ++++ contracts/solana/tests/recv-message.ts | 241 ++ contracts/solana/tests/setup.ts | 95 + contracts/solana/tests/swap.ts | 287 ++ contracts/solana/tsconfig.json | 11 + contracts/solana/yarn.lock | 1360 +++++++++ 45 files changed, 8720 insertions(+) create mode 100644 contracts/solana/.gitignore create mode 100644 contracts/solana/.prettierignore create mode 100644 contracts/solana/Anchor.toml create mode 100644 contracts/solana/Cargo.lock create mode 100644 contracts/solana/Cargo.toml create mode 100644 contracts/solana/package.json create mode 100644 contracts/solana/programs/intent/Cargo.toml create mode 100644 contracts/solana/programs/intent/Xargo.toml create mode 100644 contracts/solana/programs/intent/src/connection.rs create mode 100644 contracts/solana/programs/intent/src/constants.rs create mode 100644 contracts/solana/programs/intent/src/constatns.rs create mode 100644 contracts/solana/programs/intent/src/error.rs create mode 100644 contracts/solana/programs/intent/src/event.rs create mode 100644 contracts/solana/programs/intent/src/helpers.rs create mode 100644 contracts/solana/programs/intent/src/instructions/cancel.rs create mode 100644 contracts/solana/programs/intent/src/instructions/config.rs create mode 100644 contracts/solana/programs/intent/src/instructions/fill.rs create mode 100644 contracts/solana/programs/intent/src/instructions/mod.rs create mode 100644 contracts/solana/programs/intent/src/instructions/query_accounts.rs create mode 100644 contracts/solana/programs/intent/src/instructions/recv_message.rs create mode 100644 contracts/solana/programs/intent/src/instructions/swap.rs create mode 100644 contracts/solana/programs/intent/src/lib.rs create mode 100644 contracts/solana/programs/intent/src/state.rs create mode 100644 contracts/solana/programs/intent/src/types/misc.rs create mode 100644 contracts/solana/programs/intent/src/types/mod.rs create mode 100644 contracts/solana/programs/intent/src/types/order_cancel.rs create mode 100644 contracts/solana/programs/intent/src/types/order_fill.rs create mode 100644 contracts/solana/programs/intent/src/types/order_message.rs create mode 100644 contracts/solana/programs/intent/src/types/swap_order.rs create mode 100644 contracts/solana/scripts/setup.ts create mode 100644 contracts/solana/scripts/types/index.ts create mode 100644 contracts/solana/scripts/types/order-cancel.ts create mode 100644 contracts/solana/scripts/types/order-fill.ts create mode 100644 contracts/solana/scripts/types/order_message.ts create mode 100644 contracts/solana/scripts/types/swap-order.ts create mode 100644 contracts/solana/scripts/utils/index.ts create mode 100644 contracts/solana/scripts/utils/transaction.ts create mode 100644 contracts/solana/tests/begin.ts create mode 100644 contracts/solana/tests/cancel.ts create mode 100644 contracts/solana/tests/fill.ts create mode 100644 contracts/solana/tests/recv-message.ts create mode 100644 contracts/solana/tests/setup.ts create mode 100644 contracts/solana/tests/swap.ts create mode 100644 contracts/solana/tsconfig.json create mode 100644 contracts/solana/yarn.lock diff --git a/contracts/solana/.gitignore b/contracts/solana/.gitignore new file mode 100644 index 000000000..2e0446b07 --- /dev/null +++ b/contracts/solana/.gitignore @@ -0,0 +1,7 @@ +.anchor +.DS_Store +target +**/*.rs.bk +node_modules +test-ledger +.yarn diff --git a/contracts/solana/.prettierignore b/contracts/solana/.prettierignore new file mode 100644 index 000000000..414258343 --- /dev/null +++ b/contracts/solana/.prettierignore @@ -0,0 +1,7 @@ +.anchor +.DS_Store +target +node_modules +dist +build +test-ledger diff --git a/contracts/solana/Anchor.toml b/contracts/solana/Anchor.toml new file mode 100644 index 000000000..fb0f67767 --- /dev/null +++ b/contracts/solana/Anchor.toml @@ -0,0 +1,18 @@ +[toolchain] + +[features] +resolution = false +skip-lint = false + +[programs.localnet] +intent = "4nTmHeE5qdFuKoqeakERkMn3iFFV2X76zkvdDA26LPCG" + +[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/**/*.ts" diff --git a/contracts/solana/Cargo.lock b/contracts/solana/Cargo.lock new file mode 100644 index 000000000..86e8efcc8 --- /dev/null +++ b/contracts/solana/Cargo.lock @@ -0,0 +1,2700 @@ +# 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", + "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.94" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c1fd03a028ef38ba2276dce7e33fcd6369c158a1bca17946c4b1b701891c1ff7" + +[[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.90", +] + +[[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.20.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8b37c88a63ffd85d15b406896cc343916d7cf57838a847b3a6f2ca5d39a5695a" +dependencies = [ + "bytemuck_derive", +] + +[[package]] +name = "bytemuck_derive" +version = "1.8.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bcfcc3cd946cb52f0bbfdbbcfa2f4e24f75ebb6c0e1002f7c25904fada18b9ec" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.90", +] + +[[package]] +name = "byteorder" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" + +[[package]] +name = "bytes" +version = "1.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "325918d6fe32f23b19878fe4b34794ae41fc19ddbe53b10571a4874d44ffd39b" + +[[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.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f34d93e62b03caf570cccc334cbc6c2fceca82f39211051345108adcba3eebdc" +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.38" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a21f936df1771bf62b77f047b726c4625ff2e8aa607c01ec06e5a05bd8463401" +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.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "613f8cc01fe9cf1a3eb3d7f488fd2fa8388403e97039e2f73692932e291a770d" +dependencies = [ + "crossbeam-epoch", + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-epoch" +version = "0.9.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b82ac4a3c2ca9c3460964f020e1402edd5753411d7737aa39c3714ad1b5420e" +dependencies = [ + "crossbeam-utils", +] + +[[package]] +name = "crossbeam-utils" +version = "0.8.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "22ec99545bb0ed0ea7bb9b8e1e9122ea386ff8a48c0922e43f36d45ab09e0e80" + +[[package]] +name = "crunchy" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7a81dae078cea95a014a339291cec439d2f232ebe854a9d672b796c6afafa9b7" + +[[package]] +name = "crypto-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.90", +] + +[[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.90", +] + +[[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 = "hex" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" + +[[package]] +name = "hmac" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "126888268dcc288495a26bf004b38c5fdbb31682f992c84ceb046a1f0fe38840" +dependencies = [ + "crypto-mac", + "digest 0.9.0", +] + +[[package]] +name = "hmac" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e" +dependencies = [ + "digest 0.10.7", +] + +[[package]] +name = "hmac-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 = "intent" +version = "0.1.0" +dependencies = [ + "anchor-lang", + "anchor-spl", + "borsh 1.5.3", + "bytemuck", + "hex", + "rlp", +] + +[[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.74" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a865e038f7f6ed956f788f0d7d60c541fff74c7bd74272c5d4cf15c63743e705" +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.167" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09d6582e104315a817dff97f75133544b2e094ee22447d2acf4a74e189ba06fc" + +[[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 = "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.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed3955f1a9c7c0c15e092f9c887db08b1fc683305fdf6eb6684f22555355e202" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.90", +] + +[[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.90", +] + +[[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.90", +] + +[[package]] +name = "quote" +version = "1.0.37" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b5b9d34b8991d19d98081b46eacdd8eb58c6f2b201139f7c5f643cc155a633af" +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.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b6dfecf2c74bce2466cabf93f6664d6998a69eb21e39f4207930065b27b771f" +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 = "rlp" +version = "0.1.0" +source = "git+http://github.com/icon-project/xcall-multi?branch=main#c72a55083bdddda033e582ab36de338cc62d2529" +dependencies = [ + "bytes", + "hex", + "rustc-hex", + "serde", +] + +[[package]] +name = "rustc-hash" +version = "1.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2" + +[[package]] +name = "rustc-hex" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3e75f6a532d0fd9f7f13144f392b6ad56a32696bfcd9c78f797f16bbb6f072d6" + +[[package]] +name = "rustc_version" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cfcb3a22ef46e85b45de6ee7e79d063319ebb6594faafcf1c225ea92ab6e9b92" +dependencies = [ + "semver", +] + +[[package]] +name = "rustversion" +version = "1.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e819f2bc632f285be6d7cd36e25940d45b2391dd6d9b939e79de557f7014248" + +[[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.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61697e0a1c7e512e84a621326239844a24d8207b4669b41bc18b32ea5cbf988b" + +[[package]] +name = "serde" +version = "1.0.215" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6513c1ad0b11a9376da888e3e0baa0077f1aed55c17f50e7b2397136129fb88f" +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.215" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ad1e866f866923f252f05c889987993144fb74e722403468a4ebd70c3cd756c0" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.90", +] + +[[package]] +name = "serde_json" +version = "1.0.133" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c7fceb2473b9166b2294ef05efcb65a3db80803f0b03ef86a5fc88a2b85ee377" +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.90", +] + +[[package]] +name = "sha2" +version = "0.9.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4d58a1e1bf39749807d89cf2d98ac2dfa0ff1cb3faa38fbb64dd88ac8013d800" +dependencies = [ + "block-buffer 0.9.0", + "cfg-if", + "cpufeatures", + "digest 0.9.0", + "opaque-debug", +] + +[[package]] +name = "sha2" +version = "0.10.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest 0.10.7", +] + +[[package]] +name = "sha3" +version = "0.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.90", +] + +[[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", + "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", + "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.90", +] + +[[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", + "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", + "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.90", +] + +[[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.90", + "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", + "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.90", +] + +[[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", + "num-traits", + "num_enum", + "solana-program", + "thiserror", +] + +[[package]] +name = "spl-token-2022" +version = "3.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b01d1b2851964e257187c0bca43a0de38d0af59192479ca01ac3e2b58b1bd95a" +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", +] + +[[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.90" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "919d3b74a5dd0ccd15aeb8f93e7006bd9e14c295087c9896a110f490752bcf31" +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.90", +] + +[[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.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "445e881f4f6d382d5f27c034e25eb92edd7c784ceab92a0937db7f2e9471b938" +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.97" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d15e63b4482863c109d70a7b8706c1e364eb6ea449b201a76c5b89cedcec2d5c" +dependencies = [ + "cfg-if", + "once_cell", + "wasm-bindgen-macro", +] + +[[package]] +name = "wasm-bindgen-backend" +version = "0.2.97" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8d36ef12e3aaca16ddd3f67922bc63e48e953f126de60bd33ccc0101ef9998cd" +dependencies = [ + "bumpalo", + "log", + "once_cell", + "proc-macro2", + "quote", + "syn 2.0.90", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.97" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "705440e08b42d3e4b36de7d66c944be628d579796b8090bfa3471478a2260051" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.97" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "98c9ae5a76e46f4deecd0f0255cc223cfa18dc9b261213b8aa0c7b36f61b3f1d" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.90", + "wasm-bindgen-backend", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.97" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ee99da9c5ba11bd675621338ef6fa52296b76b83305e9b6e5c77d4c286d6d49" + +[[package]] +name = "web-sys" +version = "0.3.74" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a98bc3c33f0fe7e59ad7cd041b89034fa82a7c2d4365ca538dda6cdaf513863c" +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.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "36c1fec1a2bb5866f07c25f68c26e565c4c200aebb96d7e55710c19d3e8ac49b" +dependencies = [ + "memchr", +] + +[[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.90", +] + +[[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.90", +] diff --git a/contracts/solana/Cargo.toml b/contracts/solana/Cargo.toml new file mode 100644 index 000000000..bfaa5b850 --- /dev/null +++ b/contracts/solana/Cargo.toml @@ -0,0 +1,16 @@ +[workspace] +members = [ + "programs/*" +] +resolver = "2" + +[workspace.dependencies] + +[profile.release] +overflow-checks = true +lto = "fat" +codegen-units = 1 +[profile.release.build-override] +opt-level = 3 +incremental = false +codegen-units = 1 diff --git a/contracts/solana/package.json b/contracts/solana/package.json new file mode 100644 index 000000000..650288281 --- /dev/null +++ b/contracts/solana/package.json @@ -0,0 +1,23 @@ +{ + "license": "ISC", + "scripts": { + "lint:fix": "prettier */*.js \"*/**/*{.js,.ts}\" -w", + "lint": "prettier */*.js \"*/**/*{.js,.ts}\" --check" + }, + "dependencies": { + "@coral-xyz/anchor": "^0.30.1", + "@solana/spl-token": "^0.4.9", + "keccak": "^3.0.4", + "rlp": "^3.0.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", + "typescript": "^4.3.5" + } +} diff --git a/contracts/solana/programs/intent/Cargo.toml b/contracts/solana/programs/intent/Cargo.toml new file mode 100644 index 000000000..a909630f6 --- /dev/null +++ b/contracts/solana/programs/intent/Cargo.toml @@ -0,0 +1,27 @@ +[package] +name = "intent" +version = "0.1.0" +description = "Created with Anchor" +edition = "2021" + +[lib] +crate-type = ["cdylib", "lib"] +name = "intent" + +[features] +default = [] +cpi = ["no-entrypoint"] +no-entrypoint = [] +no-idl = [] +no-log-ix-name = [] +idl-build = ["anchor-lang/idl-build", "anchor-spl/idl-build"] + +[dependencies] +anchor-lang = { version = "0.30.1", features = ["init-if-needed"] } +bytemuck = { version = "1.7", features = ["derive"] } +anchor-spl = "0.30.1" +borsh = { version = "1.5.1" } +hex ={ version = "0.4.3", default-features = false } +rlp = { git = "http://github.com/icon-project/xcall-multi", branch = "main" } + + diff --git a/contracts/solana/programs/intent/Xargo.toml b/contracts/solana/programs/intent/Xargo.toml new file mode 100644 index 000000000..475fb71ed --- /dev/null +++ b/contracts/solana/programs/intent/Xargo.toml @@ -0,0 +1,2 @@ +[target.bpfel-unknown-unknown.dependencies.std] +features = [] diff --git a/contracts/solana/programs/intent/src/connection.rs b/contracts/solana/programs/intent/src/connection.rs new file mode 100644 index 000000000..927575ca9 --- /dev/null +++ b/contracts/solana/programs/intent/src/connection.rs @@ -0,0 +1,33 @@ +use anchor_lang::prelude::*; + +use crate::{ + error::IntentError, + event, + state::{Config, Receipt}, +}; + +pub fn send_message<'info>( + config: &mut Account<'info, Config>, + to: String, + msg: Vec, +) -> Result<()> { + let conn_sn = config.increment_conn_sn(); + + event::Message { + targetNetwork: to, + sn: conn_sn, + msg, + }; + + Ok(()) +} + +pub fn recv_message<'info>(receipt: &mut Account<'info, Receipt>, receipt_bump: u8) -> Result<()> { + if receipt.received { + return Err(IntentError::DuplicateMessage.into()); + } + + receipt.new(receipt_bump); + + Ok(()) +} diff --git a/contracts/solana/programs/intent/src/constants.rs b/contracts/solana/programs/intent/src/constants.rs new file mode 100644 index 000000000..98c33e390 --- /dev/null +++ b/contracts/solana/programs/intent/src/constants.rs @@ -0,0 +1,5 @@ +pub const ACCOUNT_DISCRIMINATOR_SIZE: usize = 8; + +pub const NATIVE_ADDRESS: &str = "11111111111111111111111111111111"; + +pub const VAULT_TOKEN_SEED_PREFIX: &'static str = "vault_token"; diff --git a/contracts/solana/programs/intent/src/constatns.rs b/contracts/solana/programs/intent/src/constatns.rs new file mode 100644 index 000000000..98c33e390 --- /dev/null +++ b/contracts/solana/programs/intent/src/constatns.rs @@ -0,0 +1,5 @@ +pub const ACCOUNT_DISCRIMINATOR_SIZE: usize = 8; + +pub const NATIVE_ADDRESS: &str = "11111111111111111111111111111111"; + +pub const VAULT_TOKEN_SEED_PREFIX: &'static str = "vault_token"; diff --git a/contracts/solana/programs/intent/src/error.rs b/contracts/solana/programs/intent/src/error.rs new file mode 100644 index 000000000..9e8304adb --- /dev/null +++ b/contracts/solana/programs/intent/src/error.rs @@ -0,0 +1,88 @@ +use anchor_lang::prelude::error_code; + +#[error_code] +pub enum IntentError { + #[msg("Only Admin")] + OnlyAdmin, + + #[msg("Only Relayer")] + OnlyRelayer, + + #[msg("Only fee handler")] + OnlyFeeHandler, + + #[msg("Network ID is misconfigured")] + NetworkIdMisconfigured, + + #[msg("Invalid network")] + InvalidNetwork, + + #[msg("Order mismatched")] + OrderMismatched, + + #[msg("Rlp decode failed")] + DecodeFailed, + + #[msg("Emitter program ID is not valid")] + InvalidEmitterAddress, + + #[msg("Destination account is not valid")] + InvalidDestinationAccount, + + #[msg("Solver account is not valid")] + InvalidSolverAccount, + + #[msg("Fee handler account is not valid")] + InvalidFeeHandler, + + #[msg("Invalid pubkey")] + InvalidPubkey, + + #[msg("Signer must be a swap creator")] + CreatorMustBeSigner, + + #[msg("Order has been already filled")] + OrderAlreadyFilled, + + #[msg("Duplicate message")] + DuplicateMessage, + + #[msg("Config account is missing")] + ConfigAccountIsMissing, + + #[msg("Mint account mismatch")] + MintAccountMismatch, + + #[msg("Order finished account is missing")] + OrderFinishedAccountIsMissing, + + #[msg("Order account is missing")] + OrderAccountIsMissing, + + #[msg("Vault native account is missing")] + VaultNativeAccountIsMissing, + + #[msg("Vault token account is missing")] + VaultTokenAccountIsMissing, + + #[msg("Signer token account is missing")] + SignerTokenAccountIsMissing, + + #[msg("Solver token account is missing")] + SolverTokenAccountIsMissing, + + #[msg("Creator token account is missing")] + CreatorTokenAccountIsMissing, + + #[msg("Fee handler token account is missing")] + FeeHandlerTokenAccountIsMissing, + + #[msg("Order finished account must not be specified")] + OrderFinishedAccountMustNotBeSpecified, + + #[msg("Vault token account must not be specified")] + VaultTokenAccountMustNotBeSpecified, + + #[msg("Config account must not be specified")] + ConfigAccountMustNotBeSpecified, +} diff --git a/contracts/solana/programs/intent/src/event.rs b/contracts/solana/programs/intent/src/event.rs new file mode 100644 index 000000000..2bdc8e179 --- /dev/null +++ b/contracts/solana/programs/intent/src/event.rs @@ -0,0 +1,66 @@ +#![allow(non_snake_case)] + +use anchor_lang::prelude::*; + +/// Emitted when a new swap intent is created +#[event] +pub struct SwapIntent { + // The ID of the swap order + pub id: u128, + // Address of emitter contract + pub emitter: String, + // The source network ID + pub srcNID: String, + // The destination network ID + pub dstNID: String, + // The address of the creator of the swap order + pub creator: String, + // The address where the swapped tokens will be sent + pub destinationAddress: String, + // The address of the token being swapped + pub token: String, + // The amount of token being swapped + pub amount: u128, + // The token to be received after the swap + pub toToken: String, + // The amount of tokens to be receive after the swap + pub toAmount: u128, + // Additional arbitrary data for the swap + pub data: Vec, +} + +// Emitted when a swap order is filled +#[event] +pub struct OrderFilled { + // The ID of the order being filled + pub id: u128, + // The source network ID of the swap order + pub srcNID: String, +} + +// Emitted when a swap order is cancelled +#[event] +pub struct OrderCancelled { + // The ID of the order being cancelled + pub id: u128, + // The source network ID where the order was created + pub srcNID: String, +} + +// Emitted when a swap order is completed +#[event] +pub struct OrderClosed { + // The ID of the order + pub id: u128, +} + +/// Emitted when a cross-chain message is sent +#[event] +pub struct Message { + // The ID of the target network + pub targetNetwork: String, + // The connection sequence number + pub sn: u128, + // The rlp encoded message being sent to other chain + pub msg: Vec, +} diff --git a/contracts/solana/programs/intent/src/helpers.rs b/contracts/solana/programs/intent/src/helpers.rs new file mode 100644 index 000000000..956cab267 --- /dev/null +++ b/contracts/solana/programs/intent/src/helpers.rs @@ -0,0 +1,87 @@ +use anchor_lang::{ + prelude::*, + solana_program::{hash, keccak, program::invoke, system_instruction}, +}; +use anchor_spl::token; + +use crate::state::Config; + +pub fn transfer_sol<'info>( + from: &AccountInfo<'info>, + to: &AccountInfo<'info>, + amount: u64, + system_program: &AccountInfo<'info>, +) -> Result<()> { + let ix = system_instruction::transfer(&from.key(), &to.key(), amount); + invoke( + &ix, + &[from.to_owned(), to.to_owned(), system_program.to_owned()], + )?; + + Ok(()) +} + +pub fn transfer_sol_signed<'info>( + from: &AccountInfo<'info>, + to: &AccountInfo<'info>, + amount: u64, +) -> Result<()> { + **from.try_borrow_mut_lamports()? -= amount; + **to.try_borrow_mut_lamports()? += amount; + + Ok(()) +} + +pub fn transfer_spl_token<'info>( + from: AccountInfo<'info>, + to: AccountInfo<'info>, + authority: AccountInfo<'info>, + amount: u64, + token_program: AccountInfo<'info>, +) -> Result<()> { + let accounts = token::Transfer { + from, + to, + authority, + }; + token::transfer(CpiContext::new(token_program, accounts), amount) +} + +pub fn transfer_spl_token_signed<'info>( + from: AccountInfo<'info>, + to: AccountInfo<'info>, + authority: AccountInfo<'info>, + amount: u64, + token_program: AccountInfo<'info>, + config: &Account<'info, Config>, +) -> Result<()> { + let accounts = token::Transfer { + from, + to, + authority, + }; + + let seeds = &[Config::SEED_PREFIX.as_bytes(), &[config.bump]]; + let signer_seeds = &[&seeds[..]]; + token::transfer( + CpiContext::new_with_signer(token_program, accounts, signer_seeds), + amount, + ) +} + +pub fn get_instruction_data(ix_name: &str, data: Vec) -> Vec { + let preimage = format!("{}:{}", "global", ix_name); + + let mut ix_discriminator = [0u8; 8]; + ix_discriminator.copy_from_slice(&hash::hash(preimage.as_bytes()).to_bytes()[..8]); + + let mut ix_data = Vec::new(); + ix_data.extend_from_slice(&ix_discriminator); + ix_data.extend_from_slice(&data); + + ix_data +} + +pub fn hash_data(data: &Vec) -> Vec { + keccak::hash(&data).to_bytes().to_vec() +} diff --git a/contracts/solana/programs/intent/src/instructions/cancel.rs b/contracts/solana/programs/intent/src/instructions/cancel.rs new file mode 100644 index 000000000..315e80e5c --- /dev/null +++ b/contracts/solana/programs/intent/src/instructions/cancel.rs @@ -0,0 +1,137 @@ +use anchor_lang::prelude::*; + +use crate::{ + connection, + error::*, + event, helpers, + state::*, + types::{ + order_cancel::Cancel, + order_fill::OrderFill, + order_message::{MessageType, OrderMessage}, + swap_order::SwapOrder, + }, +}; + +pub fn cancel_order<'info>(ctx: Context<'_, '_, '_, 'info, CancelCtx<'info>>) -> Result<()> { + let order = &ctx.accounts.order_account.order; + let config = &mut ctx.accounts.config; + + let cancel = Cancel::new(order.encode()); + + if order.src_nid() == order.dst_nid() { + let order_finished = ctx + .accounts + .order_finished + .as_mut() + .ok_or(IntentError::OrderFinishedAccountIsMissing)?; + + resolve_cancel( + config.network_id.clone(), + cancel, + order, + config, + order_finished, + ctx.bumps.order_finished.unwrap(), + )?; + return Ok(()); + } + + if ctx.accounts.order_finished.is_some() { + return Err(IntentError::OrderFinishedAccountMustNotBeSpecified.into()); + } + + let order_msg = OrderMessage::new(MessageType::CANCEL, cancel.encode()); + connection::send_message(config, order.dst_nid(), order_msg.encode()) +} + +pub fn resolve_cancel<'info>( + src_network: String, + cancel: Cancel, + order: &SwapOrder, + config: &mut Account<'info, Config>, + order_finished: &mut Account<'info, OrderFinished>, + order_finished_bump: u8, +) -> Result<()> { + if src_network != order.src_nid() { + return Err(IntentError::InvalidNetwork.into()); + } + + if order_finished.finished { + return Ok(()); + } + order_finished.new(order_finished_bump); + + let fill = OrderFill::new(order.id(), cancel.order_bytes(), order.creator()); + let order_msg = OrderMessage::new(MessageType::FILL, fill.encode()); + + connection::send_message(config, order.src_nid(), order_msg.encode())?; + event::OrderCancelled { + id: order.id(), + srcNID: order.src_nid(), + }; + + Ok(()) +} + +#[derive(Accounts)] +#[instruction(order: SwapOrder)] +pub struct CancelCtx<'info> { + #[account( + mut, + constraint = signer.key().to_string() == order_account.order.creator() @IntentError::CreatorMustBeSigner + )] + pub signer: Signer<'info>, + + pub system_program: Program<'info, System>, + + #[account(mut)] + pub config: Account<'info, Config>, + + #[account( + mut, + seeds = [ + &signer.key().to_bytes(), + order.dst_nid().as_bytes(), + &order.amount().to_be_bytes(), + &order.to_amount().to_be_bytes() + ], + bump = order_account.bump + )] + pub order_account: Account<'info, OrderAccount>, + + #[account( + init_if_needed, + space = OrderFinished::SIZE, + payer = signer, + seeds = [&order.get_hash()], + bump + )] + pub order_finished: Option>, +} + +#[derive(Accounts)] +#[instruction(src_network: String, cancel: Cancel)] +pub struct ResolveCancelCtx<'info> { + #[account(mut)] + pub signer: Signer<'info>, + + pub system_program: Program<'info, System>, + + #[account( + owner = crate::id() + )] + pub intent: Signer<'info>, + + #[account(mut)] + pub config: Account<'info, Config>, + + #[account( + init_if_needed, + space = OrderFinished::SIZE, + payer = signer, + seeds = [&helpers::hash_data(&cancel.order_bytes())], + bump + )] + pub order_finished: Account<'info, OrderFinished>, +} diff --git a/contracts/solana/programs/intent/src/instructions/config.rs b/contracts/solana/programs/intent/src/instructions/config.rs new file mode 100644 index 000000000..4734bd65c --- /dev/null +++ b/contracts/solana/programs/intent/src/instructions/config.rs @@ -0,0 +1,142 @@ +use anchor_lang::prelude::*; + +use crate::{state::*, IntentError}; + +pub fn initialize( + ctx: Context, + fee_handler: Pubkey, + network_id: String, +) -> Result<()> { + ctx.accounts.config.new( + ctx.accounts.signer.key(), + fee_handler, + network_id, + ctx.bumps.config, + ); + ctx.accounts + .vault_native_account + .new(ctx.bumps.vault_native_account); + + Ok(()) +} + +pub fn set_admin(ctx: Context, account: Pubkey) -> Result<()> { + ctx.accounts.config.set_admin(account); + + Ok(()) +} + +pub fn set_protocol_fee(ctx: Context, fee: u64) -> Result<()> { + ctx.accounts.config.set_protocol_fee(fee); + + Ok(()) +} + +pub fn set_fee_handler(ctx: Context, fee_handler: Pubkey) -> Result<()> { + ctx.accounts.config.set_fee_handler(fee_handler); + + Ok(()) +} + +#[derive(Accounts)] +pub struct InitializeCtx<'info> { + /// The configuration account, which stores important settings for the program. + /// This account is initialized only once during the lifetime of program and it will + /// throw error if tries to initialize twice + #[account( + init, + payer = signer, + space = Config::SIZE, + seeds = [Config::SEED_PREFIX.as_bytes()], + bump + )] + pub config: Account<'info, Config>, + + /// The vault account that holds native SOL token + #[account( + init, + payer = signer, + space = VaultNative::SIZE, + seeds = [VaultNative::SEED_PREFIX.as_bytes()], + bump, + )] + pub vault_native_account: Account<'info, VaultNative>, + + /// The account that signs and pays for the transaction. This account is mutable + /// because it will be debited for any fees or rent required during the transaction. + #[account(mut)] + pub signer: Signer<'info>, + + /// The solana system program account, used for creating and managing accounts. + system_program: Program<'info, System>, +} + +#[derive(Accounts)] +pub struct GetConfigCtx<'info> { + /// The configuration account, which stores important settings for the program. + #[account( + seeds = [Config::SEED_PREFIX.as_bytes()], + bump = config.bump + )] + pub config: Account<'info, Config>, +} + +#[derive(Accounts)] +pub struct SetAdminCtx<'info> { + /// The configuration account, which stores important settings for the program. + /// This account is mutable because the admin of the program will be updated. + #[account( + mut, + seeds = [Config::SEED_PREFIX.as_bytes()], + bump = config.bump + )] + pub config: Account<'info, Config>, + + /// The account that signs and pays for the transaction. This account is checked + /// against the `config.admin` to ensure it is valid. + #[account( + mut, + address = config.admin @IntentError::OnlyAdmin + )] + pub admin: Signer<'info>, +} + +#[derive(Accounts)] +pub struct SetFeeHandlerCtx<'info> { + /// The configuration account, which stores important settings for the program. + /// This account is mutable because the fee handler of the protocol will be updated. + #[account( + mut, + seeds = [Config::SEED_PREFIX.as_bytes()], + bump = config.bump + )] + pub config: Account<'info, Config>, + + /// The account that signs and pays for the transaction. This account is checked + /// against the `config.admin` to ensure it is valid. + #[account( + mut, + address = config.admin @IntentError::OnlyAdmin + )] + pub admin: Signer<'info>, +} + +#[derive(Accounts)] +pub struct SetFeeCtx<'info> { + /// The configuration account, which stores important settings for the program. + /// This account is mutable because the fee handler of the protocol will be updated. + #[account( + mut, + seeds = [Config::SEED_PREFIX.as_bytes()], + bump = config.bump + )] + pub config: Account<'info, Config>, + + /// The account that signs and pays for the transaction. This account is checked + /// against the `config.admin` to ensure it is valid. + #[account( + mut, + address = config.admin @IntentError::OnlyAdmin + )] + pub admin: Signer<'info>, +} diff --git a/contracts/solana/programs/intent/src/instructions/fill.rs b/contracts/solana/programs/intent/src/instructions/fill.rs new file mode 100644 index 000000000..ef4369818 --- /dev/null +++ b/contracts/solana/programs/intent/src/instructions/fill.rs @@ -0,0 +1,322 @@ +use std::str::FromStr; + +use anchor_lang::prelude::*; +use anchor_spl::{ + associated_token::AssociatedToken, + token::{Mint, Token, TokenAccount}, +}; + +use crate::{ + connection, + constants::*, + error::*, + event, + helpers::*, + recv_message::*, + state::*, + types::{ + order_fill::OrderFill, + order_message::{MessageType, OrderMessage}, + swap_order::SwapOrder, + }, +}; + +pub fn order_fill<'info>( + ctx: Context<'_, '_, '_, 'info, FillCtx<'info>>, + order: SwapOrder, + solver_address: String, +) -> Result<()> { + let config = &mut ctx.accounts.config; + let order_fnished = &mut ctx.accounts.order_finished; + + if order.dst_address() != ctx.accounts.destination_address.key().to_string() { + return Err(IntentError::InvalidDestinationAccount.into()); + } + + if order_fnished.finished { + return Err(IntentError::OrderAlreadyFilled.into()); + } + order_fnished.new(ctx.bumps.order_finished); + + let fee = (order.to_amount() as u64 * config.protocol_fee) / 10_000; + let to_amount = order.to_amount() as u64 - fee; + + if order.to_token() == NATIVE_ADDRESS { + transfer_sol( + &ctx.accounts.signer, + &ctx.accounts.destination_address, + to_amount, + &ctx.accounts.system_program, + )?; + transfer_sol( + &ctx.accounts.signer, + &ctx.accounts.fee_handler, + fee, + &ctx.accounts.system_program, + )?; + } else { + let signer_token_account = ctx + .accounts + .signer_token_account + .as_ref() + .ok_or(IntentError::SignerTokenAccountIsMissing)?; + + let destination_token_account = ctx + .accounts + .destination_token_account + .as_ref() + .ok_or(IntentError::CreatorTokenAccountIsMissing)?; + + let fee_handler_token_account = ctx + .accounts + .fee_handler_token_account + .as_ref() + .ok_or(IntentError::FeeHandlerTokenAccountIsMissing)?; + + transfer_spl_token( + signer_token_account.to_account_info(), + destination_token_account.to_account_info(), + ctx.accounts.signer.to_account_info(), + to_amount, + ctx.accounts.token_program.to_account_info(), + )?; + transfer_spl_token( + signer_token_account.to_account_info(), + fee_handler_token_account.to_account_info(), + ctx.accounts.signer.to_account_info(), + fee, + ctx.accounts.token_program.to_account_info(), + )?; + } + + let fill = OrderFill::new(order.id(), order.encode(), solver_address); + + if order.src_nid() == order.dst_nid() { + invoke_resolve( + config.network_id.clone(), + Some(fill), + None, + order, + &ctx.accounts.signer, + &ctx.accounts.system_program, + ctx.remaining_accounts, + &config.to_account_info(), + &[&[Config::SEED_PREFIX.as_bytes(), &[ctx.accounts.config.bump]]], + )?; + return Ok(()); + } + + let order_msg = OrderMessage::new(MessageType::FILL, fill.encode()); + connection::send_message(config, order.dst_nid(), order_msg.encode())?; + + event::OrderFilled { + id: order.id(), + srcNID: order.src_nid(), + }; + + Ok(()) +} + +pub fn resolve_fill( + ctx: Context, + src_network: String, + fill: OrderFill, +) -> Result<()> { + let order = &mut ctx.accounts.order_account.order; + + if order.get_hash() != hash_data(&fill.order_bytes()) { + return Err(IntentError::OrderMismatched.into()); + } + + if src_network != order.dst_nid() { + return Err(IntentError::InvalidNetwork.into()); + } + + if order.token() == NATIVE_ADDRESS { + let vault_native_account = ctx + .accounts + .vault_native_account + .as_mut() + .ok_or(IntentError::VaultNativeAccountIsMissing)?; + + transfer_sol_signed( + &vault_native_account.to_account_info(), + &ctx.accounts.solver.to_account_info(), + order.amount() as u64, + )?; + } else { + let vault_token_account = ctx + .accounts + .vault_token_account + .as_ref() + .ok_or(IntentError::VaultTokenAccountIsMissing)?; + + let solver_token_account = ctx + .accounts + .solver_token_account + .as_ref() + .ok_or(IntentError::SolverTokenAccountIsMissing)?; + + transfer_spl_token_signed( + vault_token_account.to_account_info(), + solver_token_account.to_account_info(), + ctx.accounts.config.to_account_info(), + order.amount() as u64, + ctx.accounts.token_program.to_account_info(), + &ctx.accounts.config, + )?; + } + + event::OrderClosed { id: order.id() }; + + Ok(()) +} + +#[derive(Accounts)] +#[instruction(order: SwapOrder)] +pub struct FillCtx<'info> { + #[account(mut)] + pub signer: Signer<'info>, + + pub system_program: Program<'info, System>, + + #[account(mut)] + pub config: Box>, + + /// CHECK: The account which receives the protocol fee for each order being filled. + /// This account is validated against `config.fee_handler`. + #[account( + mut, + address = config.fee_handler @ IntentError::InvalidFeeHandler + )] + pub fee_handler: AccountInfo<'info>, + + /// CHECK: The destination address where order creator wants to receive the intent + /// order. This account is validated in instruction. + #[account(mut)] + pub destination_address: AccountInfo<'info>, + + #[account( + init_if_needed, + space = OrderFinished::SIZE, + payer = signer, + seeds = [&order.get_hash()], + bump + )] + pub order_finished: Box>, + + /// The associated token account of fee handler of the intent program + #[account( + mut, + associated_token::mint = mint, + associated_token::authority = fee_handler + )] + pub fee_handler_token_account: Option>>, + + /// The assocaited token account derived from `destination_address`. + #[account( + init_if_needed, + payer = signer, + associated_token::mint = mint, + associated_token::authority = destination_address + )] + pub destination_token_account: Option>>, + + /// The token account of the signer + #[account( + mut, + token::mint = mint, + token::authority = signer + )] + pub signer_token_account: Option>>, + + #[account( + constraint = mint.key().to_string() == order.to_token() @IntentError::MintAccountMismatch + )] + pub mint: Option>>, + + pub token_program: Program<'info, Token>, + + pub associated_token_program: Program<'info, AssociatedToken>, +} + +#[derive(Accounts)] +#[instruction(src_network: String, fill: OrderFill, order: SwapOrder)] +pub struct ResolveFillCtx<'info> { + #[account(mut)] + pub signer: Signer<'info>, + + pub system_program: Program<'info, System>, + + #[account( + owner = crate::id() + )] + pub intent: Signer<'info>, + + #[account(mut)] + pub config: Account<'info, Config>, + + #[account( + mut, + seeds = [ + &order_creator.key().to_bytes(), + order.dst_nid().as_bytes(), + &order.amount().to_be_bytes(), + &order.to_amount().to_be_bytes() + ], + bump = order_account.bump, + close = order_creator + )] + pub order_account: Account<'info, OrderAccount>, + + /// CHECK: The order creator account validated against `order_account.order.creator` + #[account( + mut, + constraint = order_creator.key().to_string() == order.creator() + )] + pub order_creator: AccountInfo<'info>, + + /// CHECK: The account of the solver to receive fund intent locked amount in source chain + /// This account is validated against `fill.solver`. + #[account( + mut, + constraint = solver.key().to_string() == fill.solver() @IntentError::InvalidSolverAccount + )] + pub solver: AccountInfo<'info>, + + /// The solver token account + #[account( + init_if_needed, + payer = signer, + associated_token::mint = mint, + associated_token::authority = solver + )] + pub solver_token_account: Option>, + + /// Vault native account + #[account( + mut, + seeds = [VaultNative::SEED_PREFIX.as_bytes()], + bump = vault_native_account.bump + )] + pub vault_native_account: Option>, + + /// Vault token account + #[account( + mut, + token::mint = mint, + token::authority = config, + seeds = [VAULT_TOKEN_SEED_PREFIX.as_bytes(), &Pubkey::from_str(&order.token()).unwrap().to_bytes()], + bump + )] + pub vault_token_account: Option>, + + #[account( + constraint = mint.key().to_string() == order_account.order.token() @IntentError::MintAccountMismatch + )] + pub mint: Option>, + + pub token_program: Program<'info, Token>, + + pub associated_token_program: Program<'info, AssociatedToken>, +} diff --git a/contracts/solana/programs/intent/src/instructions/mod.rs b/contracts/solana/programs/intent/src/instructions/mod.rs new file mode 100644 index 000000000..9e71c3a29 --- /dev/null +++ b/contracts/solana/programs/intent/src/instructions/mod.rs @@ -0,0 +1,13 @@ +pub mod cancel; +pub mod config; +pub mod fill; +pub mod query_accounts; +pub mod recv_message; +pub mod swap; + +pub use cancel::*; +pub use config::*; +pub use fill::*; +pub use query_accounts::*; +pub use recv_message::*; +pub use swap::*; diff --git a/contracts/solana/programs/intent/src/instructions/query_accounts.rs b/contracts/solana/programs/intent/src/instructions/query_accounts.rs new file mode 100644 index 000000000..df9ff6ea2 --- /dev/null +++ b/contracts/solana/programs/intent/src/instructions/query_accounts.rs @@ -0,0 +1,376 @@ +use std::str::FromStr; + +use anchor_lang::{prelude::*, solana_program::system_program}; +use anchor_spl::{ + associated_token::{get_associated_token_address, AssociatedToken}, + token::Token, +}; + +use crate::{ + constants::*, misc::*, order_cancel::*, order_message::*, state::*, OrderFill, SwapOrder, +}; + +pub fn query_swap_accounts( + ctx: Context, + swap_order: SwapOrder, + page: u8, + limit: u8, +) -> Result { + let config = &ctx.accounts.config; + + let signer = Pubkey::from_str(&swap_order.creator()).unwrap(); + let (order_pda, _) = Pubkey::find_program_address( + &[ + &signer.to_bytes(), + swap_order.dst_nid().as_bytes(), + &swap_order.amount().to_be_bytes(), + &swap_order.to_amount().to_be_bytes(), + ], + &crate::id(), + ); + let (vault_native_pda, _) = + Pubkey::find_program_address(&[VaultNative::SEED_PREFIX.as_bytes()], &crate::id()); + + let mut account_metas = vec![ + AccountMetadata::new_readonly(system_program::id(), false), + AccountMetadata::new(config.key(), false), + AccountMetadata::new(order_pda, false), + ]; + + if swap_order.token() == NATIVE_ADDRESS { + account_metas.push(AccountMetadata::new(vault_native_pda, false)); + + account_metas.push(AccountMetadata::new_readonly(crate::id(), false)); + account_metas.push(AccountMetadata::new_readonly(crate::id(), false)); + account_metas.push(AccountMetadata::new_readonly(crate::id(), false)); + } else { + let token_mint_address = Pubkey::from_str(&swap_order.token()).unwrap(); + let signer_token = get_associated_token_address(&signer, &token_mint_address); + + let (vault_token_pda, _) = Pubkey::find_program_address( + &[ + VAULT_TOKEN_SEED_PREFIX.as_bytes(), + &token_mint_address.to_bytes(), + ], + &crate::id(), + ); + + // Vault native account (null) + account_metas.push(AccountMetadata::new_readonly(crate::id(), false)); + + // Vault token account + account_metas.push(AccountMetadata::new(vault_token_pda, false)); + + // Signer token account + account_metas.push(AccountMetadata::new(signer_token, false)); + + // Token mint + account_metas.push(AccountMetadata::new(token_mint_address, false)); + } + + // Token program + account_metas.push(AccountMetadata::new_readonly(Token::id(), false)); + + Ok(QueryAccountsPaginateResponse::new( + account_metas, + page, + limit, + )) +} + +pub fn query_recv_message_accounts( + ctx: Context, + src_network: String, + conn_sn: u128, + msg: Vec, + page: u8, + limit: u8, +) -> Result { + let config = &ctx.accounts.config; + + let (receipt_pda, _) = Pubkey::find_program_address( + &[ + Receipt::SEED_PREFIX.as_bytes(), + src_network.as_bytes(), + &conn_sn.to_be_bytes(), + ], + &crate::id(), + ); + + let mut account_metas = vec![ + AccountMetadata::new_readonly(system_program::id(), false), + AccountMetadata::new_readonly(config.key(), false), + AccountMetadata::new(receipt_pda, false), + AccountMetadata::new(config.key(), false), + ]; + + let order_msg = OrderMessage::try_from(&msg).unwrap(); + match order_msg.message_type() { + MessageType::FILL => { + let fill = OrderFill::try_from(&order_msg.message()).unwrap(); + let order = SwapOrder::try_from(&fill.order_bytes()).unwrap(); + + let order_creator = Pubkey::from_str(&order.creator()).unwrap(); + let solver = Pubkey::from_str(&fill.solver()).unwrap(); + + let (order_pda, _) = Pubkey::find_program_address( + &[ + &order_creator.to_bytes(), + order.dst_nid().as_bytes(), + &order.amount().to_be_bytes(), + &order.to_amount().to_be_bytes(), + ], + &crate::id(), + ); + + // Order account + account_metas.push(AccountMetadata::new(order_pda, false)); + + // Order creator + account_metas.push(AccountMetadata::new(order_creator, false)); + + // Solver account + account_metas.push(AccountMetadata::new(solver, false)); + + if order.token() == NATIVE_ADDRESS { + let (vault_native_pda, _) = Pubkey::find_program_address( + &[VaultNative::SEED_PREFIX.as_bytes()], + &crate::id(), + ); + + account_metas.push(AccountMetadata::new(crate::id(), false)); + + // Vault native account + account_metas.push(AccountMetadata::new(vault_native_pda, false)); + + account_metas.push(AccountMetadata::new(crate::id(), false)); + account_metas.push(AccountMetadata::new(crate::id(), false)); + } else { + let token_mint_address = Pubkey::from_str(&order.token()).unwrap(); + + let (vault_token_pda, _) = Pubkey::find_program_address( + &[ + VAULT_TOKEN_SEED_PREFIX.as_bytes(), + &token_mint_address.to_bytes(), + ], + &crate::id(), + ); + + let solver_token_account = + get_associated_token_address(&solver, &token_mint_address); + + // Solver token account + account_metas.push(AccountMetadata::new(solver_token_account, false)); + + // Vault native account (null) + account_metas.push(AccountMetadata::new(crate::id(), false)); + + // Vault token account + account_metas.push(AccountMetadata::new(vault_token_pda, false)); + + // Mint account + account_metas.push(AccountMetadata::new(token_mint_address, false)); + } + + // Token program + account_metas.push(AccountMetadata::new(Token::id(), false)); + + // Associated token program + account_metas.push(AccountMetadata::new(AssociatedToken::id(), false)); + } + MessageType::CANCEL => { + let cancel = Cancel::try_from(&order_msg.message()).unwrap(); + let order = SwapOrder::try_from(&cancel.order_bytes()).unwrap(); + + let (order_finished_pda, _) = + Pubkey::find_program_address(&[&order.get_hash()], &crate::id()); + + // Order finished account + account_metas.push(AccountMetadata::new(order_finished_pda, false)); + + account_metas.push(AccountMetadata::new(crate::id(), false)); + } + } + + Ok(QueryAccountsPaginateResponse::new( + account_metas, + page, + limit, + )) +} + +pub fn query_fill_accounts( + ctx: Context, + order: SwapOrder, + signer: Pubkey, + solver_address: String, + page: u8, + limit: u8, +) -> Result { + let config = &ctx.accounts.config; + let destination_address = Pubkey::from_str(&order.dst_address()).unwrap(); + + let (order_finished_pda, _) = Pubkey::find_program_address(&[&order.get_hash()], &crate::id()); + + let mut account_metas = vec![ + AccountMetadata::new_readonly(system_program::id(), false), + AccountMetadata::new(config.key(), false), + AccountMetadata::new(config.fee_handler, false), + AccountMetadata::new(destination_address, false), + AccountMetadata::new(order_finished_pda, false), + ]; + + if order.to_token() == NATIVE_ADDRESS { + account_metas.push(AccountMetadata::new(crate::id(), false)); + account_metas.push(AccountMetadata::new(crate::id(), false)); + account_metas.push(AccountMetadata::new(crate::id(), false)); + account_metas.push(AccountMetadata::new(crate::id(), false)); + } else { + let token_mint_address = Pubkey::from_str(&order.to_token()).unwrap(); + let fee_handler_token = + get_associated_token_address(&config.fee_handler, &token_mint_address); + let destination_token = + get_associated_token_address(&destination_address, &token_mint_address); + let signer_token = get_associated_token_address(&signer, &token_mint_address); + + // Fee handler token account + account_metas.push(AccountMetadata::new(fee_handler_token, false)); + + // Destination token account + account_metas.push(AccountMetadata::new(destination_token, false)); + + // Signer token account + account_metas.push(AccountMetadata::new(signer_token, false)); + + // Mint account + account_metas.push(AccountMetadata::new(token_mint_address, false)); + } + + // Token program + account_metas.push(AccountMetadata::new(Token::id(), false)); + + // Associated token program + account_metas.push(AccountMetadata::new(AssociatedToken::id(), false)); + + if order.src_nid() == order.dst_nid() { + let creator = Pubkey::from_str(&order.creator()).unwrap(); + let solver = Pubkey::from_str(&solver_address).unwrap(); + + let (order_pda, _) = Pubkey::find_program_address( + &[ + &creator.to_bytes(), + order.dst_nid().as_bytes(), + &order.amount().to_be_bytes(), + &order.to_amount().to_be_bytes(), + ], + &crate::id(), + ); + + // Config account + account_metas.push(AccountMetadata::new(config.key(), false)); + + // Order account + account_metas.push(AccountMetadata::new(order_pda, false)); + + // Order creator account + account_metas.push(AccountMetadata::new(creator, false)); + + // Solver account + account_metas.push(AccountMetadata::new(solver, false)); + + if order.token() == NATIVE_ADDRESS { + let (vault_native_pda, _) = + Pubkey::find_program_address(&[VaultNative::SEED_PREFIX.as_bytes()], &crate::id()); + + account_metas.push(AccountMetadata::new(crate::id(), false)); + + // Vault native account + account_metas.push(AccountMetadata::new(vault_native_pda, false)); + + account_metas.push(AccountMetadata::new(crate::id(), false)); + account_metas.push(AccountMetadata::new(crate::id(), false)); + } else { + let token_mint_address = Pubkey::from_str(&order.token()).unwrap(); + let solver_token = get_associated_token_address(&solver, &token_mint_address); + let (vault_token_pda, _) = Pubkey::find_program_address( + &[ + VAULT_TOKEN_SEED_PREFIX.as_bytes(), + &token_mint_address.to_bytes(), + ], + &crate::id(), + ); + + // Solver token account + account_metas.push(AccountMetadata::new(solver_token, false)); + + // Vault native account (null) + account_metas.push(AccountMetadata::new(crate::id(), false)); + + // Vault token account + account_metas.push(AccountMetadata::new(vault_token_pda, false)); + + // Mint account + account_metas.push(AccountMetadata::new(token_mint_address, false)); + } + + // Token program + account_metas.push(AccountMetadata::new(Token::id(), false)); + + // Associated token program + account_metas.push(AccountMetadata::new(AssociatedToken::id(), false)); + } + + Ok(QueryAccountsPaginateResponse::new( + account_metas, + page, + limit, + )) +} + +pub fn query_cancel_accounts( + ctx: Context, + order: SwapOrder, + page: u8, + limit: u8, +) -> Result { + let config = &ctx.accounts.config; + let signer = Pubkey::from_str(&order.creator()).unwrap(); + + let (order_pda, _) = Pubkey::find_program_address( + &[ + &signer.to_bytes(), + order.dst_nid().as_bytes(), + &order.amount().to_be_bytes(), + &order.to_amount().to_be_bytes(), + ], + &crate::id(), + ); + + let mut account_metas = vec![ + AccountMetadata::new_readonly(system_program::id(), false), + AccountMetadata::new(config.key(), false), + AccountMetadata::new(order_pda, false), + ]; + + if order.src_nid() == order.dst_nid() { + let (order_finished_pda, _) = + Pubkey::find_program_address(&[&order.get_hash()], &crate::id()); + + // Order finished account + account_metas.push(AccountMetadata::new(order_finished_pda, false)); + } else { + // null + account_metas.push(AccountMetadata::new(crate::id(), false)); + } + + Ok(QueryAccountsPaginateResponse::new( + account_metas, + page, + limit, + )) +} + +#[derive(Accounts)] +pub struct QueryAccountCtx<'info> { + pub config: Account<'info, Config>, +} diff --git a/contracts/solana/programs/intent/src/instructions/recv_message.rs b/contracts/solana/programs/intent/src/instructions/recv_message.rs new file mode 100644 index 000000000..992086372 --- /dev/null +++ b/contracts/solana/programs/intent/src/instructions/recv_message.rs @@ -0,0 +1,146 @@ +use anchor_lang::{ + prelude::*, + solana_program::{instruction::Instruction, program::invoke_signed}, +}; + +use crate::{ + connection, + error::IntentError, + helpers, + misc::*, + state::*, + types::{ + order_cancel::Cancel, + order_fill::OrderFill, + order_message::{MessageType, OrderMessage}, + swap_order::SwapOrder, + }, +}; + +pub fn recv_message<'info>( + ctx: Context<'_, '_, '_, 'info, RecvMessageCtx<'info>>, + src_network: String, + msg: Vec, +) -> Result<()> { + connection::recv_message(&mut ctx.accounts.receipt, ctx.bumps.receipt)?; + + let order_msg = OrderMessage::try_from(&msg)?; + match order_msg.message_type() { + MessageType::FILL => { + let fill = OrderFill::try_from(&order_msg.message())?; + let order = SwapOrder::try_from(&fill.order_bytes())?; + invoke_resolve( + src_network.clone(), + Some(fill), + None, + order, + &ctx.accounts.signer, + &ctx.accounts.system_program, + ctx.remaining_accounts, + &ctx.accounts.config.to_account_info(), + &[&[Config::SEED_PREFIX.as_bytes(), &[ctx.accounts.config.bump]]], + ) + } + MessageType::CANCEL => { + let cancel = Cancel::try_from(&order_msg.message())?; + let order = SwapOrder::try_from(&cancel.order_bytes())?; + invoke_resolve( + src_network.clone(), + None, + Some(cancel), + order, + &ctx.accounts.signer, + &ctx.accounts.system_program, + ctx.remaining_accounts, + &ctx.accounts.config.to_account_info(), + &[&[Config::SEED_PREFIX.as_bytes(), &[ctx.accounts.config.bump]]], + ) + } + } +} + +pub fn invoke_resolve<'info>( + src_network: String, + fill: Option, + cancel: Option, + order: SwapOrder, + signer: &Signer<'info>, + system_program: &Program<'info, System>, + remaining_accounts: &[AccountInfo<'info>], + program_signer: &AccountInfo<'info>, + signers_seeds: &[&[&[u8]]], +) -> Result<()> { + let mut data = vec![]; + let mut ix_name = RESOLVE_FILL_IX; + + if fill.is_some() { + let args = ResolveFillArgs { + src_network: src_network.clone(), + fill: fill.unwrap(), + order, + }; + args.serialize(&mut data)?; + } else { + let args = ResolveCancelArgs { + src_network: src_network.clone(), + cancel: cancel.unwrap(), + order, + }; + args.serialize(&mut data)?; + ix_name = RESOLVE_CANCEL_IX; + } + let ix_data = helpers::get_instruction_data(ix_name, data); + + let mut account_metas: Vec = vec![ + AccountMeta::new(signer.key(), true), + AccountMeta::new_readonly(system_program.key(), false), + AccountMeta::new_readonly(program_signer.key(), true), + ]; + let mut account_infos: Vec> = vec![ + system_program.to_account_info(), + signer.to_account_info(), + program_signer.to_account_info(), + ]; + + for account in remaining_accounts { + if account.is_writable { + account_metas.push(AccountMeta::new(account.key(), account.is_signer)) + } else { + account_metas.push(AccountMeta::new_readonly(account.key(), account.is_signer)) + } + account_infos.push(account.to_account_info()); + } + + let ix = Instruction { + program_id: crate::id(), + accounts: account_metas, + data: ix_data, + }; + + invoke_signed(&ix, &account_infos, signers_seeds)?; + + Ok(()) +} + +#[derive(Accounts)] +#[instruction(src_network: String, conn_sn: u128)] +pub struct RecvMessageCtx<'info> { + #[account( + mut, + address = config.admin @IntentError::OnlyRelayer + )] + pub signer: Signer<'info>, + + pub system_program: Program<'info, System>, + + pub config: Account<'info, Config>, + + #[account( + init_if_needed, + payer = signer, + space = Receipt::SIZE, + seeds = [Receipt::SEED_PREFIX.as_bytes(), src_network.as_bytes(), &conn_sn.to_be_bytes()], + bump + )] + pub receipt: Account<'info, Receipt>, +} diff --git a/contracts/solana/programs/intent/src/instructions/swap.rs b/contracts/solana/programs/intent/src/instructions/swap.rs new file mode 100644 index 000000000..1fb332805 --- /dev/null +++ b/contracts/solana/programs/intent/src/instructions/swap.rs @@ -0,0 +1,134 @@ +use std::str::FromStr; + +use anchor_lang::prelude::*; +use anchor_spl::token::{Mint, Token, TokenAccount}; + +use crate::{ + constants::*, error::IntentError, event, helpers, state::*, types::swap_order::SwapOrder, +}; + +pub fn swap_order(ctx: Context, order: SwapOrder) -> Result<()> { + let mut order = order; + + order.set_id(ctx.accounts.config.increment_deposit_id()); + ctx.accounts + .order_account + .new(&order, ctx.bumps.order_account); + + // Escrows amount from user + if order.token() == NATIVE_ADDRESS { + if ctx.accounts.vault_token_account.is_some() { + return Err(IntentError::VaultTokenAccountMustNotBeSpecified.into()); + } + + let vault_native_account = ctx + .accounts + .vault_native_account + .as_ref() + .ok_or(IntentError::VaultNativeAccountIsMissing)?; + + helpers::transfer_sol( + &ctx.accounts.signer, + &vault_native_account.to_account_info(), + order.amount() as u64, + &ctx.accounts.system_program, + )?; + } else { + let user_token_account = ctx + .accounts + .signer_token_account + .as_ref() + .ok_or(IntentError::SignerTokenAccountIsMissing)?; + + let vault_token_account = ctx + .accounts + .vault_token_account + .as_ref() + .ok_or(IntentError::VaultTokenAccountIsMissing)?; + + helpers::transfer_spl_token( + user_token_account.to_account_info(), + vault_token_account.to_account_info(), + ctx.accounts.signer.to_account_info(), + order.amount() as u64, + ctx.accounts.token_program.to_account_info(), + )?; + } + + event::SwapIntent { + id: order.id(), + emitter: order.emitter(), + srcNID: order.src_nid(), + dstNID: order.dst_nid(), + creator: order.creator(), + destinationAddress: order.dst_address(), + token: order.token(), + amount: order.amount(), + toToken: order.to_token(), + toAmount: order.to_amount(), + data: order.data(), + }; + + Ok(()) +} + +#[derive(Accounts)] +#[instruction(order: SwapOrder)] +pub struct SwapCtx<'info> { + #[account( + mut, + constraint = signer.key().to_string() == order.creator() @IntentError::CreatorMustBeSigner + )] + pub signer: Signer<'info>, + + pub system_program: Program<'info, System>, + + #[account( + mut, + constraint = order.src_nid() == config.network_id @IntentError::NetworkIdMisconfigured, + constraint = order.emitter() == crate::id().to_string() @IntentError::InvalidEmitterAddress + )] + pub config: Box>, + + #[account( + init, + payer = signer, + space = OrderAccount::SIZE, + seeds = [ + &signer.key().to_bytes(), + order.dst_nid().as_bytes(), + &order.amount().to_be_bytes(), + &order.to_amount().to_be_bytes() + ], + bump + )] + pub order_account: Box>, + + #[account( + mut, + seeds = [VaultNative::SEED_PREFIX.as_bytes()], + bump + )] + pub vault_native_account: Option>>, + + #[account( + init_if_needed, + payer = signer, + token::mint = mint, + token::authority = config, + seeds = [VAULT_TOKEN_SEED_PREFIX.as_bytes(), &Pubkey::from_str(&order.token()).unwrap().to_bytes()], + bump + )] + pub vault_token_account: Option>>, + + #[account( + mut, + token::mint = mint, + token::authority = signer + )] + pub signer_token_account: Option>>, + + pub mint: Option>>, + + pub token_program: Program<'info, Token>, +} diff --git a/contracts/solana/programs/intent/src/lib.rs b/contracts/solana/programs/intent/src/lib.rs new file mode 100644 index 000000000..22dec9797 --- /dev/null +++ b/contracts/solana/programs/intent/src/lib.rs @@ -0,0 +1,137 @@ +use anchor_lang::prelude::*; + +pub mod connection; +pub mod constants; +pub mod error; +pub mod event; +pub mod helpers; +pub mod instructions; +pub mod state; +pub mod types; + +use error::IntentError; +use instructions::*; +use types::*; + +declare_id!("4nTmHeE5qdFuKoqeakERkMn3iFFV2X76zkvdDA26LPCG"); + +#[program] +pub mod intent { + use super::*; + + pub fn initialize( + ctx: Context, + network_id: String, + fee_handler: Pubkey, + ) -> Result<()> { + instructions::initialize(ctx, fee_handler, network_id) + } + + pub fn set_admin(ctx: Context, account: Pubkey) -> Result<()> { + instructions::set_admin(ctx, account) + } + + pub fn set_protocol_fee(ctx: Context, fee: u64) -> Result<()> { + instructions::set_protocol_fee(ctx, fee) + } + + pub fn set_fee_handler(ctx: Context, fee_handler: Pubkey) -> Result<()> { + instructions::set_fee_handler(ctx, fee_handler) + } + + pub fn swap(ctx: Context, order: SwapOrder) -> Result<()> { + instructions::swap_order(ctx, order) + } + + pub fn fill<'info>( + ctx: Context<'_, '_, '_, 'info, FillCtx<'info>>, + order: SwapOrder, + solver_address: String, + ) -> Result<()> { + instructions::order_fill(ctx, order, solver_address) + } + + #[allow(unused_variables)] + pub fn cancel<'info>( + ctx: Context<'_, '_, '_, 'info, CancelCtx<'info>>, + order: SwapOrder, + ) -> Result<()> { + instructions::cancel_order(ctx) + } + + #[allow(unused_variables)] + pub fn recv_message<'info>( + ctx: Context<'_, '_, '_, 'info, RecvMessageCtx<'info>>, + src_network: String, + conn_sn: u128, + msg: Vec, + ) -> Result<()> { + instructions::recv_message(ctx, src_network, msg) + } + + #[allow(unused_variables)] + pub fn resolve_fill( + ctx: Context, + src_network: String, + fill: OrderFill, + order: SwapOrder, + ) -> Result<()> { + instructions::resolve_fill(ctx, src_network, fill) + } + + pub fn resolve_cancel<'info>( + ctx: Context, + src_network: String, + cancel: types::order_cancel::Cancel, + order: SwapOrder, + ) -> Result<()> { + instructions::resolve_cancel( + src_network, + cancel, + &order, + &mut ctx.accounts.config, + &mut ctx.accounts.order_finished, + ctx.bumps.order_finished, + ) + } + + pub fn query_swap_accounts<'info>( + ctx: Context<'_, '_, '_, 'info, QueryAccountCtx<'info>>, + swap_order: SwapOrder, + page: u8, + limit: u8, + ) -> Result { + instructions::query_swap_accounts(ctx, swap_order, page, limit) + } + + pub fn query_fill_accounts<'info>( + ctx: Context<'_, '_, '_, 'info, QueryAccountCtx<'info>>, + order: SwapOrder, + signer: Pubkey, + solver_address: String, + page: u8, + limit: u8, + ) -> Result { + instructions::query_fill_accounts(ctx, order, signer, solver_address, page, limit) + } + + pub fn query_cancel_accounts( + ctx: Context, + order: SwapOrder, + page: u8, + limit: u8, + ) -> Result { + instructions::query_cancel_accounts(ctx, order, page, limit) + } + + pub fn query_recv_message_accounts( + ctx: Context, + src_network: String, + conn_sn: u128, + msg: Vec, + page: u8, + limit: u8, + ) -> Result { + instructions::query_recv_message_accounts(ctx, src_network, conn_sn, msg, page, limit) + } +} diff --git a/contracts/solana/programs/intent/src/state.rs b/contracts/solana/programs/intent/src/state.rs new file mode 100644 index 000000000..c00226252 --- /dev/null +++ b/contracts/solana/programs/intent/src/state.rs @@ -0,0 +1,133 @@ +use anchor_lang::prelude::*; + +use crate::{constants::*, error::*, types::swap_order::SwapOrder}; + +#[account(zero_copy)] +pub struct RandomStruct { + id: u128, +} + +#[account] +pub struct VaultNative { + pub bump: u8, +} + +impl VaultNative { + pub const SEED_PREFIX: &'static str = "vault_native"; + + pub const SIZE: usize = ACCOUNT_DISCRIMINATOR_SIZE + 1; + + pub fn new(&mut self, bump: u8) { + self.bump = bump; + } +} + +#[account] +pub struct Config { + pub admin: Pubkey, + pub fee_handler: Pubkey, + pub network_id: String, + pub protocol_fee: u64, + pub deposit_id: u128, + pub conn_sn: u128, + pub bump: u8, +} + +impl Config { + pub const SEED_PREFIX: &'static str = "config"; + + pub const SIZE: usize = ACCOUNT_DISCRIMINATOR_SIZE + 32 + 32 + 32 + 8 + 16 + 16 + 16 + 1; + + pub fn new(&mut self, admin: Pubkey, fee_handler: Pubkey, network_id: String, bump: u8) { + self.admin = admin; + self.bump = bump; + self.fee_handler = fee_handler; + self.network_id = network_id; + self.protocol_fee = 0; + self.deposit_id = 0; + self.conn_sn = 0; + } + + pub fn ensure_admin(&self, signer: Pubkey) -> Result<()> { + if self.admin != signer { + return Err(IntentError::OnlyAdmin.into()); + } + Ok(()) + } + + pub fn ensure_fee_handler(&self, signer: Pubkey) -> Result<()> { + if self.fee_handler != signer { + return Err(IntentError::OnlyAdmin.into()); + } + Ok(()) + } + + pub fn set_admin(&mut self, account: Pubkey) { + self.admin = account + } + + pub fn set_fee_handler(&mut self, fee_handler: Pubkey) { + self.fee_handler = fee_handler + } + + pub fn set_protocol_fee(&mut self, fee: u64) { + self.protocol_fee = fee + } + + pub fn increment_deposit_id(&mut self) -> u128 { + self.deposit_id += 1; + self.deposit_id + } + + pub fn increment_conn_sn(&mut self) -> u128 { + self.conn_sn += 1; + self.conn_sn + } +} + +#[account] +pub struct OrderAccount { + pub order: SwapOrder, + pub bump: u8, +} + +impl OrderAccount { + pub const SIZE: usize = ACCOUNT_DISCRIMINATOR_SIZE + 400 + 1; + + pub fn new(&mut self, order: &SwapOrder, bump: u8) { + self.order = order.to_owned(); + self.bump = bump + } +} + +#[account] +pub struct OrderFinished { + pub finished: bool, + pub bump: u8, +} + +impl OrderFinished { + pub const SIZE: usize = ACCOUNT_DISCRIMINATOR_SIZE + 1 + 1; + + pub fn new(&mut self, bump: u8) { + self.finished = true; + self.bump = bump + } +} + +#[account] +pub struct Receipt { + pub received: bool, + pub bump: u8, +} + +impl Receipt { + pub const SEED_PREFIX: &'static str = "receipt"; + + pub const SIZE: usize = ACCOUNT_DISCRIMINATOR_SIZE + 1 + 1; + + pub fn new(&mut self, bump: u8) { + self.received = true; + self.bump = bump + } +} diff --git a/contracts/solana/programs/intent/src/types/misc.rs b/contracts/solana/programs/intent/src/types/misc.rs new file mode 100644 index 000000000..25bcdfe05 --- /dev/null +++ b/contracts/solana/programs/intent/src/types/misc.rs @@ -0,0 +1,89 @@ +use anchor_lang::prelude::*; + +use super::{order_cancel::Cancel, order_fill::OrderFill, swap_order::SwapOrder}; + +pub const RESOLVE_FILL_IX: &str = "resolve_fill"; +pub const RESOLVE_CANCEL_IX: &str = "resolve_cancel"; + +pub const QUERY_SWAP_ACCOUNTS_IX: &str = "query_swap_accounts"; +pub const QUERY_FILL_ACCOUNTS_IX: &str = "query_fill_accounts"; +pub const QUERY_CANCEL_ACCOUNTS_IX: &str = "query_cancel_accounts"; +pub const QUERY_RECV_MESSAGE_ACCOUNTS_IX: &str = "query_recv_message_accounts"; + +#[derive(Debug, Clone, AnchorSerialize, AnchorDeserialize)] +pub struct ResolveFillArgs { + pub src_network: String, + pub fill: OrderFill, + pub order: SwapOrder, +} + +#[derive(Debug, Clone, AnchorSerialize, AnchorDeserialize)] +pub struct ResolveCancelArgs { + pub src_network: String, + pub cancel: Cancel, + pub order: SwapOrder, +} + +#[derive(Debug, Default, PartialEq, Eq, Clone, AnchorSerialize, AnchorDeserialize)] +pub struct AccountMetadata { + pub pubkey: Pubkey, + pub is_writable: bool, + pub is_signer: bool, +} + +impl AccountMetadata { + pub fn new(pubkey: Pubkey, is_signer: bool) -> Self { + Self { + pubkey, + is_signer, + is_writable: true, + } + } + + pub fn new_readonly(pubkey: Pubkey, is_signer: bool) -> Self { + Self { + pubkey, + is_signer, + is_writable: false, + } + } +} + +#[derive(Debug, Clone, AnchorSerialize, AnchorDeserialize)] +pub struct QueryAccountsResponse { + pub accounts: Vec, +} + +#[derive(Debug, Clone, AnchorSerialize, AnchorDeserialize)] +pub struct QueryAccountsPaginateResponse { + pub accounts: Vec, + pub total_accounts: u8, + pub limit: u8, + pub page: u8, + pub has_next_page: bool, +} + +impl QueryAccountsPaginateResponse { + pub fn new(accounts: Vec, page: u8, limit: u8) -> Self { + let offset = ((page - 1) * limit) as usize; + let total = accounts.len(); + + let to_index = if offset + limit as usize > total { + total + } else { + offset + limit as usize + }; + + let accounts = accounts[offset..to_index].to_vec(); + let total_accounts = total as u8; + let has_next_page = total > to_index; + + QueryAccountsPaginateResponse { + accounts, + total_accounts, + limit, + page, + has_next_page, + } + } +} diff --git a/contracts/solana/programs/intent/src/types/mod.rs b/contracts/solana/programs/intent/src/types/mod.rs new file mode 100644 index 000000000..e7ca3101e --- /dev/null +++ b/contracts/solana/programs/intent/src/types/mod.rs @@ -0,0 +1,12 @@ +pub mod misc; +pub mod order_cancel; +pub mod order_fill; +pub mod order_message; +pub mod swap_order; + +use crate::IntentError; +use anchor_lang::{prelude::borsh, AnchorDeserialize, AnchorSerialize}; +pub use misc::*; +pub use order_fill::*; +use rlp::{Decodable, Encodable}; +pub use swap_order::*; diff --git a/contracts/solana/programs/intent/src/types/order_cancel.rs b/contracts/solana/programs/intent/src/types/order_cancel.rs new file mode 100644 index 000000000..6109cb582 --- /dev/null +++ b/contracts/solana/programs/intent/src/types/order_cancel.rs @@ -0,0 +1,62 @@ +use super::*; + +#[derive(Debug, Clone, PartialEq, Eq, AnchorSerialize, AnchorDeserialize)] +pub struct Cancel { + /// Encoded order data + order_bytes: Vec, +} + +impl Cancel { + pub fn new(order_bytes: Vec) -> Self { + Self { order_bytes } + } + + pub fn order_bytes(&self) -> Vec { + self.order_bytes.clone() + } + + pub fn encode(&self) -> Vec { + rlp::encode(self).to_vec() + } +} + +impl Encodable for Cancel { + fn rlp_append(&self, stream: &mut rlp::RlpStream) { + stream.begin_list(1); + stream.append(&self.order_bytes); + } +} + +impl Decodable for Cancel { + fn decode(rlp: &rlp::Rlp) -> Result { + let order_bytes: Vec = rlp.val_at(0)?; + + Ok(Self { order_bytes }) + } +} + +impl TryFrom<&Vec> for Cancel { + type Error = IntentError; + fn try_from(value: &Vec) -> Result { + let rlp = rlp::Rlp::new(value as &[u8]); + Self::decode(&rlp).map_err(|_| IntentError::DecodeFailed) + } +} + +impl TryFrom<&[u8]> for Cancel { + type Error = IntentError; + fn try_from(value: &[u8]) -> Result { + let rlp = rlp::Rlp::new(value); + Self::decode(&rlp).map_err(|_| IntentError::DecodeFailed) + } +} + +#[test] +fn test_order_cancel_decode() { + let data = Cancel::new(hex::decode("6c449988e2f33302803c93f8287dc1d8cb33848a").unwrap()); + let expected = hex::decode("d5946c449988e2f33302803c93f8287dc1d8cb33848a").unwrap(); + let decoded: Cancel = Cancel::try_from(&expected).unwrap(); + + assert_eq!(data.encode(), expected); + assert_eq!(decoded, data) +} diff --git a/contracts/solana/programs/intent/src/types/order_fill.rs b/contracts/solana/programs/intent/src/types/order_fill.rs new file mode 100644 index 000000000..ecdf6d42a --- /dev/null +++ b/contracts/solana/programs/intent/src/types/order_fill.rs @@ -0,0 +1,104 @@ +use super::*; + +#[derive(Debug, Clone, PartialEq, Eq, AnchorSerialize, AnchorDeserialize)] +pub struct OrderFill { + /// ID of the order being filled + id: u128, + /// Encoded order data + order_bytes: Vec, + /// Address of the solver filling the order + solver: String, +} + +impl OrderFill { + pub fn new(id: u128, order_bytes: Vec, solver: String) -> Self { + Self { + id, + order_bytes, + solver, + } + } + + pub fn id(&self) -> u128 { + self.id + } + + pub fn order_bytes(&self) -> Vec { + self.order_bytes.clone() + } + + pub fn solver(&self) -> String { + self.solver.clone() + } + + pub fn encode(&self) -> Vec { + rlp::encode(self).to_vec() + } +} + +impl Encodable for OrderFill { + fn rlp_append(&self, stream: &mut rlp::RlpStream) { + stream.begin_list(3); + stream.append(&self.id()); + stream.append(&self.order_bytes()); + stream.append(&self.solver()); + } +} + +impl Decodable for OrderFill { + fn decode(rlp: &rlp::Rlp) -> Result { + let id: u128 = rlp.val_at(0)?; + let order_bytes: Vec = rlp.val_at(1)?; + let solver: String = rlp.val_at(2)?; + + Ok(Self { + id, + order_bytes, + solver, + }) + } +} + +impl TryFrom<&Vec> for OrderFill { + type Error = IntentError; + fn try_from(value: &Vec) -> Result { + let rlp = rlp::Rlp::new(value as &[u8]); + Self::decode(&rlp).map_err(|_| IntentError::DecodeFailed) + } +} + +impl TryFrom<&[u8]> for OrderFill { + type Error = IntentError; + fn try_from(value: &[u8]) -> Result { + let rlp = rlp::Rlp::new(value); + Self::decode(&rlp).map_err(|_| IntentError::DecodeFailed) + } +} + +#[test] +fn test_order_fill_decode_1() { + let data = OrderFill::new( + 1, + hex::decode("6c449988e2f33302803c93f8287dc1d8cb33848a").unwrap(), + "0xcb0a6bbccfccde6be9f10ae781b9d9b00d6e63".to_string(), + ); + let expected = hex::decode("f83f01946c449988e2f33302803c93f8287dc1d8cb33848aa830786362306136626263636663636465366265396631306165373831623964396230306436653633").unwrap(); + let decoded: OrderFill = OrderFill::try_from(&expected).unwrap(); + + assert_eq!(data.encode(), expected); + assert_eq!(decoded, data) +} + +#[test] +fn test_order_fill_decode_2() { + let data = OrderFill::new( + 2, + hex::decode("cb0a6bbccfccde6be9f10ae781b9d9b00d6e63").unwrap(), + "0x6c449988e2f33302803c93f8287dc1d8cb33848a".to_string(), + ); + let expected = hex::decode("f8400293cb0a6bbccfccde6be9f10ae781b9d9b00d6e63aa307836633434393938386532663333333032383033633933663832383764633164386362333338343861").unwrap(); + let decoded: OrderFill = OrderFill::try_from(&expected).unwrap(); + + assert_eq!(data.encode(), expected); + assert_eq!(decoded, data) +} diff --git a/contracts/solana/programs/intent/src/types/order_message.rs b/contracts/solana/programs/intent/src/types/order_message.rs new file mode 100644 index 000000000..90f60c67e --- /dev/null +++ b/contracts/solana/programs/intent/src/types/order_message.rs @@ -0,0 +1,120 @@ +use super::*; +use anchor_lang::prelude::borsh; + +#[derive(Debug, Clone, PartialEq, Eq, AnchorSerialize, AnchorDeserialize)] +pub enum MessageType { + FILL = 1, + CANCEL = 2, +} + +#[derive(Debug, Clone, PartialEq, Eq, AnchorSerialize, AnchorDeserialize)] +pub struct OrderMessage { + /// Type of message (Fill or Cancel) + message_type: MessageType, + /// Encoded message data + message: Vec, +} + +impl OrderMessage { + pub fn new(message_type: MessageType, message: Vec) -> Self { + Self { + message_type, + message, + } + } + + pub fn message_type(&self) -> MessageType { + self.message_type.clone() + } + + pub fn message(&self) -> Vec { + self.message.clone() + } + + pub fn encode(&self) -> Vec { + rlp::encode(self).to_vec() + } +} + +impl From for MessageType { + fn from(value: u32) -> Self { + match value { + 1 => MessageType::FILL, + 2 => MessageType::CANCEL, + _ => panic!("Invalid message type"), + } + } +} + +impl From for u32 { + fn from(value: MessageType) -> Self { + match value { + MessageType::FILL => 1, + MessageType::CANCEL => 2, + } + } +} + +impl Encodable for OrderMessage { + fn rlp_append(&self, stream: &mut rlp::RlpStream) { + let message_type: u32 = self.message_type().into(); + + stream.begin_list(2); + stream.append(&message_type); + stream.append(&self.message()); + } +} + +impl Decodable for OrderMessage { + fn decode(rlp: &rlp::Rlp) -> Result { + let message_type: u32 = rlp.val_at(0)?; + let message: Vec = rlp.val_at(1)?; + + Ok(Self { + message_type: message_type.into(), + message, + }) + } +} + +impl TryFrom<&Vec> for OrderMessage { + type Error = IntentError; + fn try_from(value: &Vec) -> Result { + let rlp = rlp::Rlp::new(value as &[u8]); + Self::decode(&rlp).map_err(|_| IntentError::DecodeFailed) + } +} + +impl TryFrom<&[u8]> for OrderMessage { + type Error = IntentError; + fn try_from(value: &[u8]) -> Result { + let rlp = rlp::Rlp::new(value); + Self::decode(&rlp).map_err(|_| IntentError::DecodeFailed) + } +} + +#[test] +fn test_order_message_decode_1() { + let data = OrderMessage::new( + MessageType::CANCEL, + hex::decode("6c449988e2f33302803c93f8287dc1d8cb33848a").unwrap(), + ); + let expected = hex::decode("d602946c449988e2f33302803c93f8287dc1d8cb33848a").unwrap(); + let decoded: OrderMessage = OrderMessage::try_from(&expected).unwrap(); + + assert_eq!(data.encode(), expected); + assert_eq!(decoded, data) +} + +#[test] +fn test_order_message_decode_2() { + let data = OrderMessage::new( + MessageType::FILL, + hex::decode("6c449988e2f33302803c93f8287dc1d8cb33848a").unwrap(), + ); + let expected = hex::decode("d601946c449988e2f33302803c93f8287dc1d8cb33848a").unwrap(); + let decoded: OrderMessage = OrderMessage::try_from(&expected).unwrap(); + + assert_eq!(data.encode(), expected); + assert_eq!(decoded, data); +} diff --git a/contracts/solana/programs/intent/src/types/swap_order.rs b/contracts/solana/programs/intent/src/types/swap_order.rs new file mode 100644 index 000000000..349d05ad0 --- /dev/null +++ b/contracts/solana/programs/intent/src/types/swap_order.rs @@ -0,0 +1,230 @@ +use anchor_lang::solana_program::keccak; + +use super::*; + +#[derive(Debug, Clone, PartialEq, Eq, AnchorSerialize, AnchorDeserialize)] +pub struct SwapOrder { + /// Unique identifier for each order + id: u128, + /// Address of emitter contract + emitter: String, + /// Network ID of the source chain + src_nid: String, + /// Netword ID of the destination chain + dst_nid: String, + /// Address of the user who created the swap order + creator: String, + /// Address where the swapped token should be sent + destination_address: String, + /// Address of the token to be swapped + token: String, + /// Amount of the token to be swapped + amount: u128, + /// Address of the token to receive on the destination chain + to_token: String, + /// Amount of `to_token` expected to be received + to_amount: u128, + /// Additional data for the swap + data: Vec, +} + +impl SwapOrder { + pub fn new( + id: u128, + emitter: String, + src_nid: String, + dst_nid: String, + creator: String, + destination_address: String, + token: String, + amount: u128, + to_token: String, + to_amount: u128, + data: Vec, + ) -> Self { + Self { + id, + emitter, + src_nid, + dst_nid, + creator, + destination_address, + token, + amount, + to_token, + to_amount, + data, + } + } + + pub fn id(&self) -> u128 { + self.id + } + + pub fn set_id(&mut self, id: u128) { + self.id = id + } + + pub fn emitter(&self) -> String { + self.emitter.clone() + } + + pub fn src_nid(&self) -> String { + self.src_nid.clone() + } + + pub fn dst_nid(&self) -> String { + self.dst_nid.clone() + } + + pub fn creator(&self) -> String { + self.creator.clone() + } + + pub fn dst_address(&self) -> String { + self.destination_address.clone() + } + + pub fn token(&self) -> String { + self.token.clone() + } + + pub fn amount(&self) -> u128 { + self.amount + } + + pub fn to_token(&self) -> String { + self.to_token.clone() + } + + pub fn to_amount(&self) -> u128 { + self.to_amount + } + + pub fn data(&self) -> Vec { + self.data.clone() + } + + pub fn set_data(&mut self, data: Vec) { + self.data = data + } + + pub fn get_hash(&self) -> Vec { + keccak::hash(&self.encode()).to_bytes().to_vec() + } + + pub fn encode(&self) -> Vec { + rlp::encode(self).to_vec() + } +} + +impl Encodable for SwapOrder { + fn rlp_append(&self, stream: &mut rlp::RlpStream) { + stream.begin_list(11); + stream.append(&self.id()); + stream.append(&self.emitter()); + stream.append(&self.src_nid()); + stream.append(&self.dst_nid()); + stream.append(&self.creator()); + stream.append(&self.dst_address()); + stream.append(&self.token()); + stream.append(&self.amount()); + stream.append(&self.to_token()); + stream.append(&self.to_amount()); + stream.append(&self.data()); + } +} + +impl Decodable for SwapOrder { + fn decode(rlp: &rlp::Rlp) -> Result { + let id: u128 = rlp.val_at(0)?; + let emitter: String = rlp.val_at(1)?; + let src_nid: String = rlp.val_at(2)?; + let dst_nid: String = rlp.val_at(3)?; + let creator: String = rlp.val_at(4)?; + let destination_address: String = rlp.val_at(5)?; + let token: String = rlp.val_at(6)?; + let amount: u128 = rlp.val_at(7)?; + let to_token: String = rlp.val_at(8)?; + let to_amount: u128 = rlp.val_at(9)?; + let data: Vec = rlp.val_at(10)?; + + Ok(Self { + id, + emitter, + src_nid, + dst_nid, + destination_address, + creator, + token, + amount, + to_amount, + to_token, + data, + }) + } +} + +impl TryFrom<&Vec> for SwapOrder { + type Error = IntentError; + fn try_from(value: &Vec) -> Result { + let rlp = rlp::Rlp::new(value as &[u8]); + Self::decode(&rlp).map_err(|_| IntentError::DecodeFailed) + } +} + +impl TryFrom<&[u8]> for SwapOrder { + type Error = IntentError; + fn try_from(value: &[u8]) -> Result { + let rlp = rlp::Rlp::new(value); + Self::decode(&rlp).map_err(|_| IntentError::DecodeFailed) + } +} + +#[test] +fn test_swap_order_decode_1() { + let swap_order = SwapOrder::new( + 1, + "0xbe6452d4d6c61cee97d3".to_string(), + "Ethereum".to_string(), + "Polygon".to_string(), + "0x3e36eddd65e239222e7e67".to_string(), + "0xd2c6218b875457a41b6fb7964e".to_string(), + "0x14355340e857912188b7f202d550222487".to_string(), + 1000, + "0x91a4728b517484f0f610de7b".to_string(), + 900, + Vec::new(), + ); + + let expected = hex::decode("f8a601963078626536343532643464366336316365653937643388457468657265756d87506f6c79676f6e983078336533366564646436356532333932323265376536379c30786432633632313862383735343537613431623666623739363465a43078313433353533343065383537393132313838623766323032643535303232323438378203e89a307839316134373238623531373438346630663631306465376282038480").unwrap(); + let encoded = swap_order.encode(); + let decoded = SwapOrder::try_from(&expected).unwrap(); + + assert_eq!(swap_order, decoded); + assert_eq!(encoded, expected) +} + +#[test] +fn test_swap_order_decode_2() { + let swap_order = SwapOrder::new( + 1, + "0xbe6452d4d6c61cee97d3".to_string(), + "Ethereum".to_string(), + "Polygon".to_string(), + "0x3e36eddd65e239222e7e67".to_string(), + "0xd2c6218b875457a41b6fb7964e".to_string(), + "0x14355340e857912188b7f202d550222487".to_string(), + 100000 * 10_u128.pow(22), + "0x91a4728b517484f0f610de7b".to_string(), + 900 * 10_u128.pow(7), + hex::decode("6c449988e2f33302803c93f8287dc1d8cb33848a").unwrap(), + ); + + let expected = hex::decode("f8c701963078626536343532643464366336316365653937643388457468657265756d87506f6c79676f6e983078336533366564646436356532333932323265376536379c30786432633632313862383735343537613431623666623739363465a43078313433353533343065383537393132313838623766323032643535303232323438378c033b2e3c9fd0803ce80000009a3078393161343732386235313734383466306636313064653762850218711a00946c449988e2f33302803c93f8287dc1d8cb33848a").unwrap(); + let encoded = swap_order.encode(); + let decoded = SwapOrder::try_from(&expected).unwrap(); + + assert_eq!(decoded, swap_order); + assert_eq!(encoded, expected) +} diff --git a/contracts/solana/scripts/setup.ts b/contracts/solana/scripts/setup.ts new file mode 100644 index 000000000..1d2daf161 --- /dev/null +++ b/contracts/solana/scripts/setup.ts @@ -0,0 +1,224 @@ +import * as anchor from "@coral-xyz/anchor"; +import os from "os"; +import { PublicKey, Connection } from "@solana/web3.js"; + +import { keccakHash, loadKeypairFromFile, uint128ToArray } from "./utils"; +import { Intent } from "../target/types/intent"; +import intentIdl from "../target/idl/intent.json"; +import { SwapOrder } from "./types"; + +/** RPC PROVIDER */ +export const RPC_URL = "http://127.0.0.1:8899"; +export const connection = new Connection(RPC_URL, "confirmed"); + +/** WALLET KEYPAIR */ +let keypairFilePath = os.homedir + "/.config/solana/id.json"; +export const keypair = loadKeypairFromFile(keypairFilePath); +export const wallet = new anchor.Wallet(keypair); + +/** PROVIDER FOR CLIENT */ +export const provider = new anchor.AnchorProvider(connection, wallet); +anchor.setProvider(provider); + +export const intentProgram: anchor.Program = new anchor.Program( + intentIdl as anchor.Idl, + provider +) as unknown as anchor.Program; + +export class IntentPda { + constructor() {} + + static config() { + let [pda, bump] = PublicKey.findProgramAddressSync( + [Buffer.from("config")], + intentProgram.programId + ); + + return { bump, pda }; + } + + static order( + creator: PublicKey, + dstNID: string, + amount: number, + toAmount: number + ) { + let [pda, bump] = PublicKey.findProgramAddressSync( + [ + creator.toBuffer(), + Buffer.from(dstNID), + uint128ToArray(amount), + uint128ToArray(toAmount), + ], + intentProgram.programId + ); + + return { bump, pda }; + } + + static vaultToken(mint: PublicKey) { + let [pda, bump] = PublicKey.findProgramAddressSync( + [Buffer.from("vault_token"), mint.toBuffer()], + intentProgram.programId + ); + + return { bump, pda }; + } + + static vaultNative() { + let [pda, bump] = PublicKey.findProgramAddressSync( + [Buffer.from("vault_native")], + intentProgram.programId + ); + + return { bump, pda }; + } + + static orderFinished(order: SwapOrder) { + let encoded = order.encode(); + let hash = keccakHash(Buffer.from(encoded)); + + let [pda, bump] = PublicKey.findProgramAddressSync( + [Buffer.from(hash, "hex")], + intentProgram.programId + ); + + return { pda, bump }; + } + + static receipt(srcNID: string, connSn: number) { + let [pda, bump] = PublicKey.findProgramAddressSync( + [Buffer.from("receipt"), Buffer.from(srcNID), uint128ToArray(connSn)], + intentProgram.programId + ); + + return { bump, pda }; + } +} + +export const getSwapAccounts = async (order: any) => { + return await intentProgram.methods + .querySwapAccounts(order, 1, 30) + .accountsStrict({ + config: IntentPda.config().pda, + }) + .view({ commitment: "confirmed" }); +}; + +export const getFillAccounts = async ( + order: any, + signer: PublicKey, + solverAddress: string +) => { + return await intentProgram.methods + .queryFillAccounts(order, signer, solverAddress, 1, 30) + .accountsStrict({ + config: IntentPda.config().pda, + }) + .view({ commitment: "confirmed" }); +}; + +export const getCancelAccounts = async (order: any) => { + return await intentProgram.methods + .queryCancelAccounts(order, 1, 30) + .accountsStrict({ + config: IntentPda.config().pda, + }) + .view({ commitment: "confirmed" }); +}; + +export const getRecvMessageAccounts = async ( + srcNetwork: string, + connSn: number, + msg: Buffer +) => { + return await intentProgram.methods + .queryRecvMessageAccounts(srcNetwork, new anchor.BN(connSn), msg, 1, 30) + .accountsStrict({ + config: IntentPda.config().pda, + }) + .view({ commitment: "confirmed" }); +}; + +export const getSwapIx = async (swap: any) => { + const accounts = (await getSwapAccounts(swap)).accounts; + + return await intentProgram.methods + .swap(swap) + .accountsStrict({ + signer: new PublicKey(swap.creator), + systemProgram: accounts[0].pubkey, + config: accounts[1].pubkey, + orderAccount: accounts[2].pubkey, + vaultNativeAccount: accounts[3].pubkey, + vaultTokenAccount: accounts[4].pubkey, + signerTokenAccount: accounts[5].pubkey, + mint: accounts[6].pubkey, + tokenProgram: accounts[7].pubkey, + }) + .instruction(); +}; + +export const getFillIx = async ( + swap: any, + solverKey: PublicKey, + solverAddress: string +) => { + const accounts = (await getFillAccounts(swap, solverKey, solverAddress)) + .accounts; + + return await intentProgram.methods + .fill(swap, solverAddress.toString()) + .accountsStrict({ + signer: solverKey, + systemProgram: accounts[0].pubkey, + config: accounts[1].pubkey, + feeHandler: accounts[2].pubkey, + destinationAddress: accounts[3].pubkey, + orderFinished: accounts[4].pubkey, + feeHandlerTokenAccount: accounts[5].pubkey, + destinationTokenAccount: accounts[6].pubkey, + signerTokenAccount: accounts[7].pubkey, + mint: accounts[8].pubkey, + tokenProgram: accounts[9].pubkey, + associatedTokenProgram: accounts[10].pubkey, + }) + .remainingAccounts(accounts.slice(11)) + .instruction(); +}; + +export const getCancelIx = async (swap: any) => { + const accounts = (await getCancelAccounts(swap)).accounts; + + return await intentProgram.methods + .cancel(swap) + .accountsStrict({ + signer: new PublicKey(swap.creator), + systemProgram: accounts[0].pubkey, + config: accounts[1].pubkey, + orderAccount: accounts[2].pubkey, + orderFinished: accounts[3].pubkey, + }) + .instruction(); +}; + +export const getRecvMessageIx = async ( + srcNetwork: string, + connSn: number, + message: Buffer, + signer: PublicKey +) => { + const accounts = (await getRecvMessageAccounts(srcNetwork, connSn, message)) + .accounts; + + return await intentProgram.methods + .recvMessage(srcNetwork, new anchor.BN(connSn), message) + .accountsStrict({ + signer: signer, + systemProgram: accounts[0].pubkey, + config: accounts[1].pubkey, + receipt: accounts[2].pubkey, + }) + .remainingAccounts(accounts.slice(3)) + .instruction(); +}; diff --git a/contracts/solana/scripts/types/index.ts b/contracts/solana/scripts/types/index.ts new file mode 100644 index 000000000..d4e2197d1 --- /dev/null +++ b/contracts/solana/scripts/types/index.ts @@ -0,0 +1,4 @@ +export * from "./order-cancel"; +export * from "./order-fill"; +export * from "./swap-order"; +export * from "./order_message"; diff --git a/contracts/solana/scripts/types/order-cancel.ts b/contracts/solana/scripts/types/order-cancel.ts new file mode 100644 index 000000000..de0dcfa79 --- /dev/null +++ b/contracts/solana/scripts/types/order-cancel.ts @@ -0,0 +1,18 @@ +import * as rlp from "rlp"; + +export class OrderCancel { + orderBytes: Uint8Array; + + constructor(orderBytes: Uint8Array) { + this.orderBytes = orderBytes; + } + + encode() { + let rlpInput: rlp.Input = [this.orderBytes]; + return rlp.encode(rlpInput); + } + + decode(data: Uint8Array) { + return rlp.decode(data); + } +} diff --git a/contracts/solana/scripts/types/order-fill.ts b/contracts/solana/scripts/types/order-fill.ts new file mode 100644 index 000000000..78c7c8656 --- /dev/null +++ b/contracts/solana/scripts/types/order-fill.ts @@ -0,0 +1,22 @@ +import * as rlp from "rlp"; + +export class OrderFill { + id: number; + orderBytes: Uint8Array; + solver: string; + + constructor(id: number, orderBytes: Uint8Array, solver: string) { + this.id = id; + this.orderBytes = orderBytes; + this.solver = solver; + } + + encode() { + let rlpInput: rlp.Input = [this.id, this.orderBytes, this.solver]; + return rlp.encode(rlpInput); + } + + decode(data: Uint8Array) { + return rlp.decode(data); + } +} diff --git a/contracts/solana/scripts/types/order_message.ts b/contracts/solana/scripts/types/order_message.ts new file mode 100644 index 000000000..88f99f53a --- /dev/null +++ b/contracts/solana/scripts/types/order_message.ts @@ -0,0 +1,25 @@ +import * as rlp from "rlp"; + +export enum MessageType { + FILL = 1, + CANCEL = 2, +} + +export class OrderMessage { + messageType: MessageType; + message: Uint8Array; + + constructor(messageType: MessageType, message: Uint8Array) { + this.messageType = messageType; + this.message = message; + } + + encode() { + let rlpInput: rlp.Input = [this.messageType, this.message]; + return rlp.encode(rlpInput); + } + + decode(data: Uint8Array) { + return rlp.decode(data); + } +} diff --git a/contracts/solana/scripts/types/swap-order.ts b/contracts/solana/scripts/types/swap-order.ts new file mode 100644 index 000000000..cb938bd09 --- /dev/null +++ b/contracts/solana/scripts/types/swap-order.ts @@ -0,0 +1,79 @@ +import * as rlp from "rlp"; +import * as anchor from "@coral-xyz/anchor"; + +export class SwapOrder { + id: number; + emitter: string; + srcNID: string; + dstNID: string; + creator: string; + destinationAddress: string; + token: string; + amount: number; + toToken: string; + toAmount: number; + data: Buffer; + + constructor( + id: number, + emitter: string, + srcNID: string, + dstNID: string, + creator: string, + destinationAddress: string, + token: string, + amount: number, + toToken: string, + toAmount: number, + data: Buffer + ) { + this.id = id; + this.emitter = emitter; + this.srcNID = srcNID; + this.dstNID = dstNID; + this.creator = creator; + this.destinationAddress = destinationAddress; + this.token = token; + this.amount = amount; + this.toToken = toToken; + this.toAmount = toAmount; + this.data = data; + } + + static from(swap: any): SwapOrder { + return new SwapOrder( + swap.id.toNumber(), + swap.emitter, + swap.srcNid, + swap.dstNid, + swap.creator, + swap.destinationAddress, + swap.token, + swap.amount.toNumber(), + swap.toToken, + swap.toAmount.toNumber(), + swap.data + ); + } + + encode() { + let rlpInput: rlp.Input = [ + this.id, + this.emitter, + this.srcNID, + this.dstNID, + this.creator, + this.destinationAddress, + this.token, + this.amount, + this.toToken, + this.toAmount, + this.data, + ]; + return rlp.encode(rlpInput); + } + + decode(data: Buffer) { + return rlp.decode(data); + } +} diff --git a/contracts/solana/scripts/utils/index.ts b/contracts/solana/scripts/utils/index.ts new file mode 100644 index 000000000..58fef57ef --- /dev/null +++ b/contracts/solana/scripts/utils/index.ts @@ -0,0 +1,42 @@ +import fs from "fs"; +import { createHash } from "crypto"; +import createKeccakHash from "keccak"; +import { Keypair, PublicKey } from "@solana/web3.js"; + +export function loadKeypairFromFile(path: string): Keypair { + return Keypair.fromSecretKey( + Buffer.from(JSON.parse(fs.readFileSync(path, "utf-8"))) + ); +} + +export const sleep = (seconds: number) => { + return new Promise((resolve) => setTimeout(resolve, seconds * 1000)); +}; + +export const hash = (message: Uint8Array) => { + return createHash("").update(message).digest("hex"); +}; + +export const keccakHash = (message: Uint8Array) => { + return createKeccakHash("keccak256").update(message).digest("hex"); +}; + +export const uint128ToArray = (num: any) => { + if (typeof num === "string" || typeof num === "number") { + num = BigInt(num); + } else if (!(num instanceof BigInt)) { + throw new Error("Input must be a BigInt or convertible to a BigInt."); + } + + let buffer = new ArrayBuffer(16); + let view = new DataView(buffer); + + view.setBigUint64(0, num >> BigInt(64), false); + view.setBigUint64(8, num & BigInt("0xFFFFFFFFFFFFFFFF"), false); + + return new Uint8Array(buffer); +}; + +export const SYSVAR_INSTRUCTIONS_ID = new PublicKey( + "Sysvar1nstructions1111111111111111111111111" +); diff --git a/contracts/solana/scripts/utils/transaction.ts b/contracts/solana/scripts/utils/transaction.ts new file mode 100644 index 000000000..dd6e3f3bd --- /dev/null +++ b/contracts/solana/scripts/utils/transaction.ts @@ -0,0 +1,128 @@ +import { + Connection, + Keypair, + AddressLookupTableProgram, + TransactionMessage, + TransactionInstruction, + PublicKey, + VersionedTransaction, +} from "@solana/web3.js"; + +import { sleep } from "./index"; + +export class TxnHelpers { + connection: Connection; + payer: Keypair; + lookupTable: PublicKey; + + constructor(connection: Connection, payer: Keypair) { + this.connection = connection; + this.payer = payer; + } + + async createAddressLookupTable() { + let recentSlot = await this.connection.getSlot("max"); + + let [createLookupTableIx, lookupTable] = + AddressLookupTableProgram.createLookupTable({ + authority: this.payer.publicKey, + payer: this.payer.publicKey, + recentSlot, + }); + + const tx = await this.buildV0Txn([createLookupTableIx], [this.payer]); + + await this.connection.sendTransaction(tx); + return (this.lookupTable = lookupTable); + } + + async extendAddressLookupTable(addresses: PublicKey[]) { + await sleep(2); + + let extendLookupTableIx = AddressLookupTableProgram.extendLookupTable({ + addresses, + authority: this.payer.publicKey, + lookupTable: this.lookupTable, + payer: this.payer.publicKey, + }); + + const tx = await this.buildV0Txn([extendLookupTableIx], [this.payer]); + await this.connection.sendTransaction(tx); + } + + async getAddressLookupTable() { + return await this.connection + .getAddressLookupTable(this.lookupTable) + .then((res) => res.value); + } + + async printAddressLookupTable() { + await sleep(2); + + const lookupTableAccount = await this.getAddressLookupTable(); + console.log(`Lookup Table: ${this.lookupTable}`); + + for (let i = 0; i < lookupTableAccount.state.addresses.length; i++) { + const address = lookupTableAccount.state.addresses[i]; + console.log( + `Index: ${i.toString().padEnd(2)} Address: ${address.toBase58()}` + ); + } + } + + async buildV0Txn(instructions: TransactionInstruction[], signers: Keypair[]) { + let blockHash = await this.connection + .getLatestBlockhash() + .then((res) => res.blockhash); + + const messageV0 = new TransactionMessage({ + payerKey: signers[0].publicKey, + recentBlockhash: blockHash, + instructions, + }).compileToV0Message(); + + const tx = new VersionedTransaction(messageV0); + signers.forEach((s) => tx.sign([s])); + return tx; + } + + async buildTxnWithLookupTable( + instructions: TransactionInstruction[], + signers: Keypair[] + ) { + await sleep(2); + + const lookupTableAccount = await this.connection + .getAddressLookupTable(this.lookupTable) + .then((res) => res.value); + + let blockhash = await this.connection + .getLatestBlockhash() + .then((res) => res.blockhash); + + let messageV0 = new TransactionMessage({ + payerKey: signers[0].publicKey, + recentBlockhash: blockhash, + instructions, + }).compileToV0Message([lookupTableAccount]); + + const tx = new VersionedTransaction(messageV0); + signers.forEach((s) => tx.sign([s])); + return tx; + } + + async airdrop(to: PublicKey, lamports: number) { + let aridropTx = await this.connection.requestAirdrop(to, lamports); + await this.connection.confirmTransaction(aridropTx, "confirmed"); + } + + async logParsedTx(txSignature: string) { + await sleep(2); + console.log( + await this.connection.getParsedTransaction(txSignature, { + commitment: "confirmed", + maxSupportedTransactionVersion: 0, + }) + ); + } +} diff --git a/contracts/solana/tests/begin.ts b/contracts/solana/tests/begin.ts new file mode 100644 index 000000000..ff007faaf --- /dev/null +++ b/contracts/solana/tests/begin.ts @@ -0,0 +1,201 @@ +import * as anchor from "@coral-xyz/anchor"; +import { Keypair, LAMPORTS_PER_SOL } from "@solana/web3.js"; +import { assert, expect } from "chai"; + +import { TxnHelpers } from "../scripts/utils/transaction"; +import { IntentPda, intentProgram, wallet, connection } from "../scripts/setup"; +import { SYSTEM_PROGRAM_ID } from "@coral-xyz/anchor/dist/cjs/native/system"; +import { sleep } from "../scripts/utils"; +import { TestContext } from "./setup"; + +describe("Initialize", () => { + let srcNid = "solana"; + let dstNid = "icon"; + + const ctx = new TestContext(srcNid, dstNid); + const txnHelpers = new TxnHelpers(connection, wallet.payer); + + it("should initialize intent program", async () => { + let feeHandler = Keypair.generate(); + let ix = await intentProgram.methods + .initialize(srcNid, feeHandler.publicKey) + .accountsStrict({ + signer: wallet.publicKey, + systemProgram: SYSTEM_PROGRAM_ID, + config: IntentPda.config().pda, + vaultNativeAccount: IntentPda.vaultNative().pda, + }) + .instruction(); + + let tx = await txnHelpers.buildV0Txn([ix], [wallet.payer]); + await connection.sendTransaction(tx); + await sleep(2); + + let config = await ctx.getConfig(); + assert.equal(config.networkId, srcNid); + assert.equal(config.connSn.toNumber(), 0); + assert.equal(config.depositId.toNumber(), 0); + assert.equal(config.protocolFee.toNumber(), 0); + assert.equal(config.feeHandler.toString(), feeHandler.publicKey.toString()); + assert.equal(config.admin.toString(), wallet.publicKey.toString()); + }); + + it("should fail when initializing intent program two times", async () => { + try { + let feeHandler = Keypair.generate(); + let ix = await intentProgram.methods + .initialize("solana", feeHandler.publicKey) + .accountsStrict({ + signer: wallet.publicKey, + systemProgram: SYSTEM_PROGRAM_ID, + config: IntentPda.config().pda, + vaultNativeAccount: IntentPda.vaultNative().pda, + }) + .instruction(); + + let tx = await txnHelpers.buildV0Txn([ix], [wallet.payer]); + await connection.sendTransaction(tx); + } catch (err) { + expect(err.message).to.includes( + "Error processing Instruction 0: custom program error: 0x0" + ); + } + }); + + it("should set protocol fee", async () => { + let protocolFee = 100; + let ix = await intentProgram.methods + .setProtocolFee(new anchor.BN(protocolFee)) + .accountsStrict({ + admin: wallet.publicKey, + config: IntentPda.config().pda, + }) + .instruction(); + + let tx = await txnHelpers.buildV0Txn([ix], [wallet.payer]); + await connection.sendTransaction(tx); + await sleep(2); + + let config = await ctx.getConfig(); + assert.equal(config.protocolFee.toNumber(), protocolFee); + }); + + it("should fail to set protocol fee", async () => { + let protocolFee = 100; + let newKeypair = Keypair.generate(); + await txnHelpers.airdrop(newKeypair.publicKey, LAMPORTS_PER_SOL); + + try { + let ix = await intentProgram.methods + .setProtocolFee(new anchor.BN(protocolFee)) + .accountsStrict({ + admin: newKeypair.publicKey, + config: IntentPda.config().pda, + }) + .instruction(); + + let tx = await txnHelpers.buildV0Txn([ix], [newKeypair]); + await connection.sendTransaction(tx); + } catch (err) { + expect(err.message).to.includes("Only Admin"); + } + }); + + it("should set fee handler", async () => { + let newFeeHandler = Keypair.generate(); + let ix = await intentProgram.methods + .setFeeHandler(newFeeHandler.publicKey) + .accountsStrict({ + admin: wallet.publicKey, + config: IntentPda.config().pda, + }) + .instruction(); + + let tx = await txnHelpers.buildV0Txn([ix], [wallet.payer]); + await connection.sendTransaction(tx); + await sleep(2); + + let config = await ctx.getConfig(); + assert.equal( + config.feeHandler.toString(), + newFeeHandler.publicKey.toString() + ); + }); + + it("should fail to set fee handler", async () => { + let newFeeHandler = Keypair.generate(); + let newKeypair = Keypair.generate(); + await txnHelpers.airdrop(newKeypair.publicKey, LAMPORTS_PER_SOL); + + try { + let ix = await intentProgram.methods + .setFeeHandler(newFeeHandler.publicKey) + .accountsStrict({ + admin: newKeypair.publicKey, + config: IntentPda.config().pda, + }) + .instruction(); + + let tx = await txnHelpers.buildV0Txn([ix], [newKeypair]); + await connection.sendTransaction(tx); + } catch (err) { + expect(err.message).to.includes("Only Admin"); + } + }); + + it("should fail to set admin", async () => { + let newAdmin = Keypair.generate(); + let newKeypair = Keypair.generate(); + await txnHelpers.airdrop(newKeypair.publicKey, LAMPORTS_PER_SOL); + + try { + let ix = await intentProgram.methods + .setAdmin(newAdmin.publicKey) + .accountsStrict({ + admin: newKeypair.publicKey, + config: IntentPda.config().pda, + }) + .instruction(); + + let tx = await txnHelpers.buildV0Txn([ix], [newKeypair]); + await connection.sendTransaction(tx); + } catch (err) { + expect(err.message).to.includes("Only Admin"); + } + }); + + it("should set admin", async () => { + let newAdmin = Keypair.generate(); + await txnHelpers.airdrop(newAdmin.publicKey, LAMPORTS_PER_SOL); + + let ix = await intentProgram.methods + .setAdmin(newAdmin.publicKey) + .accountsStrict({ + admin: wallet.publicKey, + config: IntentPda.config().pda, + }) + .instruction(); + + let tx = await txnHelpers.buildV0Txn([ix], [wallet.payer]); + await connection.sendTransaction(tx); + await sleep(2); + + let config = await ctx.getConfig(); + assert.equal(config.admin.toString(), newAdmin.publicKey.toString()); + + let anotherIx = await intentProgram.methods + .setAdmin(wallet.publicKey) + .accountsStrict({ + admin: newAdmin.publicKey, + config: IntentPda.config().pda, + }) + .instruction(); + + let anotherTx = await txnHelpers.buildV0Txn([anotherIx], [newAdmin]); + await connection.sendTransaction(anotherTx); + await sleep(2); + + let latestConfig = await ctx.getConfig(); + assert.equal(latestConfig.admin.toString(), wallet.publicKey.toString()); + }); +}); diff --git a/contracts/solana/tests/cancel.ts b/contracts/solana/tests/cancel.ts new file mode 100644 index 000000000..ed899e3d6 --- /dev/null +++ b/contracts/solana/tests/cancel.ts @@ -0,0 +1,167 @@ +import * as anchor from "@coral-xyz/anchor"; +import { assert, expect } from "chai"; +import { Keypair, LAMPORTS_PER_SOL } from "@solana/web3.js"; +import { SYSTEM_PROGRAM_ID } from "@coral-xyz/anchor/dist/cjs/native/system"; + +import { TxnHelpers } from "../scripts/utils/transaction"; +import { + IntentPda, + intentProgram, + wallet, + connection, + getSwapIx, + getCancelIx, +} from "../scripts/setup"; +import { SwapOrder } from "../scripts/types"; +import { sleep } from "../scripts/utils"; +import { TestContext } from "./setup"; + +describe("Cancel", async () => { + let srcNid = "solana"; + let dstNid = "icon"; + + const ctx = new TestContext(srcNid, dstNid); + const txnHelpers = new TxnHelpers(connection, wallet.payer); + + it("should cancel the order", async () => { + let creator = Keypair.generate(); + await txnHelpers.airdrop(creator.publicKey, LAMPORTS_PER_SOL * 10); + + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + const swapIx = await getSwapIx(swap); + const swapTx = await txnHelpers.buildV0Txn([swapIx], [creator]); + await connection.sendTransaction(swapTx); + await sleep(2); + + const config = await ctx.getConfig(); + swap.id = config.depositId; + + const cancelIx = await getCancelIx(swap); + const cancelTx = await txnHelpers.buildV0Txn([cancelIx], [creator]); + await connection.sendTransaction(cancelTx); + await sleep(2); + + // conn_sn should be increased by one + const afterConfig = await ctx.getConfig(); + assert.equal(afterConfig.connSn.toNumber(), config.connSn.toNumber() + 1); + }); + + it("should cancel the order and resolve", async () => { + let creator = Keypair.generate(); + await txnHelpers.airdrop(creator.publicKey, LAMPORTS_PER_SOL * 10); + + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid: srcNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + const swapIx = await getSwapIx(swap); + const swapTx = await txnHelpers.buildV0Txn([swapIx], [creator]); + await connection.sendTransaction(swapTx); + await sleep(2); + + const config = await ctx.getConfig(); + swap.id = config.depositId; + const swapOrder = SwapOrder.from(swap); + + const cancelIx = await getCancelIx(swap); + const cancelTx = await txnHelpers.buildV0Txn([cancelIx], [creator]); + await connection.sendTransaction(cancelTx); + await sleep(2); + + // order finished account should be created + const orderFinished = await ctx.getOrderFinishedAccount(swapOrder); + assert.equal(orderFinished.finished, true); + + // conn_sn should be increased by one + const afterConfig = await ctx.getConfig(); + assert.equal(afterConfig.connSn.toNumber(), config.connSn.toNumber() + 1); + }); + + it("should fail if the creator is not signer", async () => { + let creator = Keypair.generate(); + await txnHelpers.airdrop(creator.publicKey, LAMPORTS_PER_SOL * 10); + + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + const swapIx = await getSwapIx(swap); + const swapTx = await txnHelpers.buildV0Txn([swapIx], [creator]); + await connection.sendTransaction(swapTx); + await sleep(2); + + const config = await ctx.getConfig(); + swap.id = config.depositId; + const swapOrder = SwapOrder.from(swap); + + let anotherSigner = Keypair.generate(); + await txnHelpers.airdrop(anotherSigner.publicKey, LAMPORTS_PER_SOL * 10); + + const cancelIx = await intentProgram.methods + .cancel(swap) + .accountsStrict({ + systemProgram: SYSTEM_PROGRAM_ID, + signer: anotherSigner.publicKey, + config: IntentPda.config().pda, + orderAccount: IntentPda.order( + creator.publicKey, + swapOrder.dstNID, + swapOrder.amount, + swapOrder.toAmount + ).pda, + orderFinished: IntentPda.orderFinished(swapOrder).pda, + }) + .instruction(); + const cancelTx = await txnHelpers.buildV0Txn([cancelIx], [anotherSigner]); + try { + await connection.sendTransaction(cancelTx); + } catch (err) { + expect(err.message).to.includes("Signer must be a swap creator"); + } + }); +}); diff --git a/contracts/solana/tests/fill.ts b/contracts/solana/tests/fill.ts new file mode 100644 index 000000000..13a79181e --- /dev/null +++ b/contracts/solana/tests/fill.ts @@ -0,0 +1,575 @@ +import * as anchor from "@coral-xyz/anchor"; +import { assert, expect } from "chai"; +import { + ComputeBudgetProgram, + Keypair, + LAMPORTS_PER_SOL, + PublicKey, +} from "@solana/web3.js"; +import { SYSTEM_PROGRAM_ID } from "@coral-xyz/anchor/dist/cjs/native/system"; +import { + ASSOCIATED_TOKEN_PROGRAM_ID, + createAssociatedTokenAccount, + getAccount, + getAssociatedTokenAddress, + TOKEN_PROGRAM_ID, +} from "@solana/spl-token"; + +import { TxnHelpers } from "../scripts/utils/transaction"; +import { + IntentPda, + intentProgram, + wallet, + connection, + getFillIx, + getSwapIx, +} from "../scripts/setup"; +import { SwapOrder } from "../scripts/types"; +import { sleep } from "../scripts/utils"; +import { TestContext } from "./setup"; + +describe("Fill", async () => { + let srcNid = "solana"; + let dstNid = "icon"; + let mintKey: PublicKey; + + const ctx = new TestContext(srcNid, dstNid); + const txnHelpers = new TxnHelpers(connection, wallet.payer); + + before(async () => { + mintKey = await ctx.createMint(); + }); + + it("should fill spl token", async () => { + let signer = Keypair.generate(); + await txnHelpers.airdrop(signer.publicKey, LAMPORTS_PER_SOL * 10); + + let solverAddress = Keypair.generate(); + let destination = Keypair.generate(); + let signerTokenBalance = 1000000000 * 100; + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + const config = await ctx.getConfig(); + const signerTokenAddress = (await ctx.mintToken(signer.publicKey, 100)) + .address; + + // Program expects this token account to be already initialized + const feeHandlerTokenAccount = await createAssociatedTokenAccount( + connection, + signer, + mintKey, + config.feeHandler + ); + // Program initializes this account if not already initialized + let destinationTokenAccount = await getAssociatedTokenAddress( + mintKey, + destination.publicKey + ); + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: signer.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: mintKey.toString(), + amount, + toToken: mintKey.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + let swapOrder = SwapOrder.from(swap); + + let fillIx = await getFillIx( + swap, + signer.publicKey, + solverAddress.publicKey.toString() + ); + const computeUintLimit = ComputeBudgetProgram.setComputeUnitLimit({ + units: 1000000, + }); + const computeBudgeLimit = ComputeBudgetProgram.setComputeUnitPrice({ + microLamports: 0, + }); + let fillTx = await txnHelpers.buildV0Txn( + [computeUintLimit, computeBudgeLimit, fillIx], + [signer] + ); + await connection.sendTransaction(fillTx); + await sleep(2); + + // check signer token balance + let expectedBalance = signerTokenBalance - toAmount.toNumber(); + let signerAccount = await getAccount(connection, signerTokenAddress); + assert.equal(signerAccount.amount.toString(), expectedBalance.toString()); + + // check fee handler balance + let fee = + (Number(swapOrder.toAmount) * config.protocolFee.toNumber()) / 10_000; + let feeHandlerToken = await getAccount(connection, feeHandlerTokenAccount); + assert.equal(feeHandlerToken.amount.toString(), fee.toString()); + + // check order creator (destination address) balance + let destinationTokenBalance = Number(swapOrder.toAmount) - fee; + let destinationToken = await getAccount( + connection, + destinationTokenAccount + ); + assert.equal( + destinationToken.amount.toString(), + destinationTokenBalance.toString() + ); + }); + + it("should fill native token", async () => { + let signer = Keypair.generate(); + await txnHelpers.airdrop(signer.publicKey, LAMPORTS_PER_SOL * 10); + + let solverAddress = Keypair.generate(); + let destination = Keypair.generate(); + let signerBalance = 1000000000 * 10; + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + const config = await ctx.getConfig(); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: signer.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: mintKey.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + let swapOrder = SwapOrder.from(swap); + + let fillIx = await getFillIx( + swap, + signer.publicKey, + solverAddress.publicKey.toString() + ); + let fillTx = await txnHelpers.buildV0Txn([fillIx], [signer]); + await connection.sendTransaction(fillTx); + await sleep(2); + + // check signer SOL balance + let expectedBalance = + signerBalance - (amount.toNumber() + ctx.orderFinishedAccountRent + 5000); + let signerNowBalance = await connection.getBalance(signer.publicKey); + assert.equal(signerNowBalance.toString(), expectedBalance.toString()); + + // check fee handler balance + let fee = + (Number(swapOrder.toAmount) * config.protocolFee.toNumber()) / 10_000; + let feeHandlerBalance = await connection.getBalance(config.feeHandler); + assert.equal(feeHandlerBalance.toString(), fee.toString()); + + // check order creator (destination address) balance + let destinationExpectedBalance = Number(swapOrder.toAmount) - fee; + let destinationBalance = await connection.getBalance(destination.publicKey); + assert.equal( + destinationBalance.toString(), + destinationExpectedBalance.toString() + ); + }); + + it("should fill SPL token and resolve", async () => { + // creator depsoit 11.8 SOL to receive 135k SPL token + + let solver = Keypair.generate(); + let creator = Keypair.generate(); + await txnHelpers.airdrop(solver.publicKey, LAMPORTS_PER_SOL * 20); + await txnHelpers.airdrop(creator.publicKey, LAMPORTS_PER_SOL * 20); + + let solverAddress = Keypair.generate(); + let destination = Keypair.generate(); + let amount = new anchor.BN(11800000000); + let toAmount = new anchor.BN(135000000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid: srcNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: mintKey.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + const swapIx = await getSwapIx(swap); + const swapTx = await txnHelpers.buildV0Txn([swapIx], [creator]); + await connection.sendTransaction(swapTx); + await sleep(2); + + const config = await ctx.getConfig(); + + // mint 135k SPL token to solver for paying creator destination address + const solverTokenAddress = (await ctx.mintToken(solver.publicKey, 135000)) + .address; + + // fee_handler token has to be created before filling the order + const feeHandlerTokenAddress = (await ctx.mintToken(config.feeHandler, 10)) + .address; + const feeHandlerBalanceBeforeFill = ( + await getAccount(connection, feeHandlerTokenAddress) + ).amount; + + swap.id = config.depositId; + const swapOrder = SwapOrder.from(swap); + + const vaultBalanceBeforeFill = await connection.getBalance( + IntentPda.vaultNative().pda + ); + const creatorBalanceBeforeFill = await connection.getBalance( + creator.publicKey + ); + + const fillIx = await getFillIx( + swap, + solver.publicKey, + solverAddress.publicKey.toString() + ); + const computeUintLimit = ComputeBudgetProgram.setComputeUnitLimit({ + units: 1000000, + }); + const computeBudgeLimit = ComputeBudgetProgram.setComputeUnitPrice({ + microLamports: 0, + }); + const fillTx = await txnHelpers.buildV0Txn( + [computeUintLimit, computeBudgeLimit, fillIx], + [solver] + ); + await connection.sendTransaction(fillTx); + await sleep(2); + + // solver token balance should be zero + const solverTokenAccount = await getAccount(connection, solverTokenAddress); + assert.equal(solverTokenAccount.amount.toString(), "0"); + + // fee handler should have received swap fees + const fee = ctx.calculateSwapFee( + config.protocolFee.toNumber(), + toAmount.toNumber() + ); + const feeHandlerBalance = ( + await getAccount(connection, feeHandlerTokenAddress) + ).amount; + assert.equal( + feeHandlerBalance.toString(), + (BigInt(fee) + feeHandlerBalanceBeforeFill).toString() + ); + + // destination token account should receive to_amount + const destinationTokenAddress = await getAssociatedTokenAddress( + mintKey, + destination.publicKey + ); + const destinationTokenBalance = ( + await getAccount(connection, destinationTokenAddress) + ).amount; + assert.equal( + destinationTokenBalance.toString(), + (toAmount.toNumber() - fee).toString() + ); + + // solver_address should receive 11.8 SOL + let solverBalance = await connection.getBalance(solverAddress.publicKey); + assert.equal(solverBalance.toString(), swapOrder.amount.toString()); + + // native vault account balance should be decreased by 11.8 SOL + const vaultBalanceAfterFill = await connection.getBalance( + IntentPda.vaultNative().pda + ); + assert.equal( + vaultBalanceAfterFill, + vaultBalanceBeforeFill - swapOrder.amount + ); + + // order creator should receive order_accont rent after the account is closed + const creatorBalanceAfterFill = await connection.getBalance( + creator.publicKey + ); + assert.equal( + creatorBalanceAfterFill, + creatorBalanceBeforeFill + ctx.orderAccountRent + ); + }); + + it("should fill native token and resolve", async () => { + // creator deposit 5600 SPL token to receive 2.1 SOL + + const toToken = SYSTEM_PROGRAM_ID; + const toAmount = new anchor.BN(2100000000); + const token = mintKey; + const amount = new anchor.BN(5600000000000); + + const solverAddress = Keypair.generate(); + const destination = Keypair.generate(); + const signerBalance = 1000000000 * 10; + + const solver = Keypair.generate(); + const creator = Keypair.generate(); + await txnHelpers.airdrop(solver.publicKey, signerBalance); + await txnHelpers.airdrop(creator.publicKey, signerBalance); + + // mint token to creator token account which is deposited in token vault account + await ctx.mintToken(creator.publicKey, amount.toNumber()); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid: srcNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: token.toString(), + amount, + toToken: toToken.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + const swapIx = await getSwapIx(swap); + const swapTx = await txnHelpers.buildV0Txn([swapIx], [creator]); + await connection.sendTransaction(swapTx); + await sleep(2); + + const config = await ctx.getConfig(); + + swap.id = config.depositId; + const swapOrder = SwapOrder.from(swap); + + const feeHandlerBalanceBeforeFill = await connection.getBalance( + config.feeHandler + ); + const creatorBalanceBeforeFill = await connection.getBalance( + creator.publicKey + ); + const vaultTokenBalanceBeforeFill = (await ctx.getVaultTokenAccount()) + .amount; + + const fillIx = await getFillIx( + swap, + solver.publicKey, + solverAddress.publicKey.toString() + ); + const computeUintLimit = ComputeBudgetProgram.setComputeUnitLimit({ + units: 1000000, + }); + const computeBudgeLimit = ComputeBudgetProgram.setComputeUnitPrice({ + microLamports: 0, + }); + const fillTx = await txnHelpers.buildV0Txn( + [computeUintLimit, computeBudgeLimit, fillIx], + [solver] + ); + await connection.sendTransaction(fillTx); + await sleep(2); + + // check destination address balance of order creator + const destinationBalance = await connection.getBalance( + destination.publicKey + ); + const swapFee = ctx.calculateSwapFee( + config.protocolFee.toNumber(), + swapOrder.toAmount + ); + assert.equal(destinationBalance, toAmount.toNumber() - swapFee); + + // check fee handler balance + const feeHandlerBalanceAfterFill = await connection.getBalance( + config.feeHandler + ); + assert.equal( + feeHandlerBalanceAfterFill, + feeHandlerBalanceBeforeFill + swapFee + ); + + // check solver balance + const solverTokenAddress = await getAssociatedTokenAddress( + mintKey, + solverAddress.publicKey + ); + const solverTokenAccount = await getAccount(connection, solverTokenAddress); + assert.equal(solverTokenAccount.amount.toString(), amount.toString()); + + // vault token account balance should be decreased after releasing to solver + const vaultTokenAccountAfterFill = await ctx.getVaultTokenAccount(); + assert.equal( + vaultTokenAccountAfterFill.amount.toString(), + (vaultTokenBalanceBeforeFill - BigInt(amount.toNumber())).toString() + ); + + // order creator should get rent fee of order_account + const creatorBalanceAfterFill = await connection.getBalance( + creator.publicKey + ); + assert.equal( + creatorBalanceAfterFill, + creatorBalanceBeforeFill + ctx.orderAccountRent + ); + }); + + it("should fail when destination account doesn't match with destination address", async () => { + const solver = Keypair.generate(); + const creator = Keypair.generate(); + await txnHelpers.airdrop(solver.publicKey, LAMPORTS_PER_SOL * 10); + + let solverAddress = Keypair.generate(); + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + const config = await ctx.getConfig(); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: mintKey.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + let swapOrder = SwapOrder.from(swap); + + let fillIx = await intentProgram.methods + .fill(swap, solverAddress.toString()) + .accountsStrict({ + systemProgram: SYSTEM_PROGRAM_ID, + signer: solver.publicKey, + config: IntentPda.config().pda, + feeHandler: config.feeHandler, + destinationAddress: Keypair.generate().publicKey, + orderFinished: IntentPda.orderFinished(swapOrder).pda, + feeHandlerTokenAccount: null, + destinationTokenAccount: null, + signerTokenAccount: null, + mint: null, + tokenProgram: TOKEN_PROGRAM_ID, + associatedTokenProgram: ASSOCIATED_TOKEN_PROGRAM_ID, + }) + .instruction(); + let tx = await txnHelpers.buildV0Txn([fillIx], [solver]); + + try { + await connection.sendTransaction(tx); + } catch (err) { + expect(err.message).to.includes("Destination account is not valid"); + } + }); + + it("should fail when we fill already filled order", async () => { + const solver = Keypair.generate(); + const creator = Keypair.generate(); + await txnHelpers.airdrop(solver.publicKey, LAMPORTS_PER_SOL * 10); + + let solverAddress = Keypair.generate(); + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: mintKey.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + const fillIx = await getFillIx( + swap, + solver.publicKey, + solverAddress.publicKey.toString() + ); + let tx = await txnHelpers.buildV0Txn([fillIx], [solver]); + await connection.sendTransaction(tx); + await sleep(2); + + try { + const newSolver = Keypair.generate(); + await txnHelpers.airdrop(newSolver.publicKey, LAMPORTS_PER_SOL * 10); + + const newFillIx = await getFillIx( + swap, + newSolver.publicKey, + newSolver.publicKey.toString() + ); + let tx = await txnHelpers.buildV0Txn([newFillIx], [newSolver]); + await connection.sendTransaction(tx); + } catch (err) { + expect(err.message).to.includes("Order has been already filled"); + } + }); + + it("should fail when fee handler account is not valid", async () => { + const solver = Keypair.generate(); + const creator = Keypair.generate(); + await txnHelpers.airdrop(solver.publicKey, LAMPORTS_PER_SOL * 10); + + let solverAddress = Keypair.generate(); + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: mintKey.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + let swapOrder = SwapOrder.from(swap); + + let fillIx = await intentProgram.methods + .fill(swap, solverAddress.toString()) + .accountsStrict({ + systemProgram: SYSTEM_PROGRAM_ID, + signer: solver.publicKey, + config: IntentPda.config().pda, + feeHandler: Keypair.generate().publicKey, + destinationAddress: destination.publicKey, + orderFinished: IntentPda.orderFinished(swapOrder).pda, + feeHandlerTokenAccount: null, + destinationTokenAccount: null, + signerTokenAccount: null, + mint: null, + tokenProgram: TOKEN_PROGRAM_ID, + associatedTokenProgram: ASSOCIATED_TOKEN_PROGRAM_ID, + }) + .instruction(); + let tx = await txnHelpers.buildV0Txn([fillIx], [solver]); + + try { + await connection.sendTransaction(tx); + } catch (err) { + expect(err.message).to.includes("Fee handler account is not valid"); + } + }); +}); diff --git a/contracts/solana/tests/recv-message.ts b/contracts/solana/tests/recv-message.ts new file mode 100644 index 000000000..e30c20038 --- /dev/null +++ b/contracts/solana/tests/recv-message.ts @@ -0,0 +1,241 @@ +import * as anchor from "@coral-xyz/anchor"; +import { assert, expect } from "chai"; +import { Keypair, LAMPORTS_PER_SOL } from "@solana/web3.js"; +import { SYSTEM_PROGRAM_ID } from "@coral-xyz/anchor/dist/cjs/native/system"; + +import { TxnHelpers } from "../scripts/utils/transaction"; +import { + intentProgram, + wallet, + connection, + getSwapIx, + getRecvMessageIx, +} from "../scripts/setup"; +import { + MessageType, + OrderCancel, + OrderFill, + OrderMessage, + SwapOrder, +} from "../scripts/types"; +import { sleep } from "../scripts/utils"; +import { TestContext } from "./setup"; + +describe("Receive Message", async () => { + let srcNid = "solana"; + let dstNid = "icon"; + + const ctx = new TestContext(srcNid, dstNid); + const txnHelpers = new TxnHelpers(connection, wallet.payer); + + it("should receive and resolve cancel", async () => { + let creator = Keypair.generate(); + await txnHelpers.airdrop(creator.publicKey, LAMPORTS_PER_SOL * 10); + + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + const swapIx = await getSwapIx(swap); + const swapTx = await txnHelpers.buildV0Txn([swapIx], [creator]); + await connection.sendTransaction(swapTx); + await sleep(2); + + const config = await ctx.getConfig(); + swap.id = config.depositId; + const swapOrder = SwapOrder.from(swap); + + const cancelMessage = new OrderCancel(swapOrder.encode()); + const orderMessage = new OrderMessage( + MessageType.CANCEL, + cancelMessage.encode() + ); + + const connSn = 1; + const recvMessageIx = await getRecvMessageIx( + srcNid, + connSn, + Buffer.from(orderMessage.encode()), + ctx.admin.publicKey + ); + const recvMessageTx = await txnHelpers.buildV0Txn( + [recvMessageIx], + [ctx.admin] + ); + await connection.sendTransaction(recvMessageTx); + await sleep(2); + + const configAfter = await ctx.getConfig(); + assert.equal(configAfter.connSn.toNumber(), config.connSn.toNumber() + 1); + }); + + it("should fail for duplicate message", async () => { + let creator = Keypair.generate(); + await txnHelpers.airdrop(creator.publicKey, LAMPORTS_PER_SOL * 10); + + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + const swapOrder = SwapOrder.from(swap); + + const cancelMessage = new OrderCancel(swapOrder.encode()); + const orderMessage = new OrderMessage( + MessageType.CANCEL, + cancelMessage.encode() + ); + + const connSn = 1; + const recvMessageIx = await getRecvMessageIx( + srcNid, + connSn, + Buffer.from(orderMessage.encode()), + ctx.admin.publicKey + ); + const recvMessageTx = await txnHelpers.buildV0Txn( + [recvMessageIx], + [ctx.admin] + ); + try { + await connection.sendTransaction(recvMessageTx); + } catch (err) { + expect(err.message).to.includes("Duplicate message"); + } + }); + + it("should fail when signer is not an admin", async () => { + let creator = Keypair.generate(); + await txnHelpers.airdrop(creator.publicKey, LAMPORTS_PER_SOL * 10); + + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + const swapOrder = SwapOrder.from(swap); + + const cancelMessage = new OrderCancel(swapOrder.encode()); + const orderMessage = new OrderMessage( + MessageType.CANCEL, + cancelMessage.encode() + ); + + const connSn = 1; + const recvMessageIx = await getRecvMessageIx( + srcNid, + connSn, + Buffer.from(orderMessage.encode()), + creator.publicKey + ); + const recvMessageTx = await txnHelpers.buildV0Txn( + [recvMessageIx], + [creator] + ); + try { + await connection.sendTransaction(recvMessageTx); + } catch (err) { + expect(err.message).to.includes("Only Relayer"); + } + }); + + it("should receive and resolve fill", async () => { + let creator = Keypair.generate(); + await txnHelpers.airdrop(creator.publicKey, LAMPORTS_PER_SOL * 10); + + let destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: creator.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: SYSTEM_PROGRAM_ID.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + const swapIx = await getSwapIx(swap); + const swapTx = await txnHelpers.buildV0Txn([swapIx], [creator]); + await connection.sendTransaction(swapTx); + await sleep(2); + + const config = await ctx.getConfig(); + swap.id = config.depositId; + const swapOrder = SwapOrder.from(swap); + const vaultBalanceBeforeResolve = (await ctx.getVaultAccount()).lamports; + + const fillMessage = new OrderFill( + swapOrder.id, + swapOrder.encode(), + creator.publicKey.toString() + ); + const orderMessage = new OrderMessage( + MessageType.FILL, + fillMessage.encode() + ); + + const connSn = 2; + const recvMessageIx = await getRecvMessageIx( + dstNid, + connSn, + Buffer.from(orderMessage.encode()), + ctx.admin.publicKey + ); + const recvMessageTx = await txnHelpers.buildV0Txn( + [recvMessageIx], + [ctx.admin] + ); + await connection.sendTransaction(recvMessageTx); + await sleep(2); + + const vaultBalanceAfterResolve = (await ctx.getVaultAccount()).lamports; + assert.equal( + vaultBalanceAfterResolve, + vaultBalanceBeforeResolve - swapOrder.amount + ); + }); +}); diff --git a/contracts/solana/tests/setup.ts b/contracts/solana/tests/setup.ts new file mode 100644 index 000000000..cdc36033d --- /dev/null +++ b/contracts/solana/tests/setup.ts @@ -0,0 +1,95 @@ +import { Keypair, LAMPORTS_PER_SOL, PublicKey } from "@solana/web3.js"; +import { + createMint, + getAccount, + getOrCreateAssociatedTokenAccount, + mintTo, +} from "@solana/spl-token"; + +import { IntentPda, intentProgram, wallet, connection } from "../scripts/setup"; +import { SwapOrder } from "../scripts/types"; + +export class TestContext { + srcNID: string; + dstNID: string; + admin: Keypair; + mintKey: PublicKey; + orderAccountRent: number; + vaultNativeAccountRent: number; + orderFinishedAccountRent: number; + + constructor(srcNID: string, dstNID: string) { + this.srcNID = srcNID; + this.dstNID = dstNID; + this.admin = wallet.payer; + + this.setup() + .then(() => {}) + .catch(() => console.log("setup error")); + } + + async setup() { + this.orderAccountRent = await connection.getMinimumBalanceForRentExemption( + 409 + ); + this.vaultNativeAccountRent = + await connection.getMinimumBalanceForRentExemption(9); + this.orderFinishedAccountRent = + await connection.getMinimumBalanceForRentExemption(10); + } + + async createMint() { + const mintKey = await createMint( + connection, + wallet.payer, + wallet.publicKey, + wallet.publicKey, + 9 + ); + + this.mintKey = mintKey; + return mintKey; + } + + async mintToken(toAddress: PublicKey, amount: number) { + const tokenAccount = await getOrCreateAssociatedTokenAccount( + connection, + wallet.payer, + this.mintKey, + toAddress + ); + await mintTo( + connection, + wallet.payer, + this.mintKey, + tokenAccount.address, + wallet.publicKey, + amount * LAMPORTS_PER_SOL + ); + return tokenAccount; + } + + async getConfig() { + return await intentProgram.account.config.fetch(IntentPda.config().pda); + } + + async getVaultAccount() { + return await intentProgram.account.vaultNative.getAccountInfo( + IntentPda.vaultNative().pda + ); + } + + async getOrderFinishedAccount(swapOrder: SwapOrder) { + return await intentProgram.account.orderFinished.fetch( + IntentPda.orderFinished(swapOrder).pda + ); + } + + async getVaultTokenAccount() { + return await getAccount(connection, IntentPda.vaultToken(this.mintKey).pda); + } + + calculateSwapFee(protocolFee: number, amount: number) { + return (amount * protocolFee) / 10_000; + } +} diff --git a/contracts/solana/tests/swap.ts b/contracts/solana/tests/swap.ts new file mode 100644 index 000000000..9d08858c6 --- /dev/null +++ b/contracts/solana/tests/swap.ts @@ -0,0 +1,287 @@ +import * as anchor from "@coral-xyz/anchor"; +import { assert, expect } from "chai"; +import { Keypair, LAMPORTS_PER_SOL, PublicKey } from "@solana/web3.js"; +import { SYSTEM_PROGRAM_ID } from "@coral-xyz/anchor/dist/cjs/native/system"; +import { getAccount, TOKEN_PROGRAM_ID } from "@solana/spl-token"; + +import { TxnHelpers } from "../scripts/utils/transaction"; +import { + IntentPda, + intentProgram, + wallet, + connection, + getSwapIx, +} from "../scripts/setup"; +import { SwapOrder } from "../scripts/types"; +import { sleep } from "../scripts/utils"; +import { TestContext } from "./setup"; + +describe("Swap", async () => { + let srcNid = "solana"; + let dstNid = "icon"; + let mintKey: PublicKey; + + const ctx = new TestContext(srcNid, dstNid); + const txnHelpers = new TxnHelpers(connection, wallet.payer); + + before(async () => { + mintKey = await ctx.createMint(); + }); + + it("swap token", async () => { + let signer = Keypair.generate(); + await txnHelpers.airdrop(signer.publicKey, LAMPORTS_PER_SOL * 10); + + const destination = Keypair.generate(); + let signerTokenBalance = 1000000000 * 100; + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + const configBefore = await ctx.getConfig(); + + const signerTokenAccount = await ctx.mintToken(signer.publicKey, 100); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: signer.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: mintKey.toString(), + amount, + toToken: mintKey.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + let swapIx = await getSwapIx(swap); + let swapTx = await txnHelpers.buildV0Txn([swapIx], [signer]); + await connection.sendTransaction(swapTx); + await sleep(2); + + // check signer token balance + let expectedBalance = signerTokenBalance - amount.toNumber(); + let signerAccount = await getAccount( + connection, + signerTokenAccount.address + ); + assert.equal(signerAccount.amount.toString(), expectedBalance.toString()); + + // check vault token balance + const vaultTokenAccount = await ctx.getVaultTokenAccount(); + assert.equal(vaultTokenAccount.amount.toString(), amount.toString()); + + // deposit id should be increased + let config = await ctx.getConfig(); + assert.equal( + config.depositId.toNumber(), + configBefore.depositId.toNumber() + 1 + ); + }); + + it("swap native token", async () => { + let signer = Keypair.generate(); + await txnHelpers.airdrop(signer.publicKey, LAMPORTS_PER_SOL * 10); + + const destination = Keypair.generate(); + let signerPrevBalance = LAMPORTS_PER_SOL * 10; + let amount = new anchor.BN(8000000000); + let toAmount = new anchor.BN(1000000000); + const configBefore = await ctx.getConfig(); + const vaultAccountBefore = await ctx.getVaultAccount(); + const vaultBalanceBefore = + vaultAccountBefore.lamports - ctx.vaultNativeAccountRent; + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: signer.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: mintKey.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + let swapIx = await getSwapIx(swap); + let swapTx = await txnHelpers.buildV0Txn([swapIx], [signer]); + await connection.sendTransaction(swapTx); + await sleep(2); + + // check signer SOL balance + let expectedBalance = + signerPrevBalance - (amount.toNumber() + ctx.orderAccountRent + 5000); + let signerBalance = await connection.getBalance(signer.publicKey); + assert.equal(signerBalance.toString(), expectedBalance.toString()); + + // check vault SOL balance + const vaultNative = await ctx.getVaultAccount(); + assert.equal( + vaultNative.lamports.toString(), + ( + amount.toNumber() + + ctx.vaultNativeAccountRent + + vaultBalanceBefore + ).toString() + ); + + // deposit id should be increased + let config = await ctx.getConfig(); + assert.equal( + config.depositId.toNumber(), + configBefore.depositId.toNumber() + 1 + ); + }); + + it("should fail when source network id is invalid", async () => { + let signer = Keypair.generate(); + await txnHelpers.airdrop(signer.publicKey, LAMPORTS_PER_SOL * 10); + + const destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid: "invalid", + dstNid, + creator: signer.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: mintKey.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + let swapIx = await getSwapIx(swap); + let swapTx = await txnHelpers.buildV0Txn([swapIx], [signer]); + try { + await connection.sendTransaction(swapTx); + } catch (err) { + expect(err.message).to.includes("Network ID is misconfigured"); + } + }); + + it("should fail when creator address doesn't match with signer", async () => { + let signer = Keypair.generate(); + await txnHelpers.airdrop(signer.publicKey, LAMPORTS_PER_SOL * 10); + + const destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: wallet.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: mintKey.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + let swapOrder = SwapOrder.from(swap); + + let swapIx = await intentProgram.methods + .swap(swap) + .accountsStrict({ + systemProgram: SYSTEM_PROGRAM_ID, + signer: signer.publicKey, + config: IntentPda.config().pda, + orderAccount: IntentPda.order( + signer.publicKey, + swapOrder.dstNID, + amount.toNumber(), + toAmount.toNumber() + ).pda, + vaultNativeAccount: IntentPda.vaultNative().pda, + vaultTokenAccount: null, + signerTokenAccount: null, + mint: null, + tokenProgram: TOKEN_PROGRAM_ID, + }) + .signers([signer]) + .instruction(); + + let swapTx = await txnHelpers.buildV0Txn([swapIx], [signer]); + try { + await connection.sendTransaction(swapTx); + } catch (err) { + expect(err.message).to.includes("Signer must be a swap creator"); + } + }); + + it("should fail when emitter address doesn't match with program ID", async () => { + let signer = Keypair.generate(); + await txnHelpers.airdrop(signer.publicKey, LAMPORTS_PER_SOL * 10); + + const randomEmitter = Keypair.generate(); + const destination = Keypair.generate(); + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + let swap = { + id: new anchor.BN(1), + emitter: randomEmitter.publicKey.toString(), + srcNid, + dstNid, + creator: signer.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: SYSTEM_PROGRAM_ID.toString(), + amount, + toToken: mintKey.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + let swapIx = await getSwapIx(swap); + let swapTx = await txnHelpers.buildV0Txn([swapIx], [signer]); + try { + await connection.sendTransaction(swapTx); + } catch (err) { + expect(err.message).to.includes("Emitter program ID is not valid"); + } + }); + + it("should fail when different mint account is passed than the specified one in SwapOrder", async () => { + let signer = Keypair.generate(); + await txnHelpers.airdrop(signer.publicKey, LAMPORTS_PER_SOL * 10); + + const destination = Keypair.generate(); + let signerTokenBalance = 1000000000 * 100; + let amount = new anchor.BN(1000000000); + let toAmount = new anchor.BN(1000000000); + + await ctx.mintToken(signer.publicKey, signerTokenBalance); + + let swap = { + id: new anchor.BN(1), + emitter: intentProgram.programId.toString(), + srcNid, + dstNid, + creator: signer.publicKey.toString(), + destinationAddress: destination.publicKey.toString(), + token: mintKey.toString(), + amount, + toToken: mintKey.toString(), + toAmount, + data: Buffer.from(new Uint8Array()), + }; + + let swapIx = await getSwapIx(swap); + let swapTx = await txnHelpers.buildV0Txn([swapIx], [signer]); + try { + await connection.sendTransaction(swapTx); + } catch (err) { + expect(err.message).to.includes("A token mint constraint was violated"); + } + }); +}); diff --git a/contracts/solana/tsconfig.json b/contracts/solana/tsconfig.json new file mode 100644 index 000000000..247d160aa --- /dev/null +++ b/contracts/solana/tsconfig.json @@ -0,0 +1,11 @@ +{ + "compilerOptions": { + "types": ["mocha", "chai"], + "typeRoots": ["./node_modules/@types"], + "lib": ["es2015"], + "module": "commonjs", + "target": "es6", + "esModuleInterop": true, + "resolveJsonModule": true + } +} diff --git a/contracts/solana/yarn.lock b/contracts/solana/yarn.lock new file mode 100644 index 000000000..ca8535fa2 --- /dev/null +++ b/contracts/solana/yarn.lock @@ -0,0 +1,1360 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@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" + +"@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.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" + +"@noble/curves@^1.4.2": + version "1.6.0" + resolved "https://registry.yarnpkg.com/@noble/curves/-/curves-1.6.0.tgz#be5296ebcd5a1730fccea4786d420f87abfeb40b" + integrity sha512-TlaHRXDehJuRNR9TfZDNQ45mMEd5dwUwmicsafcIX4SsNiqnCHKjE/1alYPd/lDRVhxdhUAlv8uEhMCI5zjIJQ== + dependencies: + "@noble/hashes" "1.5.0" + +"@noble/hashes@1.5.0", "@noble/hashes@^1.3.1", "@noble/hashes@^1.4.0": + version "1.5.0" + resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.5.0.tgz#abadc5ca20332db2b1b2aa3e496e9af1213570b0" + integrity sha512-1j6kQFb7QRru7eKN3ZDvRcP13rugwdxZqCjbiAVZfIJwgj2A65UmT4TgARXGlXgnRkORLTDTrO19ZErt7+QXgA== + +"@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-group@^0.0.7": + version "0.0.7" + resolved "https://registry.yarnpkg.com/@solana/spl-token-group/-/spl-token-group-0.0.7.tgz#83c00f0cd0bda33115468cd28b89d94f8ec1fee4" + integrity sha512-V1N/iX7Cr7H0uazWUT2uk27TMqlqedpXHRqqAbVO2gvmJyT0E0ummMEAVQeXZ05ZhQ/xF39DLSdBp90XebWEug== + dependencies: + "@solana/codecs" "2.0.0-rc.1" + +"@solana/spl-token-metadata@^0.1.6": + 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.4.9": + version "0.4.9" + resolved "https://registry.yarnpkg.com/@solana/spl-token/-/spl-token-0.4.9.tgz#24d032d2935f237925c3b058ba6bb1e1ece5428c" + integrity sha512-g3wbj4F4gq82YQlwqhPB0gHFXfgsC6UmyGMxtSLf/BozT/oKd59465DbnlUK8L8EcimKMavxsVAMoLcEdeCicg== + dependencies: + "@solana/buffer-layout" "^4.0.0" + "@solana/buffer-layout-utils" "^0.2.0" + "@solana/spl-token-group" "^0.0.7" + "@solana/spl-token-metadata" "^0.1.6" + buffer "^6.0.3" + +"@solana/web3.js@^1.32.0": + version "1.95.8" + resolved "https://registry.yarnpkg.com/@solana/web3.js/-/web3.js-1.95.8.tgz#2d49abda23f7a79a3cc499ab6680f7be11786ee1" + integrity sha512-sBHzNh7dHMrmNS5xPD1d0Xa2QffW/RXaxu/OysRXBfwTp+LYqGGmMtCYYwrHPrN5rjAmJCsQRNAwv4FM0t3B6g== + 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" + +"@solana/web3.js@^1.68.0": + version "1.95.4" + resolved "https://registry.yarnpkg.com/@solana/web3.js/-/web3.js-1.95.4.tgz#771603f60d75cf7556ad867e1fd2efae32f9ad09" + integrity sha512-sdewnNEA42ZSMxqkzdwEWi6fDgzwtJHaQa5ndUGEJYtoOnM6X5cvPmjoTUp7/k7bRrVAxfBgDnvQQHD6yhlLYw== + 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" + +"@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" + +"@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/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/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.9.0" + resolved "https://registry.yarnpkg.com/@types/node/-/node-22.9.0.tgz#b7f16e5c3384788542c72dc3d561a7ceae2c0365" + integrity sha512-vuyHg81vvWA1Z1ELfvLko2c8f34gyA0zaic0+Rllc5lbCnbSyuvb2Oxpm6TAUAC/2xZN3QGqxBNggD1nNR2AfQ== + dependencies: + undici-types "~6.19.8" + +"@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/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== + +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" + +agentkeepalive@^4.5.0: + version "4.5.0" + resolved "https://registry.yarnpkg.com/agentkeepalive/-/agentkeepalive-4.5.0.tgz#2673ad1389b3c418c5a20c5d7364f93ca04be923" + integrity sha512-5GG/5IbQQpC9FpkRGsSvZI5QYeSCzlJHdpBQntCsuTOxhKD8lqKhrleg2Yi7yvMIf82Ycmmqln9U8V9qwEiJew== + dependencies: + humanize-ms "^1.2.1" + +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@^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" + +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== + +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: + 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" + +base64-js@^1.3.1: + version "1.5.1" + resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" + integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== + +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.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== + +bindings@^1.3.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" + +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== + +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" + +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== + +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" + +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@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: + version "4.0.8" + resolved "https://registry.yarnpkg.com/bufferutil/-/bufferutil-4.0.8.tgz#1de6a71092d65d7766c4d8a522b261a6e787e8ea" + integrity sha512-4T53u4PdgsXqKaIctwF8ifXlRTTmEPJ8iEPWFdGZvcf7sbwYo6FKFEX9eNNAnzFZ7EzJAQ3CJeOtCRA4rDp7Pw== + dependencies: + node-gyp-build "^4.3.0" + +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== + +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@^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.3.0" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-5.3.0.tgz#67c20a7ebef70e7f3970a01f90fa210cb6860385" + integrity sha512-dLitG79d+GV1Nb/VYcCDFivJeK1hiukt9QjRNVOsUtTy1rR1YJsmpGGTZ3qJos+uw7WmWF4wUwBd9jxjocFC2w== + +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" + +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@^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.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== + +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== + +cross-fetch@^3.1.5: + version "3.1.8" + resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-3.1.8.tgz#0327eba65fd68a7d119f8fb2bf9334a1a7956f82" + integrity sha512-cvA+JwZoU0Xq+h6WkMvAUqPEYy92Obet6UdKLfW60qn99ftItKjB5T+BkyWOFWe2pUyfQ+IJHmpOTznqk1M6Kg== + dependencies: + node-fetch "^2.6.12" + +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== + +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" + +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== + +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" + +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== + +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" + +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== + +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== + +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== + +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: + version "2.3.3" + resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" + integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw== + +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== + +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" + +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@^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== + +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== + +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.3: + version "2.0.4" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" + integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== + +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-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-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== + +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.1.1: + version "4.1.2" + resolved "https://registry.yarnpkg.com/jayson/-/jayson-4.1.2.tgz#443c26a8658703e0b2e881117b09395d88b6982e" + integrity sha512-5nzMWDHy6f+koZOuYsArh2AXs73NfWYVlFyJJuCedr93GpY+Ku8qq10ropSXVfHK+H0T6paA88ww+/dV+1fBNA== + 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-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" + +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== + +keccak@^3.0.4: + 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" + +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" + +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" + +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== + +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.4: + 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.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" + +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== + +nanoid@3.3.1: + version "3.3.1" + resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.1.tgz#6347a18cac88af88f58af0b3594b723d5e99bb35" + integrity sha512-n6Vs/3KGyxPQd6uO0eH4Bv0ojGSUvuLlIHtC3Y0kEO23YRge8H9x1GCzLn28YX0H66pMkxuaeESFq4tKISKwdw== + +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-fetch@^2.6.12, 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: + 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== + +node-gyp-build@^4.3.0: + version "4.8.3" + resolved "https://registry.yarnpkg.com/node-gyp-build/-/node-gyp-build-4.8.3.tgz#9187216d24dbee29e44eb20d2ebf62a296bbea1a" + integrity sha512-EMS95CMJzdoSKoIiXo8pxKoL8DYxwIZXYlLmgPb8KUv794abpnLK6ynsCAWNliOjREKruYKdzbh76HHYUHX7nw== + +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== + +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" + +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== + +pathval@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/pathval/-/pathval-1.1.1.tgz#8534e77a77ce7ac5a2512ea21e0fdb8fcf6c3d8d" + integrity sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ== + +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== + +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" + +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" + +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" + +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== + +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== + +rlp@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/rlp/-/rlp-3.0.0.tgz#5a60725ca4314a3a165feecca1836e4f2c1e2343" + integrity sha512-PD6U2PGk6Vq2spfgiWZdomLvRGDreBLxi5jv5M8EpRo3pU6VEm31KO+HFxE18Q3vgqfDrQ9pZA3FP95rkijNKw== + +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" + +safe-buffer@^5.0.1, safe-buffer@^5.1.0, 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== + +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" + +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" + +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== + +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" + +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-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@^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@^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" + +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== + +"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== + +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" + +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-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.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== + +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== + +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== + +undici-types@~6.19.8: + version "6.19.8" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.19.8.tgz#35111c9d1437ab83a7cdc0abae2f26d88eda0a02" + integrity sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw== + +utf-8-validate@^5.0.2: + 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: + 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== + +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" + +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.5.10: + 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== + +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: + 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==