commit 09dea46948f6449faeab4761515081f9b41f71be Author: AidarKC Date: Mon May 4 14:19:40 2026 +0300 Initial import: shine + solana-shine-client-lib diff --git a/shine/.gitignore b/shine/.gitignore new file mode 100644 index 0000000..4c26dd3 --- /dev/null +++ b/shine/.gitignore @@ -0,0 +1,9 @@ +.anchor +.DS_Store +target +**/*.rs.bk +node_modules +test-ledger +.yarn +program-keypair.json +/old_vers/ diff --git a/shine/.idea/.gitignore b/shine/.idea/.gitignore new file mode 100644 index 0000000..7bc07ec --- /dev/null +++ b/shine/.idea/.gitignore @@ -0,0 +1,10 @@ +# Default ignored files +/shelf/ +/workspace.xml +# Editor-based HTTP Client requests +/httpRequests/ +# Environment-dependent path to Maven home directory +/mavenHomeManager.xml +# Datasource local storage ignored files +/dataSources/ +/dataSources.local.xml diff --git a/shine/.idea/modules.xml b/shine/.idea/modules.xml new file mode 100644 index 0000000..dd6793f --- /dev/null +++ b/shine/.idea/modules.xml @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/shine/.idea/shine.iml b/shine/.idea/shine.iml new file mode 100644 index 0000000..477406b --- /dev/null +++ b/shine/.idea/shine.iml @@ -0,0 +1,12 @@ + + + + + + + + + + + + \ No newline at end of file diff --git a/shine/.idea/vcs.xml b/shine/.idea/vcs.xml new file mode 100644 index 0000000..35eb1dd --- /dev/null +++ b/shine/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/shine/.prettierignore b/shine/.prettierignore new file mode 100644 index 0000000..4142583 --- /dev/null +++ b/shine/.prettierignore @@ -0,0 +1,7 @@ +.anchor +.DS_Store +target +node_modules +dist +build +test-ledger diff --git a/shine/AGENTS.md b/shine/AGENTS.md new file mode 100644 index 0000000..f8b669a --- /dev/null +++ b/shine/AGENTS.md @@ -0,0 +1,9 @@ +# AGENTS.md + +## Documentation Rule + +В проекте есть спецификация пользовательской PDA-записи: + +- `doc/SHINE_USER_PDA_V1.md` + +Если меняется формат записи, сериализация, правила подписи, `prev_hash`, экономика лимитов или связанные ограничения create/update, этот документ нужно обновлять в том же изменении. diff --git a/shine/Anchor.toml b/shine/Anchor.toml new file mode 100644 index 0000000..3494b7c --- /dev/null +++ b/shine/Anchor.toml @@ -0,0 +1,39 @@ +[toolchain] +package_manager = "yarn" + +[features] +resolution = true +skip-lint = false + +[programs.localnet] +shine_users = "5dFcWDNp42Xn9Vv4oDMJzM4obBJ8hvDuAtPX54fT5L3t" #тут надо если что обновлять +shine_payments = "92sgkgx7KHpbhQu81mNGHaKa7skJB7esArVdPM7paDSW" #тут надо если что обновлять + + +[programs.devnet] +shine_users = "5dFcWDNp42Xn9Vv4oDMJzM4obBJ8hvDuAtPX54fT5L3t" #тут надо если что обновлять +shine_payments = "92sgkgx7KHpbhQu81mNGHaKa7skJB7esArVdPM7paDSW" #тут надо если что обновлять + + +[workspace] +members = [ + "programs/shine_users", + "programs/shine_payments", +] + +[registry] +url = "https://api.apr.dev" + +[provider] +cluster = "devnet"#"http://127.0.0.1:8899" # это в какую сеть деплоит по умолчанию +wallet = "~/.config/solana/id.json" # а это с какого кошелько спишутся средства за деплой + +[scripts] +test = "yarn run ts-mocha -p ./tsconfig.json -t 1000000 tests/**/*.ts" + + + + + + + diff --git a/shine/Cargo.lock b/shine/Cargo.lock new file mode 100644 index 0000000..2ff187b --- /dev/null +++ b/shine/Cargo.lock @@ -0,0 +1,3027 @@ +# This file is automatically @generated by Cargo. +# It is not intended for manual editing. +version = 4 + +[[package]] +name = "aead" +version = "0.5.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d122413f284cf2d62fb1b7db97e02edb8cda96d769b16e443a4f6195e35662b0" +dependencies = [ + "crypto-common", + "generic-array", +] + +[[package]] +name = "aes" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0" +dependencies = [ + "cfg-if", + "cipher", + "cpufeatures", +] + +[[package]] +name = "aes-gcm-siv" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ae0784134ba9375416d469ec31e7c5f9fa94405049cf08c5ce5b4698be673e0d" +dependencies = [ + "aead", + "aes", + "cipher", + "ctr", + "polyval", + "subtle", + "zeroize", +] + +[[package]] +name = "ahash" +version = "0.8.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5a15f179cd60c4584b8a8c596927aadc462e27f2ca70c04e0071964a73ba7a75" +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 = "allocator-api2" +version = "0.2.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "683d7910e743518b0e34f1186f92494becacb047c7b6bf616c96772180fef923" + +[[package]] +name = "anchor-attribute-access-control" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f70fd141a4d18adf11253026b32504f885447048c7494faf5fa83b01af9c0cf" +dependencies = [ + "anchor-syn", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "anchor-attribute-account" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "715a261c57c7679581e06f07a74fa2af874ac30f86bd8ea07cca4a7e5388a064" +dependencies = [ + "anchor-syn", + "bs58", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "anchor-attribute-constant" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "730d6df8ae120321c5c25e0779e61789e4b70dc8297102248902022f286102e4" +dependencies = [ + "anchor-syn", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "anchor-attribute-error" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "27e6e449cc3a37b2880b74dcafb8e5a17b954c0e58e376432d7adc646fb333ef" +dependencies = [ + "anchor-syn", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "anchor-attribute-event" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d7710e4c54adf485affcd9be9adec5ef8846d9c71d7f31e16ba86ff9fc1dd49f" +dependencies = [ + "anchor-syn", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "anchor-attribute-program" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05ecfd49b2aeadeb32f35262230db402abed76ce87e27562b34f61318b2ec83c" +dependencies = [ + "anchor-lang-idl", + "anchor-syn", + "anyhow", + "bs58", + "heck", + "proc-macro2", + "quote", + "serde_json", + "syn 1.0.109", +] + +[[package]] +name = "anchor-derive-accounts" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be89d160793a88495af462a7010b3978e48e30a630c91de47ce2c1d3cb7a6149" +dependencies = [ + "anchor-syn", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "anchor-derive-serde" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "abc6ee78acb7bfe0c2dd2abc677aaa4789c0281a0c0ef01dbf6fe85e0fd9e6e4" +dependencies = [ + "anchor-syn", + "borsh-derive-internal", + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "anchor-derive-space" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "134a01c0703f6fd355a0e472c033f6f3e41fac1ef6e370b20c50f4c8d022cea7" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "anchor-lang" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6bab117055905e930f762c196e08f861f8dfe7241b92cee46677a3b15561a0a" +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", + "base64 0.21.7", + "bincode", + "borsh 0.10.4", + "bytemuck", + "solana-program", + "thiserror 1.0.69", +] + +[[package]] +name = "anchor-lang-idl" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "32e8599d21995f68e296265aa5ab0c3cef582fd58afec014d01bd0bce18a4418" +dependencies = [ + "anchor-lang-idl-spec", + "anyhow", + "heck", + "regex", + "serde", + "serde_json", + "sha2 0.10.9", +] + +[[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.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c08cb5d762c0694f74bd02c9a5b04ea53cefc496e2c27b3234acffca5cd076b" +dependencies = [ + "anchor-lang", + "spl-associated-token-account", + "spl-pod", + "spl-token 7.0.0", + "spl-token-2022", + "spl-token-group-interface", + "spl-token-metadata-interface", +] + +[[package]] +name = "anchor-syn" +version = "0.31.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5dc7a6d90cc643df0ed2744862cdf180587d1e5d28936538c18fc8908489ed67" +dependencies = [ + "anyhow", + "bs58", + "cargo_toml", + "heck", + "proc-macro2", + "quote", + "serde", + "serde_json", + "sha2 0.10.9", + "syn 1.0.109", + "thiserror 1.0.69", +] + +[[package]] +name = "anyhow" +version = "1.0.98" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e16d2d3311acee920a9eb8d33b8cbc1787ce4a264e85f964c2404b969bdcd487" + +[[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 = "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 = "base64" +version = "0.22.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "72b3254f16251a8381aa12e40e3c4d2f0199f8c6508fbecb9d91f575e0fbb8c6" + +[[package]] +name = "bincode" +version = "1.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b1f45e9417d87227c7a56d22e471c6206462cba514c7590c09aff4cf6d1ddcad" +dependencies = [ + "serde", +] + +[[package]] +name = "bitflags" +version = "2.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b8e56985ec62d17e9c1001dc89c88ecd7dc08e47eba5ec7c29c7b5eeecde967" + +[[package]] +name = "blake3" +version = "1.8.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3888aaa89e4b2a40fca9848e400f6a658a5a3978de7be858e209cafa8be9a4a0" +dependencies = [ + "arrayref", + "arrayvec", + "cc", + "cfg-if", + "constant_time_eq", + "digest 0.10.7", +] + +[[package]] +name = "block-buffer" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4152116fd6e9dadb291ae18fc1ec3575ed6d84c29642d97890f4b4a3417297e4" +dependencies = [ + "generic-array", +] + +[[package]] +name = "block-buffer" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3078c7629b62d3f0439517fa394996acacc5cbc91c5a20d8c658e77abd503a71" +dependencies = [ + "generic-array", +] + +[[package]] +name = "borsh" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "115e54d64eb62cdebad391c19efc9dce4981c690c85a33a12199d99bb9546fee" +dependencies = [ + "borsh-derive 0.10.4", + "hashbrown 0.13.2", +] + +[[package]] +name = "borsh" +version = "1.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ad8646f98db542e39fc66e68a20b2144f6a732636df7c2354e74645faaa433ce" +dependencies = [ + "borsh-derive 1.5.7", + "cfg_aliases", +] + +[[package]] +name = "borsh-derive" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "831213f80d9423998dd696e2c5345aba6be7a0bd8cd19e31c5243e13df1cef89" +dependencies = [ + "borsh-derive-internal", + "borsh-schema-derive-internal", + "proc-macro-crate 0.1.5", + "proc-macro2", + "syn 1.0.109", +] + +[[package]] +name = "borsh-derive" +version = "1.5.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fdd1d3c0c2f5833f22386f252fe8ed005c7f59fdcddeef025c01b4c3b9fd9ac3" +dependencies = [ + "once_cell", + "proc-macro-crate 3.3.0", + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[package]] +name = "borsh-derive-internal" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "65d6ba50644c98714aa2a70d13d7df3cd75cd2b523a2b452bf010443800976b3" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "borsh-schema-derive-internal" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "276691d96f063427be83e6692b86148e488ebba9f48f77788724ca027ba3b6d4" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "bs58" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf88ba1141d185c399bee5288d850d63b8369520c1eafc32a0430b5b6c287bf4" +dependencies = [ + "tinyvec", +] + +[[package]] +name = "bumpalo" +version = "3.17.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1628fb46dfa0b37568d12e5edd512553eccf6a22a78e8bde00bb4aed84d5bdbf" + +[[package]] +name = "bv" +version = "0.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8834bb1d8ee5dc048ee3124f2c7c1afcc6bc9aed03f11e9dfd8c69470a5db340" +dependencies = [ + "feature-probe", + "serde", +] + +[[package]] +name = "bytemuck" +version = "1.23.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9134a6ef01ce4b366b50689c94f82c14bc72bc5d0386829828a2e2752ef7958c" +dependencies = [ + "bytemuck_derive", +] + +[[package]] +name = "bytemuck_derive" +version = "1.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7ecc273b49b3205b83d648f0690daa588925572cc5063745bfe547fe7ec8e1a1" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[package]] +name = "byteorder" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" + +[[package]] +name = "cargo_toml" +version = "0.19.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a98356df42a2eb1bd8f1793ae4ee4de48e384dd974ce5eac8eee802edb7492be" +dependencies = [ + "serde", + "toml 0.8.22", +] + +[[package]] +name = "cc" +version = "1.2.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f4ac86a9e5bc1e2b3449ab9d7d3a6a405e3d1bb28d7b9be8614f55846ae3766" +dependencies = [ + "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 = "cipher" +version = "0.4.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad" +dependencies = [ + "crypto-common", + "inout", +] + +[[package]] +name = "common" +version = "0.1.0" +dependencies = [ + "anchor-lang", +] + +[[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.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "59ed5838eebb26a2bb2e58f6d5b5316989ae9d08bab10e0e6d103e656d1b0280" +dependencies = [ + "libc", +] + +[[package]] +name = "crunchy" +version = "0.2.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "43da5946c66ffcc7745f48db692ffbb10a83bfe0afd96235c5c2a4fb23994929" + +[[package]] +name = "crypto-common" +version = "0.1.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bfb12502f3fc46cca1bb51ac28df9d618d813cdc3d2f25b9fe775a34af26bb3" +dependencies = [ + "generic-array", + "rand_core 0.6.4", + "typenum", +] + +[[package]] +name = "ctr" +version = "0.9.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0369ee1ad671834580515889b80f2ea915f23b8be8d0daa4bbaf2ac5c7590835" +dependencies = [ + "cipher", +] + +[[package]] +name = "curve25519-dalek" +version = "3.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0b9fdf9972b2bd6af2d913799d9ebc165ea4d2e65878e329d9c6b372c4491b61" +dependencies = [ + "byteorder", + "digest 0.9.0", + "rand_core 0.5.1", + "subtle", + "zeroize", +] + +[[package]] +name = "curve25519-dalek" +version = "4.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97fb8b7c4503de7d6ae7b42ab72a5a59857b4c937ec27a3d4539dba95b5ab2be" +dependencies = [ + "cfg-if", + "cpufeatures", + "curve25519-dalek-derive", + "digest 0.10.7", + "fiat-crypto", + "rand_core 0.6.4", + "rustc_version", + "serde", + "subtle", + "zeroize", +] + +[[package]] +name = "curve25519-dalek-derive" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f46882e17999c6cc590af592290432be3bce0428cb0d5f8b6715e4dc7b383eb3" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[package]] +name = "derivation-path" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6e5c37193a1db1d8ed868c03ec7b152175f26160a5b740e5e484143877e0adf0" + +[[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 3.2.0", + "ed25519", + "sha2 0.9.9", + "zeroize", +] + +[[package]] +name = "either" +version = "1.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "48c757948c5ede0e46177b7add2e67155f70e33c07fea8284df6576da70b3719" + +[[package]] +name = "equivalent" +version = "1.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "877a4ace8713b0bcf2a4e7eec82529c029f1d0619886d18145fea96c3ffe5c0f" + +[[package]] +name = "feature-probe" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "835a3dc7d1ec9e75e2b5fb4ba75396837112d2060b03f7d43bc1897c7f7211da" + +[[package]] +name = "fiat-crypto" +version = "0.2.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "28dea519a9695b9977216879a3ebfddf92f1c08c05d984f8996aecd6ecdc811d" + +[[package]] +name = "five8" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a75b8549488b4715defcb0d8a8a1c1c76a80661b5fa106b4ca0e7fce59d7d875" +dependencies = [ + "five8_core", +] + +[[package]] +name = "five8_const" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26dec3da8bc3ef08f2c04f61eab298c3ab334523e55f076354d6d6f613799a7b" +dependencies = [ + "five8_core", +] + +[[package]] +name = "five8_core" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2551bf44bc5f776c15044b9b94153a00198be06743e262afaaa61f11ac7523a5" + +[[package]] +name = "fnv" +version = "1.0.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" + +[[package]] +name = "foldhash" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d9c4f5dac5e15c24eb999c26181a6ca40b39fe946cbe4c263c7209467bc83af2" + +[[package]] +name = "generic-array" +version = "0.14.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85649ca51fd72272d7821adaf274ad91c288277713d9c18820d8499a7ff69e9a" +dependencies = [ + "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", + "libc", + "wasi 0.9.0+wasi-snapshot-preview1", +] + +[[package]] +name = "getrandom" +version = "0.2.16" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "335ff9f135e4384c8150d6f27c6daed433577f86b4750418338c01a1a2528592" +dependencies = [ + "cfg-if", + "js-sys", + "libc", + "wasi 0.11.0+wasi-snapshot-preview1", + "wasm-bindgen", +] + +[[package]] +name = "hashbrown" +version = "0.13.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "43a3c133739dddd0d2990f9a4bdf8eb4b21ef50e4851ca85ab661199821d510e" +dependencies = [ + "ahash", +] + +[[package]] +name = "hashbrown" +version = "0.15.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bf151400ff0baff5465007dd2f3e717f3fe502074ca563069ce3a6629d07b289" +dependencies = [ + "allocator-api2", + "equivalent", + "foldhash", +] + +[[package]] +name = "heck" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6d621efb26863f0e9924c6ac577e8275e5e6b77455db64ffa6c65c904e9e132c" +dependencies = [ + "unicode-segmentation", +] + +[[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 = "indexmap" +version = "2.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cea70ddb795996207ad57735b50c5982d8844f38ba9ee5f1aedcfb708a2aa11e" +dependencies = [ + "equivalent", + "hashbrown 0.15.2", +] + +[[package]] +name = "inout" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "879f10e63c20629ecabbb64a8010319738c66a5cd0c29b02d63d272b03751d01" +dependencies = [ + "generic-array", +] + +[[package]] +name = "itertools" +version = "0.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba291022dbbd398a455acf126c1e341954079855bc60dfdda641363bd6922569" +dependencies = [ + "either", +] + +[[package]] +name = "itoa" +version = "1.0.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4a5f13b858c8d314ee3e8f639011f7ccefe71f97f96e50151fb991f267928e2c" + +[[package]] +name = "js-sys" +version = "0.3.77" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1cfaf33c695fc6e08064efbc1f72ec937429614f25eef83af942d0e227c3a28f" +dependencies = [ + "once_cell", + "wasm-bindgen", +] + +[[package]] +name = "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.172" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d750af042f7ef4f724306de029d18836c26c1765a54a6a3f094cbd23a7267ffa" + +[[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", + "libsecp256k1-core", + "libsecp256k1-gen-ecmult", + "libsecp256k1-gen-genmult", + "rand 0.7.3", + "serde", + "sha2 0.9.9", +] + +[[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 = "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.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13dc2df351e3202783a1fe0d44375f7295ffb4049267b0f3018346dc122a1d94" + +[[package]] +name = "memchr" +version = "2.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "78ca9ab1a0babb1e7d5695e3530886289c18cf2f87ec19a575a0abdce112e3a3" + +[[package]] +name = "memoffset" +version = "0.9.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "488016bfae457b036d996092f6cb448677611ce4449e970ceaf42695203f218a" +dependencies = [ + "autocfg", +] + +[[package]] +name = "merlin" +version = "3.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "58c38e2799fc0978b65dfff8023ec7843e2330bb462f19198840b34b6582397d" +dependencies = [ + "byteorder", + "keccak", + "rand_core 0.6.4", + "zeroize", +] + +[[package]] +name = "mpl-token-metadata" +version = "5.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "046f0779684ec348e2759661361c8798d79021707b1392cb49f3b5eb911340ff" +dependencies = [ + "borsh 0.10.4", + "num-derive 0.3.3", + "num-traits", + "solana-program", + "thiserror 1.0.69", +] + +[[package]] +name = "num-bigint" +version = "0.4.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a5e44f723f1133c9deac646763579fdb3ac745e418f2a7af9cd0c431da1f20b9" +dependencies = [ + "num-integer", + "num-traits", +] + +[[package]] +name = "num-derive" +version = "0.3.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "876a53fff98e03a936a674b29568b0e605f06b29372c2489ff4de23f1949743d" +dependencies = [ + "proc-macro2", + "quote", + "syn 1.0.109", +] + +[[package]] +name = "num-derive" +version = "0.4.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed3955f1a9c7c0c15e092f9c887db08b1fc683305fdf6eb6684f22555355e202" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[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.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d0bca838442ec211fa11de3a8b0e0e8f3a4522575b5c4c06ed722e005036f26" +dependencies = [ + "num_enum_derive", + "rustversion", +] + +[[package]] +name = "num_enum_derive" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "680998035259dcfcafe653688bf2aa6d3e2dc05e98be6ab46afb089dc84f1df8" +dependencies = [ + "proc-macro-crate 3.3.0", + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[package]] +name = "once_cell" +version = "1.21.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "42f5e15c9953c5e4ccceeb2e7382a716482c34515315f7b03532b8b4e8393d2d" + +[[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 = "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.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b4f627cb1b25917193a259e49bdad08f671f8d9708acfd5fe0a8c1455d87220" + +[[package]] +name = "polyval" +version = "0.6.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d1fe60d06143b2430aa532c94cfe9e29783047f06c0d7fd359a9a51b729fa25" +dependencies = [ + "cfg-if", + "cpufeatures", + "opaque-debug", + "universal-hash", +] + +[[package]] +name = "ppv-lite86" +version = "0.2.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85eae3c4ed2f50dcfe72643da4befc30deadb458a9b590d720cde2f2b1e97da9" +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.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "edce586971a4dfaa28950c6f18ed55e0406c1ab88bbce2c6f6293a7aaba73d35" +dependencies = [ + "toml_edit", +] + +[[package]] +name = "proc-macro2" +version = "1.0.95" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02b3e5e68a3a1a02aad3ec490a98007cbc13c37cbe84a3cd7b8e406d76e7f778" +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 = "quote" +version = "1.0.40" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1885c039570dc00dcb4ff087a89e185fd56bae234ddc7f056a945bf36467248d" +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.16", +] + +[[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 = "redox_syscall" +version = "0.5.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "928fca9cf2aa042393a8325b9ead81d2f0df4cb12e1e24cef072922ccd99c5af" +dependencies = [ + "bitflags", +] + +[[package]] +name = "regex" +version = "1.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b544ef1b4eac5dc2db33ea63606ae9ffcfac26c1416a2806ae0bf5f56b201191" +dependencies = [ + "aho-corasick", + "memchr", + "regex-automata", + "regex-syntax", +] + +[[package]] +name = "regex-automata" +version = "0.4.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "809e8dc61f6de73b46c85f4c96486310fe304c434cfa43669d7b40f711150908" +dependencies = [ + "aho-corasick", + "memchr", + "regex-syntax", +] + +[[package]] +name = "regex-syntax" +version = "0.8.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c" + +[[package]] +name = "rustc_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.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eded382c5f5f786b989652c49544c4877d9f015cc22e145a5ea8ea66c2921cd2" + +[[package]] +name = "ryu" +version = "1.0.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "28d3b2b1366ec20994f1fd18c3c594f05c5dd4bc44d8bb0c1c632c8d6829481f" + +[[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.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "56e6fa9c48d24d85fb3de5ad847117517440f6beceb7798af16b4a87d616b8d0" + +[[package]] +name = "serde" +version = "1.0.219" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5f0e2c6ed6606019b4e29e69dbaba95b11854410e5347d525002456dbbb786b6" +dependencies = [ + "serde_derive", +] + +[[package]] +name = "serde_bytes" +version = "0.11.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8437fd221bde2d4ca316d61b90e337e9e702b3820b87d63caa9ba6c02bd06d96" +dependencies = [ + "serde", +] + +[[package]] +name = "serde_derive" +version = "1.0.219" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b0276cf7f2c73365f7157c8123c21cd9a50fbbd844757af28ca1f5925fc2a00" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[package]] +name = "serde_json" +version = "1.0.140" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "20068b6e96dc6c9bd23e01df8827e6c7e1f2fddd43c21810382803c136b99373" +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 = "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.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7507d819769d01a365ab707794a4084392c824f54a7a6a7862f8c3d0892b283" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest 0.10.7", +] + +[[package]] +name = "sha3" +version = "0.10.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75872d278a8f37ef87fa0ddbda7802605cb18344497949862c0d4dcb291eba60" +dependencies = [ + "digest 0.10.7", + "keccak", +] + +[[package]] +name = "shine_payments" +version = "0.1.0" +dependencies = [ + "anchor-lang", + "anchor-spl", + "common", + "mpl-token-metadata", + "spl-token 4.0.2", +] + +[[package]] +name = "shine_users" +version = "0.1.0" +dependencies = [ + "anchor-lang", + "common", + "ed25519-dalek", + "sha2 0.10.9", +] + +[[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 = "smallvec" +version = "1.15.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8917285742e9f3e1683f0a9c4e6b57960b7314d0b08d30d1ecd426713ee2eee9" + +[[package]] +name = "solana-account" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0f949fe4edaeaea78c844023bfc1c898e0b1f5a100f8a8d2d0f85d0a7b090258" +dependencies = [ + "solana-account-info", + "solana-clock", + "solana-instruction", + "solana-pubkey", + "solana-sdk-ids", +] + +[[package]] +name = "solana-account-info" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e0c17d606a298a205fae325489fbed88ee6dc4463c111672172327e741c8905d" +dependencies = [ + "bincode", + "serde", + "solana-program-error", + "solana-program-memory", + "solana-pubkey", +] + +[[package]] +name = "solana-address-lookup-table-interface" +version = "2.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d1673f67efe870b64a65cb39e6194be5b26527691ce5922909939961a6e6b395" +dependencies = [ + "bincode", + "bytemuck", + "serde", + "serde_derive", + "solana-clock", + "solana-instruction", + "solana-pubkey", + "solana-sdk-ids", + "solana-slot-hashes", +] + +[[package]] +name = "solana-atomic-u64" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d52e52720efe60465b052b9e7445a01c17550666beec855cce66f44766697bc2" +dependencies = [ + "parking_lot", +] + +[[package]] +name = "solana-big-mod-exp" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "75db7f2bbac3e62cfd139065d15bcda9e2428883ba61fc8d27ccb251081e7567" +dependencies = [ + "num-bigint", + "num-traits", + "solana-define-syscall", +] + +[[package]] +name = "solana-bincode" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19a3787b8cf9c9fe3dd360800e8b70982b9e5a8af9e11c354b6665dd4a003adc" +dependencies = [ + "bincode", + "serde", + "solana-instruction", +] + +[[package]] +name = "solana-blake3-hasher" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1a0801e25a1b31a14494fc80882a036be0ffd290efc4c2d640bfcca120a4672" +dependencies = [ + "blake3", + "solana-define-syscall", + "solana-hash", + "solana-sanitize", +] + +[[package]] +name = "solana-borsh" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "718333bcd0a1a7aed6655aa66bef8d7fb047944922b2d3a18f49cbc13e73d004" +dependencies = [ + "borsh 0.10.4", + "borsh 1.5.7", +] + +[[package]] +name = "solana-clock" +version = "2.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1bb482ab70fced82ad3d7d3d87be33d466a3498eb8aa856434ff3c0dfc2e2e31" +dependencies = [ + "serde", + "serde_derive", + "solana-sdk-ids", + "solana-sdk-macro", + "solana-sysvar-id", +] + +[[package]] +name = "solana-cpi" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8dc71126edddc2ba014622fc32d0f5e2e78ec6c5a1e0eb511b85618c09e9ea11" +dependencies = [ + "solana-account-info", + "solana-define-syscall", + "solana-instruction", + "solana-program-error", + "solana-pubkey", + "solana-stable-layout", +] + +[[package]] +name = "solana-curve25519" +version = "2.3.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eae4261b9a8613d10e77ac831a8fa60b6fa52b9b103df46d641deff9f9812a23" +dependencies = [ + "bytemuck", + "bytemuck_derive", + "curve25519-dalek 4.1.3", + "solana-define-syscall", + "subtle", + "thiserror 2.0.12", +] + +[[package]] +name = "solana-decode-error" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8c781686a18db2f942e70913f7ca15dc120ec38dcab42ff7557db2c70c625a35" +dependencies = [ + "num-traits", +] + +[[package]] +name = "solana-define-syscall" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2ae3e2abcf541c8122eafe9a625d4d194b4023c20adde1e251f94e056bb1aee2" + +[[package]] +name = "solana-derivation-path" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "939756d798b25c5ec3cca10e06212bdca3b1443cb9bb740a38124f58b258737b" +dependencies = [ + "derivation-path", + "qstring", + "uriparse", +] + +[[package]] +name = "solana-epoch-rewards" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "86b575d3dd323b9ea10bb6fe89bf6bf93e249b215ba8ed7f68f1a3633f384db7" +dependencies = [ + "serde", + "serde_derive", + "solana-hash", + "solana-sdk-ids", + "solana-sdk-macro", + "solana-sysvar-id", +] + +[[package]] +name = "solana-epoch-schedule" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3fce071fbddecc55d727b1d7ed16a629afe4f6e4c217bc8d00af3b785f6f67ed" +dependencies = [ + "serde", + "serde_derive", + "solana-sdk-ids", + "solana-sdk-macro", + "solana-sysvar-id", +] + +[[package]] +name = "solana-example-mocks" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "84461d56cbb8bb8d539347151e0525b53910102e4bced875d49d5139708e39d3" +dependencies = [ + "serde", + "serde_derive", + "solana-address-lookup-table-interface", + "solana-clock", + "solana-hash", + "solana-instruction", + "solana-keccak-hasher", + "solana-message", + "solana-nonce", + "solana-pubkey", + "solana-sdk-ids", + "solana-system-interface", + "thiserror 2.0.12", +] + +[[package]] +name = "solana-feature-gate-interface" +version = "2.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "43f5c5382b449e8e4e3016fb05e418c53d57782d8b5c30aa372fc265654b956d" +dependencies = [ + "bincode", + "serde", + "serde_derive", + "solana-account", + "solana-account-info", + "solana-instruction", + "solana-program-error", + "solana-pubkey", + "solana-rent", + "solana-sdk-ids", + "solana-system-interface", +] + +[[package]] +name = "solana-fee-calculator" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d89bc408da0fb3812bc3008189d148b4d3e08252c79ad810b245482a3f70cd8d" +dependencies = [ + "log", + "serde", + "serde_derive", +] + +[[package]] +name = "solana-hash" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b5b96e9f0300fa287b545613f007dfe20043d7812bee255f418c1eb649c93b63" +dependencies = [ + "borsh 1.5.7", + "bytemuck", + "bytemuck_derive", + "five8", + "js-sys", + "serde", + "serde_derive", + "solana-atomic-u64", + "solana-sanitize", + "wasm-bindgen", +] + +[[package]] +name = "solana-instruction" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "47298e2ce82876b64f71e9d13a46bc4b9056194e7f9937ad3084385befa50885" +dependencies = [ + "bincode", + "borsh 1.5.7", + "getrandom 0.2.16", + "js-sys", + "num-traits", + "serde", + "serde_derive", + "solana-define-syscall", + "solana-pubkey", + "wasm-bindgen", +] + +[[package]] +name = "solana-instructions-sysvar" +version = "2.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e0e85a6fad5c2d0c4f5b91d34b8ca47118fc593af706e523cdbedf846a954f57" +dependencies = [ + "bitflags", + "solana-account-info", + "solana-instruction", + "solana-program-error", + "solana-pubkey", + "solana-sanitize", + "solana-sdk-ids", + "solana-serialize-utils", + "solana-sysvar-id", +] + +[[package]] +name = "solana-keccak-hasher" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c7aeb957fbd42a451b99235df4942d96db7ef678e8d5061ef34c9b34cae12f79" +dependencies = [ + "sha3", + "solana-define-syscall", + "solana-hash", + "solana-sanitize", +] + +[[package]] +name = "solana-last-restart-slot" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4a6360ac2fdc72e7463565cd256eedcf10d7ef0c28a1249d261ec168c1b55cdd" +dependencies = [ + "serde", + "serde_derive", + "solana-sdk-ids", + "solana-sdk-macro", + "solana-sysvar-id", +] + +[[package]] +name = "solana-loader-v2-interface" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d8ab08006dad78ae7cd30df8eea0539e207d08d91eaefb3e1d49a446e1c49654" +dependencies = [ + "serde", + "serde_bytes", + "serde_derive", + "solana-instruction", + "solana-pubkey", + "solana-sdk-ids", +] + +[[package]] +name = "solana-loader-v3-interface" +version = "3.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa4be76cfa9afd84ca2f35ebc09f0da0f0092935ccdac0595d98447f259538c2" +dependencies = [ + "serde", + "serde_bytes", + "serde_derive", + "solana-instruction", + "solana-pubkey", + "solana-sdk-ids", + "solana-system-interface", +] + +[[package]] +name = "solana-loader-v4-interface" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "706a777242f1f39a83e2a96a2a6cb034cb41169c6ecbee2cf09cb873d9659e7e" +dependencies = [ + "serde", + "serde_bytes", + "serde_derive", + "solana-instruction", + "solana-pubkey", + "solana-sdk-ids", + "solana-system-interface", +] + +[[package]] +name = "solana-message" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1796aabce376ff74bf89b78d268fa5e683d7d7a96a0a4e4813ec34de49d5314b" +dependencies = [ + "bincode", + "blake3", + "lazy_static", + "serde", + "serde_derive", + "solana-bincode", + "solana-hash", + "solana-instruction", + "solana-pubkey", + "solana-sanitize", + "solana-sdk-ids", + "solana-short-vec", + "solana-system-interface", + "solana-transaction-error", + "wasm-bindgen", +] + +[[package]] +name = "solana-msg" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f36a1a14399afaabc2781a1db09cb14ee4cc4ee5c7a5a3cfcc601811379a8092" +dependencies = [ + "solana-define-syscall", +] + +[[package]] +name = "solana-native-token" +version = "2.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "307fb2f78060995979e9b4f68f833623565ed4e55d3725f100454ce78a99a1a3" + +[[package]] +name = "solana-nonce" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "703e22eb185537e06204a5bd9d509b948f0066f2d1d814a6f475dafb3ddf1325" +dependencies = [ + "serde", + "serde_derive", + "solana-fee-calculator", + "solana-hash", + "solana-pubkey", + "solana-sha256-hasher", +] + +[[package]] +name = "solana-program" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "586469467e93ceb79048f8d8e3a619bf61d05396ee7de95cb40280301a589d05" +dependencies = [ + "bincode", + "blake3", + "borsh 0.10.4", + "borsh 1.5.7", + "bs58", + "bytemuck", + "console_error_panic_hook", + "console_log", + "getrandom 0.2.16", + "lazy_static", + "log", + "memoffset", + "num-bigint", + "num-derive 0.4.2", + "num-traits", + "rand 0.8.5", + "serde", + "serde_bytes", + "serde_derive", + "solana-account-info", + "solana-address-lookup-table-interface", + "solana-atomic-u64", + "solana-big-mod-exp", + "solana-bincode", + "solana-blake3-hasher", + "solana-borsh", + "solana-clock", + "solana-cpi", + "solana-decode-error", + "solana-define-syscall", + "solana-epoch-rewards", + "solana-epoch-schedule", + "solana-example-mocks", + "solana-feature-gate-interface", + "solana-fee-calculator", + "solana-hash", + "solana-instruction", + "solana-instructions-sysvar", + "solana-keccak-hasher", + "solana-last-restart-slot", + "solana-loader-v2-interface", + "solana-loader-v3-interface", + "solana-loader-v4-interface", + "solana-message", + "solana-msg", + "solana-native-token", + "solana-nonce", + "solana-program-entrypoint", + "solana-program-error", + "solana-program-memory", + "solana-program-option", + "solana-program-pack", + "solana-pubkey", + "solana-rent", + "solana-sanitize", + "solana-sdk-ids", + "solana-sdk-macro", + "solana-secp256k1-recover", + "solana-serde-varint", + "solana-serialize-utils", + "solana-sha256-hasher", + "solana-short-vec", + "solana-slot-hashes", + "solana-slot-history", + "solana-stable-layout", + "solana-stake-interface", + "solana-system-interface", + "solana-sysvar", + "solana-sysvar-id", + "solana-vote-interface", + "thiserror 2.0.12", + "wasm-bindgen", +] + +[[package]] +name = "solana-program-entrypoint" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "473ffe73c68d93e9f2aa726ad2985fe52760052709aaab188100a42c618060ec" +dependencies = [ + "solana-account-info", + "solana-msg", + "solana-program-error", + "solana-pubkey", +] + +[[package]] +name = "solana-program-error" +version = "2.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9ee2e0217d642e2ea4bee237f37bd61bb02aec60da3647c48ff88f6556ade775" +dependencies = [ + "borsh 1.5.7", + "num-traits", + "serde", + "serde_derive", + "solana-decode-error", + "solana-instruction", + "solana-msg", + "solana-pubkey", +] + +[[package]] +name = "solana-program-memory" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1b0268f6c89825fb634a34bd0c3b8fdaeaecfc3728be1d622a8ee6dd577b60d4" +dependencies = [ + "num-traits", + "solana-define-syscall", +] + +[[package]] +name = "solana-program-option" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dc677a2e9bc616eda6dbdab834d463372b92848b2bfe4a1ed4e4b4adba3397d0" + +[[package]] +name = "solana-program-pack" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "319f0ef15e6e12dc37c597faccb7d62525a509fec5f6975ecb9419efddeb277b" +dependencies = [ + "solana-program-error", +] + +[[package]] +name = "solana-pubkey" +version = "2.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9b62adb9c3261a052ca1f999398c388f1daf558a1b492f60a6d9e64857db4ff1" +dependencies = [ + "borsh 0.10.4", + "borsh 1.5.7", + "bytemuck", + "bytemuck_derive", + "curve25519-dalek 4.1.3", + "five8", + "five8_const", + "getrandom 0.2.16", + "js-sys", + "num-traits", + "serde", + "serde_derive", + "solana-atomic-u64", + "solana-decode-error", + "solana-define-syscall", + "solana-sanitize", + "solana-sha256-hasher", + "wasm-bindgen", +] + +[[package]] +name = "solana-rent" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d1aea8fdea9de98ca6e8c2da5827707fb3842833521b528a713810ca685d2480" +dependencies = [ + "serde", + "serde_derive", + "solana-sdk-ids", + "solana-sdk-macro", + "solana-sysvar-id", +] + +[[package]] +name = "solana-sanitize" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61f1bc1357b8188d9c4a3af3fc55276e56987265eb7ad073ae6f8180ee54cecf" + +[[package]] +name = "solana-sdk-ids" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c5d8b9cc68d5c88b062a33e23a6466722467dde0035152d8fb1afbcdf350a5f" +dependencies = [ + "solana-pubkey", +] + +[[package]] +name = "solana-sdk-macro" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "86280da8b99d03560f6ab5aca9de2e38805681df34e0bb8f238e69b29433b9df" +dependencies = [ + "bs58", + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[package]] +name = "solana-secp256k1-recover" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "baa3120b6cdaa270f39444f5093a90a7b03d296d362878f7a6991d6de3bbe496" +dependencies = [ + "libsecp256k1", + "solana-define-syscall", + "thiserror 2.0.12", +] + +[[package]] +name = "solana-security-txt" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "156bb61a96c605fa124e052d630dba2f6fb57e08c7d15b757e1e958b3ed7b3fe" +dependencies = [ + "hashbrown 0.15.2", +] + +[[package]] +name = "solana-seed-derivable" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3beb82b5adb266c6ea90e5cf3967235644848eac476c5a1f2f9283a143b7c97f" +dependencies = [ + "solana-derivation-path", +] + +[[package]] +name = "solana-seed-phrase" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "36187af2324f079f65a675ec22b31c24919cb4ac22c79472e85d819db9bbbc15" +dependencies = [ + "hmac", + "pbkdf2", + "sha2 0.10.9", +] + +[[package]] +name = "solana-serde-varint" +version = "2.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2a7e155eba458ecfb0107b98236088c3764a09ddf0201ec29e52a0be40857113" +dependencies = [ + "serde", +] + +[[package]] +name = "solana-serialize-utils" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "817a284b63197d2b27afdba829c5ab34231da4a9b4e763466a003c40ca4f535e" +dependencies = [ + "solana-instruction", + "solana-pubkey", + "solana-sanitize", +] + +[[package]] +name = "solana-sha256-hasher" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0037386961c0d633421f53560ad7c80675c0447cba4d1bb66d60974dd486c7ea" +dependencies = [ + "sha2 0.10.9", + "solana-define-syscall", + "solana-hash", +] + +[[package]] +name = "solana-short-vec" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5c54c66f19b9766a56fa0057d060de8378676cb64987533fa088861858fc5a69" +dependencies = [ + "serde", +] + +[[package]] +name = "solana-signature" +version = "2.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "64c8ec8e657aecfc187522fc67495142c12f35e55ddeca8698edbb738b8dbd8c" +dependencies = [ + "five8", + "solana-sanitize", +] + +[[package]] +name = "solana-signer" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7c41991508a4b02f021c1342ba00bcfa098630b213726ceadc7cb032e051975b" +dependencies = [ + "solana-pubkey", + "solana-signature", + "solana-transaction-error", +] + +[[package]] +name = "solana-slot-hashes" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0c8691982114513763e88d04094c9caa0376b867a29577939011331134c301ce" +dependencies = [ + "serde", + "serde_derive", + "solana-hash", + "solana-sdk-ids", + "solana-sysvar-id", +] + +[[package]] +name = "solana-slot-history" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97ccc1b2067ca22754d5283afb2b0126d61eae734fc616d23871b0943b0d935e" +dependencies = [ + "bv", + "serde", + "serde_derive", + "solana-sdk-ids", + "solana-sysvar-id", +] + +[[package]] +name = "solana-stable-layout" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f14f7d02af8f2bc1b5efeeae71bc1c2b7f0f65cd75bcc7d8180f2c762a57f54" +dependencies = [ + "solana-instruction", + "solana-pubkey", +] + +[[package]] +name = "solana-stake-interface" +version = "1.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5269e89fde216b4d7e1d1739cf5303f8398a1ff372a81232abbee80e554a838c" +dependencies = [ + "borsh 0.10.4", + "borsh 1.5.7", + "num-traits", + "serde", + "serde_derive", + "solana-clock", + "solana-cpi", + "solana-decode-error", + "solana-instruction", + "solana-program-error", + "solana-pubkey", + "solana-system-interface", + "solana-sysvar-id", +] + +[[package]] +name = "solana-system-interface" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "94d7c18cb1a91c6be5f5a8ac9276a1d7c737e39a21beba9ea710ab4b9c63bc90" +dependencies = [ + "js-sys", + "num-traits", + "serde", + "serde_derive", + "solana-decode-error", + "solana-instruction", + "solana-pubkey", + "wasm-bindgen", +] + +[[package]] +name = "solana-sysvar" +version = "2.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d50c92bc019c590f5e42c61939676e18d14809ed00b2a59695dd5c67ae72c097" +dependencies = [ + "base64 0.22.1", + "bincode", + "bytemuck", + "bytemuck_derive", + "lazy_static", + "serde", + "serde_derive", + "solana-account-info", + "solana-clock", + "solana-define-syscall", + "solana-epoch-rewards", + "solana-epoch-schedule", + "solana-fee-calculator", + "solana-hash", + "solana-instruction", + "solana-instructions-sysvar", + "solana-last-restart-slot", + "solana-program-entrypoint", + "solana-program-error", + "solana-program-memory", + "solana-pubkey", + "solana-rent", + "solana-sanitize", + "solana-sdk-ids", + "solana-sdk-macro", + "solana-slot-hashes", + "solana-slot-history", + "solana-stake-interface", + "solana-sysvar-id", +] + +[[package]] +name = "solana-sysvar-id" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5762b273d3325b047cfda250787f8d796d781746860d5d0a746ee29f3e8812c1" +dependencies = [ + "solana-pubkey", + "solana-sdk-ids", +] + +[[package]] +name = "solana-transaction-error" +version = "2.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "222a9dc8fdb61c6088baab34fc3a8b8473a03a7a5fd404ed8dd502fa79b67cb1" +dependencies = [ + "solana-instruction", + "solana-sanitize", +] + +[[package]] +name = "solana-vote-interface" +version = "2.2.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef4f08746f154458f28b98330c0d55cb431e2de64ee4b8efc98dcbe292e0672b" +dependencies = [ + "bincode", + "num-derive 0.4.2", + "num-traits", + "serde", + "serde_derive", + "solana-clock", + "solana-decode-error", + "solana-hash", + "solana-instruction", + "solana-pubkey", + "solana-rent", + "solana-sdk-ids", + "solana-serde-varint", + "solana-serialize-utils", + "solana-short-vec", + "solana-system-interface", +] + +[[package]] +name = "solana-zk-sdk" +version = "2.3.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "97b9fc6ec37d16d0dccff708ed1dd6ea9ba61796700c3bb7c3b401973f10f63b" +dependencies = [ + "aes-gcm-siv", + "base64 0.22.1", + "bincode", + "bytemuck", + "bytemuck_derive", + "curve25519-dalek 4.1.3", + "itertools", + "js-sys", + "merlin", + "num-derive 0.4.2", + "num-traits", + "rand 0.8.5", + "serde", + "serde_derive", + "serde_json", + "sha3", + "solana-derivation-path", + "solana-instruction", + "solana-pubkey", + "solana-sdk-ids", + "solana-seed-derivable", + "solana-seed-phrase", + "solana-signature", + "solana-signer", + "subtle", + "thiserror 2.0.12", + "wasm-bindgen", + "zeroize", +] + +[[package]] +name = "spl-associated-token-account" +version = "6.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76fee7d65013667032d499adc3c895e286197a35a0d3a4643c80e7fd3e9969e3" +dependencies = [ + "borsh 1.5.7", + "num-derive 0.4.2", + "num-traits", + "solana-program", + "spl-associated-token-account-client", + "spl-token 7.0.0", + "spl-token-2022", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-associated-token-account-client" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d6f8349dbcbe575f354f9a533a21f272f3eb3808a49e2fdc1c34393b88ba76cb" +dependencies = [ + "solana-instruction", + "solana-pubkey", +] + +[[package]] +name = "spl-discriminator" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a7398da23554a31660f17718164e31d31900956054f54f52d5ec1be51cb4f4b3" +dependencies = [ + "bytemuck", + "solana-program-error", + "solana-sha256-hasher", + "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.101", +] + +[[package]] +name = "spl-discriminator-syn" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d1dbc82ab91422345b6df40a79e2b78c7bce1ebb366da323572dd60b7076b67" +dependencies = [ + "proc-macro2", + "quote", + "sha2 0.10.9", + "syn 2.0.101", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-elgamal-registry" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ce0f668975d2b0536e8a8fd60e56a05c467f06021dae037f1d0cfed0de2e231d" +dependencies = [ + "bytemuck", + "solana-program", + "solana-zk-sdk", + "spl-pod", + "spl-token-confidential-transfer-proof-extraction", +] + +[[package]] +name = "spl-memo" +version = "6.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9f09647c0974e33366efeb83b8e2daebb329f0420149e74d3a4bd2c08cf9f7cb" +dependencies = [ + "solana-account-info", + "solana-instruction", + "solana-msg", + "solana-program-entrypoint", + "solana-program-error", + "solana-pubkey", +] + +[[package]] +name = "spl-pod" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d994afaf86b779104b4a95ba9ca75b8ced3fdb17ee934e38cb69e72afbe17799" +dependencies = [ + "borsh 1.5.7", + "bytemuck", + "bytemuck_derive", + "num-derive 0.4.2", + "num-traits", + "solana-decode-error", + "solana-msg", + "solana-program-error", + "solana-program-option", + "solana-pubkey", + "solana-zk-sdk", + "thiserror 2.0.12", +] + +[[package]] +name = "spl-program-error" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9d39b5186f42b2b50168029d81e58e800b690877ef0b30580d107659250da1d1" +dependencies = [ + "num-derive 0.4.2", + "num-traits", + "solana-program", + "spl-program-error-derive", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-program-error-derive" +version = "0.4.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e6d375dd76c517836353e093c2dbb490938ff72821ab568b545fd30ab3256b3e" +dependencies = [ + "proc-macro2", + "quote", + "sha2 0.10.9", + "syn 2.0.101", +] + +[[package]] +name = "spl-tlv-account-resolution" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "cd99ff1e9ed2ab86e3fd582850d47a739fec1be9f4661cba1782d3a0f26805f3" +dependencies = [ + "bytemuck", + "num-derive 0.4.2", + "num-traits", + "solana-account-info", + "solana-decode-error", + "solana-instruction", + "solana-msg", + "solana-program-error", + "solana-pubkey", + "spl-discriminator", + "spl-pod", + "spl-program-error", + "spl-type-length-value", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-token" +version = "4.0.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5e9e171cbcb4b1f72f6d78ed1e975cb467f56825c27d09b8dd2608e4e7fc8b3b" +dependencies = [ + "arrayref", + "bytemuck", + "num-derive 0.4.2", + "num-traits", + "num_enum", + "solana-program", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-token" +version = "7.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed320a6c934128d4f7e54fe00e16b8aeaecf215799d060ae14f93378da6dc834" +dependencies = [ + "arrayref", + "bytemuck", + "num-derive 0.4.2", + "num-traits", + "num_enum", + "solana-program", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-token-2022" +version = "6.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5b27f7405010ef816587c944536b0eafbcc35206ab6ba0f2ca79f1d28e488f4f" +dependencies = [ + "arrayref", + "bytemuck", + "num-derive 0.4.2", + "num-traits", + "num_enum", + "solana-program", + "solana-security-txt", + "solana-zk-sdk", + "spl-elgamal-registry", + "spl-memo", + "spl-pod", + "spl-token 7.0.0", + "spl-token-confidential-transfer-ciphertext-arithmetic", + "spl-token-confidential-transfer-proof-extraction", + "spl-token-confidential-transfer-proof-generation", + "spl-token-group-interface", + "spl-token-metadata-interface", + "spl-transfer-hook-interface", + "spl-type-length-value", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-token-confidential-transfer-ciphertext-arithmetic" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "170378693c5516090f6d37ae9bad2b9b6125069be68d9acd4865bbe9fc8499fd" +dependencies = [ + "base64 0.22.1", + "bytemuck", + "solana-curve25519", + "solana-zk-sdk", +] + +[[package]] +name = "spl-token-confidential-transfer-proof-extraction" +version = "0.2.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "eff2d6a445a147c9d6dd77b8301b1e116c8299601794b558eafa409b342faf96" +dependencies = [ + "bytemuck", + "solana-curve25519", + "solana-program", + "solana-zk-sdk", + "spl-pod", + "thiserror 2.0.12", +] + +[[package]] +name = "spl-token-confidential-transfer-proof-generation" +version = "0.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8627184782eec1894de8ea26129c61303f1f0adeed65c20e0b10bc584f09356d" +dependencies = [ + "curve25519-dalek 4.1.3", + "solana-zk-sdk", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-token-group-interface" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d595667ed72dbfed8c251708f406d7c2814a3fa6879893b323d56a10bedfc799" +dependencies = [ + "bytemuck", + "num-derive 0.4.2", + "num-traits", + "solana-decode-error", + "solana-instruction", + "solana-msg", + "solana-program-error", + "solana-pubkey", + "spl-discriminator", + "spl-pod", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-token-metadata-interface" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dfb9c89dbc877abd735f05547dcf9e6e12c00c11d6d74d8817506cab4c99fdbb" +dependencies = [ + "borsh 1.5.7", + "num-derive 0.4.2", + "num-traits", + "solana-borsh", + "solana-decode-error", + "solana-instruction", + "solana-msg", + "solana-program-error", + "solana-pubkey", + "spl-discriminator", + "spl-pod", + "spl-type-length-value", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-transfer-hook-interface" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4aa7503d52107c33c88e845e1351565050362c2314036ddf19a36cd25137c043" +dependencies = [ + "arrayref", + "bytemuck", + "num-derive 0.4.2", + "num-traits", + "solana-account-info", + "solana-cpi", + "solana-decode-error", + "solana-instruction", + "solana-msg", + "solana-program-error", + "solana-pubkey", + "spl-discriminator", + "spl-pod", + "spl-program-error", + "spl-tlv-account-resolution", + "spl-type-length-value", + "thiserror 1.0.69", +] + +[[package]] +name = "spl-type-length-value" +version = "0.7.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba70ef09b13af616a4c987797870122863cba03acc4284f226a4473b043923f9" +dependencies = [ + "bytemuck", + "num-derive 0.4.2", + "num-traits", + "solana-account-info", + "solana-decode-error", + "solana-msg", + "solana-program-error", + "spl-discriminator", + "spl-pod", + "thiserror 1.0.69", +] + +[[package]] +name = "subtle" +version = "2.6.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" + +[[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.101" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ce2b7fc941b3a24138a0a7cf8e858bfc6a992e7978a068a5c760deb0ed43caf" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + +[[package]] +name = "thiserror" +version = "1.0.69" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6aaf5339b578ea85b50e080feb250a3e8ae8cfcdff9a461c9ec2904bc923f52" +dependencies = [ + "thiserror-impl 1.0.69", +] + +[[package]] +name = "thiserror" +version = "2.0.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "567b8a2dae586314f7be2a752ec7474332959c6460e02bde30d702a66d488708" +dependencies = [ + "thiserror-impl 2.0.12", +] + +[[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.101", +] + +[[package]] +name = "thiserror-impl" +version = "2.0.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7f7cf42b4507d8ea322120659672cf1b9dbb93f8f2d4ecfd6e51350ff5b17a1d" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[package]] +name = "tinyvec" +version = "1.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09b3661f17e86524eccd4371ab0429194e0d7c008abb45f7a7495b1719463c71" +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.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "05ae329d1f08c4d17a59bed7ff5b5a769d062e64a62d34a3261b219e62cd5aae" +dependencies = [ + "serde", + "serde_spanned", + "toml_datetime", + "toml_edit", +] + +[[package]] +name = "toml_datetime" +version = "0.6.9" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3da5db5a963e24bc68be8b17b6fa82814bb22ee8660f192bb182771d498f09a3" +dependencies = [ + "serde", +] + +[[package]] +name = "toml_edit" +version = "0.22.26" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "310068873db2c5b3e7659d2cc35d21855dbafa50d1ce336397c666e3cb08137e" +dependencies = [ + "indexmap", + "serde", + "serde_spanned", + "toml_datetime", + "toml_write", + "winnow", +] + +[[package]] +name = "toml_write" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bfb942dfe1d8e29a7ee7fcbde5bd2b9a25fb89aa70caea2eba3bee836ff41076" + +[[package]] +name = "typenum" +version = "1.18.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1dccffe3ce07af9386bfd29e80c0ab1a8205a2fc34e4bcd40364df902cfa8f3f" + +[[package]] +name = "unicode-ident" +version = "1.0.18" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5a5f39404a5da50712a4c1eecf25e90dd62b613502b7e925fd4e4d19b5c96512" + +[[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.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fc1de2c688dc15305988b563c3854064043356019f97a4b46276fe734c4f07ea" +dependencies = [ + "crypto-common", + "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.100" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1edc8929d7499fc4e8f0be2262a241556cfc54a0bea223790e71446f2aab1ef5" +dependencies = [ + "cfg-if", + "once_cell", + "rustversion", + "wasm-bindgen-macro", +] + +[[package]] +name = "wasm-bindgen-backend" +version = "0.2.100" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2f0a0651a5c2bc21487bde11ee802ccaf4c51935d0d3d42a6101f98161700bc6" +dependencies = [ + "bumpalo", + "log", + "proc-macro2", + "quote", + "syn 2.0.101", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-macro" +version = "0.2.100" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7fe63fc6d09ed3792bd0897b314f53de8e16568c2b3f7982f468c0bf9bd0b407" +dependencies = [ + "quote", + "wasm-bindgen-macro-support", +] + +[[package]] +name = "wasm-bindgen-macro-support" +version = "0.2.100" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "8ae87ea40c9f689fc23f209965b6fb8a99ad69aeeb0231408be24920604395de" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.101", + "wasm-bindgen-backend", + "wasm-bindgen-shared", +] + +[[package]] +name = "wasm-bindgen-shared" +version = "0.2.100" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1a05d73b933a847d6cccdda8f838a22ff101ad9bf93e33684f39c1f5f0eece3d" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "web-sys" +version = "0.3.77" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "33b6dd2ef9186f1f2072e409e99cd22a975331a6b3591b12c764e0e55c60d5d2" +dependencies = [ + "js-sys", + "wasm-bindgen", +] + +[[package]] +name = "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.7.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c06928c8748d81b05c9be96aad92e1b6ff01833332f281e8cfca3be4b35fc9ec" +dependencies = [ + "memchr", +] + +[[package]] +name = "zerocopy" +version = "0.8.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a1702d9583232ddb9174e01bb7c15a2ab8fb1bc6f227aa1233858c351a3ba0cb" +dependencies = [ + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.8.25" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "28a6e20d751156648aa063f3800b706ee209a32c0b4d9f24be3d980b01be55ef" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.101", +] + +[[package]] +name = "zeroize" +version = "1.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde" +dependencies = [ + "zeroize_derive", +] + +[[package]] +name = "zeroize_derive" +version = "1.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "85a5b4158499876c763cb03bc4e49185d3cccbabb15b33c627f7884f43db852e" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.101", +] diff --git a/shine/Cargo.toml b/shine/Cargo.toml new file mode 100644 index 0000000..4adb720 --- /dev/null +++ b/shine/Cargo.toml @@ -0,0 +1,25 @@ +[workspace] +members = [ + "programs/common", + "programs/shine_users", + "programs/shine_payments", +] +resolver = "2" + +[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/shine/doc/Deploy.txt b/shine/doc/Deploy.txt new file mode 100644 index 0000000..b1242bd --- /dev/null +++ b/shine/doc/Deploy.txt @@ -0,0 +1,31 @@ +Деплой в devnet (по умолчанию у тебя уже devnet): + обе программы сразу: +anchor deploy + или по одной: +anchor deploy --program-name shine_users +anchor deploy --program-name shine_payments + + Проверка деплоя +anchor keys list +solana program show 5dFcWDNp42Xn9Vv4oDMJzM4obBJ8hvDuAtPX54fT5L3t // +solana program show GcGFR47xF7o7ztXzN4MFThmxzHn4z6VmpmELgNk8smCm // + + Апгрейд в будущем + После изменений кода: +anchor build +anchor upgrade --program-name shine_users +anchor upgrade --program-name shine_payments + + посмотреть адрес кошелька + solana address -k /home/ai/.config/solana/id.jsonanchor build --program-name shine_payments + + + + + + +anchor deploy --program-name shine_payments + +solana program show 92sgkgx7KHpbhQu81mNGHaKa7skJB7esArVdPM7paDSW --url http://127.0.0.1:8899 + +solana address -k /home/ai/work/SOLANA/shine/target/deploy/shine_payments-keypair.json diff --git a/shine/doc/READ_ME.txt b/shine/doc/READ_ME.txt new file mode 100644 index 0000000..89831f0 --- /dev/null +++ b/shine/doc/READ_ME.txt @@ -0,0 +1,39 @@ +# подключаться надо к +JSON RPC URL: http://127.0.0.1:8899 + +# Запустить саму ноду +solana-test-validator +# Удалить процесс ноды что бы запустить заново +kill -9 $(pgrep -f "solana-test-validator") + +или +ps aux | grep solana-test-validator +а потом +kill -9 1052345 +# Убивает и логи и всю базу локальной ноды +rm -rf test-ledger + + +# Удалить все данные с ноды +solana-test-validator --reset + + +# Что бы запустить просмотр логов ноды +solana logs + +# Запустить контракт +anchor deploy + +# Cкомпилировать и задеплоить новую версию +anchor build # Скомпилировать контракт и сгенерировать IDL +anchor deploy # Задеплоить контракт в сеть (указанную в Anchor.toml) +Если ты хочешь сразу убедиться, куда он деплоится — проверь Anchor.toml. +[provider] +cluster = "https://api.testnet.solana.com" # или "localnet" +wallet = "~/.config/solana/id.json" + + + + +# Создать новый проект +anchor init имя_проекта diff --git a/shine/doc/READ_ME_NEW_DEPLOY.txt b/shine/doc/READ_ME_NEW_DEPLOY.txt new file mode 100644 index 0000000..708ca0e --- /dev/null +++ b/shine/doc/READ_ME_NEW_DEPLOY.txt @@ -0,0 +1,65 @@ +https://api.devnet.solana.com + + +проверить настройки +solana config get + +настроить +solana config set --url https://api.devnet.solana.com +или +solana config set --url http://127.0.0.1:8899 + +потом +solana airdrop 2 --keypair /home/ai/.config/solana/id.json +и +solana balance --keypair /home/ai/.config/solana/id.json + + + +anchor deploy \ + --provider.cluster https://api.devnet.solana.com \ + --provider.wallet /home/ai/.config/solana/id.json + + + + + + + + + + + + + + + +Шаг 1. Создай новый ключ для новой программы + +solana-keygen new --outfile target/deploy/user_registry-testnet-keypair.json + +Шаг 2. Укажи новый ID в declare_id!: + +declare_id!("НОВЫЙ_PUBKEY_ОТСЮДА"); // получен из предыдущей команды + +Чтобы узнать pubkey: + +solana address -k target/deploy/user_registry-testnet-keypair.json + +Шаг 3. Обнови Anchor.toml: + +[programs.testnet] +user_registry = "НОВЫЙ_PUBKEY" + +[provider] +cluster = "https://api.testnet.solana.com" +wallet = "~/.config/solana/id.json" + +Шаг 4. Компиляция и деплой: + +anchor build +anchor deploy --provider.cluster testnet + +Шаг 5. Проверка: + +solana program show НОВЫЙ_PUBKEY --url https://api.testnet.solana.com \ No newline at end of file diff --git a/shine/doc/ReadMe.md b/shine/doc/ReadMe.md new file mode 100644 index 0000000..ae846df --- /dev/null +++ b/shine/doc/ReadMe.md @@ -0,0 +1 @@ +Просто разные заметки для себя \ No newline at end of file diff --git a/shine/doc/SHINE_USER_PDA_V1.md b/shine/doc/SHINE_USER_PDA_V1.md new file mode 100644 index 0000000..bda69c1 --- /dev/null +++ b/shine/doc/SHINE_USER_PDA_V1.md @@ -0,0 +1,144 @@ +# SHiNE User PDA v1.0 + +## 1. Назначение + +`SHiNE User PDA v1.0` — бинарный формат пользовательской записи в PDA Solana. + +Хранит: +- логин; +- ключи пользователя; +- номер внутренней сети; +- лимит (баланс); +- серверные данные (опционально); +- список серверов подключения; +- данные восстановления; +- связь с предыдущей версией через `prev_hash`; +- подпись владельца `root_key`. + +Размер PDA фиксированный: + +`768 bytes` + +Полезная длина записи указывается в `record_len`. + +## 2. Общие правила кодирования + +- Числа: `little-endian`. +- Строки: `UTF-8` с префиксом длины `u8`. +- Padding до 768 байт: нули `0x00`. +- Padding не входит в `record_len`. + +## 3. Структура записи + +- `magic` (5): `"SHiNE"` +- `format_major` (1): `1` +- `format_minor` (1): `0` +- `record_len` (2): длина от `magic` до `signature` включительно +- `created_at_ms` (8) +- `updated_at_ms` (8) +- `version` (4) +- `prev_hash` (32) +- `login_len` (1) +- `login` (N) +- `root_key` (32) +- `blockchain_key` (32) +- `device_key` (32) +- `chain_number` (2) +- `balance` (8) +- `is_server` (1) +- если `is_server=1`: + - `server_key` (32) + - `server_address_len` (1) + - `server_address` (N) +- `connection_servers_count` (1) +- повтор `count` раз: + - `server_login_len` (1) + - `server_login` (N) +- `trusted_count` (1) +- `reserved` (5) = `0x00 0x00 0x00 0x00 0x00` +- `signature` (64) +- `padding` до 768 + +## 4. Подпись (v1.0) + +В `v1.0` подписывается не сырой блок полей напрямую, а его SHA-256. + +1. Формируется `unsigned_bytes`: + - все поля от `magic` до `reserved` включительно; + - поле `signature` не включается; + - padding не включается. +2. Считается `msg_hash = SHA-256(unsigned_bytes)`. +3. `signature = Ed25519.sign(root_private_key, msg_hash)`. +4. Проверка: + - `Ed25519.verify(root_key, msg_hash, signature)`. + +## 5. Что входит в `prev_hash` + +Для связи версий: + +- `prev_hash = SHA-256(previous_unsigned_bytes)` + +Где `previous_unsigned_bytes` — предыдущая версия записи от `magic` до `reserved` включительно, без `signature` и без padding. + +Для первой версии: + +- `prev_hash = 32` нулевых байта. + +## 6. Правила create/update в текущей реализации + +### Create + +- PDA: seed `["login=", login]`. +- Создаётся запись версии `0`. +- `updated_at_ms = created_at_ms`. +- Стартовый лимит: + - `START_BONUS_LIMIT + additional_limit`. +- Оплата: + - регистрационная комиссия; + - пополнение `additional_limit` по курсу; + - рента PDA (плательщик транзакции). + +### Update + +- Проверка подписи новой записи по `root_key`. +- Проверка: + - `magic`, `format_major`, `format_minor`; + - корректного `record_len`; + - `prev_hash` на соответствие предыдущей версии; + - `version = old_version + 1`; + - неизменяемых полей: `login`, `created_at_ms`, `root_key`. +- `balance` не уменьшается: + - `new_balance = old_balance + additional_limit`. +- При `additional_limit > 0` берётся комиссия пополнения. + +## 7. Параметры экономики/размера (settings) + +См. `programs/shine_users/src/settings.rs`: + +- `USER_PDA_SPACE = 768` +- `REGISTRATION_FEE_RECEIVER` +- `REGISTRATION_FEE_LAMPORTS` +- `LIMIT_STEP` +- `LAMPORTS_PER_LIMIT_STEP` +- `START_BONUS_LIMIT` + +## 8. Root Key Rotation (пока не включено) + +В `v1.0` `root_key` неизменяем. + +Варианты расширения: + +1. **Dual-signature rotate tx**: + - отдельный флаг операции rotate; + - запись подписывается и старым, и новым root key; + - контракт проверяет обе подписи. + +2. **Two-step commit/confirm**: + - шаг 1: proposal смены root (`old root` подпись); + - шаг 2: confirm (`new root` подпись) в отдельной tx. + +3. **Recovery guardians**: + - отдельная PDA для доверенных лиц; + - пороговая схема (например `m-of-n`) для восстановления root. + +Для v1.0 решение отложено, но изменение root_key в update запрещено. diff --git a/shine/doc/TODO b/shine/doc/TODO new file mode 100644 index 0000000..93bbc21 --- /dev/null +++ b/shine/doc/TODO @@ -0,0 +1,7 @@ +. Сделать новые форматы для пользователей что бы там было больше информации + +. Протестировать работу и может доработать что бы можно было паралельно регистрировать 5 и более юзеров - за счёт передачи при вызове адресов PDA +1 +2 +3 +4 и т.д. + + + +. - пока не надо - Сделать что бы в файле общей информации добавилась запись для будущей миграции пользователей (хотя можно потом и добавить будет :))) \ No newline at end of file diff --git a/shine/doc/how to getback money after deploy from buffer.txt b/shine/doc/how to getback money after deploy from buffer.txt new file mode 100644 index 0000000..3501e79 --- /dev/null +++ b/shine/doc/how to getback money after deploy from buffer.txt @@ -0,0 +1,19 @@ +Как вернуть деньги + + Узнаём адрес buffer account: + +solana program show <адрес_твоей_программы> + +Там будет строчка Buffer: . + +Закрываем буфер: + + solana program close --recipient <адрес_кошелька> + + 💡 --recipient — это куда вернуть SOL (обычно твой же кошелёк из ~/.config/solana/id.json). + +Сколько вернётся + + Если бинарник весит ~400 KB, с одного буфера вернётся ~0.35 SOL. + + У тебя две программы (shine_users, shine_payments), значит можно вернуть ~0.7 SOL. \ No newline at end of file diff --git a/shine/doc/how todo minimal size.txt b/shine/doc/how todo minimal size.txt new file mode 100644 index 0000000..fe6b7f0 --- /dev/null +++ b/shine/doc/how todo minimal size.txt @@ -0,0 +1,6 @@ +✅ Чтобы максимально уменьшить .so надо будет включить флаг оптимизации в Cargo.toml: + +[profile.release] +opt-level = "z" # Максимальная компрессия +lto = true # Link Time Optimization +codegen-units = 1 # Уменьшает размер бинаря diff --git a/shine/doc/sh/restart.sh b/shine/doc/sh/restart.sh new file mode 100755 index 0000000..e13ccd0 --- /dev/null +++ b/shine/doc/sh/restart.sh @@ -0,0 +1,44 @@ +#!/bin/bash + +set -e # Завершаем при ошибке +set -o pipefail + + + + + +PROGRAM_KEYPAIR="target/deploy/shine-keypair.json" # замени на свой путь +WALLET=$(solana address) + +echo "🧹 Удаление старого ledger..." +rm -rf test-ledger + +echo "🚀 Запуск solana-test-validator в фоне..." +solana-test-validator --ledger test-ledger --reset > validator.log 2>&1 & +VALIDATOR_PID=$! + +# Убедимся, что validator запущен +echo "⏳ Ожидание запуска валидатора..." +until solana cluster-version &>/dev/null; do + sleep 1 +done +sleep 2 # На всякий случай немного подождём + +echo "💸 Airdrop 10 SOL на $WALLET..." +solana airdrop 10 HMww7YSVfwVm4i8sugqj7wyH26dqzHykzv3wzWwzEvPA + +solana airdrop 5 $WALLET + +echo "🔨 Сборка контракта..." +anchor build + +echo "📦 Деплой контракта..." +anchor deploy + +echo "✅ Готово!" + +# Не убиваем валидатор, чтобы он оставался запущенным +echo "ℹ️ Валидатор всё ещё работает (PID $VALIDATOR_PID)" + +echo "ℹ️ Запускаем логи" +solana logs diff --git a/shine/doc/sh/stop.sh b/shine/doc/sh/stop.sh new file mode 100755 index 0000000..c54d0a4 --- /dev/null +++ b/shine/doc/sh/stop.sh @@ -0,0 +1,23 @@ +#!/bin/bash + +set -e # Завершаем при ошибке +set -o pipefail + + +kill -9 $(pgrep -f "solana-test-validator") + +# 🔍 Ищем запущенный solana-test-validator +EXISTING_PID=$(pgrep -f "solana-test-validator") + +if [ -n "$EXISTING_PID" ]; then + echo "🛑 Найден работающий solana-test-validator (PID $EXISTING_PID), останавливаем..." + bash kill -9 $(pgrep -f "solana-test-validator") + echo "✅ Пытаюсь остановить старый валидатор..." + + # ждём завершения + while kill -0 "$EXISTING_PID" 2>/dev/null; do + sleep 0.5 + done + echo "✅ Старый валидатор остановлен." +fi + diff --git a/shine/doc/что содержит utils.txt b/shine/doc/что содержит utils.txt new file mode 100644 index 0000000..2ee02f1 --- /dev/null +++ b/shine/doc/что содержит utils.txt @@ -0,0 +1,145 @@ +Функции для работы с PDA: + + +🧩 create_pda(...) + + Создаёт новый PDA, если он ещё не существует. + + Проверяет, чтобы не было коллизии. + + Без записи данных. + +🧩 write_to_pda(...) + + Просто записывает байты в существующий PDA. + + Без создания. + +🧩 create_and_write_pda(...) + + Комбинированная функция. + + Сначала проверяет, есть ли PDA — если нет, создаёт. + + Затем записывает данные. + + Очень удобна для инициализации одного PDA в один вызов. + +🧩 safe_read_pda(...) + + Возвращает Vec с содержимым PDA. + + Никогда не паникует: если PDA не существует или пустой — просто отдаёт Vec::new(). + + Защита от двойного borrow'а (через try_borrow_data()). + + Отличный инструмент для безопасного считывания данных. + +💡 Да, с этим ты можешь: +Возможность Функция +📦 Создать PDA create_pda +💾 Записать в PDA write_to_pda +⚡ Создать и записать create_and_write_pda +📖 Безопасно прочитать safe_read_pda + + +🔧 create_pda(...) + +🔹 Назначение: +Создаёт новый PDA-аккаунт, если он ещё не существует. + +📥 Аргументы: + + pda_account: &AccountInfo — аккаунт, который хотим создать + + signer: &AccountInfo — аккаунт плательщика (обычно пользователь) + + system_program: &AccountInfo — системная программа + + program_id: &Pubkey — адрес текущей программы + + seeds: &[&[u8]] — массив сидов, по которым создавался PDA + + space: u64 — сколько байт выделить под данные + +📤 Возвращает: + + Result<()> — Ok если успешно, Err если PDA уже существует или при ошибке создания + +🧠 Особенности: + + Проверяет, что PDA ещё не создан (через pda_account.owner == Pubkey::default()) + + Выбрасывает ErrCode::PdaAlreadyExists, если уже существует + +🔧 write_to_pda(...) + +🔹 Назначение: +Записывает бинарные данные в существующий PDA. + +📥 Аргументы: + + pda_account: &AccountInfo — аккаунт, в который пишем + + data: &[u8] — массив байт, которые нужно записать + +📤 Возвращает: + + Result<()> — Ok при успехе, Err если не удалось получить доступ к данным + +🧠 Особенности: + + ⚠️ Только пишет, не создаёт PDA + + Записывает в начало data-секции аккаунта + +🔧 create_and_write_pda(...) + +🔹 Назначение: +Если PDA ещё не существует — создаёт, затем сразу записывает данные. + +📥 Аргументы: + + pda_account: &AccountInfo — аккаунт для создания/записи + + signer: &AccountInfo — кто оплачивает создание + + system_program: &AccountInfo — системная программа + + program_id: &Pubkey — адрес текущей программы + + seeds: &[&[u8]] — сиды PDA + + data: Vec — данные для записи + + space: u64 — сколько байт выделить (при создании) + +📤 Возвращает: + + Result<()> — Ok при успехе, Err при ошибке создания или записи + +🧠 Особенности: + + Безопасно создаёт и пишет за один вызов + + Не выбрасывает ошибку, если PDA уже существует — просто пишет + +🔧 safe_read_pda(...) + +🔹 Назначение: +Безопасно считывает байты из PDA. Никогда не паникует. + +📥 Аргументы: + + pda_account: &AccountInfo — аккаунт для чтения + +📤 Возвращает: + + Vec — массив байт с содержимым PDA + → Если аккаунт не инициализирован или пустой, возвращает Vec::new() + +🧠 Особенности: + + Не выбрасывает ошибки — только логирует + + Полностью безопасно: подходит для чтения read-only PDA в любой ситуации \ No newline at end of file diff --git a/shine/migrations/deploy.ts b/shine/migrations/deploy.ts new file mode 100644 index 0000000..439431e --- /dev/null +++ b/shine/migrations/deploy.ts @@ -0,0 +1,12 @@ +// Migrations are an early feature. Currently, they're nothing more than this +// single deploy script that's invoked from the CLI, injecting a provider +// configured from the workspace's Anchor.toml. + +import * as anchor from "@coral-xyz/anchor"; + +module.exports = async function (provider: anchor.AnchorProvider) { + // Configure client to use the provider. + anchor.setProvider(provider); + + // Add your deploy script here. +}; diff --git a/shine/mu.sh b/shine/mu.sh new file mode 100755 index 0000000..90132e9 --- /dev/null +++ b/shine/mu.sh @@ -0,0 +1,29 @@ +#!/bin/bash + +# Кол-во адресов для генерации +NUM_KEYS=5 + +# RPC endpoint (можешь поменять) +#RPC_URL="https://api.testnet.solana.com" +RPC_URL="http://127.0.0.1:8899" + +echo "👉 Используем RPC: $RPC_URL" + +for i in $(seq 1 $NUM_KEYS); do + KEYPAIR="temp-key-$i.json" + echo "🔐 Генерирую ключ №$i: $KEYPAIR" + solana-keygen new --outfile "$KEYPAIR" --no-bip39-passphrase --silent + + PUBKEY=$(solana-keygen pubkey "$KEYPAIR") + echo "🪙 Публичный ключ: $PUBKEY" + + echo "💸 Запрашиваю airdrop на $PUBKEY..." + solana airdrop 1 "$PUBKEY" --url "$RPC_URL" + + echo "🔍 Проверяю баланс:" + solana balance "$PUBKEY" --url "$RPC_URL" + echo "-----------------------------" +done + +echo "✅ Готово. Удаляю временные ключи..." +rm temp-key-*.json diff --git a/shine/package.json b/shine/package.json new file mode 100644 index 0000000..7d765e5 --- /dev/null +++ b/shine/package.json @@ -0,0 +1,20 @@ +{ + "license": "ISC", + "scripts": { + "lint:fix": "prettier */*.js \"*/**/*{.js,.ts}\" -w", + "lint": "prettier */*.js \"*/**/*{.js,.ts}\" --check" + }, + "dependencies": { + "@coral-xyz/anchor": "^0.31.1" + }, + "devDependencies": { + "chai": "^4.3.4", + "mocha": "^9.0.3", + "ts-mocha": "^10.0.0", + "@types/bn.js": "^5.1.0", + "@types/chai": "^4.3.0", + "@types/mocha": "^9.0.0", + "typescript": "^5.7.3", + "prettier": "^2.6.2" + } +} diff --git a/shine/programs/common/Cargo.toml b/shine/programs/common/Cargo.toml new file mode 100644 index 0000000..2b3ee68 --- /dev/null +++ b/shine/programs/common/Cargo.toml @@ -0,0 +1,10 @@ +[package] +name = "common" +version = "0.1.0" +edition = "2021" + +[dependencies] +anchor-lang = "0.31.1" + + +[features] diff --git a/shine/programs/common/src/lib.rs b/shine/programs/common/src/lib.rs new file mode 100644 index 0000000..b5614dd --- /dev/null +++ b/shine/programs/common/src/lib.rs @@ -0,0 +1 @@ +pub mod utils; diff --git a/shine/programs/common/src/utils.rs b/shine/programs/common/src/utils.rs new file mode 100644 index 0000000..77eb884 --- /dev/null +++ b/shine/programs/common/src/utils.rs @@ -0,0 +1,402 @@ +use anchor_lang::prelude::*; +use anchor_lang::solana_program::{ + program::invoke_signed, + system_instruction, + system_program +}; + + + + + + +/// сдесь коды всех ошибок + +#[error_code] +pub enum ErrCode { + /// Система уже инициализирована и не может быть инициализирована повторно! + #[msg("Система уже инициализирована и не может быть инициализирована повторно!")] + SystemAlreadyInitialized = 1000, + + #[msg("PDA не содержит данных или не инициализирован")] + EmptyPdaData = 1002, + + #[msg("Пользователь уже зарегистрирован")] + UserAlreadyExists = 1003, + + #[msg("Некорректный логин")] + InvalidLogin = 1004, + + #[msg("Не совпадает PDA адрес")] + InvalidPdaAddress = 1006, + + #[msg("Формат данных не поддерживается")] + UnsupportedFormat = 1011, + + #[msg("Ошибка при десериализации")] + DeserializationError = 1012, + + /// PDA уже существует, создание невозможно + #[msg("PDA-аккаунт уже существует и не может быть создан повторно.")] + PdaAlreadyExists = 1009, + + + #[msg("Подписавший не совпадает с ожидаемым пользователем (это потому что пока временно можно регистрировать пользователя с другово аккаунта")] + InvalidSigner = 1005, + + /// Не получилось создат ьпользователя, система уже перегружена, попробуйте поззже!" + #[msg("Не получилось создать пользователя, система уже перегружена, попробуйте поззже!")] + NoSuitableIdPda = 1010, + + #[msg("Невалидная цифровая подпись записи")] + InvalidSignature = 1013, + + #[msg("Невалидный формат записи")] + InvalidRecordFormat = 1014, + + #[msg("Невалидная длина записи")] + InvalidRecordLength = 1015, + + #[msg("Невалидные данные записи")] + InvalidRecordData = 1016, + + #[msg("Невалидный хэш предыдущей версии")] + InvalidPrevHash = 1017, + + #[msg("Попытка изменить неизменяемое поле")] + ImmutableFieldChanged = 1018, + + #[msg("Попытка уменьшить лимит/баланс")] + BalanceDecrease = 1019, + + #[msg("Невалидная версия записи")] + InvalidVersion = 1020, + + #[msg("Размер записи превышает допустимый")] + RecordTooLarge = 1021, + + #[msg("Переполнение при вычислении")] + MathOverflow = 1022, + + #[msg("Неверный адрес получателя комиссии")] + InvalidFeeReceiver = 1023, + + #[msg("Пополнение лимита должно быть кратно шагу")] + InvalidLimitIncrement = 1024, + + #[msg("Невалидная magic-сигнатура записи")] + InvalidRecordMagic = 1025, + + +} + + + + + + + + + + + + + + + + + + + + +///---------------------------------------------------------------------------------------------------------- +/// Базовые функции для работы с PDA +///---------------------------------------------------------------------------------------------------------- + +/// Создаёт PDA аккаунт (если его ещё нет), и записывает в него массив байт. +/// +/// Аргументы: +/// - `pda_account`: аккаунт, куда записываем +/// - `signer`: кто платит за создание (обычно пользователь) +/// - `program_id`: адрес текущей программы +/// - `seeds`: слайс сидов, по которым создавался PDA +/// - `data`: байты для записи +/// - `space`: желаемый размер аккаунта +pub fn create_and_write_pda<'info>( + pda_account: &AccountInfo<'info>, + signer: &AccountInfo<'info>, + system_program: &AccountInfo<'info>, + program_id: &Pubkey, + seeds: &[&[u8]], + data: Vec, + space: u64, +) -> Result<()> { + // ─────────────────────────────────────────────── + // 1. Проверяем, создан ли аккаунт (если нет — owner = default) + if pda_account.owner == &Pubkey::default() { + msg!("Создаём PDA с размером {} байт", space); + + let space = space; //+ 128; // Добавляется запас под метаданные + // Вычисляем необходимую арендную плату + let lamports = Rent::get()?.minimum_balance(space as usize); + + // Формируем инструкцию + let create_instr = system_instruction::create_account( + signer.key, + pda_account.key, + lamports, + space, + program_id, + ); + + // Выполняем инструкцию с подписью от PDA + invoke_signed( + &create_instr, + &[ + signer.clone(), + pda_account.clone(), + system_program.clone(), + ], + &[&seeds], + )?; + } + + // ─────────────────────────────────────────────── + // 2. Пишем данные в аккаунт + let mut account_data = pda_account.try_borrow_mut_data()?; + + let copy_len = std::cmp::min(account_data.len(), data.len()); + account_data[..copy_len].copy_from_slice(&data[..copy_len]); + + // Если хочешь дополнить оставшееся нулями — раскомментируй: + // for i in copy_len..account_data.len() { + // account_data[i] = 0; + // } + + msg!("Успешно записано {} байт в PDA", copy_len); + Ok(()) +} + + + + +/// Создаёт PDA аккаунт (если его ещё нет). +/// +/// ⚠️ Если аккаунт уже существует, выбрасывается ошибка. +/// Используется внутри инструкций смарт-контракта. +/// +/// Аргументы: +/// - `pda_account`: аккаунт, который хотим создать (PDA) +/// - `signer`: кто оплачивает создание аккаунта (обычно пользователь) +/// - `system_program`: системная программа (`111...111`) +/// - `program_id`: адрес текущей программы (используется для подписи PDA) +/// - `seeds`: массив сидов, по которым вычислялся PDA +/// - `space`: желаемый размер аккаунта в байтах (только данных, без метаданных) +pub fn create_pda<'info>( + pda_account: &AccountInfo<'info>, + signer: &AccountInfo<'info>, + system_program: &AccountInfo<'info>, + program_id: &Pubkey, + seeds: &[&[u8]], + space: u64, +) -> Result<()> { + // ─────────────────────────────────────────────── + // 1. Проверяем, существует ли аккаунт + if pda_account.owner != &Pubkey::default() { + // Если владелец не равен Pubkey::default, значит аккаунт уже создан + // Возвращаем ошибку с пояснением + return Err(error!(ErrCode::PdaAlreadyExists)); + } + + // ─────────────────────────────────────────────── + // 2. Логируем, что будем создавать PDA + msg!("Создаём PDA-аккаунт на {} байт", space); + + // Добавляем запас под метаданные Solana (примерно 128 байт) + let full_space = space; + + // Получаем минимальный баланс для аренды (чтобы аккаунт не удалили) + let lamports = Rent::get()?.minimum_balance(full_space as usize); + + // ─────────────────────────────────────────────── + // 3. Создаём инструкцию system_program для создания аккаунта + let create_instr = system_instruction::create_account( + signer.key, // от имени кого + pda_account.key, // для какого PDA + lamports, // сколько лампортов перевести + full_space, // сколько байт выделить + program_id, // кто будет владельцем PDA + ); + + // ─────────────────────────────────────────────── + // 4. Выполняем инструкцию с подписью PDA (через сиды) + invoke_signed( + &create_instr, + &[ + signer.clone(), + pda_account.clone(), + system_program.clone(), + ], + &[&seeds], // PDA сиды → для подписи + )?; + + Ok(()) +} + +/// Записывает массив байт в PDA аккаунт (в начало data-секции). +/// +/// ⚠️ Убедись, что PDA был передан как `#[account(mut)]` +/// ⚠️ Эта функция ничего не создаёт, только пишет. +/// +/// Аргументы: +/// - `pda_account`: аккаунт, в который пишем (должен быть mut) +/// - `data`: бинарный массив, который нужно записать +pub fn write_to_pda<'info>( + pda_account: &AccountInfo<'info>, + data: &[u8], +) -> Result<()> { + // ─────────────────────────────────────────────── + // 1. Получаем доступ к данным PDA (на запись) + let mut account_data = pda_account.try_borrow_mut_data()?; + + // ─────────────────────────────────────────────── + // 2. Вычисляем сколько байт реально можно записать + // (на случай, если data длиннее, чем выделено место) + let copy_len = std::cmp::min(account_data.len(), data.len()); + + // ─────────────────────────────────────────────── + // 3. Копируем данные в аккаунт (с самого начала) + account_data[..copy_len].copy_from_slice(&data[..copy_len]); + + // Логируем, сколько байт записано + msg!("Успешно записано {} байт в PDA", copy_len); + + Ok(()) +} + + + + + + + + + + +/// ------------------------------------------------------------------------ +/// safe_read_pda ‒ «безопасное чтение PDA» +/// ------------------------------------------------------------------------ +/// +/// * Принимает: ссылку на `AccountInfo<'info>` PDA-аккаунта. +/// * Возвращает: `Vec` с данными аккаунта. +/// Если аккаунта нет или его данные пусты — возвращается `Vec::new()` +/// длиной 0 байт. +/// +/// Как работает ─────────────────────────────────────────────────────────── +/// 1. Проверяем, что аккаунт **инициализирован**: у не-инициализированного +/// owner = Pubkey::default(). Если owner нулевой — сразу отдаём пустой вектор. +/// 2. Если длина буфера == 0 (Anchor helper `data_is_empty()`), тоже отдаём пустой. +/// 3. Пытаемся безопасно (`try_borrow_data`) получить ссылку на данные. +/// - Успех → копируем их в Vec и возвращаем. +/// - Ошибка (например, конфликт borrow) → логируем и возвращаем пустой Vec. +/// +/// пример использования +/// let raw_bytes = safe_read_pda(&ctx.accounts.readonly_pda); +/// require!(!raw_bytes.is_empty(), ErrCode::EmptyPdaData); +/// msg!("Размер считанных данных: {}", raw_bytes.len()); +/// ------------------------------------------------------------------------ +pub fn safe_read_pda<'info>(pda_account: &AccountInfo<'info>) -> Vec { + // ───────────────────────────────────────────────────────────────────── + // 1) Аккаунт Н*Е* СУЩЕСТВУЕТ или не инициализирован: + // owner == Pubkey::default() (в Solana нулевой owner у пустого счёта) + // ───────────────────────────────────────────────────────────────────── + if pda_account.owner == &Pubkey::default() { + msg!("safe_read_pda: аккаунт не инициализирован ‒ возвращаем пустой массив"); + return Vec::new(); // [] + } + + // ───────────────────────────────────────────────────────────────────── + // 2) У аккаунта нет данных (длина 0) — тоже считаем «пустым» + // ───────────────────────────────────────────────────────────────────── + if pda_account.data_is_empty() { + msg!("safe_read_pda: у аккаунта data_len == 0 ‒ возвращаем пустой массив"); + return Vec::new(); + } + + // ───────────────────────────────────────────────────────────────────── + // 3) Пытаемся безопасно забрать буфер данных; ошибки перехватываем + // ───────────────────────────────────────────────────────────────────── + match pda_account.try_borrow_data() { + Ok(data_ref) => { + // to_vec() копирует bytes → Vec, чтобы дальше работать без borrow-лифа + data_ref.to_vec() + } + Err(e) => { + // Ошибка при borrow (например, уже есть активное мутабельное заимствование) + msg!("safe_read_pda: ошибка borrow_data ({:?}) ‒ возвращаем пустой массив", e); + Vec::new() + } + } +} + + + + + + +/// ------------------------------------------------------------------------ +/// delete_pda_with_assign — закрыть PDA, вернуть ренту и освободить адрес +/// ------------------------------------------------------------------------ +/// +/// Параметры: +/// - `pda_account` : PDA-аккаунт (mut), который закрываем (owned вашей программой) +/// - `recipient` : счёт, на который возвращаем лампорты (обычно пользователь) +/// - `system_program`: системная программа (111...111) +/// - `program_id` : Pubkey вашей программы (проверка владельца) +/// - `seeds` : сиды PDA (в том же порядке, как при создании), чтобы PDA «подписал» assign +/// +/// Делает: +/// 1) Проверяет, что PDA принадлежит вашей программе. +/// 2) Обнуляет данные и сжимает их до 0 байт (realloc(0)). +/// 3) Переводит все лампорты PDA на `recipient`. +/// 4) Делает `assign` владельца на System Program (через `invoke_signed`). +/// +/// Результат: +/// — В конце транзакции аккаунт с lamports=0 и data_len=0 будет удалён рантаймом, +/// владелец = System Program (чисто/ожидаемо). +/// — В следующей транзакции можно снова создать PDA с тем же сидом. +/// ------------------------------------------------------------------------ + +pub fn delete_pda_return_rent<'info>( + pda_account: &AccountInfo<'info>, + recipient: &AccountInfo<'info>, + program_id: &Pubkey, +) -> Result<()> { + // 0) проверки + require!(pda_account.owner != &Pubkey::default(), ErrCode::EmptyPdaData); + require!(pda_account.owner == program_id, ErrCode::InvalidPdaAddress); + + // 1) Переложить все лампорты с PDA на получателя (мы владелец, это разрешено) + let amount = **pda_account.lamports.borrow(); + if amount > 0 { + **recipient.lamports.borrow_mut() = recipient + .lamports() + .checked_add(amount) + .ok_or(ProgramError::InsufficientFunds)?; + **pda_account.lamports.borrow_mut() = 0; + } + + // 2) Нулим данные (если были) + if !pda_account.data_is_empty() { + let mut data = pda_account.try_borrow_mut_data()?; + for b in data.iter_mut() { *b = 0; } + } + + // 3) Сжать до 0 байт + pda_account.realloc(0, false)?; + + // Никаких assign/transfer больше не делаем — это надёжнее. + msg!("PDA закрыт: рента отправлена на {}", recipient.key); + Ok(()) +} + diff --git a/shine/programs/shine_payments/Cargo.toml b/shine/programs/shine_payments/Cargo.toml new file mode 100644 index 0000000..add95e4 --- /dev/null +++ b/shine/programs/shine_payments/Cargo.toml @@ -0,0 +1,33 @@ +[package] +name = "shine_payments" +version = "0.1.0" +description = "Payments and investments smart contract" +edition = "2021" + +[lib] +crate-type = ["cdylib", "lib"] +name = "shine_payments" +test = false +doctest = false +bench = false + +[dependencies] +anchor-lang = "0.31.1" +common = { path = "../common" } + +# ==== добавлено для NFT-функционала ==== +anchor-spl = { version = "0.31.1", features = ["associated_token", "token"] } +mpl-token-metadata = "5.1.1" +spl-token = { version = "4.0.0", features = ["no-entrypoint"] } +# ====================================== + +[features] +default = [] +no-entrypoint = [] +no-idl = [] +no-log-ix-name = [] +anchor-debug = [] +custom-heap = [] +custom-panic = [] +cpi = [] +idl-build = ["anchor-lang/idl-build"] diff --git a/shine/programs/shine_payments/dApp/copyToServer.sh b/shine/programs/shine_payments/dApp/copyToServer.sh new file mode 100755 index 0000000..55f4dfd --- /dev/null +++ b/shine/programs/shine_payments/dApp/copyToServer.sh @@ -0,0 +1,30 @@ +#!/bin/bash + +# Скрипт для копирования dApp на тестовый сервер + +# Настройки +LOCAL_FILE="init.html" +REMOTE_USER="aidar" +REMOTE_HOST="shineup.me" +REMOTE_PATH="/home/aidar/Docker_server/site/dApp" + +# Проверка, что файл существует +if [ ! -f "$LOCAL_FILE" ]; then + echo "Ошибка: файл $LOCAL_FILE не найден." + exit 1 +fi + +# Копирование файла +scp "$LOCAL_FILE" "${REMOTE_USER}@${REMOTE_HOST}:${REMOTE_PATH}" + +# Проверка результата +if [ $? -eq 0 ]; then + echo "Файл успешно загружен на сервер." +else + echo "Ошибка при загрузке файла на сервер." + exit 1 +fi + +#echo +#echo "Нажмите Enter, чтобы закрыть..." +#read diff --git a/shine/programs/shine_payments/dApp/init.html b/shine/programs/shine_payments/dApp/init.html new file mode 100644 index 0000000..3e2b8ac --- /dev/null +++ b/shine/programs/shine_payments/dApp/init.html @@ -0,0 +1,404 @@ + + + + + + Shine Payments — Phantom demo (devnet, deep logs) + + + + + + + +

Shine Payments — Phantom wallet (devnet)

+ +
+ + + + + +
+ +
+ В Phantom выбери сеть Devnet. Логи ниже и в консоли (F12 → Console). +
+ +

+
+
+
+
+
+
diff --git a/shine/programs/shine_payments/src/investments.rs b/shine/programs/shine_payments/src/investments.rs
new file mode 100644
index 0000000..b9d71fa
--- /dev/null
+++ b/shine/programs/shine_payments/src/investments.rs
@@ -0,0 +1,326 @@
+use anchor_lang::prelude::*;
+
+use anchor_lang::solana_program::{program::invoke_signed, system_instruction};
+use common::utils::*; // тянем общие PDA-хелперы из programs/common
+
+// === добавлено: используем наш NFT-модуль ===
+use crate::nft::{CreateNftParams, create_nft_with_freeze};
+// ============================================
+
+/// Утилита чтения структуры из PDA: читает байты и десериализует.
+/// Возвращает ошибку, если данных нет/пустые/неверный формат.
+fn read_state_from_pda(pda: &AccountInfo) -> Result {
+    let raw = safe_read_pda(pda);                         // ← берём Vec (или пустой)
+    require!(!raw.is_empty(), ErrCode::EmptyPdaData);     // ← пусто — ошибка
+    let st = deserialize_invest_state(&raw)?;             // ← десериализуем по формату
+    require!(st.format == INVEST_STATE_FORMAT_V1, ErrCode::UnsupportedFormat); // ← проверяем версию
+    Ok(st)
+}
+
+/// Утилита записи структуры в PDA: сериализует и пишет.
+/// Важно: сам аккаунт уже должен существовать и быть #[account(mut)].
+fn write_state_to_pda(pda: &AccountInfo, s: &InvestState) -> Result<()> {
+    let raw = serialize_invest_state_v1(s); // ← 24 байта
+    write_to_pda(pda, &raw)                 // ← записываем в начало data
+}
+
+/// ==============================================
+/// Контексты инструкций (минимально необходимые)
+/// ==============================================
+
+/// init: создаём PDA и кладём в него PayStateV1 {format=1, coef=10, ...0}
+#[derive(Accounts)]
+pub struct Init<'info> {
+    /// Плательщик аренды за PDA; подписант транзакции.
+    #[account(mut)]
+    pub payer: Signer<'info>,
+
+    /// Наш PDA (с произвольным типом, чтобы работать через AccountInfo).
+    /// Проверку адреса делаем в handler (по seed + bump), чтобы избежать подмены.
+    /// CHECK: проверяется вручную по адресу
+    #[account(mut)]
+    pub state_pda: UncheckedAccount<'info>,
+
+    /// Системная программа.
+    pub system_program: Program<'info, System>,
+}
+
+/// Общие аккаунты для invest/add_bonus/claim:
+/// Везде просто читаем/пишем одно и то же состояние из того же PDA.
+#[derive(Accounts)]
+pub struct UseState<'info> {
+    /// Любой платящий/подписант (в реальном коде — свои проверки).
+    pub signer: Signer<'info>,
+
+    /// Тот же PDA с состоянием (должен уже существовать).
+    /// CHECK: проверяется вручную по адресу
+    #[account(mut)]
+    pub state_pda: UncheckedAccount<'info>,
+
+    /// Системная программа (на всякий случай; может не понадобиться).
+    pub system_program: Program<'info, System>,
+}
+
+/// ==============================================
+/// Программа
+/// ==============================================
+
+use super::*;
+use anchor_lang::prelude::*;
+
+
+/// ------------------------------------------
+/// init: создаёт PDA и записывает в него дефолтное состояние.
+/// format = 1, coef = 10, остальные поля = 0.
+/// ------------------------------------------
+pub fn init(ctx: Context) -> Result<()> {
+    let program_id = ctx.program_id; // ← адрес этой программы
+
+    // 1. Проверка что вызывает именно разрешённый ключ
+    /* todo   пока все могут вызыватьно                                         !! но в итоге будет добавленна проверка что бы только дао могло вызвать эту функцию один раз
+    require_keys_eq!(
+        ctx.accounts.payer.key(),
+        ALLOWED_INIT_CALLER,
+        ErrCode::InvalidSigner
+    );
+*/
+
+    // 2. Проверка что PDA ещё не создан
+    if ctx.accounts.state_pda.data_len() > 0 && ctx.accounts.state_pda.owner != &System::id() {
+        return Err(error!(ErrCode::PdaAlreadyExists));
+    }
+    
+    // 2. Ещё раз Проверка что PDA ещё не создан  
+    if ctx.accounts.state_pda.owner != &System::id()
+        || ctx.accounts.state_pda.lamports() > 0
+    {
+        // Если аккаунт уже создан и не пустой
+        return Err(error!(ErrCode::PdaAlreadyExists));
+    }
+    
+    let pda_key_expected = Pubkey::find_program_address(&[crate::PDA_SEED_PREFIX], program_id).0; // ← вычисляем PDA
+    require_keys_eq!(
+        pda_key_expected,
+        ctx.accounts.state_pda.key(),
+        ErrCode::InvalidPdaAddress
+    ); // ← убеждаемся, что нам подали именно правильный PDA
+
+    // Конструируем дефолтную структуру состояния.
+    let state = InvestState {
+        format: INVEST_STATE_FORMAT_V1,  // ← 1
+        coef: crate::DEFAULT_COEF,       // ← 10
+        q1_tokens: 0,                    // ← нули
+        sum1_bonus: 0,
+        q1_paid_tokens: 0,
+        sum1_paid_bonus: 0,
+    };
+
+    // Сериализуем в 24 байта.
+    let data = serialize_invest_state_v1(&state);
+
+    // Для подписи PDA нужен bump; здесь получим (ключ, bump).
+    let (_pda_key, bump) = Pubkey::find_program_address(&[crate::PDA_SEED_PREFIX], program_id);
+
+    // Сиды для invoke_signed: [seed, bump]
+    let seeds: [&[u8]; 2] = [crate::PDA_SEED_PREFIX, &[bump]];
+
+    // Создаём и сразу записываем, арендный минимум оплачивает payer.
+    create_and_write_pda(
+        &ctx.accounts.state_pda.to_account_info(),   // куда пишем
+        &ctx.accounts.payer.to_account_info(),       // кто платит
+        &ctx.accounts.system_program.to_account_info(),
+        program_id,
+        &seeds,
+        data,
+        crate::PAY_STATE_SPACE,                      // резерв с запасом
+    )?;
+
+    Ok(())
+}
+
+/// ------------------------------------------
+/// invest: «внос инвестиций».
+/// По заданию: в начале читаем состояние, в конце сохраняем.
+/// (Здесь логика модификации не задана — оставляем как заглушку.)
+/// ------------------------------------------
+pub fn invest(ctx: Context, _amount: u64) -> Result<()> {
+    // 1) читаем
+    let mut st = read_state_from_pda(&ctx.accounts.state_pda.to_account_info())?; // ← PayStateV1
+
+    // --- тут можно модифицировать st по твоей бизнес-логике ---
+    // Например, ничего не меняем сейчас (заглушка).
+    let _ = &mut st; // чтоб компилятор не ругался, если пока не используем
+
+    // 2) сохраняем
+    write_state_to_pda(&ctx.accounts.state_pda.to_account_info(), &st)?;
+    Ok(())
+}
+
+/// ------------------------------------------
+/// add_bonus: «начисление бонусов» (обычно вызывать от DAO).
+/// По заданию: читаем в начале, создаём/добавляем NFT в очередь, сохраняем в конце.
+/// Для операций с NFT используем расширенный контекст AddBonusCtx (см. lib.rs).
+/// ------------------------------------------
+pub fn add_bonus(ctx: Context, investor: Pubkey, amount: u64) -> Result<()> {
+    // 1) читаем состояние
+    let mut st = read_state_from_pda(&ctx.accounts.state_pda.to_account_info())?;
+
+    // 2) создаём/добавляем NFT через модуль nft (создание metadata, mint 1, freeze, master edition, verify)
+    let next_index = st.q1_tokens as u64 + 1;
+    let params = CreateNftParams {
+        name: format!("Bonus #{}", next_index),
+        symbol: "BN".to_string(),
+        uri: "https://example.com/nft.json".to_string(), // заглушка для devnet-теста
+        index: next_index,
+        recipient: investor,
+    };
+
+    // ВАЖНО: mint_pda должен быть создан ТЕСТОМ заранее с decimals=0, mint_authority=signer, freeze_authority=signer.
+    create_nft_with_freeze(&ctx, params)?;
+
+    // 3) обновляем агрегаты очереди (минимально: увеличим счётчик и сумму бонусов)
+    st.q1_tokens = st.q1_tokens.saturating_add(1);
+    let add = u32::try_from(core::cmp::min(amount, u64::from(u32::MAX))).unwrap_or(u32::MAX);
+    st.sum1_bonus = st.sum1_bonus.saturating_add(add);
+
+    // 4) сохраняем
+    write_state_to_pda(&ctx.accounts.state_pda.to_account_info(), &st)?;
+    Ok(())
+}
+
+/// ------------------------------------------
+/// claim: «выплата».
+/// По заданию: читаем в начале, сохраняем в конце.
+/// ------------------------------------------
+pub fn claim(ctx: Context) -> Result<()> {
+    // 1) читаем
+    let mut st = read_state_from_pda(&ctx.accounts.state_pda.to_account_info())?;
+
+    // --- тут твоя логика списаний/выплат ---
+    let _ = &mut st; // заглушка
+
+    // 2) сохраняем
+    write_state_to_pda(&ctx.accounts.state_pda.to_account_info(), &st)?;
+    Ok(())
+}
+
+
+
+
+
+//todo
+
+
+
+
+
+/// ==============================================
+/// Коды ошибок (берём из твоего блока; можно расширять)
+/// ==============================================
+
+#[error_code]
+pub enum ErrCode {
+    /// Система уже инициализирована и не может быть инициализирована повторно!
+    #[msg("Система уже инициализирована и не может быть инициализирована повторно!")]
+    SystemAlreadyInitialized = 1000,
+
+    #[msg("PDA не содержит данных или не инициализирован")]
+    EmptyPdaData = 1002,
+
+    #[msg("Пользователь уже зарегистрирован")]
+    UserAlreadyExists = 1003,
+
+    #[msg("Некорректный логин")]
+    InvalidLogin = 1004,
+
+    #[msg("Не совпадает PDA адрес")]
+    InvalidPdaAddress = 1006,
+
+    #[msg("Формат данных не поддерживается")]
+    UnsupportedFormat = 1011,
+
+    #[msg("Ошибка при десериализации")]
+    DeserializationError = 1012,
+
+    /// PDA уже существует, создание невозможно
+    #[msg("PDA-аккаунт уже существует и не может быть создан повторно.")]
+    PdaAlreadyExists = 1009,
+
+    #[msg("Подписавший не совпадает с ожидаемым пользователем (временное ограничение)")]
+    InvalidSigner = 1005,
+
+    /// Не получилось создать пользователя
+    #[msg("Не получилось создать пользователя, система уже перегружена, попробуйте позже!")]
+    NoSuitableIdPda = 1010,
+}
+
+
+
+
+
+use anchor_lang::prelude::*;
+
+/// ================================
+/// КОНСТАНТЫ ФОРМАТА / ДЛИНЫ ДАННЫХ
+/// ================================
+
+/// Версия формата хранения состояния.
+/// Мы жёстко фиксируем «1», чтобы код мог отличать будущие версии.
+pub const INVEST_STATE_FORMAT_V1: u32 = 1;
+
+/// Сырые данные состояния V1 занимают ровно 6 * 4 = 24 байта.
+pub const INVEST_STATE_RAW_LEN_V1: usize = 24; // байт
+
+/// ================================
+/// ОПИСАНИЕ СТРУКТУРЫ СОСТОЯНИЯ (V1)
+/// ================================
+#[derive(Clone, Copy, Debug, Default)]
+pub struct InvestState {
+    pub format: u32,
+    pub coef: u32,
+    pub q1_tokens: u32,
+    pub sum1_bonus: u32,
+    pub q1_paid_tokens: u32,
+    pub sum1_paid_bonus: u32,
+}
+
+/// ========================================
+/// СЕРИАЛИЗАЦИЯ (структура -> массив байт)
+/// ========================================
+pub fn serialize_invest_state_v1(s: &InvestState) -> Vec {
+    let mut out = Vec::with_capacity(INVEST_STATE_RAW_LEN_V1);
+    out.extend_from_slice(&INVEST_STATE_FORMAT_V1.to_le_bytes()); // [0..4)
+    out.extend_from_slice(&s.coef.to_le_bytes());            // [4..8)
+    out.extend_from_slice(&s.q1_tokens.to_le_bytes());       // [8..12)
+    out.extend_from_slice(&s.sum1_bonus.to_le_bytes());      // [12..16)
+    out.extend_from_slice(&s.q1_paid_tokens.to_le_bytes());  // [16..20)
+    out.extend_from_slice(&s.sum1_paid_bonus.to_le_bytes()); // [20..24)
+    debug_assert_eq!(out.len(), INVEST_STATE_RAW_LEN_V1);
+    out
+}
+
+/// ===========================================
+/// ДЕСЕРИАЛИЗАЦИЯ (массив байт -> структура)
+/// ===========================================
+pub fn deserialize_invest_state(data: &[u8]) -> Result {
+    if data.len() < INVEST_STATE_RAW_LEN_V1 {
+        return Err(error!(ErrCode::DeserializationError));
+    }
+    fn read_u32_le(slice: &[u8], start: usize) -> u32 {
+        let bytes: [u8; 4] = slice[start..start + 4]
+            .try_into()
+            .expect("slice has enough length due to pre-check");
+        u32::from_le_bytes(bytes)
+    }
+    let format = read_u32_le(data, 0);
+    if format != INVEST_STATE_FORMAT_V1 {
+        return Err(error!(ErrCode::UnsupportedFormat));
+    }
+    let coef            = read_u32_le(data, 4);
+    let q1_tokens       = read_u32_le(data, 8);
+    let sum1_bonus      = read_u32_le(data, 12);
+    let q1_paid_tokens  = read_u32_le(data, 16);
+    let sum1_paid_bonus = read_u32_le(data, 20);
+
+    Ok(InvestState { format, coef, q1_tokens, sum1_bonus, q1_paid_tokens, sum1_paid_bonus })
+}
diff --git a/shine/programs/shine_payments/src/lib.rs b/shine/programs/shine_payments/src/lib.rs
new file mode 100644
index 0000000..aa1bfee
--- /dev/null
+++ b/shine/programs/shine_payments/src/lib.rs
@@ -0,0 +1,158 @@
+use anchor_lang::prelude::*;
+
+declare_id!("6Hes38UKFGF8cfQDQDVWoMGcSzGMUAgamWG31hCVhyPY");
+
+
+/// Подключаем модуль с полной реализацией.
+pub mod investments;
+use investments::*; // импортируем всё в корень
+
+// === модуль NFT ===
+pub mod nft;
+
+// ==============================================
+// Константы формата / сидов / размеров
+// ==============================================
+
+/// Префикс (seed) для PDA, где храним глобальное состояние выплат.
+/// Важно: сид — это просто набор байт; здесь он фиксированный.
+pub const PDA_SEED_PREFIX: &[u8] = b"shine_investments_state";
+
+/// Значение коэффициента «по умолчанию» при инициализации.
+pub const DEFAULT_COEF: u32 = 10; // ← «коэффициент» = 10 при init
+
+/// Ровно столько байт резервируем под PDA-данные.
+/// (Можно добавить запас на будущее, но по заданию — только 28.)
+pub const PAY_STATE_SPACE: u64 = 50; // просто сделал с запасом
+
+// ==============================================
+// Программа
+// ==============================================
+
+#[program]
+pub mod shine_payments {
+    use super::*;
+    // Явно подтягиваем типы и функции, чтобы не было путаницы после предыдущих ошибок парсера
+    use crate::investments::{Init, UseState};
+    use crate::investments::{
+        add_bonus as inv_add_bonus, claim as inv_claim, init as inv_init, invest as inv_invest,
+        ErrCode,
+    };
+
+    /// init — создаёт PDA и кладёт дефолтное состояние.
+    pub fn init(ctx: Context) -> Result<()> {
+        inv_init(ctx)
+    }
+
+    /// invest — в начале читает состояние, в конце сохраняет (логика внутри модуля).
+    pub fn invest(ctx: Context, amount: u64) -> Result<()> {
+        inv_invest(ctx, amount)
+    }
+
+    /// add_bonus — начисление бонусов (обычно от DAO).
+    /// Для NFT используем расширенный контекст AddBonusCtx (с аккаунтами коллекции и т.п.).
+    pub fn add_bonus(ctx: Context, investor: Pubkey, amount: u64) -> Result<()> {
+        inv_add_bonus(ctx, investor, amount)
+    }
+
+    /// claim — выплата.
+    pub fn claim(ctx: Context) -> Result<()> {
+        inv_claim(ctx)
+    }
+
+    /// ВРЕМЕННАЯ ФУНКЦИЯ только для тестов (в итоговой версии её не будет):
+    /// deleteInit — удалить PDA из init и вернуть ренту подписанту.
+    pub fn delete_init(ctx: Context) -> Result<()> {
+        let program_id = ctx.program_id;
+
+        // PDA по тем же сид/бамп, что и в init
+        let (expected_pda, _bump) = Pubkey::find_program_address(&[PDA_SEED_PREFIX], program_id);
+        require_keys_eq!(
+            expected_pda,
+            ctx.accounts.state_pda.key(),
+            ErrCode::InvalidPdaAddress
+        );
+
+        // Рента уйдёт на счёт подписанта (signer)
+        common::utils::delete_pda_return_rent(
+            &ctx.accounts.state_pda.to_account_info(),
+            &ctx.accounts.signer.to_account_info(),
+            program_id,
+        )
+    }
+}
+
+// ==============================================
+// Контексты вне #[program]
+// ==============================================
+
+/// Контекст для deleteInit (временный для тестов)
+#[derive(Accounts)]
+pub struct DeleteInit<'info> {
+    /// Подписант транзакции — ПОЛУЧАТЕЛЬ ренты
+    #[account(mut)]
+    pub signer: Signer<'info>,
+
+    /// Тот самый PDA из init
+    /// CHECK: адрес валидируем в хендлере по сид-у
+    #[account(mut)]
+    pub state_pda: UncheckedAccount<'info>,
+
+    /// Системная программа
+    pub system_program: Program<'info, System>,
+}
+
+/// Контекст для add_bonus: полный набор аккаунтов для операций с NFT и коллекцией.
+/// (Комменты по стилю проекта оставлены.)
+#[derive(Accounts)]
+pub struct AddBonusCtx<'info> {
+    /// Любой платящий/подписант (в реальном коде — свои проверки).
+    #[account(mut)]
+    pub signer: Signer<'info>,
+
+    /// Тот же PDA с состоянием (должен уже существовать).
+    /// CHECK: проверяется вручную по адресу
+    #[account(mut)]
+    pub state_pda: UncheckedAccount<'info>,
+
+    // --- аккаунты минтимого NFT ---
+    /// Mint создаваемого NFT (должен быть создан заранее: decimals=0, mint_authority=signer, freeze_authority=signer)
+    /// CHECK
+    #[account(mut)]
+    pub mint_pda: UncheckedAccount<'info>,
+
+    /// ATA получателя (может быть предсоздан тестом)
+    /// CHECK
+    #[account(mut)]
+    pub recipient_ata: UncheckedAccount<'info>,
+    /// Владелец ATA (инвестор)
+    /// CHECK
+    pub recipient_owner: UncheckedAccount<'info>,
+
+    // --- аккаунты коллекции (уже созданной заранее) ---
+    /// CHECK
+    pub collection_mint: UncheckedAccount<'info>,
+    /// CHECK
+    #[account(mut)]
+    pub collection_metadata_pda: UncheckedAccount<'info>,
+    /// CHECK
+    #[account(mut)]
+    pub collection_master_edition_pda: UncheckedAccount<'info>,
+    /// Апдейтер коллекции (update authority)
+    pub collection_update_authority: Signer<'info>,
+
+    // --- metadata + master edition для создаваемого NFT ---
+    /// CHECK
+    #[account(mut)]
+    pub metadata_pda: UncheckedAccount<'info>,
+    /// CHECK
+    #[account(mut)]
+    pub master_edition_pda: UncheckedAccount<'info>,
+
+    // --- программы ---
+    /// CHECK: проверяется по ID внутри nft.rs
+    pub token_metadata_program: UncheckedAccount<'info>,
+    pub token_program: Program<'info, anchor_spl::token::Token>,
+    pub associated_token_program: Program<'info, anchor_spl::associated_token::AssociatedToken>,
+    pub system_program: Program<'info, System>,
+}
diff --git a/shine/programs/shine_payments/src/nft.rs b/shine/programs/shine_payments/src/nft.rs
new file mode 100644
index 0000000..f78fa9c
--- /dev/null
+++ b/shine/programs/shine_payments/src/nft.rs
@@ -0,0 +1,190 @@
+use anchor_lang::prelude::*;
+use anchor_lang::solana_program::{program::invoke, program::invoke_signed};
+use anchor_spl::{associated_token::AssociatedToken, token::Token};
+
+use mpl_token_metadata::{
+    ID as TM_ID,
+    instructions::{
+        CreateMasterEditionV3Builder,
+        CreateMetadataAccountV3Builder,
+        SetAndVerifySizedCollectionItemBuilder,
+    },
+    types::{Collection, Creator, DataV2, Uses, UseMethod},
+};
+
+use spl_token::instruction as spl_ix;
+
+/// Параметры для минта NFT
+#[derive(Clone)]
+pub struct CreateNftParams {
+    pub name: String,
+    pub symbol: String,
+    pub uri: String,
+    pub index: u64,
+    pub recipient: Pubkey,
+}
+
+/// Создание metadata, чеканка 1 токена, freeze ATA, создание master edition, verify в коллекции.
+pub fn create_nft_with_freeze(
+    ctx: &Context,
+    params: CreateNftParams,
+) -> Result<()> {
+    let a = &ctx.accounts;
+
+    // Проверяем что это именно программа Metaplex Token Metadata
+    require_keys_eq!(a.token_metadata_program.key(), TM_ID, CustomError::InvalidMetadataProgram);
+
+    // 1) Создание Metadata для нового NFT
+    let creators = Some(vec![Creator {
+        address: a.collection_update_authority.key(),
+        verified: true,
+        share: 100,
+    }]);
+
+    let data = DataV2 {
+        name: truncate(¶ms.name, 32),
+        symbol: truncate(¶ms.symbol, 10),
+        uri: truncate(¶ms.uri, 256),
+        seller_fee_basis_points: 0,
+        creators,
+        collection: Some(Collection {
+            verified: false, // отметим как часть коллекции позже через verify
+            key: a.collection_mint.key(),
+        }),
+        uses: Some(Uses {
+            use_method: UseMethod::Burn,
+            remaining: 1,
+            total: 1,
+        }),
+    };
+
+    // В mpl-token-metadata v5 update_authority(pubkey, is_signer: bool)
+    let ix_md = CreateMetadataAccountV3Builder::new()
+        .metadata(a.metadata_pda.key())
+        .mint(a.mint_pda.key())
+        .mint_authority(a.signer.key())
+        .payer(a.signer.key())
+        .update_authority(a.collection_update_authority.key(), true)
+        .system_program(a.system_program.key())
+        .data(data)
+        .is_mutable(true)
+        .instruction();
+
+    invoke_signed(
+        &ix_md,
+        &[
+            a.metadata_pda.to_account_info(),
+            a.mint_pda.to_account_info(),
+            a.signer.to_account_info(),
+            a.collection_update_authority.to_account_info(),
+            a.system_program.to_account_info(),
+            a.token_metadata_program.to_account_info(),
+        ],
+        &[],
+    )?;
+
+    // 2) Чеканим 1 токен на ATA получателя
+    let ix_mint_to = spl_ix::mint_to(
+        &a.token_program.key(),
+        &a.mint_pda.key(),
+        &a.recipient_ata.key(),
+        &a.signer.key(),
+        &[],
+        1,
+    )?;
+    invoke(
+        &ix_mint_to,
+        &[
+            a.mint_pda.to_account_info(),
+            a.recipient_ata.to_account_info(),
+            a.signer.to_account_info(),
+            a.token_program.to_account_info(),
+        ],
+    )?;
+
+    // 3) Замораживаем ATA получателя (freeze authority = signer)
+    let ix_freeze = spl_ix::freeze_account(
+        &a.token_program.key(),
+        &a.recipient_ata.key(),
+        &a.mint_pda.key(),
+        &a.signer.key(),
+        &[],
+    )?;
+    invoke(
+        &ix_freeze,
+        &[
+            a.recipient_ata.to_account_info(),
+            a.mint_pda.to_account_info(),
+            a.signer.to_account_info(),
+            a.token_program.to_account_info(),
+        ],
+    )?;
+
+    // 4) Создаём Master Edition
+    let ix_me = CreateMasterEditionV3Builder::new()
+        .edition(a.master_edition_pda.key())
+        .mint(a.mint_pda.key())
+        .update_authority(a.collection_update_authority.key())
+        .mint_authority(a.signer.key())
+        .payer(a.signer.key())
+        .metadata(a.metadata_pda.key())
+        .token_program(a.token_program.key())
+        .system_program(a.system_program.key())
+        .max_supply(0)
+        .instruction();
+
+    invoke_signed(
+        &ix_me,
+        &[
+            a.master_edition_pda.to_account_info(),
+            a.mint_pda.to_account_info(),
+            a.collection_update_authority.to_account_info(),
+            a.signer.to_account_info(),
+            a.metadata_pda.to_account_info(),
+            a.token_program.to_account_info(),
+            a.system_program.to_account_info(),
+            a.token_metadata_program.to_account_info(),
+        ],
+        &[],
+    )?;
+
+    // 5) Verify как часть коллекции
+    // Метод называется collection_master_edition_account(...)
+    let ix_verify = SetAndVerifySizedCollectionItemBuilder::new()
+        .metadata(a.metadata_pda.key())
+        .collection_authority(a.collection_update_authority.key())
+        .payer(a.signer.key())
+        .update_authority(a.collection_update_authority.key())
+        .collection_mint(a.collection_mint.key())
+        .collection(a.collection_metadata_pda.key())
+        .collection_master_edition_account(a.collection_master_edition_pda.key())
+        .instruction();
+
+    invoke_signed(
+        &ix_verify,
+        &[
+            a.metadata_pda.to_account_info(),
+            a.collection_update_authority.to_account_info(),
+            a.signer.to_account_info(),
+            a.collection_update_authority.to_account_info(),
+            a.collection_mint.to_account_info(),
+            a.collection_metadata_pda.to_account_info(),
+            a.collection_master_edition_pda.to_account_info(),
+            a.token_metadata_program.to_account_info(),
+        ],
+        &[],
+    )?;
+
+    msg!("NFT создан, заморожен, мастер-издание создано и верифицировано в коллекции (index={})", params.index);
+    Ok(())
+}
+
+fn truncate(s: &str, max: usize) -> String {
+    if s.len() <= max { s.to_string() } else { s.chars().take(max).collect() }
+}
+
+#[error_code]
+pub enum CustomError {
+    #[msg("Invalid Token Metadata program account")]
+    InvalidMetadataProgram,
+}
diff --git a/shine/programs/shine_users/Cargo.toml b/shine/programs/shine_users/Cargo.toml
new file mode 100644
index 0000000..0406bdc
--- /dev/null
+++ b/shine/programs/shine_users/Cargo.toml
@@ -0,0 +1,30 @@
+[package]
+name = "shine_users"
+version = "0.1.0"
+description = "User registration smart contract"
+edition = "2021"
+
+[lib]
+crate-type = ["cdylib", "lib"]
+name = "shine_users"
+test = false
+doctest = false
+bench = false
+
+[dependencies]
+anchor-lang = "0.31.1"
+common = { path = "../common" }
+ed25519-dalek = { version = "1.0.1", default-features = false, features = ["u64_backend"] }
+sha2 = "0.10"
+
+
+[features]
+default = []
+no-entrypoint = []
+no-idl = []
+no-log-ix-name = []
+anchor-debug = []
+custom-heap = []
+custom-panic = []
+cpi = []
+idl-build = ["anchor-lang/idl-build"]
diff --git a/shine/programs/shine_users/src/lib.rs b/shine/programs/shine_users/src/lib.rs
new file mode 100644
index 0000000..d9829b5
--- /dev/null
+++ b/shine/programs/shine_users/src/lib.rs
@@ -0,0 +1,23 @@
+use anchor_lang::prelude::*;
+
+pub mod users;
+pub mod settings;
+
+use users::*;
+
+
+declare_id!("5dFcWDNp42Xn9Vv4oDMJzM4obBJ8hvDuAtPX54fT5L3t");
+
+
+#[program]
+pub mod shine {
+    use super::*;
+
+    pub fn create_user_pda(ctx: Context, args: CreateUserPdaArgs) -> Result<()> {
+        users::create_user_pda(ctx, args)
+    }
+
+    pub fn update_user_pda(ctx: Context, args: UpdateUserPdaArgs) -> Result<()> {
+        users::update_user_pda(ctx, args)
+    }
+}
diff --git a/shine/programs/shine_users/src/settings.rs b/shine/programs/shine_users/src/settings.rs
new file mode 100644
index 0000000..e1b8bfe
--- /dev/null
+++ b/shine/programs/shine_users/src/settings.rs
@@ -0,0 +1,12 @@
+pub const USER_PDA_SEED_PREFIX: &str = "login=";
+// Увеличили размер PDA, чтобы оставить запас под будущие расширения формата
+// (в частности, сценарии ротации root key с дополнительной подписью старого ключа).
+pub const USER_PDA_SPACE: usize = 768;
+
+pub const REGISTRATION_FEE_RECEIVER: &str = "6bFc5Gz5qF172GQhK5HpDbWs8F6qcSxdHn5XqAstf1fY";
+pub const REGISTRATION_FEE_LAMPORTS: u64 = 10_000_000; // 0.01 SOL
+
+pub const LIMIT_STEP: u64 = 10_000;
+pub const LAMPORTS_PER_LIMIT_STEP: u64 = 100_000; // 0.0001 SOL за 10_000 лимита
+
+pub const START_BONUS_LIMIT: u64 = 100_000;
diff --git a/shine/programs/shine_users/src/users.rs b/shine/programs/shine_users/src/users.rs
new file mode 100644
index 0000000..7f0585f
--- /dev/null
+++ b/shine/programs/shine_users/src/users.rs
@@ -0,0 +1,592 @@
+use crate::settings;
+use anchor_lang::prelude::*;
+use anchor_lang::solana_program::{program::invoke, system_instruction};
+use common::utils::{create_pda, safe_read_pda, write_to_pda, ErrCode};
+use ed25519_dalek::{PublicKey, Signature, Verifier};
+use sha2::{Digest, Sha256};
+use std::str::FromStr;
+
+const MAGIC: &[u8; 5] = b"SHiNE";
+const FORMAT_MAJOR: u8 = 1;
+const FORMAT_MINOR: u8 = 0;
+const RESERVED_BYTES: [u8; 5] = [0, 0, 0, 0, 0];
+const ZERO_HASH: [u8; 32] = [0; 32];
+
+#[derive(AnchorSerialize, AnchorDeserialize, Clone, Debug)]
+pub struct UserMutableFields {
+    pub blockchain_key: Pubkey,
+    pub device_key: Pubkey,
+    pub chain_number: u16,
+    pub is_server: bool,
+    pub server_key: Pubkey,
+    pub server_address: String,
+    pub connection_servers: Vec,
+    pub trusted_count: u8,
+}
+
+#[derive(AnchorSerialize, AnchorDeserialize, Clone, Debug)]
+pub struct CreateUserPdaArgs {
+    pub login: String,
+    pub root_key: Pubkey,
+    pub created_at_ms: u64,
+    pub additional_limit: u64,
+    pub fields: UserMutableFields,
+    pub signature: Vec,
+}
+
+#[derive(AnchorSerialize, AnchorDeserialize, Clone, Debug)]
+pub struct UpdateUserPdaArgs {
+    pub login: String,
+    pub root_key: Pubkey,
+    pub created_at_ms: u64,
+    pub updated_at_ms: u64,
+    pub version: u32,
+    pub prev_hash: Vec,
+    pub additional_limit: u64,
+    pub fields: UserMutableFields,
+    pub signature: Vec,
+}
+
+pub struct UserRecord {
+    pub created_at_ms: u64,
+    pub updated_at_ms: u64,
+    pub version: u32,
+    pub prev_hash: [u8; 32],
+    pub login: String,
+    pub root_key: Pubkey,
+    pub blockchain_key: Pubkey,
+    pub device_key: Pubkey,
+    pub chain_number: u16,
+    pub balance: u64,
+    pub is_server: bool,
+    pub server_key: Pubkey,
+    pub server_address: String,
+    pub connection_servers: Vec,
+    pub trusted_count: u8,
+    pub signature: [u8; 64],
+}
+
+#[derive(Accounts)]
+pub struct CreateUserPda<'info> {
+    /// CHECK: подписант транзакции, валидируется Anchor как signer и mut.
+    #[account(mut, signer)]
+    pub signer: AccountInfo<'info>,
+    /// CHECK: PDA пользователя, адрес проверяется вручную через seed в обработчике.
+    #[account(mut)]
+    pub user_pda: AccountInfo<'info>,
+    pub system_program: Program<'info, System>,
+    /// CHECK: адрес получателя комиссии проверяется вручную с константой settings.
+    #[account(mut)]
+    pub fee_receiver: AccountInfo<'info>,
+}
+
+#[derive(Accounts)]
+pub struct UpdateUserPda<'info> {
+    /// CHECK: подписант транзакции, валидируется Anchor как signer и mut.
+    #[account(mut, signer)]
+    pub signer: AccountInfo<'info>,
+    /// CHECK: PDA пользователя, адрес проверяется вручную через seed в обработчике.
+    #[account(mut)]
+    pub user_pda: AccountInfo<'info>,
+    pub system_program: Program<'info, System>,
+    /// CHECK: адрес получателя комиссии проверяется вручную с константой settings.
+    #[account(mut)]
+    pub fee_receiver: AccountInfo<'info>,
+}
+
+pub fn create_user_pda(ctx: Context, args: CreateUserPdaArgs) -> Result<()> {
+    validate_login(&args.login)?;
+    validate_fields(&args.fields)?;
+    validate_fee_receiver(&ctx.accounts.fee_receiver)?;
+    require!(
+        args.additional_limit % settings::LIMIT_STEP == 0,
+        ErrCode::InvalidLimitIncrement
+    );
+
+    let (expected_pda, bump) = find_user_pda(ctx.program_id, &args.login);
+    require_keys_eq!(
+        expected_pda,
+        ctx.accounts.user_pda.key(),
+        ErrCode::InvalidPdaAddress
+    );
+    require!(
+        ctx.accounts.user_pda.owner == &Pubkey::default(),
+        ErrCode::UserAlreadyExists
+    );
+
+    let start_balance = settings::START_BONUS_LIMIT
+        .checked_add(args.additional_limit)
+        .ok_or(error!(ErrCode::MathOverflow))?;
+
+    let mut record = UserRecord {
+        created_at_ms: args.created_at_ms,
+        updated_at_ms: args.created_at_ms,
+        version: 0,
+        prev_hash: ZERO_HASH,
+        login: args.login.clone(),
+        root_key: args.root_key,
+        blockchain_key: args.fields.blockchain_key,
+        device_key: args.fields.device_key,
+        chain_number: args.fields.chain_number,
+        balance: start_balance,
+        is_server: args.fields.is_server,
+        server_key: args.fields.server_key,
+        server_address: args.fields.server_address.clone(),
+        connection_servers: args.fields.connection_servers.clone(),
+        trusted_count: args.fields.trusted_count,
+        signature: [0; 64],
+    };
+
+    let unsigned = serialize_unsigned_record(&record)?;
+    verify_record_signature(&record.root_key, &args.signature, &unsigned)?;
+    record.signature = vec_to_signature(&args.signature)?;
+
+    let serialized = serialize_full_record(&record)?;
+    require!(
+        serialized.len() <= settings::USER_PDA_SPACE,
+        ErrCode::RecordTooLarge
+    );
+    let padded = pad_to_fixed_size(serialized, settings::USER_PDA_SPACE)?;
+
+    let pda_seeds: &[&[u8]] = &[
+        settings::USER_PDA_SEED_PREFIX.as_bytes(),
+        args.login.as_bytes(),
+        &[bump],
+    ];
+    create_pda(
+        &ctx.accounts.user_pda,
+        &ctx.accounts.signer,
+        &ctx.accounts.system_program.to_account_info(),
+        ctx.program_id,
+        pda_seeds,
+        settings::USER_PDA_SPACE as u64,
+    )?;
+    write_to_pda(&ctx.accounts.user_pda, &padded)?;
+
+    let total_fee = settings::REGISTRATION_FEE_LAMPORTS
+        .checked_add(limit_fee_lamports(args.additional_limit)?)
+        .ok_or(error!(ErrCode::MathOverflow))?;
+    transfer_lamports(
+        &ctx.accounts.signer,
+        &ctx.accounts.fee_receiver,
+        &ctx.accounts.system_program.to_account_info(),
+        total_fee,
+    )?;
+
+    Ok(())
+}
+
+pub fn update_user_pda(ctx: Context, args: UpdateUserPdaArgs) -> Result<()> {
+    validate_login(&args.login)?;
+    validate_fields(&args.fields)?;
+    validate_fee_receiver(&ctx.accounts.fee_receiver)?;
+    require!(
+        args.additional_limit % settings::LIMIT_STEP == 0,
+        ErrCode::InvalidLimitIncrement
+    );
+
+    let (expected_pda, _) = find_user_pda(ctx.program_id, &args.login);
+    require_keys_eq!(
+        expected_pda,
+        ctx.accounts.user_pda.key(),
+        ErrCode::InvalidPdaAddress
+    );
+    require!(
+        ctx.accounts.user_pda.owner == ctx.program_id,
+        ErrCode::InvalidPdaAddress
+    );
+
+    let raw = safe_read_pda(&ctx.accounts.user_pda);
+    require!(!raw.is_empty(), ErrCode::EmptyPdaData);
+    let old_record = deserialize_record_from_pda(&raw)?;
+
+    require!(
+        old_record.login == args.login,
+        ErrCode::ImmutableFieldChanged
+    );
+    require!(
+        old_record.created_at_ms == args.created_at_ms,
+        ErrCode::ImmutableFieldChanged
+    );
+    require_keys_eq!(old_record.root_key, args.root_key, ErrCode::ImmutableFieldChanged);
+    require!(
+        args.version == old_record.version.saturating_add(1),
+        ErrCode::InvalidVersion
+    );
+
+    let expected_prev_hash = hash_unsigned_record(&old_record)?;
+    let provided_prev_hash = vec_to_hash32(&args.prev_hash)?;
+    require!(
+        expected_prev_hash == provided_prev_hash,
+        ErrCode::InvalidPrevHash
+    );
+
+    let new_balance = old_record
+        .balance
+        .checked_add(args.additional_limit)
+        .ok_or(error!(ErrCode::MathOverflow))?;
+    require!(new_balance >= old_record.balance, ErrCode::BalanceDecrease);
+
+    let mut new_record = UserRecord {
+        created_at_ms: old_record.created_at_ms,
+        updated_at_ms: args.updated_at_ms,
+        version: args.version,
+        prev_hash: provided_prev_hash,
+        login: old_record.login.clone(),
+        root_key: old_record.root_key,
+        blockchain_key: args.fields.blockchain_key,
+        device_key: args.fields.device_key,
+        chain_number: args.fields.chain_number,
+        balance: new_balance,
+        is_server: args.fields.is_server,
+        server_key: args.fields.server_key,
+        server_address: args.fields.server_address.clone(),
+        connection_servers: args.fields.connection_servers.clone(),
+        trusted_count: args.fields.trusted_count,
+        signature: [0; 64],
+    };
+
+    let unsigned = serialize_unsigned_record(&new_record)?;
+    verify_record_signature(&new_record.root_key, &args.signature, &unsigned)?;
+    new_record.signature = vec_to_signature(&args.signature)?;
+
+    let serialized = serialize_full_record(&new_record)?;
+    require!(
+        serialized.len() <= settings::USER_PDA_SPACE,
+        ErrCode::RecordTooLarge
+    );
+    let padded = pad_to_fixed_size(serialized, settings::USER_PDA_SPACE)?;
+    write_to_pda(&ctx.accounts.user_pda, &padded)?;
+
+    let topup_fee = limit_fee_lamports(args.additional_limit)?;
+    if topup_fee > 0 {
+        transfer_lamports(
+            &ctx.accounts.signer,
+            &ctx.accounts.fee_receiver,
+            &ctx.accounts.system_program.to_account_info(),
+            topup_fee,
+        )?;
+    }
+
+    Ok(())
+}
+
+fn serialize_unsigned_record(record: &UserRecord) -> Result> {
+    let login_bytes = record.login.as_bytes();
+    require!(login_bytes.len() <= u8::MAX as usize, ErrCode::InvalidLogin);
+
+    let server_address_bytes = record.server_address.as_bytes();
+    require!(
+        server_address_bytes.len() <= u8::MAX as usize,
+        ErrCode::InvalidRecordData
+    );
+    require!(
+        record.connection_servers.len() <= u8::MAX as usize,
+        ErrCode::InvalidRecordData
+    );
+
+    let mut out = Vec::new();
+    out.extend_from_slice(MAGIC);
+    out.push(FORMAT_MAJOR);
+    out.push(FORMAT_MINOR);
+    out.extend_from_slice(&0u16.to_le_bytes());
+
+    out.extend_from_slice(&record.created_at_ms.to_le_bytes());
+    out.extend_from_slice(&record.updated_at_ms.to_le_bytes());
+    out.extend_from_slice(&record.version.to_le_bytes());
+    out.extend_from_slice(&record.prev_hash);
+
+    out.push(login_bytes.len() as u8);
+    out.extend_from_slice(login_bytes);
+
+    out.extend_from_slice(record.root_key.as_ref());
+    out.extend_from_slice(record.blockchain_key.as_ref());
+    out.extend_from_slice(record.device_key.as_ref());
+
+    out.extend_from_slice(&record.chain_number.to_le_bytes());
+    out.extend_from_slice(&record.balance.to_le_bytes());
+
+    out.push(if record.is_server { 1 } else { 0 });
+    if record.is_server {
+        out.extend_from_slice(record.server_key.as_ref());
+        out.push(server_address_bytes.len() as u8);
+        out.extend_from_slice(server_address_bytes);
+    }
+
+    out.push(record.connection_servers.len() as u8);
+    for login in &record.connection_servers {
+        let bytes = login.as_bytes();
+        require!(bytes.len() <= u8::MAX as usize, ErrCode::InvalidRecordData);
+        out.push(bytes.len() as u8);
+        out.extend_from_slice(bytes);
+    }
+
+    out.push(record.trusted_count);
+    out.extend_from_slice(&RESERVED_BYTES);
+
+    let record_len = out
+        .len()
+        .checked_add(64)
+        .ok_or(error!(ErrCode::MathOverflow))?;
+    require!(record_len <= u16::MAX as usize, ErrCode::RecordTooLarge);
+    let len_bytes = (record_len as u16).to_le_bytes();
+    out[7] = len_bytes[0];
+    out[8] = len_bytes[1];
+
+    Ok(out)
+}
+
+fn serialize_full_record(record: &UserRecord) -> Result> {
+    let mut out = serialize_unsigned_record(record)?;
+    out.extend_from_slice(&record.signature);
+    Ok(out)
+}
+
+fn deserialize_record_from_pda(raw: &[u8]) -> Result {
+    require!(raw.len() >= 9, ErrCode::InvalidRecordData);
+    require!(&raw[0..5] == MAGIC, ErrCode::InvalidRecordMagic);
+    require!(
+        raw[5] == FORMAT_MAJOR && raw[6] == FORMAT_MINOR,
+        ErrCode::InvalidRecordFormat
+    );
+
+    let record_len = u16::from_le_bytes([raw[7], raw[8]]) as usize;
+    require!(record_len >= 9 + 64, ErrCode::InvalidRecordLength);
+    require!(record_len <= raw.len(), ErrCode::InvalidRecordLength);
+
+    let useful = &raw[..record_len];
+    let mut cursor = 9usize;
+
+    let created_at_ms = read_u64(useful, &mut cursor)?;
+    let updated_at_ms = read_u64(useful, &mut cursor)?;
+    let version = read_u32(useful, &mut cursor)?;
+    let prev_hash = read_fixed_32(useful, &mut cursor)?;
+    let login = read_len_prefixed_string(useful, &mut cursor)?;
+
+    let root_key = Pubkey::new_from_array(read_fixed_32(useful, &mut cursor)?);
+    let blockchain_key = Pubkey::new_from_array(read_fixed_32(useful, &mut cursor)?);
+    let device_key = Pubkey::new_from_array(read_fixed_32(useful, &mut cursor)?);
+
+    let chain_number = read_u16(useful, &mut cursor)?;
+    let balance = read_u64(useful, &mut cursor)?;
+
+    let is_server = read_u8(useful, &mut cursor)? == 1;
+    let (server_key, server_address) = if is_server {
+        (
+            Pubkey::new_from_array(read_fixed_32(useful, &mut cursor)?),
+            read_len_prefixed_string(useful, &mut cursor)?,
+        )
+    } else {
+        (Pubkey::default(), String::new())
+    };
+
+    let connections_count = read_u8(useful, &mut cursor)? as usize;
+    let mut connection_servers = Vec::with_capacity(connections_count);
+    for _ in 0..connections_count {
+        connection_servers.push(read_len_prefixed_string(useful, &mut cursor)?);
+    }
+
+    let trusted_count = read_u8(useful, &mut cursor)?;
+    require!(
+        useful.get(cursor..cursor + 5) == Some(&RESERVED_BYTES),
+        ErrCode::InvalidRecordData
+    );
+    cursor += 5;
+
+    let signature = read_fixed_64(useful, &mut cursor)?;
+    require!(cursor == useful.len(), ErrCode::InvalidRecordLength);
+
+    Ok(UserRecord {
+        created_at_ms,
+        updated_at_ms,
+        version,
+        prev_hash,
+        login,
+        root_key,
+        blockchain_key,
+        device_key,
+        chain_number,
+        balance,
+        is_server,
+        server_key,
+        server_address,
+        connection_servers,
+        trusted_count,
+        signature,
+    })
+}
+
+fn hash_unsigned_record(record: &UserRecord) -> Result<[u8; 32]> {
+    let unsigned = serialize_unsigned_record(record)?;
+    let digest = Sha256::digest(unsigned);
+    let mut out = [0u8; 32];
+    out.copy_from_slice(&digest);
+    Ok(out)
+}
+
+fn verify_record_signature(root_key: &Pubkey, signature: &[u8], unsigned: &[u8]) -> Result<()> {
+    let sig_arr = vec_to_signature(signature)?;
+    let hash = Sha256::digest(unsigned);
+    let verify_key =
+        PublicKey::from_bytes(root_key.as_ref()).map_err(|_| error!(ErrCode::InvalidSignature))?;
+    let sig = Signature::from_bytes(&sig_arr).map_err(|_| error!(ErrCode::InvalidSignature))?;
+    verify_key
+        .verify(hash.as_slice(), &sig)
+        .map_err(|_| error!(ErrCode::InvalidSignature))?;
+    Ok(())
+}
+
+fn validate_login(login: &str) -> Result<()> {
+    require!(!login.is_empty(), ErrCode::InvalidLogin);
+    require!(login.len() <= 30, ErrCode::InvalidLogin);
+    for ch in login.chars() {
+        if !(ch.is_ascii_lowercase() || ch.is_ascii_digit() || ch == '_') {
+            return Err(error!(ErrCode::InvalidLogin));
+        }
+    }
+    Ok(())
+}
+
+fn validate_fields(fields: &UserMutableFields) -> Result<()> {
+    if fields.is_server {
+        require!(!fields.server_address.is_empty(), ErrCode::InvalidRecordData);
+        require!(
+            fields.server_address.as_bytes().len() <= u8::MAX as usize,
+            ErrCode::InvalidRecordData
+        );
+    } else {
+        require!(fields.server_address.is_empty(), ErrCode::InvalidRecordData);
+    }
+    require!(
+        fields.connection_servers.len() <= u8::MAX as usize,
+        ErrCode::InvalidRecordData
+    );
+    for login in &fields.connection_servers {
+        require!(!login.is_empty(), ErrCode::InvalidRecordData);
+        require!(login.as_bytes().len() <= u8::MAX as usize, ErrCode::InvalidRecordData);
+    }
+    Ok(())
+}
+
+fn validate_fee_receiver(fee_receiver: &AccountInfo) -> Result<()> {
+    let expected = Pubkey::from_str(settings::REGISTRATION_FEE_RECEIVER)
+        .map_err(|_| error!(ErrCode::InvalidFeeReceiver))?;
+    require_keys_eq!(expected, *fee_receiver.key, ErrCode::InvalidFeeReceiver);
+    Ok(())
+}
+
+fn transfer_lamports<'info>(
+    payer: &AccountInfo<'info>,
+    recipient: &AccountInfo<'info>,
+    system_program: &AccountInfo<'info>,
+    lamports: u64,
+) -> Result<()> {
+    if lamports == 0 {
+        return Ok(());
+    }
+    let ix = system_instruction::transfer(payer.key, recipient.key, lamports);
+    invoke(&ix, &[payer.clone(), recipient.clone(), system_program.clone()])?;
+    Ok(())
+}
+
+fn limit_fee_lamports(limit_delta: u64) -> Result {
+    let units = limit_delta / settings::LIMIT_STEP;
+    units
+        .checked_mul(settings::LAMPORTS_PER_LIMIT_STEP)
+        .ok_or(error!(ErrCode::MathOverflow))
+}
+
+fn find_user_pda(program_id: &Pubkey, login: &str) -> (Pubkey, u8) {
+    Pubkey::find_program_address(
+        &[settings::USER_PDA_SEED_PREFIX.as_bytes(), login.as_bytes()],
+        program_id,
+    )
+}
+
+fn pad_to_fixed_size(mut bytes: Vec, target_size: usize) -> Result> {
+    require!(bytes.len() <= target_size, ErrCode::RecordTooLarge);
+    bytes.resize(target_size, 0);
+    Ok(bytes)
+}
+
+fn vec_to_signature(input: &[u8]) -> Result<[u8; 64]> {
+    require!(input.len() == 64, ErrCode::InvalidSignature);
+    let mut out = [0u8; 64];
+    out.copy_from_slice(input);
+    Ok(out)
+}
+
+fn vec_to_hash32(input: &[u8]) -> Result<[u8; 32]> {
+    require!(input.len() == 32, ErrCode::InvalidPrevHash);
+    let mut out = [0u8; 32];
+    out.copy_from_slice(input);
+    Ok(out)
+}
+
+fn read_u8(data: &[u8], cursor: &mut usize) -> Result {
+    let v = *data.get(*cursor).ok_or(error!(ErrCode::InvalidRecordData))?;
+    *cursor += 1;
+    Ok(v)
+}
+
+fn read_u16(data: &[u8], cursor: &mut usize) -> Result {
+    let end = cursor
+        .checked_add(2)
+        .ok_or(error!(ErrCode::InvalidRecordData))?;
+    let slice = data.get(*cursor..end).ok_or(error!(ErrCode::InvalidRecordData))?;
+    *cursor = end;
+    Ok(u16::from_le_bytes([slice[0], slice[1]]))
+}
+
+fn read_u32(data: &[u8], cursor: &mut usize) -> Result {
+    let end = cursor
+        .checked_add(4)
+        .ok_or(error!(ErrCode::InvalidRecordData))?;
+    let slice = data.get(*cursor..end).ok_or(error!(ErrCode::InvalidRecordData))?;
+    *cursor = end;
+    Ok(u32::from_le_bytes([slice[0], slice[1], slice[2], slice[3]]))
+}
+
+fn read_u64(data: &[u8], cursor: &mut usize) -> Result {
+    let end = cursor
+        .checked_add(8)
+        .ok_or(error!(ErrCode::InvalidRecordData))?;
+    let slice = data.get(*cursor..end).ok_or(error!(ErrCode::InvalidRecordData))?;
+    *cursor = end;
+    Ok(u64::from_le_bytes([
+        slice[0], slice[1], slice[2], slice[3], slice[4], slice[5], slice[6], slice[7],
+    ]))
+}
+
+fn read_fixed_32(data: &[u8], cursor: &mut usize) -> Result<[u8; 32]> {
+    let end = cursor
+        .checked_add(32)
+        .ok_or(error!(ErrCode::InvalidRecordData))?;
+    let slice = data.get(*cursor..end).ok_or(error!(ErrCode::InvalidRecordData))?;
+    *cursor = end;
+    let mut out = [0u8; 32];
+    out.copy_from_slice(slice);
+    Ok(out)
+}
+
+fn read_fixed_64(data: &[u8], cursor: &mut usize) -> Result<[u8; 64]> {
+    let end = cursor
+        .checked_add(64)
+        .ok_or(error!(ErrCode::InvalidRecordData))?;
+    let slice = data.get(*cursor..end).ok_or(error!(ErrCode::InvalidRecordData))?;
+    *cursor = end;
+    let mut out = [0u8; 64];
+    out.copy_from_slice(slice);
+    Ok(out)
+}
+
+fn read_len_prefixed_string(data: &[u8], cursor: &mut usize) -> Result {
+    let len = read_u8(data, cursor)? as usize;
+    let end = cursor
+        .checked_add(len)
+        .ok_or(error!(ErrCode::InvalidRecordData))?;
+    let slice = data.get(*cursor..end).ok_or(error!(ErrCode::InvalidRecordData))?;
+    *cursor = end;
+    let value = std::str::from_utf8(slice).map_err(|_| error!(ErrCode::InvalidRecordData))?;
+    Ok(value.to_string())
+}
diff --git a/shine/scripts/devnet/README.md b/shine/scripts/devnet/README.md
new file mode 100644
index 0000000..96d6e02
--- /dev/null
+++ b/shine/scripts/devnet/README.md
@@ -0,0 +1,76 @@
+Devnet E2E тест: NFT-модуль + add_bonus
+
+Ветка содержит скрипты для проверки (NFT + add_bonus) в devnet.
+
+Скрипты:
+
+quick_devnet_e2e.js — создаёт 1 NFT и вызывает add_bonus.
+
+quick_devnet_e2e_multi.js — создаёт N NFT в коллекции.
+
+
+
+Подготовка окружения
+
+Установите зависимости:
+
+npm i @coral-xyz/anchor @solana/web3.js @solana/spl-token
+
+
+Создайте файл .env с переменными:
+
+
+ANCHOR_PROVIDER_URL=https://api.devnet.solana.com
+ANCHOR_WALLET=/Users//.config/solana/id.json
+PROGRAM_ID=qpgnAKhsXgPPaqQWfXhpme7UnG8GyStssuoSjF6Fzy3
+COLLECTION_MINT=
+
+ANCHOR_PROVIDER_URL=https://api.devnet.solana.com
+ANCHOR_WALLET=/Users//.config/solana/id.json
+PROGRAM_ID=<адрес shine_payments в devnet>
+COLLECTION_MINT=
+
+
+Пополните кошелёк тестовыми SOL:
+
+solana config set --url https://api.devnet.solana.com
+solana airdrop 2
+
+Запуск тестов:
+
+одиночный NFT:
+
+node quick_devnet_e2e.js
+
+
+несколько NFT:
+
+node quick_devnet_e2e_multi.js 3
+
+
+
+Проверка результата
+
+В выводе будут строки:
+
+add_bonus() tx: 
+
+NFT mint: 
+ + + +Откройте транзакцию или mint в Solana Explorer + + +Убедитесь: + +Verified Collection совпадает с вашим COLLECTION_MINT + +У каждого NFT Supply = 1 + +Аккаунт получателя (ATA) помечен как frozen + + + + +⚠️ Повторный запуск может вернуть ошибку PdaAlreadyExists это нормально, так как PDA уже инициализирован. diff --git a/shine/scripts/devnet/quick_devnet_e2e.js b/shine/scripts/devnet/quick_devnet_e2e.js new file mode 100644 index 0000000..d981c13 --- /dev/null +++ b/shine/scripts/devnet/quick_devnet_e2e.js @@ -0,0 +1,285 @@ +const anchor = require("@coral-xyz/anchor"); +const { + Connection, + PublicKey, + SystemProgram, + Transaction, + TransactionInstruction, +} = require("@solana/web3.js"); +const { + getAssociatedTokenAddress, + createAssociatedTokenAccountInstruction, + TOKEN_PROGRAM_ID, + ASSOCIATED_TOKEN_PROGRAM_ID, + createMint, + getAccount, +} = require("@solana/spl-token"); +const crypto = require("crypto"); + +// Адрес программы метаданных Metaplex (фиксированный) +const METAPLEX_TOKEN_METADATA_PROGRAM_ID = new PublicKey( + "metaqbxxUerdq28cj1RbAWkYQm3ybzjb6a8bt518x1s" +); + +// ──────────────────────────────── +// Утилиты +// ──────────────────────────────── +const BASE58_RE = /[1-9A-HJ-NP-Za-km-z]{32,}/g; + +function mustEnv(name) { + const v = (process.env[name] || "").trim(); + if (!v) throw new Error(`Переменная окружения ${name} не задана`); + return v; +} + +function pickBase58(raw, name) { + const m = (raw || "").toString().match(BASE58_RE); + if (!m) throw new Error(`${name} не найден/невалиден: "${raw}"`); + return m[0]; +} + +// Anchor discriminator: sha256("global:") первые 8 байт +function disc8(ixName) { + const preimage = `global:${ixName}`; + const h = crypto.createHash("sha256").update(preimage).digest(); + return h.subarray(0, 8); +} + +function u64le(n) { + const bn = BigInt(n.toString()); + const buf = Buffer.alloc(8); + buf.writeBigUInt64LE(bn); + return buf; +} + +// Надёжная отправка транзакций с ретраями при «Blockhash not found» +async function sendTx(provider, tx, signers = []) { + const conn = provider.connection; + let lastErr; + + for (let attempt = 0; attempt < 3; attempt++) { + try { + const { blockhash, lastValidBlockHeight } = await conn.getLatestBlockhash( + "confirmed" + ); + tx.recentBlockhash = blockhash; + tx.feePayer = provider.wallet.publicKey; + + for (const s of signers) tx.partialSign(s); + + const signed = await provider.wallet.signTransaction(tx); + + const sig = await conn.sendRawTransaction(signed.serialize(), { + skipPreflight: false, + preflightCommitment: "confirmed", + maxRetries: 3, + }); + + await conn.confirmTransaction( + { signature: sig, blockhash, lastValidBlockHeight }, + "confirmed" + ); + + return sig; + } catch (e) { + lastErr = e; + const msg = String(e?.message || e).toLowerCase(); + if (msg.includes("blockhash not found") || msg.includes("expired")) { + // пробуем ещё раз со свежим blockhash + continue; + } + throw e; + } + } + + throw lastErr; +} + +// ──────────────────────────────── +// Основной сценарий +// ──────────────────────────────── +(async () => { + // Провайдер / окружение + const RPC = mustEnv("ANCHOR_PROVIDER_URL"); + const WALLET_PATH = mustEnv("ANCHOR_WALLET"); + const PROGRAM_ID = new PublicKey( + pickBase58(mustEnv("PROGRAM_ID"), "PROGRAM_ID") + ); + const COLLECTION_MINT = new PublicKey( + pickBase58(mustEnv("COLLECTION_MINT"), "COLLECTION_MINT") + ); + + const provider = anchor.AnchorProvider.env(); // читает из ENV + anchor.setProvider(provider); + const conn = provider.connection; + const wallet = provider.wallet; + + console.log("────────────────────────────────────────────────────────"); + console.log("RPC :", RPC); + console.log("Wallet :", wallet.publicKey.toBase58()); + console.log("Program ID :", PROGRAM_ID.toBase58()); + console.log("Collection mint :", COLLECTION_MINT.toBase58()); + console.log( + "TokenMetadata PID :", + METAPLEX_TOKEN_METADATA_PROGRAM_ID.toBase58() + ); + console.log("ATA Program PID :", ASSOCIATED_TOKEN_PROGRAM_ID.toBase58()); + console.log("────────────────────────────────────────────────────────"); + + // 1) INIT (создаёт PDA состояния) + const [statePda] = PublicKey.findProgramAddressSync( + [Buffer.from("shine_investments_state")], + PROGRAM_ID + ); + + const initIx = new TransactionInstruction({ + programId: PROGRAM_ID, + keys: [ + { pubkey: wallet.publicKey, isSigner: true, isWritable: true }, // payer + { pubkey: statePda, isSigner: false, isWritable: true }, // state_pda + { pubkey: SystemProgram.programId, isSigner: false, isWritable: false }, + ], + data: Buffer.from([...disc8("init")]), // без аргументов + }); + + try { + const sigInit = await sendTx(provider, new Transaction().add(initIx)); + console.log( + "init() tx:", + sigInit, + `https://explorer.solana.com/tx/${sigInit}?cluster=devnet` + ); + } catch (e) { + console.log("init(): возможно уже выполнен ->", e.message); + } + + // 2) Локально создаём mint нового NFT и ATA получателя + const mintPubkey = await createMint( + conn, + wallet.payer, + wallet.publicKey, + wallet.publicKey, + 0 + ); + console.log("NFT mint:", mintPubkey.toBase58()); + + const recipientOwner = wallet.publicKey; + const recipientAta = await getAssociatedTokenAddress( + mintPubkey, + recipientOwner + ); + const ataInfo = await conn.getAccountInfo(recipientAta); + if (!ataInfo) { + const createAtaIx = createAssociatedTokenAccountInstruction( + wallet.publicKey, + recipientAta, + recipientOwner, + mintPubkey + ); + const sigAta = await sendTx( + provider, + new Transaction().add(createAtaIx) + ); + console.log("Created ATA:", recipientAta.toBase58(), sigAta); + } else { + console.log("ATA exists:", recipientAta.toBase58()); + } + + // PDA для metadata/master edition нашего нового NFT + const [metadataPda] = PublicKey.findProgramAddressSync( + [ + Buffer.from("metadata"), + METAPLEX_TOKEN_METADATA_PROGRAM_ID.toBuffer(), + mintPubkey.toBuffer(), + ], + METAPLEX_TOKEN_METADATA_PROGRAM_ID + ); + const [masterEditionPda] = PublicKey.findProgramAddressSync( + [ + Buffer.from("metadata"), + METAPLEX_TOKEN_METADATA_PROGRAM_ID.toBuffer(), + mintPubkey.toBuffer(), + Buffer.from("edition"), + ], + METAPLEX_TOKEN_METADATA_PROGRAM_ID + ); + + // PDA коллекции (metadata/master edition) + const [collectionMetadataPda] = PublicKey.findProgramAddressSync( + [ + Buffer.from("metadata"), + METAPLEX_TOKEN_METADATA_PROGRAM_ID.toBuffer(), + COLLECTION_MINT.toBuffer(), + ], + METAPLEX_TOKEN_METADATA_PROGRAM_ID + ); + const [collectionMasterEditionPda] = PublicKey.findProgramAddressSync( + [ + Buffer.from("metadata"), + METAPLEX_TOKEN_METADATA_PROGRAM_ID.toBuffer(), + COLLECTION_MINT.toBuffer(), + Buffer.from("edition"), + ], + METAPLEX_TOKEN_METADATA_PROGRAM_ID + ); + + // 3) add_bonus(investor: Pubkey, amount: u64) — raw-инструкция + // Порядок аккаунтов должен совпасть с #[derive(Accounts)] AddBonusCtx: + // signer(Signer), state_pda(mut), mint_pda(mut), recipient_ata(mut), recipient_owner, + // collection_mint, collection_metadata_pda(mut), collection_master_edition_pda(mut), + // collection_update_authority(Signer), metadata_pda(mut), master_edition_pda(mut), + // token_metadata_program, token_program, associated_token_program, system_program + const investor = recipientOwner; + const amount = 123_000_000n; // u64 + + const addBonusData = Buffer.concat([ + disc8("add_bonus"), + investor.toBuffer(), + u64le(amount), + ]); + + const addBonusIx = new TransactionInstruction({ + programId: PROGRAM_ID, + keys: [ + { pubkey: wallet.publicKey, isSigner: true, isWritable: false }, // signer + { pubkey: statePda, isSigner: false, isWritable: true }, // state_pda + { pubkey: mintPubkey, isSigner: false, isWritable: true }, // mint_pda + { pubkey: recipientAta, isSigner: false, isWritable: true }, // recipient_ata + { pubkey: recipientOwner, isSigner: false, isWritable: false }, // recipient_owner + { pubkey: COLLECTION_MINT, isSigner: false, isWritable: false }, // collection_mint + { pubkey: collectionMetadataPda, isSigner: false, isWritable: true }, // collection_metadata_pda + { pubkey: collectionMasterEditionPda, isSigner: false, isWritable: true }, // collection_master_edition_pda + { pubkey: wallet.publicKey, isSigner: true, isWritable: false }, // collection_update_authority + { pubkey: metadataPda, isSigner: false, isWritable: true }, // metadata_pda + { pubkey: masterEditionPda, isSigner: false, isWritable: true }, // master_edition_pda + { pubkey: METAPLEX_TOKEN_METADATA_PROGRAM_ID, isSigner: false, isWritable: false }, + { pubkey: TOKEN_PROGRAM_ID, isSigner: false, isWritable: false }, + { pubkey: ASSOCIATED_TOKEN_PROGRAM_ID, isSigner: false, isWritable: false }, + { pubkey: SystemProgram.programId, isSigner: false, isWritable: false }, + ], + data: addBonusData, + }); + + const sig = await sendTx(provider, new Transaction().add(addBonusIx)); + console.log( + "add_bonus() tx:", + sig, + `https://explorer.solana.com/tx/${sig}?cluster=devnet` + ); + + // 4) простые проверки + const acc = await getAccount(conn, recipientAta); + console.log("isFrozen (ATA):", acc.isFrozen); + if (!acc.isFrozen) throw new Error("Ожидали заморозку ATA после add_bonus()"); + + const mdInfo = await conn.getAccountInfo(metadataPda); + if (!mdInfo || mdInfo.data.length === 0) + throw new Error("Metadata PDA отсутствует или пуст"); + + console.log( + "Готово: raw-инструкции прошли, NFT создан/верифицирован и ATA заморожен" + ); +})().catch((e) => { + console.error("Ошибка e2e:", e); + process.exit(1); +}); diff --git a/shine/scripts/devnet/quick_devnet_e2e_multi.js b/shine/scripts/devnet/quick_devnet_e2e_multi.js new file mode 100644 index 0000000..37bf6e9 --- /dev/null +++ b/shine/scripts/devnet/quick_devnet_e2e_multi.js @@ -0,0 +1,205 @@ +const anchor = require("@coral-xyz/anchor"); +const { + Connection, + PublicKey, + SystemProgram, + Transaction, + TransactionInstruction, +} = require("@solana/web3.js"); +const { + getAssociatedTokenAddress, + createAssociatedTokenAccountInstruction, + TOKEN_PROGRAM_ID, + ASSOCIATED_TOKEN_PROGRAM_ID, + createMint, + getAccount, +} = require("@solana/spl-token"); +const crypto = require("crypto"); + +// Programs +const MPL = new PublicKey("metaqbxxUerdq28cj1RbAWkYQm3ybzjb6a8bt518x1s"); + +// utils +const BASE58_RE = /[1-9A-HJ-NP-Za-km-z]{32,}/g; +function mustEnv(name) { + const v = (process.env[name] || "").trim(); + if (!v) throw new Error(`ENV ${name} is required`); + return v; +} +function pickBase58(raw, name) { + const m = (raw || "").toString().match(BASE58_RE); + if (!m) throw new Error(`${name} invalid: "${raw}"`); + return m[0]; +} +function disc8(name) { + const preimage = `global:${name}`; + const h = crypto.createHash("sha256").update(preimage).digest(); + return h.subarray(0, 8); +} +function u64le(n) { + const bn = BigInt(n.toString()); + const buf = Buffer.alloc(8); + buf.writeBigUInt64LE(bn); + return buf; +} +async function sendTx(provider, tx, signers = []) { + const conn = provider.connection; + const { blockhash, lastValidBlockHeight } = await conn.getLatestBlockhash("confirmed"); + tx.recentBlockhash = blockhash; + tx.feePayer = provider.wallet.publicKey; + for (const s of signers) tx.partialSign(s); + const raw = await provider.wallet.signTransaction(tx); + const sig = await conn.sendRawTransaction(raw.serialize(), { + skipPreflight: false, + preflightCommitment: "confirmed", + maxRetries: 3, + }); + await conn.confirmTransaction({ signature: sig, blockhash, lastValidBlockHeight }, "confirmed"); + return sig; +} + +(async () => { + const count = Number(process.argv[2] || "3"); // сколько NFT сделать + const RPC = mustEnv("ANCHOR_PROVIDER_URL"); + const PROGRAM_ID = new PublicKey(pickBase58(mustEnv("PROGRAM_ID"), "PROGRAM_ID")); + const COLLECTION_MINT = new PublicKey(pickBase58(mustEnv("COLLECTION_MINT"), "COLLECTION_MINT")); + + const provider = anchor.AnchorProvider.env(); + anchor.setProvider(provider); + const conn = provider.connection; + const wallet = provider.wallet; + + console.log("────────────────────────────────────────────────────────"); + console.log("RPC :", RPC); + console.log("Wallet :", wallet.publicKey.toBase58()); + console.log("Program ID :", PROGRAM_ID.toBase58()); + console.log("Collection mint :", COLLECTION_MINT.toBase58()); + console.log("TokenMetadata PID :", MPL.toBase58()); + console.log("ATA Program PID :", ASSOCIATED_TOKEN_PROGRAM_ID.toBase58()); + console.log("Count :", count); + console.log("────────────────────────────────────────────────────────"); + + // ensure init (если уже есть — просто пропустим) + const [statePda] = PublicKey.findProgramAddressSync( + [Buffer.from("shine_investments_state")], + PROGRAM_ID + ); + const initIx = new TransactionInstruction({ + programId: PROGRAM_ID, + keys: [ + { pubkey: wallet.publicKey, isSigner: true, isWritable: true }, // payer + { pubkey: statePda, isSigner: false, isWritable: true }, // state_pda + { pubkey: SystemProgram.programId, isSigner: false, isWritable: false }, + ], + data: Buffer.from([...disc8("init")]), + }); + try { + const sigInit = await sendTx(provider, new Transaction().add(initIx)); + console.log("init() tx:", sigInit, `https://explorer.solana.com/tx/${sigInit}?cluster=devnet`); + } catch (e) { + console.log("init(): возможно уже выполнен ->", e.message); + } + + const minted = []; + + for (let i = 0; i < count; i++) { + // 1) создаём новый mint (NFT) + const mintPubkey = await createMint(conn, wallet.payer, wallet.publicKey, wallet.publicKey, 0); + console.log(`\n[${i + 1}/${count}] NFT mint:`, mintPubkey.toBase58()); + + // 2) создаём ATA при необходимости + const recipientOwner = wallet.publicKey; + const recipientAta = await getAssociatedTokenAddress(mintPubkey, recipientOwner); + const ataInfo = await conn.getAccountInfo(recipientAta); + if (!ataInfo) { + const createAtaIx = createAssociatedTokenAccountInstruction( + wallet.publicKey, recipientAta, recipientOwner, mintPubkey + ); + const sigAta = await sendTx(provider, new Transaction().add(createAtaIx)); + console.log(" ATA created:", recipientAta.toBase58(), sigAta); + } else { + console.log(" ATA exists:", recipientAta.toBase58()); + } + + // PDA для нашего NFT (metadata/master edition) + const [metadataPda] = PublicKey.findProgramAddressSync( + [Buffer.from("metadata"), MPL.toBuffer(), mintPubkey.toBuffer()], + MPL + ); + const [masterEditionPda] = PublicKey.findProgramAddressSync( + [Buffer.from("metadata"), MPL.toBuffer(), mintPubkey.toBuffer(), Buffer.from("edition")], + MPL + ); + + // PDA коллекции + const [collectionMetadataPda] = PublicKey.findProgramAddressSync( + [Buffer.from("metadata"), MPL.toBuffer(), COLLECTION_MINT.toBuffer()], + MPL + ); + const [collectionMasterEditionPda] = PublicKey.findProgramAddressSync( + [Buffer.from("metadata"), MPL.toBuffer(), COLLECTION_MINT.toBuffer(), Buffer.from("edition")], + MPL + ); + + // 3) add_bonus(investor, amount) + const investor = recipientOwner; + // Для наглядности — разные суммы + const amount = BigInt(100_000_000 + i * 10_000_000); // 100M, 110M, 120M... + + const addBonusData = Buffer.concat([ + disc8("add_bonus"), + investor.toBuffer(), + u64le(amount), + ]); + + const addBonusIx = new TransactionInstruction({ + programId: PROGRAM_ID, + keys: [ + { pubkey: wallet.publicKey, isSigner: true, isWritable: false }, // signer + { pubkey: statePda, isSigner: false, isWritable: true }, // state_pda + { pubkey: mintPubkey, isSigner: false, isWritable: true }, // mint_pda + { pubkey: recipientAta, isSigner: false, isWritable: true }, // recipient_ata + { pubkey: recipientOwner, isSigner: false, isWritable: false }, // recipient_owner + { pubkey: COLLECTION_MINT, isSigner: false, isWritable: false }, // collection_mint + { pubkey: collectionMetadataPda, isSigner: false, isWritable: true }, + { pubkey: collectionMasterEditionPda, isSigner: false, isWritable: true }, + { pubkey: wallet.publicKey, isSigner: true, isWritable: false }, // collection_update_authority + { pubkey: metadataPda, isSigner: false, isWritable: true }, + { pubkey: masterEditionPda, isSigner: false, isWritable: true }, + { pubkey: MPL, isSigner: false, isWritable: false }, + { pubkey: TOKEN_PROGRAM_ID, isSigner: false, isWritable: false }, + { pubkey: ASSOCIATED_TOKEN_PROGRAM_ID,isSigner: false, isWritable: false }, + { pubkey: SystemProgram.programId, isSigner: false, isWritable: false }, + ], + data: addBonusData, + }); + + const sig = await sendTx(provider, new Transaction().add(addBonusIx)); + console.log(" add_bonus() tx:", sig, `https://explorer.solana.com/tx/${sig}?cluster=devnet`); + + // 4) проверки + const acc = await getAccount(conn, recipientAta); + console.log(" ATA frozen:", acc.isFrozen); + + minted.push({ + mint: mintPubkey.toBase58(), + ata: recipientAta.toBase58(), + addBonusSig: sig, + metadataPda: metadataPda.toBase58(), + }); + } + + console.log("\n==================== SUMMARY ===================="); + console.log("Wallet:", wallet.publicKey.toBase58()); + console.log("Collection:", COLLECTION_MINT.toBase58()); + console.table(minted); + console.log("Открой каждую сигнатуру (add_bonus tx) и mint в Explorer:"); + minted.forEach((m, i) => { + console.log(`[${i + 1}] Mint: https://explorer.solana.com/address/${m.mint}?cluster=devnet`); + console.log(` TX : https://explorer.solana.com/tx/${m.addBonusSig}?cluster=devnet`); + }); + console.log("================================================="); +})().catch((e) => { + console.error("Ошибка:", e); + process.exit(1); +}); diff --git a/shine/tests/shine.ts b/shine/tests/shine.ts new file mode 100644 index 0000000..69b91f3 --- /dev/null +++ b/shine/tests/shine.ts @@ -0,0 +1,16 @@ +import * as anchor from "@coral-xyz/anchor"; +import { Program } from "@coral-xyz/anchor"; +import { Shine } from "../target/types/shine"; + +describe("shine", () => { + // Configure the client to use the local cluster. + anchor.setProvider(anchor.AnchorProvider.env()); + + const program = anchor.workspace.shine as Program; + + it("Is initialized!", async () => { + // Add your test here. + const tx = await program.methods.initialize().rpc(); + console.log("Your transaction signature", tx); + }); +}); diff --git a/shine/tsconfig.json b/shine/tsconfig.json new file mode 100644 index 0000000..cd5d2e3 --- /dev/null +++ b/shine/tsconfig.json @@ -0,0 +1,10 @@ +{ + "compilerOptions": { + "types": ["mocha", "chai"], + "typeRoots": ["./node_modules/@types"], + "lib": ["es2015"], + "module": "commonjs", + "target": "es6", + "esModuleInterop": true + } +} diff --git a/shine/validator.log b/shine/validator.log new file mode 100644 index 0000000..9d2fb12 --- /dev/null +++ b/shine/validator.log @@ -0,0 +1,3 @@ +Ledger location: test-ledger +Log: test-ledger/validator.log +Initializing... diff --git a/shine/yarn.lock b/shine/yarn.lock new file mode 100644 index 0000000..0b6fad0 --- /dev/null +++ b/shine/yarn.lock @@ -0,0 +1,1144 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"@babel/runtime@^7.25.0": + version "7.27.1" + resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.27.1.tgz#9fce313d12c9a77507f264de74626e87fd0dc541" + integrity sha512-1x3D2xEk2fRo3PAhwQwu5UubzgiVWSXTBfWpVd2Mx2AzRqJuDJCsgaDVZ7HB5iGzDW1Hl1sWN2mFyKjmR9uAog== + +"@coral-xyz/anchor-errors@^0.31.1": + version "0.31.1" + resolved "https://registry.yarnpkg.com/@coral-xyz/anchor-errors/-/anchor-errors-0.31.1.tgz#d635cbac2533973ae6bfb5d3ba1de89ce5aece2d" + integrity sha512-NhNEku4F3zzUSBtrYz84FzYWm48+9OvmT1Hhnwr6GnPQry2dsEqH/ti/7ASjjpoFTWRnPXrjAIT1qM6Isop+LQ== + +"@coral-xyz/anchor@^0.31.1": + version "0.31.1" + resolved "https://registry.yarnpkg.com/@coral-xyz/anchor/-/anchor-0.31.1.tgz#0fdeebf45a3cb2e47e8ebbb815ca98542152962c" + integrity sha512-QUqpoEK+gi2S6nlYc2atgT2r41TT3caWr/cPUEL8n8Md9437trZ68STknq897b82p5mW0XrTBNOzRbmIRJtfsA== + dependencies: + "@coral-xyz/anchor-errors" "^0.31.1" + "@coral-xyz/borsh" "^0.31.1" + "@noble/hashes" "^1.3.1" + "@solana/web3.js" "^1.69.0" + bn.js "^5.1.2" + bs58 "^4.0.1" + buffer-layout "^1.2.2" + camelcase "^6.3.0" + cross-fetch "^3.1.5" + eventemitter3 "^4.0.7" + pako "^2.0.3" + superstruct "^0.15.4" + toml "^3.0.0" + +"@coral-xyz/borsh@^0.31.1": + version "0.31.1" + resolved "https://registry.yarnpkg.com/@coral-xyz/borsh/-/borsh-0.31.1.tgz#5328e1e0921b75d7f4a62dd3f61885a938bc7241" + integrity sha512-9N8AU9F0ubriKfNE3g1WF0/4dtlGXoBN/hd1PvbNBamBNwRgHxH4P+o3Zt7rSEloW1HUs6LfZEchlx9fW7POYw== + dependencies: + bn.js "^5.1.2" + buffer-layout "^1.2.0" + +"@noble/curves@^1.4.2": + version "1.9.1" + resolved "https://registry.yarnpkg.com/@noble/curves/-/curves-1.9.1.tgz#9654a0bc6c13420ae252ddcf975eaf0f58f0a35c" + integrity sha512-k11yZxZg+t+gWvBbIswW0yoJlu8cHOC7dhunwOzoWH/mXGBiYyR4YY6hAEK/3EUs4UpB8la1RfdRpeGsFHkWsA== + dependencies: + "@noble/hashes" "1.8.0" + +"@noble/hashes@1.8.0", "@noble/hashes@^1.3.1", "@noble/hashes@^1.4.0": + version "1.8.0" + resolved "https://registry.yarnpkg.com/@noble/hashes/-/hashes-1.8.0.tgz#cee43d801fcef9644b11b8194857695acd5f815a" + integrity sha512-jCs9ldd7NwzpgXDIf6P3+NrHh9/sD6CQdxHyjQI+h/6rDNo88ypBxxz45UDuZHz9r3tNz7N/VInSVoVdtXEI4A== + +"@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.1.1": + version "2.1.1" + resolved "https://registry.yarnpkg.com/@solana/codecs-core/-/codecs-core-2.1.1.tgz#5d09d7f35b0266789d7c1f9306c08051128a6a64" + integrity sha512-iPQW3UZ2Vi7QFBo2r9tw0NubtH8EdrhhmZulx6lC8V5a+qjaxovtM/q/UW2BTNpqqHLfO0tIcLyBLrNH4HTWPg== + dependencies: + "@solana/errors" "2.1.1" + +"@solana/codecs-numbers@^2.1.0": + version "2.1.1" + resolved "https://registry.yarnpkg.com/@solana/codecs-numbers/-/codecs-numbers-2.1.1.tgz#b7a69024e2397e236bbfb11b75ff4a077236b9d2" + integrity sha512-m20IUPJhPUmPkHSlZ2iMAjJ7PaYUvlMtFhCQYzm9BEBSI6OCvXTG3GAPpAnSGRBfg5y+QNqqmKn4QHU3B6zzCQ== + dependencies: + "@solana/codecs-core" "2.1.1" + "@solana/errors" "2.1.1" + +"@solana/errors@2.1.1": + version "2.1.1" + resolved "https://registry.yarnpkg.com/@solana/errors/-/errors-2.1.1.tgz#009ebf387b0c014a8fc60a59d65757fef942e4fd" + integrity sha512-sj6DaWNbSJFvLzT8UZoabMefQUfSW/8tXK7NTiagsDmh+Q87eyQDDC9L3z+mNmx9b6dEf6z660MOIplDD2nfEw== + dependencies: + chalk "^5.4.1" + commander "^13.1.0" + +"@solana/web3.js@^1.69.0": + version "1.98.2" + resolved "https://registry.yarnpkg.com/@solana/web3.js/-/web3.js-1.98.2.tgz#45167a5cfb64436944bf4dc1e8be8482bd6d4c14" + integrity sha512-BqVwEG+TaG2yCkBMbD3C4hdpustR4FpuUFRPUmqRZYYlPI9Hg4XMWxHWOWRzHE9Lkc9NDjzXFX7lDXSgzC7R1A== + dependencies: + "@babel/runtime" "^7.25.0" + "@noble/curves" "^1.4.2" + "@noble/hashes" "^1.4.0" + "@solana/buffer-layout" "^4.0.1" + "@solana/codecs-numbers" "^2.1.0" + agentkeepalive "^4.5.0" + 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.17" + resolved "https://registry.yarnpkg.com/@swc/helpers/-/helpers-0.5.17.tgz#5a7be95ac0f0bf186e7e6e890e7a6f6cda6ce971" + integrity sha512-5IKx/Y13RsYd+sauPb2x+U/xZikHjolzfuDgTAl/Tdf3Q8rslRvC19NKDLgAJQ6wsqADk10ntlv08nPFw/gO/A== + 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.15.19" + resolved "https://registry.yarnpkg.com/@types/node/-/node-22.15.19.tgz#ba9f321675243af0456d607fa82a4865931e0cef" + integrity sha512-3vMNr4TzNQyjHcRZadojpRaD9Ofr6LsonZAoQ+HMUa/9ORTPoxVIw0e0mpqWpdjj8xybyCM+oKOUH2vwFu/oEw== + dependencies: + undici-types "~6.21.0" + +"@types/node@^12.12.54": + version "12.20.55" + resolved "https://registry.yarnpkg.com/@types/node/-/node-12.20.55.tgz#c329cbd434c42164f846b909bd6f85b5537f6240" + integrity sha512-J8xLz7q2OFulZ2cyGTLE1TbbZcjpno7FaN6zdJNrgAdrJ+DZzh/uFR6YrTb4C+nXakvud8Q4+rbhoIWlYQbUFQ== + +"@types/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.18.1" + resolved "https://registry.yarnpkg.com/@types/ws/-/ws-8.18.1.tgz#48464e4bf2ddfd17db13d845467f6070ffea4aa9" + integrity sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg== + 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== + +agentkeepalive@^4.5.0: + version "4.6.0" + resolved "https://registry.yarnpkg.com/agentkeepalive/-/agentkeepalive-4.6.0.tgz#35f73e94b3f40bf65f105219c623ad19c136ea6a" + integrity sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ== + dependencies: + humanize-ms "^1.2.1" + +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.11" + resolved "https://registry.yarnpkg.com/base-x/-/base-x-3.0.11.tgz#40d80e2a1aeacba29792ccc6c5354806421287ff" + integrity sha512-xz7wQ8xDhdyP7tQxwdteLYeFfS68tSMNCZ/Y37WJ4bhGfKPpqEIlmIyueQHqOyoPhE6xNUqjzRr8ra0eF9VRvA== + 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== + +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== + +bn.js@^5.1.2, bn.js@^5.2.0, bn.js@^5.2.1: + version "5.2.2" + resolved "https://registry.yarnpkg.com/bn.js/-/bn.js-5.2.2.tgz#82c09f9ebbb17107cd72cb7fd39bd1f9d0aaa566" + integrity sha512-v2YAxEmKaBLahNwE1mjp4WON6huMNeuDvagFZW+ASCuA/ku0bXR9hSMw0XpiqMoA3+rmnyck/tPRSFQkoC9Cuw== + +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.9" + resolved "https://registry.yarnpkg.com/bufferutil/-/bufferutil-4.0.9.tgz#6e81739ad48a95cad45a279588e13e95e24a800a" + integrity sha512-WDtdLmJvAuNNPzByAYpRo2rF1Mmradw6gvWsQKf63476DDXmomT9zUiGypLcG4ibIM67vhAj8jJRdbmEws2Aqw== + 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.4.1: + version "5.4.1" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-5.4.1.tgz#1b48bf0963ec158dce2aacf69c093ae2dd2092d8" + integrity sha512-zgVZuo2WcZgfUEmsn6eO3kINexW8RAE4maiQ8QNs8CtpPCSyMiYsULR3HQYkm3w8FIA3SberyMJMSldGsW+U3w== + +check-error@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/check-error/-/check-error-1.0.3.tgz#a6502e4312a7ee969f646e83bb3ddd56281bd694" + integrity sha512-iKEoDYaRmd1mxM90a2OEfWhjsjPpYPuQ+lMYsoxB126+t8fw7ySEO48nmDg5COTjxDI65/Y2OWpeEHk3ZOe8zg== + dependencies: + get-func-name "^2.0.2" + +chokidar@3.5.3: + version "3.5.3" + resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.5.3.tgz#1cf37c8707b932bd1af1ae22c0432e2acd1903bd" + integrity sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw== + dependencies: + anymatch "~3.1.2" + braces "~3.0.2" + glob-parent "~5.1.2" + is-binary-path "~2.1.0" + is-glob "~4.0.1" + normalize-path "~3.0.0" + readdirp "~3.6.0" + optionalDependencies: + fsevents "~2.3.2" + +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@^13.1.0: + version "13.1.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-13.1.0.tgz#776167db68c78f38dcce1f9b8d7b8b9a488abf46" + integrity sha512-/rFeCpNJQbhSZjGVwO9RFV3xPqbnERS8MmIQzCtD/zl6gpJuV/bMLuN92oG3F7d8oDEHHRrujSXNUr8fpjntKw== + +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.2.0" + resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-3.2.0.tgz#34e9192f53bc757d6614304d9e5e6fb4edb782e3" + integrity sha512-Q+xVJLoGOeIMXZmbUK4HYk+69cQH6LudR0Vu/pRm2YlU/hDV9CiS0gKUMaWY5f2NeUH9C1nV3bsTlCo0FsTV1Q== + dependencies: + node-fetch "^2.7.0" + +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== + +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== + +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: + 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.2.0" + resolved "https://registry.yarnpkg.com/jayson/-/jayson-4.2.0.tgz#b71762393fa40bc9637eaf734ca6f40d3b8c0c93" + integrity sha512-VfJ9t1YLwacIubLhONk0KFeosUBwstRWQ0IRT1KDjEjnVnSOVHC3uwugyV7L0c7R9lpVyrUGT2XWiBA1UTtpyg== + dependencies: + "@types/connect" "^3.4.33" + "@types/node" "^12.12.54" + "@types/ws" "^7.4.4" + 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" + stream-json "^1.9.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" + +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" + +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== + +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.3.0: + version "4.8.4" + resolved "https://registry.yarnpkg.com/node-gyp-build/-/node-gyp-build-4.8.4.tgz#8a70ee85464ae52327772a90d66c6077a900cfc8" + integrity sha512-LA4ZjwlnUblHVgq0oBF3Jl/6h/Nvs5fzBLwdEF4nuxnFdsfajde4WfxtJr3CaiH+F6ewcIB/q4jQ4UzPyid+CQ== + +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" + +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" + +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== + +rpc-websockets@^9.0.2: + version "9.1.1" + resolved "https://registry.yarnpkg.com/rpc-websockets/-/rpc-websockets-9.1.1.tgz#5764336f3623ee1c5cc8653b7335183e3c0c78bd" + integrity sha512-1IXGM/TfPT6nfYMIXkJdzn+L4JEsmb0FL1O2OBjaH03V3yuUDdKFulGLMFG6ErV+8pZ5HVC0limve01RyO+saA== + 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: + 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" + +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== + +stream-chain@^2.2.5: + version "2.2.5" + resolved "https://registry.yarnpkg.com/stream-chain/-/stream-chain-2.2.5.tgz#b30967e8f14ee033c5b9a19bbe8a2cba90ba0d09" + integrity sha512-1TJmBx6aSWqZ4tx7aTpBDXK0/e2hhcNSTV8+CbFJtDjbb+I1mZ8lHit0Grw9GRT+6JbIrrDd8esncgBi8aBXGA== + +stream-json@^1.9.1: + version "1.9.1" + resolved "https://registry.yarnpkg.com/stream-json/-/stream-json-1.9.1.tgz#e3fec03e984a503718946c170db7d74556c2a187" + integrity sha512-uWkjJ+2Nt/LO9Z/JyKZbMusL8Dkh97uUBTv3AJQ74y07lVahLY4eEFsPsE97pxYBwr8nnjMAIch5eqI0gPShyw== + dependencies: + stream-chain "^2.2.5" + +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" + +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== + +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.1.0" + resolved "https://registry.yarnpkg.com/ts-mocha/-/ts-mocha-10.1.0.tgz#17a1c055f5f7733fd82447c4420740db87221bc8" + integrity sha512-T0C0Xm3/WqCuF2tpa0GNGESTBoKZaiqdUP8guNv4ZY316AFXlyidnrzQ1LUrCT0Wb1i3J0zFTgOh/55Un44WdA== + 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.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@^5.7.3: + version "5.8.3" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.8.3.tgz#92f8a3e5e3cf497356f4178c34cd65a7f5e8440e" + integrity sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ== + +undici-types@~6.21.0: + version "6.21.0" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.21.0.tgz#691d00af3909be93a7faa13be61b3a5b50ef12cb" + integrity sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ== + +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" + +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.2" + resolved "https://registry.yarnpkg.com/ws/-/ws-8.18.2.tgz#42738b2be57ced85f46154320aabb51ab003705a" + integrity sha512-DMricUmwGZUVr++AEAe2uiVM7UoO9MAVZMDu05UQOaUII0lp+zOzLLU4Xqh/JvTqklB1T4uELaaPBKyjE1r4fQ== + +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== diff --git a/solana-shine-client-lib/.gitignore b/solana-shine-client-lib/.gitignore new file mode 100644 index 0000000..b63da45 --- /dev/null +++ b/solana-shine-client-lib/.gitignore @@ -0,0 +1,42 @@ +.gradle +build/ +!gradle/wrapper/gradle-wrapper.jar +!**/src/main/**/build/ +!**/src/test/**/build/ + +### IntelliJ IDEA ### +.idea/modules.xml +.idea/jarRepositories.xml +.idea/compiler.xml +.idea/libraries/ +*.iws +*.iml +*.ipr +out/ +!**/src/main/**/out/ +!**/src/test/**/out/ + +### Eclipse ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache +bin/ +!**/src/main/**/bin/ +!**/src/test/**/bin/ + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ + +### VS Code ### +.vscode/ + +### Mac OS ### +.DS_Store \ No newline at end of file diff --git a/solana-shine-client-lib/.idea/.gitignore b/solana-shine-client-lib/.idea/.gitignore new file mode 100644 index 0000000..7bc07ec --- /dev/null +++ b/solana-shine-client-lib/.idea/.gitignore @@ -0,0 +1,10 @@ +# Default ignored files +/shelf/ +/workspace.xml +# Editor-based HTTP Client requests +/httpRequests/ +# Environment-dependent path to Maven home directory +/mavenHomeManager.xml +# Datasource local storage ignored files +/dataSources/ +/dataSources.local.xml diff --git a/solana-shine-client-lib/.idea/gradle.xml b/solana-shine-client-lib/.idea/gradle.xml new file mode 100644 index 0000000..d19c938 --- /dev/null +++ b/solana-shine-client-lib/.idea/gradle.xml @@ -0,0 +1,18 @@ + + + + + + + \ No newline at end of file diff --git a/solana-shine-client-lib/.idea/misc.xml b/solana-shine-client-lib/.idea/misc.xml new file mode 100644 index 0000000..fe0b0da --- /dev/null +++ b/solana-shine-client-lib/.idea/misc.xml @@ -0,0 +1,10 @@ + + + + + + + + + + \ No newline at end of file diff --git a/solana-shine-client-lib/.idea/vcs.xml b/solana-shine-client-lib/.idea/vcs.xml new file mode 100644 index 0000000..94a25f7 --- /dev/null +++ b/solana-shine-client-lib/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/solana-shine-client-lib/build.gradle b/solana-shine-client-lib/build.gradle new file mode 100644 index 0000000..203cd31 --- /dev/null +++ b/solana-shine-client-lib/build.gradle @@ -0,0 +1,46 @@ +plugins { + id 'java' +} + +apply plugin: 'java' + +group = 'com.shine' +version = '1.0.0' + +java { + sourceCompatibility = JavaVersion.VERSION_1_8 + targetCompatibility = JavaVersion.VERSION_1_8 +} + +tasks.withType(Jar) { + manifest { + attributes( + 'Implementation-Title': 'solana-shine-lib', // или solana-shine-client-lib + 'Implementation-Version': version + ) + } +} + +repositories { + mavenCentral() + flatDir { + dirs 'libs' + } +} + +dependencies { + testImplementation platform('org.junit:junit-bom:5.10.0') + testImplementation 'org.junit.jupiter:junit-jupiter' + + implementation project(':solana-shine-lib') + implementation 'com.google.code.gson:gson:2.10.1' + // implementation "com.mmorrell:solanaj:1.15.1" + +} + +test { + useJUnitPlatform() +} + + + diff --git a/solana-shine-client-lib/gradle/wrapper/gradle-wrapper.jar b/solana-shine-client-lib/gradle/wrapper/gradle-wrapper.jar new file mode 100644 index 0000000..249e583 Binary files /dev/null and b/solana-shine-client-lib/gradle/wrapper/gradle-wrapper.jar differ diff --git a/solana-shine-client-lib/gradle/wrapper/gradle-wrapper.properties b/solana-shine-client-lib/gradle/wrapper/gradle-wrapper.properties new file mode 100644 index 0000000..2aa7c28 --- /dev/null +++ b/solana-shine-client-lib/gradle/wrapper/gradle-wrapper.properties @@ -0,0 +1,6 @@ +#Fri Jun 13 15:16:43 MSK 2025 +distributionBase=GRADLE_USER_HOME +distributionPath=wrapper/dists +distributionUrl=https\://services.gradle.org/distributions/gradle-8.13-bin.zip +zipStoreBase=GRADLE_USER_HOME +zipStorePath=wrapper/dists diff --git a/solana-shine-client-lib/gradlew b/solana-shine-client-lib/gradlew new file mode 100755 index 0000000..1b6c787 --- /dev/null +++ b/solana-shine-client-lib/gradlew @@ -0,0 +1,234 @@ +#!/bin/sh + +# +# Copyright © 2015-2021 the original authors. +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +############################################################################## +# +# Gradle start up script for POSIX generated by Gradle. +# +# Important for running: +# +# (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is +# noncompliant, but you have some other compliant shell such as ksh or +# bash, then to run this script, type that shell name before the whole +# command line, like: +# +# ksh Gradle +# +# Busybox and similar reduced shells will NOT work, because this script +# requires all of these POSIX shell features: +# * functions; +# * expansions «$var», «${var}», «${var:-default}», «${var+SET}», +# «${var#prefix}», «${var%suffix}», and «$( cmd )»; +# * compound commands having a testable exit status, especially «case»; +# * various built-in commands including «command», «set», and «ulimit». +# +# Important for patching: +# +# (2) This script targets any POSIX shell, so it avoids extensions provided +# by Bash, Ksh, etc; in particular arrays are avoided. +# +# The "traditional" practice of packing multiple parameters into a +# space-separated string is a well documented source of bugs and security +# problems, so this is (mostly) avoided, by progressively accumulating +# options in "$@", and eventually passing that to Java. +# +# Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, +# and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; +# see the in-line comments for details. +# +# There are tweaks for specific operating systems such as AIX, CygWin, +# Darwin, MinGW, and NonStop. +# +# (3) This script is generated from the Groovy template +# https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt +# within the Gradle project. +# +# You can find Gradle at https://github.com/gradle/gradle/. +# +############################################################################## + +# Attempt to set APP_HOME + +# Resolve links: $0 may be a link +app_path=$0 + +# Need this for daisy-chained symlinks. +while + APP_HOME=${app_path%"${app_path##*/}"} # leaves a trailing /; empty if no leading path + [ -h "$app_path" ] +do + ls=$( ls -ld "$app_path" ) + link=${ls#*' -> '} + case $link in #( + /*) app_path=$link ;; #( + *) app_path=$APP_HOME$link ;; + esac +done + +APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit + +APP_NAME="Gradle" +APP_BASE_NAME=${0##*/} + +# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' + +# Use the maximum available, or set MAX_FD != -1 to use that value. +MAX_FD=maximum + +warn () { + echo "$*" +} >&2 + +die () { + echo + echo "$*" + echo + exit 1 +} >&2 + +# OS specific support (must be 'true' or 'false'). +cygwin=false +msys=false +darwin=false +nonstop=false +case "$( uname )" in #( + CYGWIN* ) cygwin=true ;; #( + Darwin* ) darwin=true ;; #( + MSYS* | MINGW* ) msys=true ;; #( + NONSTOP* ) nonstop=true ;; +esac + +CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar + + +# Determine the Java command to use to start the JVM. +if [ -n "$JAVA_HOME" ] ; then + if [ -x "$JAVA_HOME/jre/sh/java" ] ; then + # IBM's JDK on AIX uses strange locations for the executables + JAVACMD=$JAVA_HOME/jre/sh/java + else + JAVACMD=$JAVA_HOME/bin/java + fi + if [ ! -x "$JAVACMD" ] ; then + die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." + fi +else + JAVACMD=java + which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. + +Please set the JAVA_HOME variable in your environment to match the +location of your Java installation." +fi + +# Increase the maximum file descriptors if we can. +if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then + case $MAX_FD in #( + max*) + MAX_FD=$( ulimit -H -n ) || + warn "Could not query maximum file descriptor limit" + esac + case $MAX_FD in #( + '' | soft) :;; #( + *) + ulimit -n "$MAX_FD" || + warn "Could not set maximum file descriptor limit to $MAX_FD" + esac +fi + +# Collect all arguments for the java command, stacking in reverse order: +# * args from the command line +# * the main class name +# * -classpath +# * -D...appname settings +# * --module-path (only if needed) +# * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. + +# For Cygwin or MSYS, switch paths to Windows format before running java +if "$cygwin" || "$msys" ; then + APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) + CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) + + JAVACMD=$( cygpath --unix "$JAVACMD" ) + + # Now convert the arguments - kludge to limit ourselves to /bin/sh + for arg do + if + case $arg in #( + -*) false ;; # don't mess with options #( + /?*) t=${arg#/} t=/${t%%/*} # looks like a POSIX filepath + [ -e "$t" ] ;; #( + *) false ;; + esac + then + arg=$( cygpath --path --ignore --mixed "$arg" ) + fi + # Roll the args list around exactly as many times as the number of + # args, so each arg winds up back in the position where it started, but + # possibly modified. + # + # NB: a `for` loop captures its iteration list before it begins, so + # changing the positional parameters here affects neither the number of + # iterations, nor the values presented in `arg`. + shift # remove old arg + set -- "$@" "$arg" # push replacement arg + done +fi + +# Collect all arguments for the java command; +# * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of +# shell script including quotes and variable substitutions, so put them in +# double quotes to make sure that they get re-expanded; and +# * put everything else in single quotes, so that it's not re-expanded. + +set -- \ + "-Dorg.gradle.appname=$APP_BASE_NAME" \ + -classpath "$CLASSPATH" \ + org.gradle.wrapper.GradleWrapperMain \ + "$@" + +# Use "xargs" to parse quoted args. +# +# With -n1 it outputs one arg per line, with the quotes and backslashes removed. +# +# In Bash we could simply go: +# +# readarray ARGS < <( xargs -n1 <<<"$var" ) && +# set -- "${ARGS[@]}" "$@" +# +# but POSIX shell has neither arrays nor command substitution, so instead we +# post-process each arg (as a line of input to sed) to backslash-escape any +# character that might be a shell metacharacter, then use eval to reverse +# that process (while maintaining the separation between arguments), and wrap +# the whole thing up as a single "set" statement. +# +# This will of course break if any of these variables contains a newline or +# an unmatched quote. +# + +eval "set -- $( + printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | + xargs -n1 | + sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | + tr '\n' ' ' + )" '"$@"' + +exec "$JAVACMD" "$@" diff --git a/solana-shine-client-lib/gradlew.bat b/solana-shine-client-lib/gradlew.bat new file mode 100644 index 0000000..107acd3 --- /dev/null +++ b/solana-shine-client-lib/gradlew.bat @@ -0,0 +1,89 @@ +@rem +@rem Copyright 2015 the original author or authors. +@rem +@rem Licensed under the Apache License, Version 2.0 (the "License"); +@rem you may not use this file except in compliance with the License. +@rem You may obtain a copy of the License at +@rem +@rem https://www.apache.org/licenses/LICENSE-2.0 +@rem +@rem Unless required by applicable law or agreed to in writing, software +@rem distributed under the License is distributed on an "AS IS" BASIS, +@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +@rem See the License for the specific language governing permissions and +@rem limitations under the License. +@rem + +@if "%DEBUG%" == "" @echo off +@rem ########################################################################## +@rem +@rem Gradle startup script for Windows +@rem +@rem ########################################################################## + +@rem Set local scope for the variables with windows NT shell +if "%OS%"=="Windows_NT" setlocal + +set DIRNAME=%~dp0 +if "%DIRNAME%" == "" set DIRNAME=. +set APP_BASE_NAME=%~n0 +set APP_HOME=%DIRNAME% + +@rem Resolve any "." and ".." in APP_HOME to make it shorter. +for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi + +@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. +set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" + +@rem Find java.exe +if defined JAVA_HOME goto findJavaFromJavaHome + +set JAVA_EXE=java.exe +%JAVA_EXE% -version >NUL 2>&1 +if "%ERRORLEVEL%" == "0" goto execute + +echo. +echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:findJavaFromJavaHome +set JAVA_HOME=%JAVA_HOME:"=% +set JAVA_EXE=%JAVA_HOME%/bin/java.exe + +if exist "%JAVA_EXE%" goto execute + +echo. +echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% +echo. +echo Please set the JAVA_HOME variable in your environment to match the +echo location of your Java installation. + +goto fail + +:execute +@rem Setup the command line + +set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar + + +@rem Execute Gradle +"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* + +:end +@rem End local scope for the variables with windows NT shell +if "%ERRORLEVEL%"=="0" goto mainEnd + +:fail +rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of +rem the _cmd.exe /c_ return code! +if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 +exit /b 1 + +:mainEnd +if "%OS%"=="Windows_NT" endlocal + +:omega diff --git a/solana-shine-client-lib/libs/tmp/solanaj-1.20.4.jar b/solana-shine-client-lib/libs/tmp/solanaj-1.20.4.jar new file mode 100644 index 0000000..a385a68 Binary files /dev/null and b/solana-shine-client-lib/libs/tmp/solanaj-1.20.4.jar differ diff --git a/solana-shine-client-lib/settings.gradle b/solana-shine-client-lib/settings.gradle new file mode 100644 index 0000000..0168e29 --- /dev/null +++ b/solana-shine-client-lib/settings.gradle @@ -0,0 +1,2 @@ +rootProject.name = 'solana-shine-client-lib' +include 'solana-shine-lib' diff --git a/solana-shine-client-lib/solana-shine-lib/build.gradle b/solana-shine-client-lib/solana-shine-lib/build.gradle new file mode 100644 index 0000000..2716a73 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/build.gradle @@ -0,0 +1,56 @@ +plugins { + id 'java' +} + +group = 'me.shineup' +version = '1.0' + + +java { + sourceCompatibility = JavaVersion.VERSION_1_8 + targetCompatibility = JavaVersion.VERSION_1_8 +} +repositories { + mavenCentral() + flatDir { + dirs 'libs' + } +} + +dependencies { + // были стандартные + testImplementation platform('org.junit:junit-bom:5.10.0') + testImplementation 'org.junit.jupiter:junit-jupiter' + + // шифрование нужна + implementation 'org.bouncycastle:bcprov-jdk15on:1.70' + +// implementation 'com.squareup.okhttp3:okhttp:4.12.0' // запросы по сети + //солана + implementation "com.mmorrell:solanaj:1.15.1" + +// implementation 'org.bitcoinj:bitcoinj-core:0.15.10' +// implementation 'com.squareup.moshi:moshi:1.13.0' + + +// implementation 'com.mmorrell:solanaj:1.20.4' - старые соланы не нужны +// implementation name: 'solanaj-1.20.4' - старые соланы не нужны + + + + // Logging + implementation 'org.slf4j:slf4j-api:1.7.36' + implementation 'ch.qos.logback:logback-classic:1.2.11' + + + implementation 'com.google.code.gson:gson:2.10.1' + +} + +test { + useJUnitPlatform() +} + + + + diff --git a/solana-shine-client-lib/solana-shine-lib/libs/solanaj-1.20.4.jar b/solana-shine-client-lib/solana-shine-lib/libs/solanaj-1.20.4.jar new file mode 100644 index 0000000..a385a68 Binary files /dev/null and b/solana-shine-client-lib/solana-shine-lib/libs/solanaj-1.20.4.jar differ diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/TODO.txt b/solana-shine-client-lib/solana-shine-lib/src/main/java/TODO.txt new file mode 100644 index 0000000..a07225c --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/TODO.txt @@ -0,0 +1,9 @@ +Что ещё надо сделать по библиотеке + +1. Сделать более рандомное создание пар ключей. (тк я три из трёх угадал в девнет!!!) + +2. Доделат обработку ошибок при вызове функции ( наверно уже в UI надо отлавливать ексепшены которые возвращает нода при препроверки вызова функции) + + 2.5 метод проверки что транзакция прям точно добавлена в систему. (хотя нужен ли он??. Тк пользователь добавился значет уже всё хорошо :) ) + +3. Исправить перевод денег (тк он не работает после перехода на старую библиотеку) \ No newline at end of file diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaSettings.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaSettings.java new file mode 100644 index 0000000..97dbcaa --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaSettings.java @@ -0,0 +1,42 @@ +package me.shineup.solana; + +import me.shineup.solana.config.Const; + +/** + * Настройки подключения к Solana. + * Позволяет выбирать RPC-сервер (локальный, тестовая сеть или произвольный). + */ +public class SolanaSettings { + + /** + * Устанавливает локальный RPC-адрес (например, http://127.0.0.1:8899). + */ + public static void setRpcUrlLocal() { + Const.RPC_URL = Const.LOCAL_RPC_URL; + } + + /** + * Устанавливает RPC-адрес тестовой сети Solana (https://api.testnet.solana.com). + */ + public static void setRpcUrlTestNet() { + Const.RPC_URL = "https://api.testnet.solana.com"; // или Const.TESTNET_RPC_URL + } + + /** + * Устанавливает пользовательский RPC-адрес. + * + * @param url Строка с адресом RPC-сервера + */ + public static void setRpcUrl(String url) { + Const.RPC_URL = url; + } + + /** + * Получает текущий установленный RPC-адрес. + * + * @return строка с текущим RPC-URL + */ + public static String getRpcUrl() { + return Const.RPC_URL; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaTxWatcher.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaTxWatcher.java new file mode 100644 index 0000000..6d1f8c3 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaTxWatcher.java @@ -0,0 +1,218 @@ +package me.shineup.solana; + + +import me.shineup.solana.internal.utils.resultChecker.TransactionStatusHelper; +import me.shineup.solana.model.TxStatus; + +import java.time.Instant; +import java.util.HashMap; +import java.util.Map; + +/** + *

SolanaTxWatcher

+ *

+ * Класс-наблюдатель за одной транзакцией сети Solana. + * Он: + *

    + *
  • Хранит подпись (signature) и фиксирует момент создания объекта.
  • + *
  • Через {@link #updateStatus()} опрашивает RPC (используя {@link TransactionStatusHelper}) + * и обновляет внутренний статус.
  • + *
  • Через {@link #shouldRetry()} сообщает, надо ли продолжать опрос (с учётом таймаута, + * лимита неудачных попыток и финального статуса).
  • + *
  • Через {@link #isSuccess()} указывает, прошла ли транзакция успешно.
  • + *
+ * + *

Пример использования

+ * + *
{@code
+ * SolanaTxWatcher watcher = new SolanaTxWatcher("5QgV...sig");
+ *
+ * while (watcher.shouldRetry()) {        // ◀ проверяем, нужно ли ещё опрашивать
+ *     watcher.updateStatus();            // ◀ запрашиваем статус через RPC
+ *     System.out.println(
+ *         watcher.getStatus() + " | " +  // ◀ печатаем статус
+ *         watcher.getStatusComment());
+ *     Thread.sleep(SolanaTxWatcher.getRetryIntervalMs()); // ◀ ждём секунду
+ * }
+ *
+ * if (watcher.isSuccess()) {
+ *     System.out.println("✅ Транзакция прошла успешно!");
+ * } else {
+ *     System.out.println("⛔ Завершили слежение без успеха.");
+ * }
+ * }
+ + */ +public class SolanaTxWatcher { + + /* ---------- НАСТРАИВАЕМЫЕ КОНСТАНТЫ ---------- */ + + /** Максимальное время слежения (мс). */ + private static final long TIMEOUT_MS = 30_000; + + /** Допустимое количество подряд статусов UNKNOWN / NETWORK_ERROR. */ + private static final int MAX_FAILED_ATTEMPTS = 3; + + /** Рекомендуемый интервал (мс) между вызовами {@link #updateStatus()}. */ + private static final long RETRY_INTERVAL_MS = 1_000; + + /* ---------- ПОЛЯ ЭКЗЕМПЛЯРА ---------- */ + + /** Подпись (signature) транзакции. */ + private final String signature; + + /** Время создания объекта (Unix-millis). */ + private final long startTimeMs; + + /** Кол-во подряд «слабых» ошибок (UNKNOWN / NETWORK_ERROR). */ + private int failedAttempts; + + /** Флаг: нужно ли ещё опрашивать RPC. */ + private boolean needRetry; + + /** Флаг: успешна ли транзакция (устанавливается при FINALIZED_SUCCESS). */ + private boolean success; + + /** Текущий статус из {@link TxStatus}. */ + private TxStatus status; + + /* ---------- ЧЕЛОВЕКО-ЧИТАЕМЫЕ ОПИСАНИЯ СТАТУСОВ ---------- */ + + private static final Map COMMENTS = new HashMap<>(); + static { + COMMENTS.put(TxStatus.NOT_FOUND, + "Подпись не дошла до RPC — ждём появления."); + COMMENTS.put(TxStatus.PROCESSED, + "Принята в обработку — ожидаем включения в блок."); + COMMENTS.put(TxStatus.CONFIRMED, + "Уже в блоке — ждём финализации."); + COMMENTS.put(TxStatus.FINALIZED_SUCCESS, + "Финализирована успешно."); + COMMENTS.put(TxStatus.FINALIZED_ERROR, + "Финализирована с ошибкой."); + COMMENTS.put(TxStatus.UNKNOWN, + "Неизвестная ошибка RPC/парсинга."); + COMMENTS.put(TxStatus.NETWORK_ERROR, + "Сбой сети или RPC недоступен."); + } + + /* ---------- КОНСТРУКТОР ---------- */ + + /** + * Создаёт watcher для указанной подписи. + * + * @param signature подпись транзакции (Base58). + */ + public SolanaTxWatcher(String signature) { + this.signature = signature; + this.startTimeMs = System.currentTimeMillis(); + this.failedAttempts = 0; + this.needRetry = true; // по умолчанию пытаемся + this.success = false; // успех пока не достигнут + this.status = TxStatus.NOT_FOUND; // стартовый + } + + /* ---------- ГЕТТЕРЫ ---------- */ + + /** @return подпись транзакции. */ + public String getSignature() { return signature; } + + /** @return время создания watcher’а (Unix-millis). */ + public long getStartTimeMs() { return startTimeMs; } + + /** @return текущий статус. */ + public TxStatus getStatus() { return status; } + + /** @return true, если транзакция финализирована без ошибок. */ + public boolean isSuccess() { return success; } + + /** @return кол-во подряд неудачных (UNKNOWN/NETWORK_ERROR) попыток. */ + public int getFailedAttempts() { return failedAttempts; } + + /** @return человеко-читаемый комментарий к текущему статусу. */ + public String getStatusComment() { + return COMMENTS.getOrDefault(status, ""); + } + + /* ---------- ОСНОВНОЙ МЕТОД ОПРОСА ---------- */ + + /** + * Запрашивает актуальный статус транзакции через {@link TransactionStatusHelper} + * и обновляет внутренние поля. + *

— При промежуточных статусах (NOT_FOUND / PROCESSED / CONFIRMED) + * счётчик ошибок сбрасывается.
+ * — При {@code FINALIZED_SUCCESS} или {@code FINALIZED_ERROR} + * флаг {@link #needRetry} переводится в {@code false}.
+ * — При {@code UNKNOWN} или {@code NETWORK_ERROR} + * счётчик ошибок увеличивается; если превышен лимит — дальнейший опрос прекращается. + */ + public void updateStatus() { + if (!needRetry) return; // уже решено не опрашивать + + TxStatus newStatus = + TransactionStatusHelper.getTxStatus(signature); + + switch (newStatus) { + + case NOT_FOUND: + case PROCESSED: + case CONFIRMED: + failedAttempts = 0; // успешное промежуточное обновление + break; + + case FINALIZED_SUCCESS: + success = true; + needRetry = false; // финальный успех + break; + + case FINALIZED_ERROR: + success = false; + needRetry = false; // финальный провал + break; + + case UNKNOWN: + case NETWORK_ERROR: + failedAttempts++; + if (failedAttempts > MAX_FAILED_ATTEMPTS) { + needRetry = false; // слишком много ошибок — прекращаем + } + break; + } + + status = newStatus; // сохраняем новый статус + } + + /* ---------- РЕШЕНИЕ: НУЖНО ЛИ ПОВТОРЯТЬ ---------- */ + + /** + * @return {@code true}, если можно и стоит делать ещё один запрос статуса.
+ * {@code false} — если достигнут финальный статус, превышен таймаут + * или лимит неудачных попыток. + */ + public boolean shouldRetry() { + if (!needRetry) return false; // наш флаг запрещает + long elapsed = System.currentTimeMillis() - startTimeMs; + if (elapsed > TIMEOUT_MS) { // вышли за таймаут + needRetry = false; + return false; + } + return true; + } + + /* ---------- ВСПОМОГАТЕЛЬНОЕ ---------- */ + + /** @return рекомендуемую задержку (мс) между опросами. */ + public static long getRetryIntervalMs() { return RETRY_INTERVAL_MS; } + + @Override + public String toString() { + return String.format("[%s] sig=%s | status=%s | needRetry=%s | success=%s | attempts=%d | %s", + Instant.ofEpochMilli(startTimeMs), + signature, + status, + needRetry, + success, + failedAttempts, + getStatusComment()); + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaWrapper.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaWrapper.java new file mode 100644 index 0000000..73a14fc --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/SolanaWrapper.java @@ -0,0 +1,88 @@ +package me.shineup.solana; + +import me.shineup.solana.internal.callSolanaFunc.RegisterUser.RegisterUserWithOneDev; +import me.shineup.solana.model.TxStatus; +import me.shineup.solana.model.UserById; +import me.shineup.solana.internal.readFromSolana.userById.UserByIdReader; +import me.shineup.solana.model.UserByLogin; +import me.shineup.solana.internal.readFromSolana.userByLogin.UserByLoginReader; +import me.shineup.solana.internal.standartActions.airDrops.SolanaAirdrop; +import me.shineup.solana.internal.standartActions.balanse.SolanaBalanceChecker; +import me.shineup.solana.internal.standartActions.keysGenerator.KeyPairBase58; +import me.shineup.solana.internal.standartActions.keysGenerator.SolanaKeyGeneratorManual; +import me.shineup.solana.internal.standartActions.transfer.SolanaTransfer; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.utils.resultChecker.TransactionStatusHelper; + +public class SolanaWrapper { + + /** Получает баланс по публичному ключу */ + public static long getBalance(String publicKey) throws Exception{ + return SolanaBalanceChecker.getBalance(publicKey); + } + + /** Запрашивает Airdrop на указанный публичный ключ */ + public static String requestAirdrop(String publicKey, long lamports) throws Exception{ + return SolanaAirdrop.requestAirdrop(publicKey, lamports); + } + + /** Обёртка для перевода lamports между двумя публичными ключами (оба ключа — в base58, приватный — отправителя). */ +// public static String sendLamports(String fromBase58Secret, String toBase58Pubkey, long lamports) throws Exception { +// return SolanaTransfer.sendSol(fromBase58Secret, toBase58Pubkey, lamports); +// } + + /** Генерирует новый Ed25519-кошелёк (ключи Solana) */ + public static KeyPairBase58 generateNewWallet() throws Exception{ // todo возмаожно ключи генерируются недостаточно рандомно + return SolanaKeyGeneratorManual.generateKeyPair(); + } + + /** Генерирует новeую пару ключей X25519 */ + public static KeyPairBase58 generateNewKeyPairX25519() throws Exception{ // todo это пока заглушка + return SolanaKeyGeneratorManual.generateKeyPair(); + } + + /** Проверяет статус транзакции по её подписи (signature) */ + public static TxStatus getTransactionStatus(String signature) throws Exception { + return TransactionStatusHelper.getTxStatus(signature); + } + + /** + * Обёртка для перевода SOL с одного аккаунта на другой. + * + * @param fromBase58Secret приватный ключ отправителя (в base58) + * @param toAddressBase58 публичный ключ получателя (в base58) + * @param lamports сумма в лампортах (1 SOL = 1_000_000_000 лампортов) + */ + public static String sendSol(String fromBase58Secret, String toAddressBase58, long lamports) throws Exception{ + return SolanaTransfer.sendSol(fromBase58Secret, toAddressBase58, lamports); + } + + /** Выполняет регистрацию пользователя и одного устройства */ + public static String registerUserWithOneDev( + String payerPubkeyB58, + String payerPrivkeyB58, + String login, + String deviceSignPubkeyB58, + String deviceX25519PubkeyB58 + ) throws Exception { + return RegisterUserWithOneDev.callRegisterUserWithOneDev( + payerPubkeyB58, + payerPrivkeyB58, + login, + deviceSignPubkeyB58, + deviceX25519PubkeyB58 + ); + } + + /** Получить объект UserByLogin по логину */ + public static UserByLogin getUserByLogin(String login) throws Exception { + return UserByLoginReader.getUserByLogin(login); + } + + /** Получить объект UserById по числовому ID */ + public static UserById getUserById(long id) throws Exception { + return UserByIdReader.getUserById(id); + } + + +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/actions.md b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/actions.md new file mode 100644 index 0000000..49e4fbc --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/actions.md @@ -0,0 +1,19 @@ +InitializeUserCounter - инициализирует счётчик пользователь (вызывается один раз) + +RegisterUserWithOneDev - регистрирует пользователя с одним устройством +- оплачивает деньги +- решистрирует акаунт (PDA) по ЛОГИНУ и (PDA) по id +- регистрирует ключи пользователя и устройствва +- увеличивает количество пользователей в ситеме ++1 + +UserByLoginReader - читает данные пользователя по логину +UserCounterReader - читает количество пользователей в системе + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - +SolanaKeyGeneratorManual - генерирует пары ключей + - - - - +SolanaBalanceChecker - показывает баланс акаунта +SolanaAirdrop - запрашивает Airdrop +SolanaTransfer - перевод со счёта на счёт +----------------------------------------- +Const - хранит константы настройки +ResultChecker(sig) - проверяет результат транзакции diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/config/Const.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/config/Const.java new file mode 100644 index 0000000..26d937b --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/config/Const.java @@ -0,0 +1,76 @@ +package me.shineup.solana.config; + +import me.shineup.solana.internal.utils.KeyPair; +import org.p2p.solanaj.core.PublicKey; + +import java.text.DecimalFormat; + + + +public class Const { + + /** Program ID из declare_id! */ + + public static final String PROGRAM_ID_str = "5dFcWDNp42Xn9Vv4oDMJzM4obBJ8hvDuAtPX54fT5L3t"; // shine + + public static final String userSeedsPrefix = "u="; // префикс для Seed адреса пользователя по логину + + public static String USER_COUNTER_SEED = "user_counter"; // Seed Адрес PDA счётчика пользователей + + public static final PublicKey PROGRAM_ID_key = new PublicKey(PROGRAM_ID_str);//"BmCgGmQbSjkE6Zg8WAwhxDMNHiTknMYqTF4ZVMrPdTpz"); // shine + + + + public static PublicKey ADMIN_FEE_ACCOUNT = new PublicKey("6bFc5Gz5qF172GQhK5HpDbWs8F6qcSxdHn5XqAstf1fY"); + + + public static final String LOCAL_RPC_URL = "http://127.0.0.1:8899"; + public static final String LOCAL_ANDROID_TEST_RPC_URL = "http://10.0.2.2:8899"; + public static final String TESTNET_RPC_URL = "https://api.testnet.solana.com"; + public static final String DEVNET_RPC_URL = "https://api.devnet.solana.com"; + + // RPC URL для используемой ноды Solana + public static String RPC_URL = DEVNET_RPC_URL; + // Запись для хранения ключей +// public record KeyPair(String name, String publicKey, String privateKey) {} не надо больше + + // Массив пар ключей + public static final KeyPair[] KEYS = { + new KeyPair("key1", "HMww7YSVfwVm4i8sugqj7wyH26dqzHykzv3wzWwzEvPA", // есть в дев нет!!! --url https://api.devnet.solana.com + "5pbFo9Zq1VsNheHwbEp6AZKa6R62CZHoGkJFZnugpMEtCmkQFjuUP7TgA5hSPqv4NABGmPP62qVnDPHmRqEAwvJc"), + new KeyPair("key2", "E3ZDHbWv1qiFvDTmaRc9wjFCgbQw6UmKJLJYbaTNvjAh", + "5qm1GJGXB1fFJ3YsU5Y3XXgTiQfaimqBWk79oEveFASH9D2of3jqUoT7dumBvS449fW5j5Sw8MgAMH2QBMmFPdry"), + new KeyPair("key3", "6bFc5Gz5qF172GQhK5HpDbWs8F6qcSxdHn5XqAstf1fY", + "3VYfYZZ3ugmgwisiQQAfcimX9T65AE9BmwmYVixAUj4jyneccSE9rzbC3g5twvH7ECZ8xgp7emJo3pR4yQqCwjGn") + }; + + // Метод для определения ключа + public static String identifyKey(String key) { + for (KeyPair kp : KEYS) { + if (kp.getPublicKey().equals(key)) { + return kp.getName() + "(public)"; + } + if (kp.getPrivateKey().equals(key)) { + return kp.getName() + "(private)"; + } + } + return key; // если не найдено + } + + // Метод для получения KeyPair по имени + public static KeyPair getKeyByName(String name) { + for (KeyPair kp : KEYS) { + if (kp.getName().equals(name)) { + return kp; + } + } + return null; // если не найдено + } + + // Метод форматирования лампортов в SOL + public static String lamportsToSol(long lamports) { + double sol = lamports / 1_000_000_000.0; + DecimalFormat df = new DecimalFormat("0.00000000"); + return df.format(sol); + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/ExeptionList.txt b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/ExeptionList.txt new file mode 100644 index 0000000..f23c3a0 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/ExeptionList.txt @@ -0,0 +1,35 @@ + +Exeption - если что то полетело в коде + +SolanaRpcConnectionException - Не удалось подключиться к RPC Solana или получить ответ + +SolanaLibLogicException - Базовое исключение всех ошибок, связанных с Solana + Выкидываем во всех не стандартных случаях из библиотеки вместо стандартного Exception + +SolanaProgramException - Исключение, выбрасываемое при кастомной ошибке от Solana-программы (смарт контракта). Например: "custom program error: 0x1771" + +SolanaInsufficientFundsForFeeException - Недостаточно SOL для оплаты комиссии (InsufficientFundsForFee). + это если вызвали регистрацию без средств + (но получается тоже не надо так как - прога не вызовет её без проверки баланса) + + + + +нет такого пользователя + + + + + + + + + + + + + + + SolanaIncorrectProgramIdException Неверный programId — вызывается не та программа (IncorrectProgramId) +возможно надо что бы быстро находить новый програм Ид +(но по факту он не меняется если кто то не потеряет пароль!!! ) ну или потребуется один раз при переходе на DAO \ No newline at end of file diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaErrorHandler.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaErrorHandler.java new file mode 100644 index 0000000..e53c095 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaErrorHandler.java @@ -0,0 +1,43 @@ +package me.shineup.solana.exceptions; + + +import com.google.gson.JsonObject; +import com.google.gson.JsonParser; + +public class SolanaErrorHandler { + + public static void handleRpcJsonError(String json) throws SolanaException { + try { + JsonObject obj = JsonParser.parseString(json).getAsJsonObject(); + if (!obj.has("error")) return; + + JsonObject error = obj.getAsJsonObject("error"); + String msg = error.has("message") ? error.get("message").getAsString() : ""; + + handleSolanaError(msg); + + } catch (Exception e) { + // fallback + throw new SolanaException("Ошибка обработки RPC-ошибки", e); + } + } + + public static void handleSolanaError(String errorMessage) throws SolanaException { + if (errorMessage == null || errorMessage.isEmpty()) return; + + if (errorMessage.contains("custom program error: 0x")) { + String hex = errorMessage.substring(errorMessage.indexOf("0x")).split(" ")[0]; + throw new SolanaException_InProgram(hex); + } + + if (errorMessage.contains("InsufficientFundsForFee")) { + throw new SolanaException_InsufficientFundsForFee(); + } + + if (errorMessage.contains("IncorrectProgramId")) { + throw new SolanaException_IncorrectProgramId(); + } + + throw new SolanaException("Неизвестная ошибка Solana: " + errorMessage); + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException.java new file mode 100644 index 0000000..b054ba4 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException.java @@ -0,0 +1,13 @@ +package me.shineup.solana.exceptions; + + +/** Базовое исключение всех ошибок, связанных с Solana */ +public class SolanaException extends Exception { + public SolanaException(String message) { + super(message); + } + + public SolanaException(String message, Throwable cause) { + super(message, cause); + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_InProgram.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_InProgram.java new file mode 100644 index 0000000..4367d56 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_InProgram.java @@ -0,0 +1,41 @@ +package me.shineup.solana.exceptions; + +/** + * Исключение, выбрасываемое при кастомной ошибке от Solana-программы. + * Например: "custom program error: 0x1771" или 10001 + */ +public class SolanaException_InProgram extends SolanaException { + private final int errorCode; + + /** + * Создаёт исключение на основе шестнадцатеричного кода (например, "0x1771"). + */ + public SolanaException_InProgram(String errorCodeHex) { + super("Ошибка от смарт-контракта. Код: " + parseHex(errorCodeHex)); + this.errorCode = parseHex(errorCodeHex); + } + + /** + * Создаёт исключение на основе десятичного кода (например, 10001). + */ + public SolanaException_InProgram(int errorCodeDecimal) { + super("Ошибка от смарт-контракта. Код: " + errorCodeDecimal); + this.errorCode = errorCodeDecimal; + } + + public int getErrorCodeDecimal() { + return errorCode; + } + + public String getErrorCodeHex() { + return "0x" + Integer.toHexString(errorCode); + } + + private static int parseHex(String hex) { + try { + return Integer.parseInt(hex.replace("0x", ""), 16); + } catch (NumberFormatException e) { + return -1; + } + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_IncorrectProgramId.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_IncorrectProgramId.java new file mode 100644 index 0000000..4f4c6ad --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_IncorrectProgramId.java @@ -0,0 +1,12 @@ +package me.shineup.solana.exceptions; + +/** + * Неверный programId — вызывается не та программа (IncorrectProgramId). + */ +public class SolanaException_IncorrectProgramId extends SolanaException { + public SolanaException_IncorrectProgramId() { + super("Указан неверный programId — не соответствует контракту"); + } +} + + diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_InsufficientFundsForFee.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_InsufficientFundsForFee.java new file mode 100644 index 0000000..64f10ee --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_InsufficientFundsForFee.java @@ -0,0 +1,10 @@ +package me.shineup.solana.exceptions; + +/** + * Недостаточно SOL для оплаты комиссии (InsufficientFundsForFee). + */ +public class SolanaException_InsufficientFundsForFee extends SolanaException { + public SolanaException_InsufficientFundsForFee() { + super("Недостаточно средств на балансе для оплаты комиссии (InsufficientFundsForFee)"); + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_LibLogic.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_LibLogic.java new file mode 100644 index 0000000..2680816 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_LibLogic.java @@ -0,0 +1,21 @@ +package me.shineup.solana.exceptions; + +/** + * Исключение, выбрасываемое вручную при логических проверках в библиотеке. + * К сообщению автоматически добавляется строка вызова (класс и номер строки). + */ +public class SolanaException_LibLogic extends SolanaException { + + public SolanaException_LibLogic(String userMessage) { + super(userMessage + getSourceSuffix()); + } + + private static String getSourceSuffix() { + StackTraceElement[] stack = Thread.currentThread().getStackTrace(); + if (stack.length > 3) { + StackTraceElement caller = stack[3]; + return " (исходный код: " + caller.getFileName() + ":" + caller.getLineNumber() + ")"; + } + return ""; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_RpcConnection.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_RpcConnection.java new file mode 100644 index 0000000..73e7ff2 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_RpcConnection.java @@ -0,0 +1,8 @@ +package me.shineup.solana.exceptions; + +/** Не удалось подключиться к RPC или получить ответ */ +public class SolanaException_RpcConnection extends SolanaException { + public SolanaException_RpcConnection(String message) { + super(message); + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_UserNotFound.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_UserNotFound.java new file mode 100644 index 0000000..04bf799 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/exceptions/SolanaException_UserNotFound.java @@ -0,0 +1,14 @@ +package me.shineup.solana.exceptions; + +/** + * Пользователь с указанным идентификатором не найден в системе. + */ +public class SolanaException_UserNotFound extends SolanaException { + public SolanaException_UserNotFound() { + super("Пользователь не найден в системе."); + } + + public SolanaException_UserNotFound(String msg) { + super("Пользователь не найден в системе: " + msg); + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/InitializeUserCounter/InitializeUserCounter.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/InitializeUserCounter/InitializeUserCounter.java new file mode 100644 index 0000000..677fa0e --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/InitializeUserCounter/InitializeUserCounter.java @@ -0,0 +1,98 @@ +package me.shineup.solana.internal.callSolanaFunc.InitializeUserCounter; + + +import org.p2p.solanaj.core.AccountMeta; +import org.p2p.solanaj.core.PublicKey; +import org.p2p.solanaj.programs.SystemProgram; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.internal.utils.resultChecker.ResultChecker; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.utils.SolanaProgramCaller; + +import java.nio.charset.StandardCharsets; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; + +/** + * Вызывает Anchor-функцию `initialize_user_counter` смарт-контракта на Solana. + * + * Эта функция предназначена для одноразовой инициализации PDA-аккаунта, в котором будет храниться + * счётчик пользователей. Аккаунт создаётся с сидом "user_counter" и содержит 8 байт, представляющих число. + * + * Повторный вызов приведёт к ошибке (если PDA уже существует). + * + */ + +public class InitializeUserCounter { + + private static final Logger LOG = LoggerFactory.getLogger(InitializeUserCounter.class); + +/** + * Вызывает Anchor-функцию `initialize_user_counter` смарт-контракта на Solana. + * + * Эта функция предназначена для одноразовой инициализации PDA-аккаунта, в котором будет храниться + * счётчик пользователей. Аккаунт создаётся с сидом "user_counter" и содержит 8 байт, представляющих число. + * Вызвать её в принципе может кто угодно, кто оплатит создание этого пда + * + * После того как PDA будет создан, любой повторный вызов приведёт к ошибке (если PDA уже существует). + */ + public static String callInitializeUserCounter( + String publicKeyB58, + String privateKeyB58 + ) { + try { + // ───────────────────────────────────────────────────────────── + // 1. Генерируем PDA-адрес для сидов ["user_counter"] + // ───────────────────────────────────────────────────────────── + String seed = "user_counter"; + PublicKey counterPda = PublicKey.findProgramAddress( + Collections.singletonList(seed.getBytes(StandardCharsets.UTF_8)), + Const.PROGRAM_ID_key + ).getAddress(); + + // ───────────────────────────────────────────────────────────── + // 2. Аргументы Anchor-функции: initialize_user_counter не требует входных данных + // ───────────────────────────────────────────────────────────── + byte[] serializedArgs = new byte[0]; // нет параметров + + // ───────────────────────────────────────────────────────────── + // 3. Список аккаунтов + // ───────────────────────────────────────────────────────────── + List accounts = Arrays.asList( + new AccountMeta(new PublicKey(publicKeyB58), true, true), // payer / signer + new AccountMeta(counterPda, false, true), // pda: user_counter + new AccountMeta(SystemProgram.PROGRAM_ID, false, false) // system_program + ); + + // ───────────────────────────────────────────────────────────── + // 4. Вызов Anchor-функции + // ───────────────────────────────────────────────────────────── + return SolanaProgramCaller.callAnchorFunction( + publicKeyB58, + privateKeyB58, + "initialize_user_counter", // имя функции Anchor + Const.PROGRAM_ID_key, + accounts, + serializedArgs + ); + + } catch (Exception e) { + LOG.error("❌ Ошибка вызова initialize_user_counter", e); + return null; + } + } + + + // Для теста + public static void main(String[] args) { +// Const.RPC_URL=Const.DEVNET_RPC_URL; + String sig = InitializeUserCounter.callInitializeUserCounter( + Const.getKeyByName("key1").getPublicKey(), + Const.getKeyByName("key1").getPrivateKey() + ); + ResultChecker.check(sig); + } +} + diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/RegisterUser/RegisterUserStepOne.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/RegisterUser/RegisterUserStepOne.java new file mode 100644 index 0000000..c2bef03 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/RegisterUser/RegisterUserStepOne.java @@ -0,0 +1,104 @@ +package me.shineup.solana.internal.callSolanaFunc.RegisterUser; + +import org.p2p.solanaj.core.AccountMeta; +import org.p2p.solanaj.core.PublicKey; +import org.p2p.solanaj.programs.SystemProgram; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.internal.utils.resultChecker.ResultChecker; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.utils.SolanaProgramCaller; + +import java.nio.charset.StandardCharsets; +import java.util.Arrays; +import java.util.Collections; +import java.util.List; + +/** + * Вызывает Anchor-функцию `register_user_step_one`. !!!!!!!!! ЭТО УСТАРЕЛО И БОЛЬШЕ НЕ НАДО + * + * Функция создаёт нового пользователя: + * - проверяет логин, + * - создаёт PDA-аккаунт пользователя, + * - переводит 0.01 SOL на счёт администрации, + * - сохраняет логин, ID, pubkey и статус 0 в PDA, + * - обновляет счётчик пользователей. + */ + + +public class RegisterUserStepOne { + + private static final Logger LOG = LoggerFactory.getLogger(RegisterUserStepOne.class); + + public static String callRegisterUserStepOne( + String payerPubkeyB58, + String payerPrivkeyB58, + String login, + String userPubkeyB58 + ) { + try { + // 1. Адрес получателя комиссии + PublicKey feeReceiver = Const.ADMIN_FEE_ACCOUNT; + + // 2. PDA-адрес по логину: seed = "u=" + login + String seed = Const.userSeedsPrefix + login; + PublicKey userPda = PublicKey.findProgramAddress( + Collections.singletonList(seed.getBytes(StandardCharsets.UTF_8)), + Const.PROGRAM_ID_key + ).getAddress(); + + // 3. Адрес PDA счётчика пользователей + String counterSeed = Const.USER_COUNTER_SEED; + PublicKey counterPda = PublicKey.findProgramAddress( + Collections.singletonList(counterSeed.getBytes(StandardCharsets.UTF_8)), + Const.PROGRAM_ID_key + ).getAddress(); + + // 4. Сериализация аргументов Anchor + byte[] loginBytes = login.getBytes(StandardCharsets.UTF_8); + byte[] userPubkeyBytes = new PublicKey(userPubkeyB58).toByteArray(); + + byte[] serializedArgs = SolanaProgramCaller.encodeAnchorArgs( + Arrays.asList("string", "pubkey"), + Arrays.asList(loginBytes, userPubkeyBytes) + ); + + // 5. Аккаунты, требуемые для вызова + List accounts = Arrays.asList( + new AccountMeta(new PublicKey(payerPubkeyB58), true, true), // signer + new AccountMeta(counterPda, false, true), // user_counter + new AccountMeta(userPda, false, true), // user_by_login_pda + new AccountMeta(SystemProgram.PROGRAM_ID, false, false), // system_program + new AccountMeta(feeReceiver, false, true) // fee_receiver + ); + + // 6. Вызов Anchor-функции + return SolanaProgramCaller.callAnchorFunction( + payerPubkeyB58, + payerPrivkeyB58, + "register_user_step_one", + Const.PROGRAM_ID_key, + accounts, + serializedArgs + ); + + } catch (Exception e) { + LOG.error("❌ Ошибка вызова register_user_step_one", e); + return null; + } + } + + // Для теста + public static void main(String[] args) { + Const.RPC_URL = Const.LOCAL_RPC_URL; + + String sig = RegisterUserStepOne.callRegisterUserStepOne( + Const.getKeyByName("key1").getPublicKey(), // кто платит + Const.getKeyByName("key1").getPrivateKey(), // + "testlogin", // логин пользователя + Const.getKeyByName("key2").getPublicKey() // публичный ключ пользователя + ); + ResultChecker.check(sig); + } + +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/RegisterUser/RegisterUserWithOneDev.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/RegisterUser/RegisterUserWithOneDev.java new file mode 100644 index 0000000..4d4ec4b --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/callSolanaFunc/RegisterUser/RegisterUserWithOneDev.java @@ -0,0 +1,156 @@ +package me.shineup.solana.internal.callSolanaFunc.RegisterUser; + +import me.shineup.solana.SolanaWrapper; +import me.shineup.solana.internal.readFromSolana.userCounter.UserCounterReader; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.standartActions.keysGenerator.KeyPairBase58; +import me.shineup.solana.internal.utils.SolanaProgramCaller; +import me.shineup.solana.internal.utils.resultChecker.ResultChecker; +import org.p2p.solanaj.core.AccountMeta; +import org.p2p.solanaj.core.PublicKey; +import org.p2p.solanaj.programs.SystemProgram; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.nio.charset.StandardCharsets; +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +import static me.shineup.solana.internal.standartActions.keysGenerator.SolanaKeyGeneratorManual.generateKeyPair; + +/** + * Вызывает Anchor-функцию `register_user_with_one_dev`. + * + * Выполняет регистрацию пользователя и одного устройства. + */ +public class RegisterUserWithOneDev { + + private static final Logger LOG = LoggerFactory.getLogger(RegisterUserWithOneDev.class); + + // Пример вызова + public static void main(String[] args) { +// Const.RPC_URL = Const.LOCAL_RPC_URL; + Const.RPC_URL=Const.DEVNET_RPC_URL; + try { + String publicKey = "2fppzT84GoDqQe2RCxuK2gjZUrMhkzKgVTY6BzfLF9RX"; + String privateKey = "2qTERJQ2EBPsHWNXxhAWW4pBk1beo1BWtifCPsDbrDhw9z5riNLsUUj6BNQ9UbprJq398Zk3Fv21ZGUjRrAU4T73"; + SolanaWrapper.getBalance(Const.getKeyByName("key1").publicKey); +// SolanaWrapper.sendSol(Const.getKeyByName("key1").getPrivateKey(),publicKey, 1000000); + SolanaWrapper.getBalance(publicKey); +// KeyPairBase58 keys = generateKeyPair(); + String sig = RegisterUserWithOneDev.callRegisterUserWithOneDev( + Const.getKeyByName("key1").getPublicKey(), // payer + Const.getKeyByName("key1").getPrivateKey(), // + "testlogin", // логин + Const.getKeyByName("key2").getPublicKey(), // подпись устройства + Const.getKeyByName("key2").getPublicKey() // x25519 + ); + ResultChecker.check(sig); + } catch (Exception e) { + LOG.error(e.getMessage()); + } + + + } + + + public static String callRegisterUserWithOneDev( + String payerPubkeyB58, + String payerPrivkeyB58, + String login, +// String userPubkeyB58, пока userPubkeyB58=payerPubkeyB58 + String deviceSignPubkeyB58, + String deviceX25519PubkeyB58 + ) throws Exception { + + // Адреса + PublicKey payer = new PublicKey(payerPubkeyB58); +/** PublicKey userPub = new PublicKey(userPubkeyB58); + * тут в принципе можно передавать на смарт контракт разные параметры + * и получиться что платит один а создаёт аккаунт на другое имя + * это может актуально при расширении, если дописать додумать и т.д. + * но пока это отключено и кто оплатил тна того и оформляем аккаунт + * пока так работает только за себя и можно отпраавлять транзакции + */ + PublicKey userPub = new PublicKey(payerPubkeyB58); // пока тот кто подписал транзакцию, на него же и аккаунт создаём + + + PublicKey devSignPub = new PublicKey(deviceSignPubkeyB58); + PublicKey devX25519Pub = new PublicKey(deviceX25519PubkeyB58); + + // Читаем текущий ID + long currentId = UserCounterReader.getUserCount(); + long newId = currentId + 1; + + // Счётчик пользователей + + PublicKey counterPda = PublicKey.findProgramAddress( + Arrays.asList("user_counter".getBytes(StandardCharsets.UTF_8)), + Const.PROGRAM_ID_key + ).getAddress(); + + // PDA по логину: ["login=", login] + PublicKey loginPda = PublicKey.findProgramAddress( + Arrays.asList( + "login=".getBytes(StandardCharsets.UTF_8), + login.getBytes(StandardCharsets.UTF_8) + ), + Const.PROGRAM_ID_key + ).getAddress(); + + // 5 возможных PDA по ID: ["userId=", String.valueOf(newId)] + String idSeedStr = String.valueOf(newId); + List idCandidates = new ArrayList<>(); + for (int i = 0; i < 5; i++) { + byte[] prefix = "userId=".getBytes(StandardCharsets.UTF_8); + byte[] id = idSeedStr.getBytes(StandardCharsets.UTF_8); + byte[] seed1 = prefix; + byte[] seed2 = id; + PublicKey pda = PublicKey.findProgramAddress(Arrays.asList(seed1, seed2), Const.PROGRAM_ID_key).getAddress(); + idCandidates.add(pda); + } + + // Комиссионный адрес + PublicKey feeReceiver = Const.ADMIN_FEE_ACCOUNT; + + // Аргументы Anchor + byte[] loginBytes = login.getBytes(StandardCharsets.UTF_8); + byte[] pubkeyBytes = userPub.toByteArray(); + byte[] devSignBytes = devSignPub.toByteArray(); + byte[] devX25519Bytes = devX25519Pub.toByteArray(); + + byte[] serializedArgs = SolanaProgramCaller.encodeAnchorArgs( + Arrays.asList("string", "pubkey", "pubkey", "pubkey"), + Arrays.asList(loginBytes, pubkeyBytes, devSignBytes, devX25519Bytes) + ); + + // Список аккаунтов + List accounts = new ArrayList(); + accounts.add(new AccountMeta(payer, true, true)); + accounts.add(new AccountMeta(counterPda, false, true)); + accounts.add(new AccountMeta(loginPda, false, true)); + + for (PublicKey idPda : idCandidates) { + accounts.add(new AccountMeta(idPda, false, true)); + } + + accounts.add(new AccountMeta(SystemProgram.PROGRAM_ID, false, false)); + accounts.add(new AccountMeta(feeReceiver, false, true)); + + // Вызов Anchor-функции + return SolanaProgramCaller.callAnchorFunction( + payerPubkeyB58, + payerPrivkeyB58, + "register_user_with_one_dev", + Const.PROGRAM_ID_key, + accounts, + serializedArgs + ); + + } + +} + + + diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userById/UserByIdParser.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userById/UserByIdParser.java new file mode 100644 index 0000000..6a06e9d --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userById/UserByIdParser.java @@ -0,0 +1,103 @@ +package me.shineup.solana.internal.readFromSolana.userById; + +import me.shineup.solana.exceptions.SolanaException; +import me.shineup.solana.model.UserById; +import org.bitcoinj.core.Base58; + +import java.nio.ByteBuffer; +import java.nio.ByteOrder; +import java.nio.charset.StandardCharsets; +import java.util.ArrayList; +import java.util.List; + +/** + * Десериализует сырые байты из PDA в объект {@link UserById}. + * + * Сейчас поддерживается только формат 1, но общий метод + * parse(...) готов к расширению версий. + */ +class UserByIdParser { + + /** Точка входа: определяем формат по первым 4 байтам (LE u32). */ + public static UserById parse(byte[] data) throws Exception { + if (data.length < 4) + throw new SolanaException("Недостаточно данных для чтения format_type"); + + int fmt = ByteBuffer.wrap(data, 0, 4) + .order(ByteOrder.LITTLE_ENDIAN) + .getInt(); + + switch (fmt) { + case 1: + return parseFormat1(data); + default: + throw new SolanaException("Неподдерживаемый формат: " + fmt); + } + } + + /** + * Формат 1 (см. Rust-комментарии): + * [0..4] format + * [4..12] id (u64 LE) + * [12] len(login) u8 + * [13..] login + * [...] pubkey 32 байта + * [...] deviceCount u8 + * [...] devices ×65 байт (type + 32 + 32) + */ + private static UserById parseFormat1(byte[] data) throws Exception { + int offset = 4; + + // id + if (data.length < offset + 8) throw new Exception("Мало байт для id"); + long id = ByteBuffer.wrap(data, offset, 8) + .order(ByteOrder.LITTLE_ENDIAN).getLong(); + offset += 8; + + // login + int loginLen = data[offset] & 0xFF; + offset += 1; + if (data.length < offset + loginLen) throw new Exception("Мало байт для login"); + String login = new String(data, offset, loginLen, StandardCharsets.UTF_8); + offset += loginLen; + + // pubkey + if (data.length < offset + 32) throw new Exception("Мало байт для pubkey"); + byte[] pubkeyBytes = new byte[32]; + System.arraycopy(data, offset, pubkeyBytes, 0, 32); + String pubkey58 = Base58.encode(pubkeyBytes); + offset += 32; + + // deviceCount + if (data.length < offset + 1) throw new Exception("Мало байт для deviceCount"); + int devCount = data[offset] & 0xFF; + offset += 1; + + // devices + List devices = new ArrayList<>(); + for (int i = 0; i < devCount; i++) { + if (data.length < offset + 65) + throw new Exception("Мало байт для devices[" + i + "]"); + UserById.DeviceInfo d = new UserById.DeviceInfo(); + d.deviceType = data[offset] & 0xFF; + byte[] devPub = new byte[32]; + byte[] x25519 = new byte[32]; + System.arraycopy(data, offset + 1, devPub, 0, 32); + System.arraycopy(data, offset + 33, x25519, 0, 32); + d.devicePubkey = Base58.encode(devPub); + d.x25519Pubkey = Base58.encode(x25519); + devices.add(d); + offset += 65; + } + + // собираем объект + UserById u = new UserById(); + u.format = 1; + u.id = id; + u.login = login; + u.pubkey = pubkey58; + u.deviceCount = devCount; + u.devices = devices; + return u; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userById/UserByIdReader.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userById/UserByIdReader.java new file mode 100644 index 0000000..67d5a26 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userById/UserByIdReader.java @@ -0,0 +1,48 @@ +package me.shineup.solana.internal.readFromSolana.userById; + +import me.shineup.solana.exceptions.SolanaException_UserNotFound; +import me.shineup.solana.model.UserById; +import me.shineup.solana.internal.utils.reader.PdaReader; +import me.shineup.solana.config.Const; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.nio.charset.StandardCharsets; + +/** + * Читает из PDA данные пользователя по числовому ID + * и десериализует их в {@link UserById}. + */ +public class UserByIdReader { + + private static final Logger LOG = LoggerFactory.getLogger(UserByIdReader.class); + + /** Получить UserById по ID. */ + public static UserById getUserById(long id) throws Exception{ + byte[] seed1 = "userId=".getBytes(StandardCharsets.UTF_8); + byte[] seed2 = Long.toString(id).getBytes(StandardCharsets.UTF_8); + String programId = Const.PROGRAM_ID_str; + + byte[] data = PdaReader.readTwoSeeds(seed1, seed2, programId); + + if (data == null) { + LOG.warn("⚠️ Нет данных в PDA для id={}", id); + throw new SolanaException_UserNotFound(); + } + + return UserByIdParser.parse(data); // Передаём данные в парсер + } + + /** Быстрый тест чтения. */ + public static void main(String[] args) { + long id = 10; // поменять на существующий ID + try { + UserById u = UserByIdReader.getUserById(id); + System.out.println("✅ Найден: " + u); + } catch (SolanaException_UserNotFound e) { + System.out.println("⚠️ Пользователь с id=" + id + " не найден"); + } catch (Exception e) { + e.printStackTrace(); + } + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userByLogin/UserByLoginParser.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userByLogin/UserByLoginParser.java new file mode 100644 index 0000000..9c20a74 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userByLogin/UserByLoginParser.java @@ -0,0 +1,73 @@ +package me.shineup.solana.internal.readFromSolana.userByLogin; + + +import me.shineup.solana.model.UserByLogin; + +/** + * Парсер для десериализации массива байт в объект UserByLogin. + * Поддерживает несколько форматов, определяемых по первым 4 байтам (LE u32). + */ +class UserByLoginParser { + + /** + * Основной метод: определяет формат по первым 4 байтам, вызывает нужный парсер. + * @param data байты, полученные из PDA + * @return объект UserByLogin + * @throws Exception если формат неизвестен или ошибка разбора + */ + public static UserByLogin parse(byte[] data) throws Exception { + if (data.length < 4) throw new Exception("Недостаточно данных для чтения format_type"); + + int format = java.nio.ByteBuffer.wrap(data, 0, 4) + .order(java.nio.ByteOrder.LITTLE_ENDIAN).getInt(); + + switch (format) { + case 1: + return parseFormat1(data); + default: + throw new Exception("Неподдерживаемый формат данных: " + format); + } + } + + /** + * Парсит формат 1: + * [0..4] — формат + * [4] — длина логина (u8) + * [5..X] — логин + * [X..X+8] — id (u64 LE) + * [X..X+32] — pubkey (32 байта) + * [X+32..X+36] — status (u32 LE) + */ + public static UserByLogin parseFormat1(byte[] data) throws Exception { + int offset = 4; // после format_type + + int loginLen = data[offset] & 0xFF; + offset += 1; + + if (data.length < offset + loginLen + 8 + 32 + 4) + throw new Exception("Недостаточно байт для парсинга format 1"); + + String login = new String(data, offset, loginLen, java.nio.charset.StandardCharsets.UTF_8); + offset += loginLen; + + long id = java.nio.ByteBuffer.wrap(data, offset, 8) + .order(java.nio.ByteOrder.LITTLE_ENDIAN).getLong(); + offset += 8; + + byte[] pubkeyBytes = new byte[32]; + System.arraycopy(data, offset, pubkeyBytes, 0, 32); + String pubkey = org.bitcoinj.core.Base58.encode(pubkeyBytes); + offset += 32; + + int status = java.nio.ByteBuffer.wrap(data, offset, 4) + .order(java.nio.ByteOrder.LITTLE_ENDIAN).getInt(); + + UserByLogin result = new UserByLogin(); + result.format = 1; + result.login = login; + result.id = id; + result.pubkey = pubkey; + result.status = status; + return result; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userByLogin/UserByLoginReader.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userByLogin/UserByLoginReader.java new file mode 100644 index 0000000..ff227e4 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userByLogin/UserByLoginReader.java @@ -0,0 +1,58 @@ +package me.shineup.solana.internal.readFromSolana.userByLogin; + + +import me.shineup.solana.exceptions.SolanaException_UserNotFound; +import me.shineup.solana.model.UserByLogin; +import me.shineup.solana.internal.utils.reader.PdaReader; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.config.Const; + +import java.nio.charset.StandardCharsets; + +/** + * Класс для получения объекта UserByLogin по логину из PDA + */ +public class UserByLoginReader { + + private static final Logger LOG = LoggerFactory.getLogger(UserByLoginReader.class); + + /** + * Получает объект UserByLogin по логину + * @param login Логин пользователя (например, "sol_user") + * @return Объект UserByLogin или null, если данных нет + * @throws Exception если ошибка соединения или парсинга + */ + public static UserByLogin getUserByLogin(String login) throws Exception { + byte[] seed1 = "login=".getBytes(StandardCharsets.UTF_8); + byte[] seed2 = login.getBytes(StandardCharsets.UTF_8); + String programId = Const.PROGRAM_ID_str; + + byte[] data = PdaReader.readTwoSeeds(seed1, seed2, programId); + + if (data == null) { + LOG.warn("⚠️ Нет данных в PDA для логина '{}'", login); + throw new SolanaException_UserNotFound(); + } + + // Передаём данные в парсер + return UserByLoginParser.parse(data); + } + + /** + * Тестовый запуск чтения информации о пользователе + */ + public static void main(String[] args) { + String login = "testlogin3"; // замените на нужный логин + try { + UserByLogin user = UserByLoginReader.getUserByLogin(login); + System.out.println("✅ Найден пользователь: " + login); + } catch (SolanaException_UserNotFound e) { + System.out.println("⚠️ Пользователь с id=" + login + " не найден"); + } catch (Exception e) { + System.err.println("❌ Ошибка при получении информации о пользователе: " + e.getMessage()); + e.printStackTrace(); + } + } +} + diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userCounter/UserCounterReader.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userCounter/UserCounterReader.java new file mode 100644 index 0000000..addb2a6 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/readFromSolana/userCounter/UserCounterReader.java @@ -0,0 +1,60 @@ +package me.shineup.solana.internal.readFromSolana.userCounter; + + + +import me.shineup.solana.exceptions.SolanaException; +import me.shineup.solana.internal.utils.reader.PdaReader; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.config.Const; + +import java.nio.ByteBuffer; +import java.nio.ByteOrder; + +/** + * Утилита для получения значения счётчика пользователей из PDA. + */ +public class UserCounterReader { + + private static final Logger LOG = LoggerFactory.getLogger(UserCounterReader.class); + + /** + * Считывает текущее количество пользователей из PDA "user_counter". + * + * @return количество пользователей (long), либо -1 если нет данных + * @throws Exception при ошибке подключения или чтения + */ + public static long getUserCount() throws Exception { + String seed = "user_counter"; + String programId = Const.PROGRAM_ID_str; + + byte[] data = PdaReader.readOneSeed(seed, programId); + if (data == null || data.length < 8) { + throw new SolanaException("⚠️ Не удалось прочитать счётчик пользователей — PDA пуст или слишком короткий"); // этого не должно случаться + } + + // Считываем первые 8 байт как u64 (Little Endian) + ByteBuffer buffer = ByteBuffer.wrap(data).order(ByteOrder.LITTLE_ENDIAN); + long count = buffer.getLong(); + + LOG.debug("👥 На данный момент в системе зарегистрировано: {} пользователей", count); + return count; + } + + /** + * Тестовый запуск + */ + public static void main(String[] args) { + try { + long count = getUserCount(); + if (count >= 0) { + System.out.println("✅ 👥 Количество пользователей системы на данный момент: " + count); + } else { + System.out.println("⚠️ Счётчик не найден или пуст"); + } + } catch (Exception e) { + System.err.println("❌ Ошибка при чтении счётчика пользователей: " + e.getMessage()); + e.printStackTrace(); + } + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/airDrops/SolanaAirdrop.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/airDrops/SolanaAirdrop.java new file mode 100644 index 0000000..22bb9e5 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/airDrops/SolanaAirdrop.java @@ -0,0 +1,68 @@ +package me.shineup.solana.internal.standartActions.airDrops; + + +import com.google.gson.JsonObject; +import com.google.gson.JsonParser; +import me.shineup.solana.exceptions.SolanaException_LibLogic; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.utils.SolanaRpcClient; + + +public class SolanaAirdrop { + private static final Logger log = LoggerFactory.getLogger(SolanaAirdrop.class); + + + // + public static void main(String[] args) { + + // Кол-во лампортов (1 SOL = 1_000_000_000 лампортов) + long lamports = 1_000_000_000L; + try { + + // Вызываем airdrop + String trx = SolanaAirdrop.requestAirdrop(Const.getKeyByName("key1").getPublicKey(), lamports); + + + // Вывод результата + log.info("Баланс должен скоро обновиться. " + trx); + } catch (Exception e) { + log.error("Airdrop не удался." + e.getMessage()); + } + } + + /** запрашивает AirDrop на счёт */ + public static String requestAirdrop(String publicKey, long lamports) throws Exception{ +// String requestJson = """ +// { +// "jsonrpc": "2.0", +// "id": 1, +// "method": "requestAirdrop", +// "params": ["%s", %d] +// } +// """.formatted(publicKey, lamports); + + String requestJson = String.format( + "{\n" + + " \"jsonrpc\": \"2.0\",\n" + + " \"id\": 1,\n" + + " \"method\": \"requestAirdrop\",\n" + + " \"params\": [\"%s\", %d]\n" + + "}", publicKey, lamports); + + + String response = SolanaRpcClient.getInstance().sendRequest(requestJson); + + JsonObject json = JsonParser.parseString(response).getAsJsonObject(); + + if (json.has("result")) { + String txSignature = json.get("result").getAsString(); + log.info("✅ Airdrop успешно запрошен. Tx: " + txSignature); + return txSignature; + } + throw new SolanaException_LibLogic("Неизвестный ответ: " + response); + } + + +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/balanse/SolanaBalanceChecker.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/balanse/SolanaBalanceChecker.java new file mode 100644 index 0000000..31d2c0b --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/balanse/SolanaBalanceChecker.java @@ -0,0 +1,72 @@ +package me.shineup.solana.internal.standartActions.balanse; + +import com.google.gson.Gson; +import com.google.gson.JsonObject; +import com.google.gson.JsonParser; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.utils.SolanaRpcClient; + +public class SolanaBalanceChecker { + + // для теста + public static void main(String[] args) { + try { + +// Const.RPC_URL = Const.LOCAL_RPC_URL; + Const.RPC_URL = "https://api.devnet.solana.com"; + //Long balance1 = SolanaBalanceChecker.getBalance(Const.getKeyByName("key1").getPublicKey()); + Long balance1 = SolanaBalanceChecker.getBalance("HMww7YSVfwVm4i8sugqj7wyH26dqzHykzv3wzWwzEvPA"); + Long balance2 = SolanaBalanceChecker.getBalance(Const.getKeyByName("key2").getPublicKey()); + Long balance3 = SolanaBalanceChecker.getBalance(Const.getKeyByName("key3").getPublicKey()); + } catch (Exception e) { + log.error( e.getMessage()); + } + } + + private static final Logger log = LoggerFactory.getLogger(SolanaBalanceChecker.class); + private final Gson gson = new Gson(); + + /** показывает баланс счёта */ + public static long getBalance(String publicKey) throws Exception { + + String requestJson = String.format( + "{\n" + + " \"jsonrpc\": \"2.0\",\n" + + " \"id\": 1,\n" + + " \"method\": \"getBalance\",\n" + + " \"params\": [\"%s\"]\n" + + "}", publicKey); +// try { + String responseJson = SolanaRpcClient.getInstance().sendRequest(requestJson); + + // Парсим строку JSON в дерево объектов + JsonObject root = JsonParser.parseString(responseJson).getAsJsonObject(); + +// if (root.has("error")) { +// log.error("❌ Не удалось получить баланс для " + publicKey); +// log.error("Ошибка от RPC: " + root.get("error")); +// new SolanaLibLogicException() ; +// } + + log.debug("📥 Ответ от RPC: " + responseJson); + + long balance = root.getAsJsonObject("result").get("value").getAsLong(); + double sol = balance / 1_000_000_000.0; + + log.info("✅ Баланс кошелька " + Const.identifyKey(publicKey) + ": " + Const.lamportsToSol(balance));// + " SOL или лампортов " + balance ); + + return balance; + +// } catch (Exception e) { +// log.error("❌ Не удалось получить баланс для " + publicKey); +// log.error("Ошибка при получении баланса: " + e.getMessage()); +// log.error("Ошибка при получении баланса: " + e.getStackTrace()[0] ); +// log.error("Ошибка при получении баланса: " + e.getStackTrace()[1] ); +// log.error("Ошибка при получении баланса: " + e.getStackTrace()[2] ); +// return -1; +// } + + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/keysGenerator/KeyPairBase58.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/keysGenerator/KeyPairBase58.java new file mode 100644 index 0000000..3eef7b4 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/keysGenerator/KeyPairBase58.java @@ -0,0 +1,26 @@ +package me.shineup.solana.internal.standartActions.keysGenerator; + + + +/** + * Объект, представляющий пару ключей Solana в формате Base58: + * - публичный ключ (32 байта) + * - приватный ключ (64 байта, включает публичный) + */ +public class KeyPairBase58 { + public final String publicKey; + public final String privateKey; + + public KeyPairBase58(String publicKey, String privateKey) { + this.publicKey = publicKey; + this.privateKey = privateKey; + } + + @Override + public String toString() { + return "KeyPairBase58{\n" + + " publicKey='" + publicKey + "',\n" + + " privateKey='" + privateKey + "'\n" + + '}'; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/keysGenerator/SolanaKeyGeneratorManual.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/keysGenerator/SolanaKeyGeneratorManual.java new file mode 100644 index 0000000..f9cdd46 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/keysGenerator/SolanaKeyGeneratorManual.java @@ -0,0 +1,101 @@ +package me.shineup.solana.internal.standartActions.keysGenerator; + + + +import org.bouncycastle.crypto.params.Ed25519PrivateKeyParameters; +import org.bouncycastle.crypto.generators.Ed25519KeyPairGenerator; +import org.bouncycastle.crypto.params.Ed25519PublicKeyParameters; +import org.bouncycastle.crypto.KeyGenerationParameters; +import org.bouncycastle.crypto.AsymmetricCipherKeyPair; + +import java.security.SecureRandom; + +public class SolanaKeyGeneratorManual { + + // ✅ Новый метод: возвращает пару ключей (в Base58) + public static KeyPairBase58 generateKeyPair() { + // Генератор ключей Ed25519 + Ed25519KeyPairGenerator keyGen = new Ed25519KeyPairGenerator(); + keyGen.init(new KeyGenerationParameters(new SecureRandom(), 256)); + AsymmetricCipherKeyPair keyPair = keyGen.generateKeyPair(); + + Ed25519PrivateKeyParameters privateKeyParams = (Ed25519PrivateKeyParameters) keyPair.getPrivate(); + Ed25519PublicKeyParameters publicKeyParams = (Ed25519PublicKeyParameters) keyPair.getPublic(); + + byte[] privateKey = privateKeyParams.getEncoded(); // 32 байта + byte[] publicKey = publicKeyParams.getEncoded(); // 32 байта + + byte[] solanaSecretKey = new byte[64]; + System.arraycopy(privateKey, 0, solanaSecretKey, 0, 32); + System.arraycopy(publicKey, 0, solanaSecretKey, 32, 32); + + return new KeyPairBase58( + Base58.encode(publicKey), + Base58.encode(solanaSecretKey) + ); + } + + + + public static void main(String[] args) { + KeyPairBase58 keys = generateKeyPair(); + + System.out.println("✅ Сгенерирован новый кошелёк Solana:"); + System.out.println("Публичный ключ (Base58): " + keys.publicKey); + System.out.println("Приватный ключ (Base58, 64 байта): " + keys.privateKey); + + System.out.println(); + System.out.println("String publicKey = \"" + keys.publicKey + "\";"); + System.out.println("String privateKey = \"" + keys.privateKey + "\";"); + } + +// +// // 👇 Встроенная реализация Base58 (без bitcoinj) + static class Base58 { + private static final char[] ALPHABET = "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz".toCharArray(); + private static final int BASE_58 = ALPHABET.length; + + public static String encode(byte[] input) { + if (input.length == 0) return ""; + + // Count leading zeros. + int zeros = 0; + while (zeros < input.length && input[zeros] == 0) { + ++zeros; + } + + // Convert base-256 digits to base-58 digits (plus conversion to ASCII characters) + int size = input.length * 2; + int[] encoded = new int[size]; + int length = 0; + + for (byte b : input) { + int carry = b & 0xFF; + int i = 0; + for (int j = size - 1; (carry != 0 || i < length) && j >= 0; j--, i++) { + carry += 256 * encoded[j]; + encoded[j] = carry % BASE_58; + carry /= BASE_58; + } + length = i; + } + + // Skip leading zeros in encoded. + int encodedStart = size - length; + while (encodedStart < size && encoded[encodedStart] == 0) { + encodedStart++; + } + + // Translate the result into a string. + StringBuilder result = new StringBuilder(zeros + size - encodedStart); + for (int i = 0; i < zeros; i++) { + result.append('1'); + } + for (int i = encodedStart; i < size; i++) { + result.append(ALPHABET[encoded[i]]); + } + + return result.toString(); + } + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/transfer/SolanaTransfer.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/transfer/SolanaTransfer.java new file mode 100644 index 0000000..0f1d738 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/standartActions/transfer/SolanaTransfer.java @@ -0,0 +1,73 @@ +package me.shineup.solana.internal.standartActions.transfer; + +import me.shineup.solana.exceptions.SolanaException_RpcConnection; +import me.shineup.solana.internal.standartActions.balanse.SolanaBalanceChecker; +import org.p2p.solanaj.core.Account; +import org.p2p.solanaj.core.PublicKey; +import org.p2p.solanaj.core.Transaction; +import org.p2p.solanaj.rpc.RpcClient; +import org.p2p.solanaj.rpc.RpcException; +import org.p2p.solanaj.programs.SystemProgram; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.utils.KeyUtils; + +import static me.shineup.solana.internal.utils.SolanaProgramCaller.getLatestBlockhash; +import static me.shineup.solana.internal.utils.SolanaProgramCaller.sendTransactionWithBlockhash; + +public class SolanaTransfer { + private static final Logger log = LoggerFactory.getLogger(SolanaTransfer.class); + + public static void main(String[] args) { + try { + Const.RPC_URL = Const.LOCAL_RPC_URL; + SolanaBalanceChecker.getBalance(Const.getKeyByName("key1").getPublicKey()); + SolanaTransfer.sendSol(Const.getKeyByName("key1").getPrivateKey(), Const.getKeyByName("key2").getPublicKey(),100_000_000); + SolanaBalanceChecker.getBalance(Const.getKeyByName("key1").getPublicKey()); + SolanaBalanceChecker.getBalance(Const.getKeyByName("key2").getPublicKey()); + } catch (Exception e) { + log.error( e.getMessage()); + } + } + + + + /** + * Переводит lamports (1 SOL = 1_000_000_000 лампортов) + * @param fromBase58Secret - приватный ключ в виде массива из 64 байт + * @param toAddressBase58 - публичный ключ получателя + * @param lamports - сумма в лампортах + */ + public static String sendSol(String fromBase58Secret, String toAddressBase58, long lamports) throws Exception{ + try { + + RpcClient rpc = new RpcClient(Const.RPC_URL); + + byte[] senderSecretKey58 = KeyUtils.base58ToBytes(fromBase58Secret); + + // Загружаем отправителя + Account from = new Account(senderSecretKey58); + PublicKey to = new PublicKey(toAddressBase58); + + // Создаём транзакцию + Transaction transaction = new Transaction(); + transaction.addInstruction( + SystemProgram.transfer(from.getPublicKey(), to, lamports) + ); + + // Получаем blockhash + String recentBlockhash = getLatestBlockhash(rpc); + + // отправляем транзакцию + String signature = sendTransactionWithBlockhash(rpc, transaction, recentBlockhash, from); + + + log.info("✅ Перевод " + Const.lamportsToSol(lamports) + " отправлен с " + Const.identifyKey(fromBase58Secret) + " на " + Const.identifyKey(toAddressBase58) + " . Подпись: " + signature); + return signature; + } catch (RpcException e) { + throw new SolanaException_RpcConnection("❌ Ошибка RPC: " + e.getMessage()); + } + + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/KeyPair.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/KeyPair.java new file mode 100644 index 0000000..f47a04f --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/KeyPair.java @@ -0,0 +1,55 @@ +package me.shineup.solana.internal.utils; + +public class KeyPair { + + public final String name; + public final String publicKey; + public final String privateKey; + + public KeyPair(String name, String publicKey, String privateKey) { + this.name = name; + this.publicKey = publicKey; + this.privateKey = privateKey; + } + + public String getName() { + return name; + } + + public String getPublicKey() { + return publicKey; + } + + public String getPrivateKey() { + return privateKey; + } + + @Override + public String toString() { + return "KeyPair{" + + "name='" + name + '\'' + + ", publicKey='" + publicKey + '\'' + + ", privateKey='" + privateKey + '\'' + + '}'; + } + + @Override + public boolean equals(Object o) { + if (this == o) return true; + if (o == null || getClass() != o.getClass()) return false; + + KeyPair keyPair = (KeyPair) o; + + if (!name.equals(keyPair.name)) return false; + if (!publicKey.equals(keyPair.publicKey)) return false; + return privateKey.equals(keyPair.privateKey); + } + + @Override + public int hashCode() { + int result = name.hashCode(); + result = 31 * result + publicKey.hashCode(); + result = 31 * result + privateKey.hashCode(); + return result; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/KeyUtils.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/KeyUtils.java new file mode 100644 index 0000000..15eaf54 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/KeyUtils.java @@ -0,0 +1,39 @@ +package me.shineup.solana.internal.utils; + + +import java.util.Arrays; + +public class KeyUtils { + + private static final String BASE58_ALPHABET = "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz"; + + public static byte[] base58ToBytes(String base58) { + // Простейший декодер Base58 + int[] indexes = new int[128]; + Arrays.fill(indexes, -1); + for (int i = 0; i < BASE58_ALPHABET.length(); i++) { + indexes[BASE58_ALPHABET.charAt(i)] = i; + } + + byte[] input = new byte[base58.length()]; + for (int i = 0; i < base58.length(); i++) { + input[i] = (byte) indexes[base58.charAt(i)]; + } + + byte[] result = new byte[64]; // длина ключа + int length = 0; + + for (byte b : input) { + int carry = b & 0xFF; + int i = 0; + for (int j = result.length - 1; (carry != 0 || i < length) && j >= 0; j--, i++) { + carry += 58 * (result[j] & 0xFF); + result[j] = (byte) (carry % 256); + carry /= 256; + } + length = i; + } + + return result; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/SolanaProgramCaller.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/SolanaProgramCaller.java new file mode 100644 index 0000000..e8ac18f --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/SolanaProgramCaller.java @@ -0,0 +1,378 @@ +package me.shineup.solana.internal.utils; + +import me.shineup.solana.config.Const; +import me.shineup.solana.exceptions.SolanaException; +import me.shineup.solana.exceptions.SolanaException_InProgram; +import me.shineup.solana.exceptions.SolanaException_InsufficientFundsForFee; +import me.shineup.solana.exceptions.SolanaException_RpcConnection; +import org.bitcoinj.core.Base58; +import org.p2p.solanaj.core.*; +import org.p2p.solanaj.rpc.RpcClient; +import org.p2p.solanaj.rpc.RpcException; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +import java.io.IOException; +import java.net.ConnectException; +import java.net.SocketTimeoutException; +import java.nio.ByteBuffer; +import java.nio.ByteOrder; +import java.nio.charset.StandardCharsets; +import java.security.MessageDigest; +import java.util.*; + +import java.util.Map; +import java.util.List; +import java.util.regex.Matcher; +import java.util.regex.Pattern; + + +public class SolanaProgramCaller { + + // Логгер для вывода отладочной информации + private static final Logger log = LoggerFactory.getLogger(SolanaProgramCaller.class); + + /** + * Универсальный метод вызова Anchor-функции на Solana. + * + * @param publicKeyB58 Публичный ключ вызывающего аккаунта в формате base58 + * @param privateKeyB58 Приватный ключ в формате base58 (только первые 32 байта или полный 64-байтный seed) + * @param functionName Имя функции Anchor (например: "register_user") + * @param programId Адрес Anchor-программы, в которую делается вызов + * @param accounts Список аккаунтов, необходимых для вызова (AccountMeta) + * @param serializedArgs Сериализованные аргументы вызова (без дискриминатора) + * @return base58-подпись транзакции или null при ошибке + */ + public static String callAnchorFunction( + String publicKeyB58, + String privateKeyB58, + String functionName, + PublicKey programId, + List accounts, + byte[] serializedArgs + ) throws Exception { + + // Создаём RPC клиент для взаимодействия с Solana (через URL, заданный в конфиге) + RpcClient rpc = new RpcClient(Const.RPC_URL); + + // Декодируем публичный ключ вызывающего аккаунта из Base58 строки + PublicKey pubKey = new PublicKey(publicKeyB58); + + // Декодируем приватный ключ из Base58 строки (ожидается 32 байта seed или 64 байта полного ключа) + byte[] priv = Base58.decode(privateKeyB58); + + // Если длина приватного ключа всего 32 байта (seed), то дополняем его 32 байтами публичного ключа + if (priv.length == 32) { + byte[] full = new byte[64]; // 64-байтный ключ для Ed25519 (32 priv + 32 pub) + System.arraycopy(priv, 0, full, 0, 32); // копируем 32 байта приватного ключа + System.arraycopy(pubKey.toByteArray(), 0, full, 32, 32); // дописываем 32 байта публичного ключа + priv = full; // теперь priv содержит полный 64-байтный ключ + } + + // Создаём объект аккаунта для подписи транзакции + Account signer = new Account(priv); + + // Логируем вызов функции и аккаунт, от имени которого он будет происходить + log.debug("Вызов Anchor-функции '{}' от аккаунта {}", functionName, Const.identifyKey(pubKey.toBase58())); + + // Генерируем Anchor-дискриминатор — первые 8 байт SHA-256 хеша строки "global:имя_функции" + byte[] discriminator = MessageDigest + .getInstance("SHA-256") // используем SHA-256 + .digest(("global:" + functionName).getBytes(StandardCharsets.UTF_8)); // хешируем + discriminator = Arrays.copyOf(discriminator, 8); // берём только первые 8 байт (Anchor-дискриминатор) + + // Формируем payload: сначала дискриминатор, потом сериализованные параметры + byte[] data = new byte[discriminator.length + serializedArgs.length]; + System.arraycopy(discriminator, 0, data, 0, discriminator.length); // копируем дискриминатор + System.arraycopy(serializedArgs, 0, data, discriminator.length, serializedArgs.length); // копируем параметры + + // Создаём инструкцию Solana-транзакции: + // - указываем ID программы (куда отправляем) + // - список аккаунтов, задействованных в вызове (AccountMeta) + // - подготовленные данные (дискриминатор + параметры) + TransactionInstruction instruction = new TransactionInstruction( + programId, // ID Solana-программы (Anchor) + accounts, // список аккаунтов + data // бинарные данные вызова + ); + + // Создаём Solana транзакцию и добавляем в неё инструкцию + Transaction tx = new Transaction(); + tx.addInstruction(instruction); + + + // Получаем blockhash + String recentBlockhash = getLatestBlockhash(rpc); + + // отправляем транзакцию + String sig = sendTransactionWithBlockhash(rpc, tx, recentBlockhash, signer); + + log.info("✅ Tx отправлена: {}", sig); + + return sig; // возвращаем base58-подпись транзакции + + } + + // Вспомогательная склейка массивов + public static byte[] concat(byte[]... arrays) { + int length = 0; + for (byte[] a : arrays) length += a.length; + byte[] result = new byte[length]; + int pos = 0; + for (byte[] a : arrays) { + System.arraycopy(a, 0, result, pos, a.length); + pos += a.length; + } + return result; + } + + + /** + * Сериализует аргументы Anchor/Borsh. + * Сейчас поддерживает: string, pubkey. + * - string → 4-байтная длина (LE) + UTF-8 + * - pubkey → 32 байта + */ + public static byte[] encodeAnchorArgs(List types, List values) { + if (types.size() != values.size()) + throw new IllegalArgumentException("Количество типов и значений должно совпадать"); + + ByteBuffer buf = ByteBuffer.allocate(1024).order(ByteOrder.LITTLE_ENDIAN); + + for (int i = 0; i < types.size(); i++) { + String t = types.get(i); + Object v = values.get(i); + + + switch (t) { + case "string": { + byte[] s = (byte[]) v; // уже UTF-8 байты + buf.putInt(s.length); // 4-byte length LE + buf.put(s); + break; + } + case "pubkey": { + byte[] pk = (byte[]) v; + if (pk.length != 32) + throw new IllegalArgumentException("Pubkey должен быть 32 байта"); + buf.put(pk); + break; + } + default: + throw new UnsupportedOperationException("Неизвестный тип: " + t); + } + } + byte[] out = new byte[buf.position()]; + buf.flip(); + buf.get(out); + return out; + } + + + /*------------------------------------------------------------ + * 1. Получаем актуальный blockhash + *-----------------------------------------------------------*/ + @SuppressWarnings("unchecked") + public static String getLatestBlockhash(RpcClient rpc) throws Exception { + + Map commitment = new HashMap(); + commitment.put("commitment", "finalized"); + + Map bhJson; + try { + bhJson = rpc.call( + "getLatestBlockhash", + Collections.singletonList(commitment), + Map.class + ); + // остальной код + } catch (Exception e) { + throw new SolanaException_RpcConnection("Неудаётся соедениться при попытке получить blockhash: " + e.getMessage()); + } + + // --- проверка ответа + поддержка обоих форматов --- + if (bhJson == null) { + throw new SolanaException("При получении номера последнего блока RPC вернул null"); + } + if (bhJson.containsKey("error")) { + throw new SolanaException("При получении номера последнего блока: RPC error: " + bhJson.get("error")); + } + + Map value; + if (bhJson.containsKey("result")) { + Map result = (Map) bhJson.get("result"); + value = (Map) result.get("value"); + } else { + value = (Map) bhJson.get("value"); + } + + if (value == null) { + throw new SolanaException("При получении номера последнего блока: Поле \"value\" отсутствует. Ответ: " + bhJson); + } + String recentBlockhash = (String) value.get("blockhash"); + if (recentBlockhash == null) { + throw new SolanaException("При получении номера последнего блока: Поле \"blockhash\" отсутствует. Ответ: " + bhJson); + } + + return recentBlockhash; + } + + /*------------------------------------------------------------ + * 2. Подписываем и отправляем транзакцию, зная blockhash + *-----------------------------------------------------------*/ + public static String sendTransactionWithBlockhash( + RpcClient rpc, + Transaction tx, + String recentBlockhash, + Account... signers + ) throws Exception { + + /* ---- подставляем hash ---- */ + tx.setRecentBlockHash(recentBlockhash); + + /* ---- подписываем ---- */ + if (signers.length == 0) { + throw new IllegalArgumentException("Нужен хотя бы один подписант"); + } else if (signers.length == 1) { + tx.sign(signers[0]); + } else { + tx.sign(Arrays.asList(signers)); + } + + +/** // можно сериализовывать в базу 64 и в базу 58 // пока оставил 58 +// // ---- сериализация в база 64 ---- +// String b64Tx = Base64.getEncoder() +// .encodeToString(tx.serialize()); +// +// // ---- отправка ---- +// Map cfg = new HashMap(); +// cfg.put("skipPreflight", Boolean.FALSE); //Boolean.TRUE); +// +// cfg.put("encoding", "base64"); // ← ключевое добавление! +// +// +// return rpc.call( +// "sendTransaction", +// Arrays.asList(b64Tx, cfg), +// String.class +// ); + */ + + /* ---- сериализация и отправка в база 58 ---- */ + byte[] txBytes = tx.serialize(); + String base58EncodedTx = Base58.encode(txBytes); + + Map cfg = new HashMap<>(); + cfg.put("skipPreflight", false); // если false - то транзакцию сразу проверяют на ошибки + + try { + return rpc.call( + "sendTransaction", + Arrays.asList(base58EncodedTx, cfg), + String.class + ); + } catch (RpcException e) { + /* ---------- низкоуровневые сетевые ошибки (не достучались до RPC) */ + Throwable cause = e.getCause(); + if (cause instanceof ConnectException) { + throw new SolanaException_RpcConnection("Не удалось подключиться к RPC-узлу: " + cause.getMessage()); + } + if (cause instanceof SocketTimeoutException) { + throw new SolanaException_RpcConnection("Время ожидания ответа RPC истекло"); + } + + /* ---------- парсинг сообщения об ошибке от RPC/симуляции -------- */ + String err = Objects.toString(e.getMessage(), ""); + + /* 1) Стандартные InstructionError-ы */ + if (err.contains("\"InstructionError\"")) { + // Чаще всего приходит: "InstructionError": [0,"InsufficientFunds"] + if (err.contains("InsufficientFunds")) { + throw new SolanaException_InsufficientFundsForFee(); + } +// if (err.contains("AccountNotFound")) { +// throw new SolanaException_AccountNotFound(); +// } +// if (err.contains("InvalidAccountData")) { +// throw new SolanaException_InvalidAccountData(); +// } +// if (err.contains("InstructionMissing")) { +// throw new SolanaException_InstructionMissing(); +// } +// if (err.contains("UninitializedAccount")) { +// throw new SolanaException_UninitializedAccount(); +// } +// if (err.contains("IncorrectProgramId")) { +// throw new SolanaException_IncorrectProgramId(); +// } + } + + /* ) Отдельная проверка на ошибку дебета с пустого аккаунта */ + if (err.contains("insufficient funds for rent") // или такое сообщение если акаунт пуст + || err.matches("(?i).*custom program error: 0x0*1\\b.*")) { // или такое если денег не хватает + throw new SolanaException_InsufficientFundsForFee(); + } + + /* 2) Кастомная ошибка из твоей программы: "custom program error: 0xXXXX" */ + Pattern p = Pattern.compile("custom program error: (0x[0-9a-fA-F]+)"); + Matcher m = p.matcher(err); + if (m.find()) { + String hexCode = m.group(1); + int decimalCode = Integer.parseInt(hexCode.substring(2), 16); + throw new SolanaException_InProgram(decimalCode); + } + + /* 3) Прочие симуляционные ошибки (можно расширять по вкусу) */ +// if (err.contains("Transaction simulation failed")) { +// throw new SolanaException( +// "Симуляция транзакции завершилась ошибкой: " + err +// ); +// } + + /* 4) Всё остальное – оборачиваем как «неизвестную» */ + throw new SolanaException("RPC-ошибка: " + err, e); + + } + } + + +} + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/SolanaRpcClient.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/SolanaRpcClient.java new file mode 100644 index 0000000..6709444 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/SolanaRpcClient.java @@ -0,0 +1,86 @@ +package me.shineup.solana.internal.utils; + +import me.shineup.solana.config.Const; +import me.shineup.solana.exceptions.SolanaException_RpcConnection; +import me.shineup.solana.exceptions.SolanaErrorHandler; +import okhttp3.MediaType; +import okhttp3.OkHttpClient; +import okhttp3.Request; +import okhttp3.RequestBody; +import okhttp3.Response; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import java.io.IOException; + +public class SolanaRpcClient { + + private static final Logger log = LoggerFactory.getLogger(SolanaRpcClient.class); + private static SolanaRpcClient instance; + + private final OkHttpClient httpClient; + private final String rpcUrl; + + private static final MediaType JSON = MediaType.get("application/json; charset=utf-8"); + + private SolanaRpcClient(String rpcUrl) { + this.rpcUrl = rpcUrl; + this.httpClient = new OkHttpClient(); + } + + public static SolanaRpcClient getInstance() { + if (instance == null) { + instance = new SolanaRpcClient(Const.RPC_URL); + } + return instance; + } + + public static SolanaRpcClient withCustomUrl(String customUrl) { + instance = new SolanaRpcClient(customUrl); + return instance; + } + + /** + * Выполняет JSON-RPC запрос и возвращает ответ как строку. + * В случае ошибки — выбрасывает RpcConnectionException. + */ + public String sendRequest(String jsonRequest) throws SolanaException_RpcConnection { + log.debug("📤 Отправка RPC-запроса: {}", jsonRequest); + + RequestBody body = RequestBody.create(jsonRequest, JSON); + + Request request = new Request.Builder() + .url(rpcUrl) + .post(body) + .addHeader("Content-Type", "application/json") + .build(); + + try (Response response = httpClient.newCall(request).execute()) { + if (!response.isSuccessful()) { + log.error("❌ RPC ответ с ошибкой: {} {}", response.code(), response.message()); + throw new SolanaException_RpcConnection("RPC ответ с ошибкой: " + response.code() + " " + response.message()); + } + + if (response.body() == null) { + log.error("❌ RPC вернул пустое тело"); + throw new SolanaException_RpcConnection("Пустой ответ от RPC"); + } + + String responseText = response.body().string(); + log.debug("📥 Получен ответ от RPC: {}", responseText); + + // ✅ Обработка ошибок, если в ответе есть error + if (responseText.contains("\"error\"")) { + SolanaErrorHandler.handleRpcJsonError(responseText); + } + + return responseText; + + } catch (IOException e) { + log.error("❌ Ошибка подключения к RPC: {}", e.toString()); + throw new SolanaException_RpcConnection("Ошибка подключения к RPC" + e.toString()); + } catch (Exception e) { + log.error("❌ Непредвиденная ошибка при вызове RPC: {}", e.toString()); + throw new SolanaException_RpcConnection("Непредвиденная ошибка RPC" + e.toString()); + } + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/jsonrpc/JsonRpcRequest.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/jsonrpc/JsonRpcRequest.java new file mode 100644 index 0000000..66310a8 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/jsonrpc/JsonRpcRequest.java @@ -0,0 +1,19 @@ +package me.shineup.solana.internal.utils.jsonrpc; + + + +import java.util.List; + + +/** странная фигня котыль который пришлось добавить при переходе на старую библиотеку solanaj для джава 8 */ +public class JsonRpcRequest { + public String jsonrpc = "2.0"; + public String method; + public List params; + public int id = 1; + + public JsonRpcRequest(String method, List params) { + this.method = method; + this.params = params; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/reader/PdaReader.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/reader/PdaReader.java new file mode 100644 index 0000000..1351eea --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/reader/PdaReader.java @@ -0,0 +1,157 @@ +package me.shineup.solana.internal.utils.reader; + + +import me.shineup.solana.exceptions.SolanaException_RpcConnection; +import me.shineup.solana.exceptions.SolanaException; +import org.p2p.solanaj.core.PublicKey; +import org.p2p.solanaj.rpc.RpcClient; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.config.Const; + +import java.nio.charset.StandardCharsets; +import java.util.*; + +/** + * Чтение произвольного PDA-аккаунта в актуальных кластерах Solana (v1.18+). + * + * ─────────────────────────────────────────────────────────────────────────────── + * ▸ Solana HTTP RPC: getAccountInfo + * + * { + * "jsonrpc":"2.0", + * "id":1, + * "result":{ + * "context":{"slot":123456}, + * "value":{ + * "data":[ // 0 - base64-строка, 1 - "base64" + * "BASE64_STRING", + * "base64" + * ], + * "executable":false, + * "lamports":2039280, + * "owner":"BPFLoaderUpgradeab1e11111111111111111111111", + * "rentEpoch":"18446744073709551615" // - строка-u64 (v1.18+) + * } + * } + * } + * + * ▸ Мы читаем только value.data[0] (Base64) – остальное не трогаем, + * поэтому не важен тип rentEpoch (строка или число). + * ─────────────────────────────────────────────────────────────────────────────── + */ +public final class PdaReader { + + /* --------------------------------------------------------------------- */ + /* CONFIG & LOG */ + /* --------------------------------------------------------------------- */ + private static final Logger LOG = LoggerFactory.getLogger(PdaReader.class); + private static final RpcClient RPC = new RpcClient(Const.RPC_URL); // один клиент на класс + + private PdaReader() {} // запретить new + + /* --------------------------------------------------------------------- */ + /* PUBLIC API */ + /* --------------------------------------------------------------------- */ + + /** Чтение PDA, вычисленного из одного текстового сида. */ + public static byte[] readOneSeed(String seed, String programId) throws Exception { + PublicKey pda = derivePda(Collections.singletonList(seed.getBytes(StandardCharsets.UTF_8)), + programId); + return fetchAccountData(pda); + } + + /** Чтение PDA, вычисленного из двух произвольных сид-массивов. */ + public static byte[] readTwoSeeds(byte[] seed1, byte[] seed2, String programId) throws Exception { + PublicKey pda = derivePda(Arrays.asList(seed1, seed2), programId); + return fetchAccountData(pda); + } + + /* --------------------------------------------------------------------- */ + /* INTERNALS */ + /* --------------------------------------------------------------------- */ + + /** Высчитываем адрес PDA для списка сидов. */ + private static PublicKey derivePda(List seeds, String programId) throws Exception { + PublicKey program = new PublicKey(programId); + PublicKey pda = PublicKey.findProgramAddress(seeds, program).getAddress(); + LOG.info("📡 PDA адрес: {}", pda.toBase58()); + return pda; + } + + /** + * Достаём бинарные данные аккаунта.
+ * Возвращает null, если аккаунт отсутствует или пуст. + */ + + + @SuppressWarnings("unchecked") + private static byte[] fetchAccountData(PublicKey pda) throws Exception { + + // 1) getAccountInfo c base64-энкодингом + Map cfg = new HashMap<>(); + cfg.put("encoding", "base64"); + cfg.put("commitment", "confirmed"); + +// Map resp = RPC.call( +// "getAccountInfo", +// Arrays.asList(pda.toBase58(), cfg), +// Map.class // сырое дерево +// ); + + + + Map resp; + try { + resp = RPC.call( + "getAccountInfo", + Arrays.asList(pda.toBase58(), cfg), + Map.class + ); + } catch (Exception e) { // solanaj бросает RuntimeException/IOException + throw new SolanaException_RpcConnection("Не удалось выполнить getAccountInfo"); + } + + // Если RPC вернул стандартное поле error — разбираем его централизованно + if (resp.get("error") != null) { + throw new SolanaException("RPC вернул поле error"); //тут можно добавить вывод что за конкретная ошибка случилась + } + + // 2) Достаём value → data[0] +// Map result = (Map) resp.get("result"); +// if (result == null) return null; + + + Map value = (Map) resp.get("value");//result.get("value"); + if (value == null) return null; + + List dataArr = (List) value.get("data"); + if (dataArr == null || dataArr.isEmpty()) return null; + + String b64 = (String) dataArr.get(0); // ← вот он payload + if (b64 == null || b64.isEmpty()) return null; + + return Base64.getDecoder().decode(b64); + } + + @SuppressWarnings("unchecked") + private static void debugDumpJson(Object node, int indent) { + String pad = String.join("", Collections.nCopies(indent, " ")); // два пробела × indent + if (node instanceof Map) { + Map map = (Map) node; + for (Map.Entry e : map.entrySet()) { + System.out.println(pad + e.getKey() + ":"); + debugDumpJson(e.getValue(), indent + 1); + } + } else if (node instanceof Iterable) { + Iterable it = (Iterable) node; + for (Object val : it) { + debugDumpJson(val, indent + 1); + } + } else { + System.out.println(pad + String.valueOf(node)); + } + } + + +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/ResultChecker.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/ResultChecker.java new file mode 100644 index 0000000..5ff55ab --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/ResultChecker.java @@ -0,0 +1,29 @@ +package me.shineup.solana.internal.utils.resultChecker; + + +/** + * Утилита для проверки результата выполнения транзакции по подписи. + * + * Данный класс: + * - выводит подпись транзакции в консоль + * - ждёт подтверждения транзакции через Solana RPC + * - выводит статус транзакции после завершения + */ +public class ResultChecker { + + /** + * Проверяет статус транзакции по её подписи. + * + * @param sig Подпись (signature) транзакции в base58-формате. + */ + public static void check(String sig) { + System.out.println("📦 Signature: " + sig); + + if (sig != null) { + TransactionStatusChecker.waitForConfirmation(sig); + SolanaTransactionStatusChecker.getTransactionStatus(sig); + } else { + System.out.println("⚠️ Подпись транзакции пуста или null"); + } + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/SolanaTransactionStatusChecker.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/SolanaTransactionStatusChecker.java new file mode 100644 index 0000000..a0d7cb5 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/SolanaTransactionStatusChecker.java @@ -0,0 +1,210 @@ +package me.shineup.solana.internal.utils.resultChecker; + +import me.shineup.solana.config.Const; +import com.google.gson.Gson; +import com.google.gson.JsonArray; +import com.google.gson.JsonElement; +import com.google.gson.JsonObject; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.internal.utils.SolanaRpcClient; + +public class SolanaTransactionStatusChecker { + + private static final Logger log = LoggerFactory.getLogger(SolanaTransactionStatusChecker.class); + private static final Gson gson = new Gson(); + + + public static boolean getTransactionStatus(String signature) { + boolean isOk = false; + String requestJson = String.format( + "{\n" + + " \"jsonrpc\": \"2.0\",\n" + + " \"id\": 1,\n" + + " \"method\": \"getTransaction\",\n" + + " \"params\": [\"%s\", { \"encoding\": \"jsonParsed\", \"commitment\": \"finalized\" }]\n" + + "}", signature); + +// String requestJson = """ +// { +// "jsonrpc": "2.0", +// "id": 1, +// "method": "getTransaction", +// "params": ["%s", { "encoding": "jsonParsed", "commitment": "finalized" }] +// } +// """.formatted(signature); + + + try { + String responseJson = SolanaRpcClient.getInstance().sendRequest(requestJson); + JsonObject root = gson.fromJson(responseJson, JsonObject.class); + + if (root.has("error")) { + log.error("❌ Ошибка при запросе транзакции: {}", root.get("error")); + return false; + } + + JsonElement resultElement = root.get("result"); + + if (resultElement == null || resultElement.isJsonNull()) { + log.warn("⚠️ Транзакция с сигнатурой {} не найдена или ещё не финализирована", signature); + return false; + } + + JsonObject result = root.getAsJsonObject("result"); + + if (result == null) { + log.warn("⚠️ Транзакция с сигнатурой {} не найдена", signature); + return false; + } + + JsonObject meta = result.getAsJsonObject("meta"); + if (meta == null) { + log.warn("⚠️ Нет информации о результате выполнения"); + return false; + } + + JsonElement err = meta.get("err"); + if (err == null || err.isJsonNull()) { + log.info("✅ Транзакция завершилась успешно"); + isOk = true; + } else { + log.warn("⚠️ Транзакция завершилась с ошибкой. Код ошибки: {}", extractCustomErrorCode(err)); + isOk = false; + } + + + // 💸 Вывод комиссии + if (meta.has("fee")) { + long fee = meta.get("fee").getAsLong(); + log.info("💸 Комиссия за транзакцию: {} лампортов", fee); + } + + // Получаем message → accountKeys + JsonObject message = result.getAsJsonObject("transaction").getAsJsonObject("message"); + JsonArray accountKeys = message.getAsJsonArray("accountKeys"); + + // 🔄 Выводим изменения балансов + logBalanceChanges(meta, accountKeys); + + // 💸 Общая сумма списанных лампортов (не только комиссия) + long totalSpent = calculateTotalSpent(meta); + log.info("💸 Общая сумма списанных лампортов (включая переводы и аренду аккаунтов): {}", totalSpent); + + + // Выводим лог исполнения (если есть) + if (meta.has("logMessages")) { + JsonElement logsElement = meta.get("logMessages"); + if (logsElement != null && logsElement.isJsonArray()) { + JsonArray logs = logsElement.getAsJsonArray(); + log.info("📝 Логи исполнения:"); + for (JsonElement logLine : logs) { + log.info(" → " + logLine.getAsString()); + } + } else if (logsElement != null && logsElement.isJsonNull()) { + log.info("📝 Логи исполнения: отсутствуют (null)"); + } else { + log.warn("📝 Логи исполнения: неожиданный формат данных"); + } + } + + } catch (Exception e) { + log.error("❌ Ошибка при запросе статуса транзакции", e); + return false; + } + return isOk; + } + + + + + /** + * Извлекает значение Custom Anchor ошибки из поля "err", если оно есть. + * + * Пример ожидаемого JSON: + * { + * "InstructionError": [0, { "Custom": 10000 }] + * } + * + * @param errJson поле "err" из ответа Solana + * @return числовой код ошибки (например, 10000) или null, если не найден + */ + public static Integer extractCustomErrorCode(JsonElement errJson) { + if (errJson == null || errJson.isJsonNull()) return null; + + try { + JsonObject errObj = errJson.getAsJsonObject(); + if (errObj.has("InstructionError")) { + JsonArray instrError = errObj.getAsJsonArray("InstructionError"); + + if (instrError.size() == 2 && instrError.get(1).isJsonObject()) { + JsonObject customObj = instrError.get(1).getAsJsonObject(); + if (customObj.has("Custom")) { + return customObj.get("Custom").getAsInt(); + } + } + } + } catch (Exception e) { + // безопасно возвращаем null + } + + return null; + } + + + /** + * Логирует изменения балансов всех аккаунтов, участвующих в транзакции. + * + * @param meta JSON-объект "meta" из транзакции (включает preBalances и postBalances) + * @param accountKeys Список accountKeys из message (содержит pubkey для каждого аккаунта) + */ + public static void logBalanceChanges(JsonObject meta, JsonArray accountKeys) { + JsonArray preBalances = meta.getAsJsonArray("preBalances"); + JsonArray postBalances = meta.getAsJsonArray("postBalances"); + + log.info("💰 Изменения балансов по аккаунтам:"); + + for (int i = 0; i < preBalances.size(); i++) { + long pre = preBalances.get(i).getAsLong(); + long post = postBalances.get(i).getAsLong(); + long delta = post - pre; + + String pubkey = accountKeys.get(i) + .getAsJsonObject() + .get("pubkey") + .getAsString(); + + double usdValue = delta * 150.0 / 1_000_000_000; + String usdFormatted = String.format("%.5f", usdValue); + + log.info("🔄 {}: {} → {} (Δ: {} лампортов) ≈ {} $", Const.identifyKey(pubkey), pre, post, delta, usdFormatted); + + + } + } + + /** + * Считает общую сумму списанных лампортов по всем аккаунтам, + * т.е. где postBalance < preBalance. + * + * @param meta JSON-объект "meta" из транзакции + * @return общее количество списанных лампортов + */ + public static long calculateTotalSpent(JsonObject meta) { + JsonArray preBalances = meta.getAsJsonArray("preBalances"); + JsonArray postBalances = meta.getAsJsonArray("postBalances"); + + long totalSpent = 0; + + for (int i = 0; i < preBalances.size(); i++) { + long pre = preBalances.get(i).getAsLong(); + long post = postBalances.get(i).getAsLong(); + + if (post < pre) { + totalSpent += (pre - post); + } + } + + return totalSpent; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/TransactionStatusChecker.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/TransactionStatusChecker.java new file mode 100644 index 0000000..44bca8f --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/TransactionStatusChecker.java @@ -0,0 +1,68 @@ +package me.shineup.solana.internal.utils.resultChecker; + +import org.p2p.solanaj.rpc.RpcClient; +import org.p2p.solanaj.rpc.types.SignatureStatuses; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; +import me.shineup.solana.config.Const; + +import java.util.Collections; +import java.util.List; + +public class TransactionStatusChecker { + + private static final Logger LOG = LoggerFactory.getLogger(TransactionStatusChecker.class); + + /** + * Проверяет статус транзакции по подписи, делает до 10 попыток с паузой. + * + * @param signature Подпись транзакции (Base58) + * @return true, если транзакция прошла успешно, иначе false + */ + public static boolean waitForConfirmation(String signature) { + RpcClient rpc = new RpcClient(Const.RPC_URL); + + try { + for (int attempt = 1; attempt <= 10; attempt++) { + LOG.info("🔍 Попытка {} проверки транзакции {}", attempt, signature); + + SignatureStatuses statuses = rpc.getApi().getSignatureStatuses(Collections.singletonList(signature), true); + List infoList = statuses.getValue(); + + if (infoList != null && !infoList.isEmpty()) { + SignatureStatuses.Value info = infoList.get(0); + + if (info != null) { + String status = info.getConfirmationStatus(); // Или getConfirmationStatusString() в других версиях + + LOG.info("⏳ Статус: {}", status); + if ("finalized".equals(status)) { + LOG.info("🎉 Финализирована большинством"); + //todo + return true; + } else if ("processed".equals(status)) { + LOG.info("Транзакция принята в пул"); + } else if ("confirmed".equals(status)) { + LOG.info("Транзакция вошла в блок, но не финализирована"); + } else { + LOG.info("Хер его знает "); + } + + } else { + LOG.info("Пока такой транзакции нету"); + } + } else { + LOG.info("Запрос вообще не удался"); + } + + Thread.sleep(3000); // Ждать 3 секунды перед следующей попыткой + } + + LOG.warn("❌ Транзакция не подтвердилась за 10 попыток: {}", signature); + } catch (Exception e) { + LOG.error("❌ Ошибка при проверке транзакции", e); + } + + return false; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/TransactionStatusHelper.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/TransactionStatusHelper.java new file mode 100644 index 0000000..cf6bb4b --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/internal/utils/resultChecker/TransactionStatusHelper.java @@ -0,0 +1,126 @@ +package me.shineup.solana.internal.utils.resultChecker; + +// Демонстрационный класс: единая точка проверки статуса транзакции Solana +// ---------------------------------------------------------------------- +// Содержит: +// 1. Enum TxStatus – перечень возможных состояний +// 2. Статический метод getTxStatus(...) – собственно проверка +// 3. Метод main(...) – пример использования +// ---------------------------------------------------------------------- + +import java.util.Collections; + +import com.google.gson.JsonObject; +import com.google.gson.JsonParser; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.utils.SolanaRpcClient; +import me.shineup.solana.model.TxStatus; +import org.p2p.solanaj.rpc.RpcClient; +import org.p2p.solanaj.rpc.types.SignatureStatuses; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; + +public class TransactionStatusHelper { + +// // -------------------------------------------------- +// // 1. Enum – возможные состояния транзакции +// // -------------------------------------------------- +// public enum TxStatus { +// // ✅ Повторять запрос, если лимит времени не исчерпан: +// NOT_FOUND, // RPC‑узел ещё не видел подпись */ +// PROCESSED, // Подпись обработана, но не попала в блок +// CONFIRMED, // Транзакция в блоке, но блок не финализирован +// +// //❌ Не повторять, даже если лимит не исчерпан: +// FINALIZED_SUCCESS, // Блок финализирован и meta.err == null +// FINALIZED_ERROR, // Блок финализирован, но meta.err содержит ошибку +// //⚠️ Не повторять или повторять ограниченно (1–3 раза, или ≤ N секунд): +// UNKNOWN, // Не удалось определить статус (RPC‑ошибка, исключение и т.д.) */ +// NETWORK_ERROR // Не удаётся подключиться по сете +// } + + // -------------------------------------------------- + // 2. Метод проверки статуса + // -------------------------------------------------- + private static final Logger log = LoggerFactory.getLogger(TransactionStatusHelper.class); + + /** + * Универсальная проверка статуса транзакции. + * @param signature подпись (transaction signature / id) + * @return текущее состояние {@link TxStatus} + */ + public static TxStatus getTxStatus(String signature) { + RpcClient rpc = new RpcClient(Const.RPC_URL); + + try { + // --- 1. Быстрый запрос STATUSES + // Пробуем получить статус транзакции + SignatureStatuses.Value info; + try { + info = rpc.getApi() + .getSignatureStatuses(Collections.singletonList(signature), true) + .getValue() + .get(0); + } catch (Exception e) { + log.error("🔌 Ошибка подключения к RPC или сети", e); + return TxStatus.NETWORK_ERROR; // Тут можно вернуть специальный статус NETWORK_ERROR, если нужно точнее + } + + + + if (info == null) { + return TxStatus.NOT_FOUND; // подпись ещё не дошла до RPC + } + + String commit = info.getConfirmationStatus(); // processed / confirmed / finalized + + switch (commit) { + case "processed": + return TxStatus.PROCESSED; + case "confirmed": + return TxStatus.CONFIRMED; + case "finalized": + // --- 2. Дошли до финала – нужен getTransaction, чтобы узнать err + String reqJson = String.format( + "{\"jsonrpc\":\"2.0\",\"id\":1,\"method\":\"getTransaction\"," + + "\"params\":[\"%s\",{\"encoding\":\"json\",\"commitment\":\"finalized\"}]}", + signature); + + // SolanaRpcClient – ваша внутренняя обёртка. Замените на свой http‑клиент. + String raw = SolanaRpcClient.getInstance().sendRequest(reqJson); + + JsonObject meta = JsonParser.parseString(raw) + .getAsJsonObject() + .getAsJsonObject("result") + .getAsJsonObject("meta"); + + return meta.get("err").isJsonNull() ? TxStatus.FINALIZED_SUCCESS : TxStatus.FINALIZED_ERROR; + default: + return TxStatus.UNKNOWN; + } + + } catch (Exception e) { + log.error("RPC error while checking {}", signature, e); + return TxStatus.UNKNOWN; + } + } + + // -------------------------------------------------- + // 3. Пример использования + // -------------------------------------------------- + /** + * Точка входа для быстрого ручного теста. + * Запускайте так: + * java TransactionStatusHelper + * Если аргумент не передан, используется демо‑подпись. + */ + public static void main(String[] args) { + // Подпись можно передать первым аргументом + String sig = (args.length > 0) + ? args[0] + : "4bxeRu4pNk9UzN6QgTPy6Q3DLJ6ZQt3xMkUQnDzohBpxjMVqRyba2Riqm8o7MBYo2YfSfvqbMFxRRWwu1XbbeiKf"; + + TxStatus status = getTxStatus(sig); + System.out.println("Статус транзакции " + sig + " → " + status); + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/TxStatus.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/TxStatus.java new file mode 100644 index 0000000..5c86b9a --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/TxStatus.java @@ -0,0 +1,22 @@ +package me.shineup.solana.model; + +/** + * Статус транзакции в Solana. + * Используется для отслеживания состояния по сигнатуре. + */ +// -------------------------------------------------- +// 1. Enum – возможные состояния транзакции +// -------------------------------------------------- +public enum TxStatus { + // ✅ Повторять запрос, если лимит времени не исчерпан: + NOT_FOUND, // RPC‑узел ещё не видел подпись */ + PROCESSED, // Подпись обработана, но не попала в блок + CONFIRMED, // Транзакция в блоке, но блок не финализирован + + //❌ Не повторять, даже если лимит не исчерпан: + FINALIZED_SUCCESS, // Блок финализирован и meta.err == null + FINALIZED_ERROR, // Блок финализирован, но meta.err содержит ошибку + //⚠️ Не повторять или повторять ограниченно (1–3 раза, или ≤ N секунд): + UNKNOWN, // Не удалось определить статус (RPC‑ошибка, исключение и т.д.) */ + NETWORK_ERROR // Не удаётся подключиться по сете +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/UserById.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/UserById.java new file mode 100644 index 0000000..8dea416 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/UserById.java @@ -0,0 +1,52 @@ +package me.shineup.solana.model; + + +import java.util.List; + +/** + * Java-представление Solana-структуры UserById. + * + * Содержит: + * • format – версия сериализации; + * • id – числовой ID пользователя; + * • login – строковый логин; + * • pubkey – публичная подпись пользователя (Base58); + * • deviceCount – сколько устройств хранится; + * • devices – список устройств (DeviceInfo). + */ +public class UserById { + + /* ---------- поля, идущие в сериализации ---------- */ + public int format; + public long id; + public String login; + public String pubkey; + public int deviceCount; + public List devices; + + /** Вложенный класс описания одного устройства. */ + public static class DeviceInfo { + public int deviceType; // 1 байт в on-chain + public String devicePubkey; // 32 байта (Base58) + public String x25519Pubkey; // 32 байта (Base58) + + @Override + public String toString() { + return "DeviceInfo{type=" + deviceType + + ", devPub=" + devicePubkey + + ", x25519=" + x25519Pubkey + '}'; + } + } + + @Override + public String toString() { + return "UserById{" + + "format=" + format + + ", id=" + id + + ", login='" + login + '\'' + + ", pubkey='" + pubkey + '\'' + + ", deviceCount=" + deviceCount + + ", devices=" + devices + + '}'; + } +} diff --git a/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/UserByLogin.java b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/UserByLogin.java new file mode 100644 index 0000000..a53e420 --- /dev/null +++ b/solana-shine-client-lib/solana-shine-lib/src/main/java/me/shineup/solana/model/UserByLogin.java @@ -0,0 +1,23 @@ +package me.shineup.solana.model; + +/** + * Класс описывающий объект UserByLogin, аналогичный структуре на стороне Solana. + */ +public class UserByLogin { + public int format; // формат сериализации (например, 1) + public String login; // логин + public long id; // числовой ID + public String pubkey; // публичный ключ (base58) + public int status; // статус (например, 0) + + @Override + public String toString() { + return "UserByLogin{" + + "format=" + format + + ", login='" + login + '\'' + + ", id=" + id + + ", pubkey='" + pubkey + '\'' + + ", status=" + status + + '}'; + } +} \ No newline at end of file diff --git a/solana-shine-client-lib/src/main/Build.txt b/solana-shine-client-lib/src/main/Build.txt new file mode 100644 index 0000000..ef4c9fd --- /dev/null +++ b/solana-shine-client-lib/src/main/Build.txt @@ -0,0 +1,2 @@ +./gradlew :solana-shine-lib:build +- команда что бы сбилдить библиотеку \ No newline at end of file diff --git a/solana-shine-client-lib/src/main/airDrop.txt b/solana-shine-client-lib/src/main/airDrop.txt new file mode 100644 index 0000000..a27d170 --- /dev/null +++ b/solana-shine-client-lib/src/main/airDrop.txt @@ -0,0 +1 @@ +solana airdrop 1 HMww7YSVfwVm4i8sugqj7wyH26dqzHykzv3wzWwzEvPA --url https://api.devnet.solana.com diff --git a/solana-shine-client-lib/src/main/java/com/shine/solana/test/AddUserExemple.java b/solana-shine-client-lib/src/main/java/com/shine/solana/test/AddUserExemple.java new file mode 100644 index 0000000..cec2458 --- /dev/null +++ b/solana-shine-client-lib/src/main/java/com/shine/solana/test/AddUserExemple.java @@ -0,0 +1,118 @@ +package com.shine.solana.test; + +import me.shineup.solana.SolanaWrapper; +import me.shineup.solana.internal.standartActions.keysGenerator.KeyPairBase58; +import me.shineup.solana.config.Const; +import me.shineup.solana.SolanaTxWatcher; +import me.shineup.solana.model.TxStatus; + +public class AddUserExemple { + public static void main(String[] args) { +try { +// SolanaWrapper.setRPC_URL_testNet(); +// SolanaWrapper.requestAirdrop(Const.getKeyByName("key3").getPublicKey(), 1); + SolanaWrapper.getBalance(Const.getKeyByName("key3").getPublicKey()); + +// long id =1; +// try { +// UserById u = SolanaWrapper.getUserById(id); +// System.out.println(u); +// UserByLogin u2 =SolanaWrapper.getUserByLogin(u.login); +// System.out.println(u2); +// } catch (Exception e) { +// e.printStackTrace(); +// } + + KeyPairBase58 k = SolanaWrapper.generateNewWallet(); + + String sig; +// sig = SolanaWrapper.sendSol(Const.getKeyByName("key1").getPrivateKey(), k.publicKey, 100000000); + SolanaWrapper.getBalance(Const.getKeyByName("key1").getPublicKey()); +// SolanaWrapper.getBalance(k.publicKey); +// SolanaWrapper.getBalance(k.publicKey); +// SolanaWrapper.getBalance(k.publicKey); + + +// String sig = SolanaWrapper.registerUserWithOneDev(k.publicKey,k.privateKey,"ivan11263456",k.publicKey, k.publicKey); + + sig = SolanaWrapper.registerUserWithOneDev(Const.getKeyByName("key1").getPublicKey(), Const.getKeyByName("key1").getPrivateKey(), + "ivan1261", Const.getKeyByName("key1").getPublicKey(), Const.getKeyByName("key1").getPublicKey()); + + + SolanaTxWatcher watcher = new SolanaTxWatcher(sig); + while (watcher.shouldRetry()) { // ◀ проверяем, нужно ли ещё опрашивать + watcher.updateStatus(); // ◀ запрашиваем статус через RPC + System.out.println( + watcher.getStatus() + " | " + // ◀ печатаем статус + watcher.getStatusComment()); + try { + Thread.sleep(SolanaTxWatcher.getRetryIntervalMs()); // ◀ ждём секунду + } catch (InterruptedException e) { + e.printStackTrace(); + } + } + + if (watcher.isSuccess()) { + System.out.println("✅ Транзакция прошла успешно!"); + } else { + System.out.println("❌ Завершили слежение без успеха."); + } + + +// TxStatus st = SolanaWrapper.getTransactionStatus(sig); +// System.out.println("Статус транзакции " + sig + " → " + st); +// TxStatus result = waitTransactionFinalized(sig); +// +// System.out.println("Итоговый статус: " + result); +// if (result == TxStatus.FINALIZED_SUCCESS) { +// System.out.println("✅ Транзакция прошла успешно!"); +// } else if (result == TxStatus.FINALIZED_ERROR) { +// System.out.println("❌ Транзакция завершилась с ошибкой."); +// } + +} catch (Exception e) { + throw new RuntimeException(e); +} + } + + + /** + * Ожидает финализации транзакции, проверяя статус каждую секунду. + * Максимальное время ожидания — 20 секунд. + */ + public static TxStatus waitTransactionFinalized(String signature) { + final int maxSeconds = 20; + TxStatus st; + for (int i = 0; i < maxSeconds; i++) { + try { + st = SolanaWrapper.getTransactionStatus(signature); + } catch (Exception e) { + throw new RuntimeException(e); //и тут дописывать проверки если надо + } + System.out.println("Статус транзакции " + signature + " → " + st); + + // Если достигли финального состояния — возвращаем + if (st == TxStatus.FINALIZED_SUCCESS || st == TxStatus.FINALIZED_ERROR) { + return st; + } + + // Если транзакция не найдена или в неизвестном состоянии — тоже выходим + if (st == TxStatus.UNKNOWN) { + System.out.println("Ошибка или подпись не найдена. Прерываем ожидание."); + return st; + } + + // Подождать 1 секунду + try { + Thread.sleep(1000); + } catch (InterruptedException e) { + System.out.println("Ожидание прервано."); + return TxStatus.UNKNOWN; + } + } + + System.out.println("⏱ Превышен лимит ожидания (20 сек)."); + return TxStatus.UNKNOWN; + } + +} diff --git a/solana-shine-client-lib/src/main/java/com/shine/solana/test/AirDrop.java b/solana-shine-client-lib/src/main/java/com/shine/solana/test/AirDrop.java new file mode 100644 index 0000000..a85cfe8 --- /dev/null +++ b/solana-shine-client-lib/src/main/java/com/shine/solana/test/AirDrop.java @@ -0,0 +1,32 @@ +package com.shine.solana.test; + +import me.shineup.solana.SolanaWrapper; +import me.shineup.solana.config.Const; + +public class AirDrop { + public static void main(String[] args) { + // Если первый аргумент равен строке "1" → запрашиваем airdrop, + // иначе просто выводим баланс. + try { +// SolanaWrapper.setRPC_URL("https://api.testnet.solana.com"); + // SolanaWrapper.setRPC_URL("https://api.devnet.solana.com"); + boolean needAirdrop = args.length > 0 && "1".equals(args[0]); + // needAirdrop = true; + if (needAirdrop) { + long oneSolLamports = 1_000_000_000L; // 1 SOL в лампортах + SolanaWrapper.requestAirdrop( + Const.getKeyByName("key1").getPublicKey(), + oneSolLamports); + } else { + SolanaWrapper.getBalance(Const.getKeyByName("key1").getPublicKey()); + SolanaWrapper.getBalance(Const.getKeyByName("key2").getPublicKey()); + SolanaWrapper.getBalance(Const.getKeyByName("key3").getPublicKey()); + } + + } catch (Exception e) { + throw new RuntimeException(e); + } + + } + +} diff --git a/solana-shine-client-lib/src/main/java/com/shine/solana/test/Main.java b/solana-shine-client-lib/src/main/java/com/shine/solana/test/Main.java new file mode 100644 index 0000000..22c4ecf --- /dev/null +++ b/solana-shine-client-lib/src/main/java/com/shine/solana/test/Main.java @@ -0,0 +1,47 @@ +package com.shine.solana.test; + + +import me.shineup.solana.SolanaWrapper; +import me.shineup.solana.config.Const; +import me.shineup.solana.internal.callSolanaFunc.InitializeUserCounter.InitializeUserCounter; + +import java.util.concurrent.ExecutorService; +import java.util.concurrent.Executors; + +public class Main { + private static final ExecutorService executorService = Executors.newCachedThreadPool(); + + public static void main(String[] args) { +// SolanaWrapper.setRPC_URL("https://api.devnet.solana.com"); +// InitializeUserCounter + + long oneSolLamports = 5_000_000_000L; // 1 SOL в лампортах + + try { + SolanaWrapper.requestAirdrop( + "FUc28vNixp7F3nnkpGVt6nuJbgvJ4429v4B5wS52Df6P", + oneSolLamports); + System.out.println(SolanaWrapper.getBalance("FUc28vNixp7F3nnkpGVt6nuJbgvJ4429v4B5wS52Df6P"));//H6q58ytZk5sd3KQisC57R6urKUjn5PaWKLCv7sNZtj4i")); + System.out.println("Баланс Shine" + SolanaWrapper.getBalance( "jZnbqzrbKaksVomiqAxsKGiz5ct8rHPgcNDiiKyTZDD")); + } catch (Exception e) { + System.out.println(e.getMessage()); + } + + // 5RpEoxRKSr2norQP3vEnq9XokQGh9EbGN8q8xUUVAdm1M5mTD1vMuyJPYJfViMWFf6c8qT5mj2bt64gLE2zm6VG3 - тестовые в фантом валет + // FUc28vNixp7F3nnkpGVt6nuJbgvJ4429v4B5wS52Df6P + + // +// executorService.submit(() -> { +// try { +// System.out.println(SolanaWrapper.getUserById(1).login); +// } catch (Exception e) { +// System.out.println(e.getMessage()); +// } +// }); + + + + + } + +} diff --git a/solana-shine-client-lib/src/main/java/com/shine/utils/CreateToken_story.txt b/solana-shine-client-lib/src/main/java/com/shine/utils/CreateToken_story.txt new file mode 100644 index 0000000..b2c0623 --- /dev/null +++ b/solana-shine-client-lib/src/main/java/com/shine/utils/CreateToken_story.txt @@ -0,0 +1,58 @@ +ai@ai-home:~$ ~/.cargo/bin/spl-token --version +spl-token-cli 5.3.0 + +//создание +ai@ai-home:~$ ~/.cargo/bin/spl-token create-token +Creating token Dt7t2kKXYT8UqdioTQZeH4SRXqDA2kh2dscqZQmfr1Rr under program TokenkegQfeZyiNwAJbNbGKPFXCWuBvf9Ss623VQ5DA + +Address: Dt7t2kKXYT8UqdioTQZeH4SRXqDA2kh2dscqZQmfr1Rr +Decimals: 9 + +Signature: 656MfR7x4K5fZJEsf1nJTk5gorMkuZtSCw2YGzvtAEHLyftDsYhoXspGspEhD1Jto8pNNfkyNVb2jRVZPBg5gi9u + +------------------------------------------------------------------ +//Эмиссия +ai@ai-home:~$ ~/.cargo/bin/spl-token mint Dt7t2kKXYT8UqdioTQZeH4SRXqDA2kh2dscqZQmfr1Rr 10 +Minting 10 tokens + Token: Dt7t2kKXYT8UqdioTQZeH4SRXqDA2kh2dscqZQmfr1Rr + Recipient: AcyCkqNEdR78s6wsuhwEeqfSe4BHGuK1EkzGcUbArifq + +Signature: 5SWjE1cHrcWEnArZuUsx4efSVXKWsADxPsuEZL2KbeWfUGy4DqQe4Mdg3AhKtXMLLmRMeQYSSX1mB7VgzLcbEGao + +ai@ai-home:~$ ~/.cargo/bin/spl-token balance Dt7t2kKXYT8UqdioTQZeH4SRXqDA2kh2dscqZQmfr1Rr +10 +------------------------------------------------------------------------------ +✅ 2. Перевод токенов на существующий аккаунт +Шаг 1. Создаём токен-аккаунт получателя (один раз) +Вариант A: вручную + +~/.cargo/bin/spl-token create-account <адрес_токена> --owner <публичный_адрес_получателя> + +Вариант B: автоматически при переводе + +~/.cargo/bin/spl-token transfer <адрес_токена> 1 <публичный_адрес_получателя> --fund-recipient + +Флаг --fund-recipient автоматически создаст токен-аккаунт получателя и оплатит аренду за счёт твоего кошелька. + +----- + +ai@ai-home:~$ ~/.cargo/bin/spl-token transfer Dt7t2kKXYT8UqdioTQZeH4SRXqDA2kh2dscqZQmfr1Rr 1 HMww7YSVfwVm4i8sugqj7wyH26dqzHykzv3wzWwzEvPA --fund-recipient +Transfer 1 tokens + Sender: AcyCkqNEdR78s6wsuhwEeqfSe4BHGuK1EkzGcUbArifq + Recipient: HMww7YSVfwVm4i8sugqj7wyH26dqzHykzv3wzWwzEvPA + Recipient associated token account: GJQPLqNXZNXATm4J1sGgtiKKCk9wFced2NXhwGxc5hfA + Funding recipient: GJQPLqNXZNXATm4J1sGgtiKKCk9wFced2NXhwGxc5hfA + +Signature: 5R3bGiqC49GBHJFMTGMRvBxNt12nkvmKrwRxUQVcvhyRn23rmr7YAaWxzvedkneuSr3YUmAQXk7UFyorhdr5WvhY + +ai@ai-home:~$ ~/.cargo/bin/spl-token accounts +Token Balance +----------------------------------------------------- +Dt7t2kKXYT8UqdioTQZeH4SRXqDA2kh2dscqZQmfr1Rr 9 + +ai@ai-home:~$ ~/.cargo/bin/spl-token accounts --owner ^C +ai@ai-home:~$ ~/.cargo/bin/spl-token accounts --owner HMww7YSVfwVm4i8sugqj7wyH26dqzHykzv3wzWwzEvPA +Token Balance +----------------------------------------------------- +Dt7t2kKXYT8UqdioTQZeH4SRXqDA2kh2dscqZQmfr1Rr 1 + diff --git a/solana-shine-client-lib/src/main/java/com/shine/utils/SolanaKeyBase58Converter.java b/solana-shine-client-lib/src/main/java/com/shine/utils/SolanaKeyBase58Converter.java new file mode 100644 index 0000000..8450c94 --- /dev/null +++ b/solana-shine-client-lib/src/main/java/com/shine/utils/SolanaKeyBase58Converter.java @@ -0,0 +1,110 @@ +package com.shine.utils; + +import java.util.ArrayList; +import java.util.List; + +/** + * Класс для конвертации приватного ключа Solana из Base58 в формат JSON-массива чисел, + * который понимает Solana CLI. + * + * Использование: + * 1. Вставьте свой приватный ключ в Base58 в main(). + * 2. Запустите программу — она выведет строку JSON, которую можно сохранить в файл. + * 3. С помощью этого файла можно подключить кошелек в Solana CLI. + */ +public class SolanaKeyBase58Converter { + + // ─────────────────────────────────────────────── + // 🔹 ТЕСТОВЫЙ ЗАПУСК + // ─────────────────────────────────────────────── + public static void main(String[] args) { + // Вставьте сюда свой приватный ключ в Base58 + String base58Key = "5RpEoxRKSr2norQP3vEnq9XokQGh9EbGN8q8xUUVAdm1M5mTD1vMuyJPYJfViMWFf6c8qT5mj2bt64gLE2zm6VG3"; //"ВСТАВЬ_СЮДА_СВОЙ_ПРИВАТНЫЙ_КЛЮЧ"; + + // Конвертация в JSON-массив чисел + String jsonKey = base58ToJson(base58Key); + + // Вывод в консоль + System.out.println("Скопируйте эту строку и сохраните в файл (например, mywallet.json):"); + System.out.println(jsonKey); + } + + // ─────────────────────────────────────────────── + // 🔹 ОСНОВНОЙ МЕТОД КОНВЕРТАЦИИ + // ─────────────────────────────────────────────── + + /** + * Преобразует приватный ключ из Base58 в строку JSON-массива чисел, + * которую можно сохранить в файл для Solana CLI. + * + * @param base58Key приватный ключ в формате Base58 (строка) + * @return строка вида "[25,156,13,48,...,147]" + */ + public static String base58ToJson(String base58Key) { + byte[] bytes = decodeBase58(base58Key); + + // Формируем JSON-массив из байт + StringBuilder json = new StringBuilder("["); + for (int i = 0; i < bytes.length; i++) { + json.append(bytes[i] & 0xFF); // "& 0xFF" нужно, чтобы убрать знак и получить 0–255 + if (i < bytes.length - 1) { + json.append(","); + } + } + json.append("]"); + return json.toString(); + } + + // ─────────────────────────────────────────────── + // 🔹 ДЕКОДЕР BASE58 → БАЙТЫ + // ─────────────────────────────────────────────── + + // Алфавит Base58 (стандарт Solana/Bitcoin) + private static final String BASE58_ALPHABET = "123456789ABCDEFGHJKLMNPQRSTUVWXYZabcdefghijkmnopqrstuvwxyz"; + + /** + * Декодирует строку Base58 в байтовый массив. + * + * @param input приватный ключ в формате Base58 + * @return байтовый массив приватного ключа + */ + private static byte[] decodeBase58(String input) { + java.math.BigInteger num = java.math.BigInteger.ZERO; + java.math.BigInteger base = java.math.BigInteger.valueOf(58); + + // Переводим Base58 в большое число + for (char c : input.toCharArray()) { + int digit = BASE58_ALPHABET.indexOf(c); + if (digit < 0) { + throw new IllegalArgumentException("Недопустимый символ в Base58: " + c); + } + num = num.multiply(base).add(java.math.BigInteger.valueOf(digit)); + } + + // Конвертируем большое число в байты + List byteList = new ArrayList<>(); + byte[] numBytes = num.toByteArray(); + + // Убираем возможный ведущий 0x00 (для BigInteger) + int startIndex = (numBytes.length > 1 && numBytes[0] == 0) ? 1 : 0; + for (int i = startIndex; i < numBytes.length; i++) { + byteList.add(numBytes[i]); + } + + // Добавляем ведущие нули для каждого символа '1' в Base58 + for (char c : input.toCharArray()) { + if (c == '1') { + byteList.add(0, (byte) 0); + } else { + break; + } + } + + // Преобразуем в обычный массив + byte[] result = new byte[byteList.size()]; + for (int i = 0; i < byteList.size(); i++) { + result[i] = byteList.get(i); + } + return result; + } +} diff --git a/solana-shine-client-lib/src/main/java/com/shine/utils/Wallets.txt b/solana-shine-client-lib/src/main/java/com/shine/utils/Wallets.txt new file mode 100644 index 0000000..80105e2 --- /dev/null +++ b/solana-shine-client-lib/src/main/java/com/shine/utils/Wallets.txt @@ -0,0 +1,37 @@ +solana/phantomWallet.json + // 5RpEoxRKSr2norQP3vEnq9XokQGh9EbGN8q8xUUVAdm1M5mTD1vMuyJPYJfViMWFf6c8qT5mj2bt64gLE2zm6VG3 - тестовые в фантом валет + // FUc28vNixp7F3nnkpGVt6nuJbgvJ4429v4B5wS52Df6P + + +Баланс + + + + +✅ 2. Инструкция: как сохранить и подключить ключ в Ubuntu +1. Сохраняем JSON в файл + +После запуска программы скопируй результат, например: + +[25,156,13,48,203,...,147] + +Сохрани в файл mywallet.json: + +echo '[25,156,13,48,203,...,147]' > mywallet.json + +2. Временно подключаем кошелёк в Solana CLI + +Подключаем как текущий кошелёк: + +solana config set --keypair mywallet.json + +Проверяем: + +solana address + +Должен показать публичный ключ, соответствующий этому приватному. +3. Возвращаем обратно свой основной ключ + +После работы верни основной ключ: + +solana config set --keypair ~/.config/solana/id.json