From dcb973fe5c0ec8a738584a41fcf9534c14bd1fec Mon Sep 17 00:00:00 2001 From: billythedummy Date: Fri, 22 Dec 2023 17:42:47 +0800 Subject: [PATCH 1/4] solores 0.5.0 and workspace dependencies --- .pre-commit-config.yaml | 1 + .vscode/settings.json | 13 + Cargo.lock | 2 +- Cargo.toml | 33 + cli-rust/Cargo.toml | 40 +- cli-rust/src/subcmd/deactivate_all.rs | 21 +- .../src/subcmd/deactivate_stake_account.rs | 21 +- cli-rust/src/subcmd/init_protocol_fee.rs | 15 +- cli-rust/src/subcmd/reclaim_all.rs | 25 +- cli-rust/src/subcmd/reclaim_stake_account.rs | 25 +- cli-rust/src/subcmd/set_fee_authority.rs | 15 +- programs/unstake/Cargo.toml | 21 +- unstake_interface/Cargo.toml | 24 +- unstake_interface/README.md | 20 + unstake_interface/src/accounts.rs | 155 +- unstake_interface/src/errors.rs | 41 +- unstake_interface/src/instructions.rs | 3255 ++++++++++++----- unstake_interface/src/lib.rs | 6 +- unstake_interface/src/typedefs.rs | 8 +- 19 files changed, 2771 insertions(+), 970 deletions(-) create mode 100644 .vscode/settings.json create mode 100644 unstake_interface/README.md diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index 8c249ae..e2cd6c7 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -9,3 +9,4 @@ repos: hooks: - id: fmt - id: clippy + args: ["--tests", "--", "-Dwarnings"] diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..b124eaf --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,13 @@ +{ + "[json]": { + "editor.tabSize": 2 + }, + "[jsonc]": { + "editor.tabSize": 2 + }, + "editor.formatOnSave": true, + "editor.defaultFormatter": "esbenp.prettier-vscode", + "[rust]": { + "editor.defaultFormatter": "rust-lang.rust-analyzer" + } +} diff --git a/Cargo.lock b/Cargo.lock index 3334cb7..d8294da 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -4424,7 +4424,7 @@ dependencies = [ [[package]] name = "unstake_interface" -version = "1.0.0" +version = "2.0.0" dependencies = [ "borsh 0.9.3", "num-derive", diff --git a/Cargo.toml b/Cargo.toml index fd9a4c4..2ce4173 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,2 +1,35 @@ [workspace] members = ["programs/*", "cli-rust", "unstake_interface"] + +[workspace.dependencies] +anchor-lang = "0.28.0" +anchor-spl = "0.28.0" +base64 = "0.21.2" +clap = { version = "^4.0" } +clap2 = { package = "clap", version = "^2.0" } +borsh = "^0.9" # lock to 0.9.3 for prod +derive_more = "^0.99" +mpl-token-metadata = "^1.13" +num-derive = ">=0.1" +num-traits = ">=0.1" +proptest = "^1" +serde = "^1" +serde_json = "^1" +spl-associated-token-account = "^1.1" # required for anchor-spl token +spl-token = "^3.0" +spl-math = ">=0.1" +thiserror = "^1" + +# solana deps +# Lock to 1.14 for prod, but ^1 for flexibility for library users +solana-account-decoder = "^1" +solana-clap-utils = "^1" +solana-cli-config = "^1" +solana-client = "^1" +solana-sdk = "^1" +solana-program = "^1" +solana-stake-program = "^1" + +# workspace members +unstake = { path = "./programs/unstake" } +unstake_interface = { path = "./unstake_interface" } diff --git a/cli-rust/Cargo.toml b/cli-rust/Cargo.toml index d3aaccc..4e126a2 100644 --- a/cli-rust/Cargo.toml +++ b/cli-rust/Cargo.toml @@ -8,23 +8,23 @@ name = "unstakeit" path = "src/main.rs" [dependencies] -anchor-lang = "0.28.0" -base64 = "0.21.2" -borsh = "^0.9.1" -clap = { version = "^4.0", features = ["derive"] } -clap2 = { package = "clap", version = "^2.0" } # required for solana-clap-utils -derive_more = "^0.99" -mpl-token-metadata = "^1.13" -serde = { version = "1.0.171", features = ["derive"] } -serde_json = "1.0.103" -solana-account-decoder = "~1.14" -solana-clap-utils = "~1.14" -solana-cli-config = "~1.14" -solana-client = "~1.14" -solana-sdk = "~1.14" -solana-program = "~1.14" -solana-stake-program = "~1.14" -spl-associated-token-account = "1.1.1" -spl-token = "^3.0" -unstake = { path = "../programs/unstake" } -unstake_interface = { path = "../unstake_interface", features = ["serde"] } +anchor-lang = { workspace = true } +base64 = { workspace = true } +borsh = { workspace = true } +clap = { workspace = true, features = ["derive"] } +clap2 = { workspace = true } # required for solana-clap-utils +derive_more = { workspace = true } +mpl-token-metadata = { workspace = true } +serde = { workspace = true } +serde_json = { workspace = true } +solana-account-decoder = { workspace = true } +solana-clap-utils = { workspace = true } +solana-cli-config = { workspace = true } +solana-client = { workspace = true } +solana-sdk = { workspace = true } +solana-program = { workspace = true } +solana-stake-program = { workspace = true } +spl-associated-token-account = { workspace = true } +spl-token = { workspace = true } +unstake = { workspace = true } +unstake_interface = { workspace = true, features = ["serde"] } diff --git a/cli-rust/src/subcmd/deactivate_all.rs b/cli-rust/src/subcmd/deactivate_all.rs index 2c2613a..4e33f76 100644 --- a/cli-rust/src/subcmd/deactivate_all.rs +++ b/cli-rust/src/subcmd/deactivate_all.rs @@ -4,9 +4,7 @@ use clap::Args; use solana_program::{pubkey::Pubkey, sysvar}; use unstake::ID; -use unstake_interface::{ - deactivate_stake_account_ix, DeactivateStakeAccountIxArgs, DeactivateStakeAccountKeys, -}; +use unstake_interface::{deactivate_stake_account_ix, DeactivateStakeAccountKeys}; use crate::{ tx_utils::{batch_ixs, chunk_array, send_or_sim_tx}, @@ -58,16 +56,13 @@ impl SubcmdExec for DeactivateAllArgs { println!("Generating txs..."); let mut deactivate_ixs = Vec::new(); for stake_account in stake_accounts_to_deactivate.iter() { - let ix = deactivate_stake_account_ix( - DeactivateStakeAccountKeys { - pool_account, - pool_sol_reserves: pool_sol_reserves.0, - clock: sysvar::clock::id(), - stake_account: *stake_account, - stake_program: solana_stake_program::id(), - }, - DeactivateStakeAccountIxArgs {}, - ) + let ix = deactivate_stake_account_ix(DeactivateStakeAccountKeys { + pool_account, + pool_sol_reserves: pool_sol_reserves.0, + clock: sysvar::clock::id(), + stake_account: *stake_account, + stake_program: solana_stake_program::id(), + }) .unwrap(); deactivate_ixs.push(ix); } diff --git a/cli-rust/src/subcmd/deactivate_stake_account.rs b/cli-rust/src/subcmd/deactivate_stake_account.rs index 0c10a76..6b909ab 100644 --- a/cli-rust/src/subcmd/deactivate_stake_account.rs +++ b/cli-rust/src/subcmd/deactivate_stake_account.rs @@ -5,9 +5,7 @@ use clap::Args; use solana_program::{message::Message, pubkey::Pubkey, sysvar}; use solana_sdk::transaction::Transaction; use unstake::ID; -use unstake_interface::{ - deactivate_stake_account_ix, DeactivateStakeAccountIxArgs, DeactivateStakeAccountKeys, -}; +use unstake_interface::{deactivate_stake_account_ix, DeactivateStakeAccountKeys}; use crate::tx_utils::send_or_sim_tx; @@ -32,16 +30,13 @@ impl SubcmdExec for DeactivateStakeAccountArgs { let stake_account = Pubkey::from_str(&self.stake_account).unwrap(); - let ix = deactivate_stake_account_ix( - DeactivateStakeAccountKeys { - pool_account, - pool_sol_reserves: pool_sol_reserves.0, - clock: sysvar::clock::id(), - stake_account, - stake_program: solana_stake_program::id(), - }, - DeactivateStakeAccountIxArgs {}, - ) + let ix = deactivate_stake_account_ix(DeactivateStakeAccountKeys { + pool_account, + pool_sol_reserves: pool_sol_reserves.0, + clock: sysvar::clock::id(), + stake_account, + stake_program: solana_stake_program::id(), + }) .unwrap(); let payer_pk = payer.pubkey(); diff --git a/cli-rust/src/subcmd/init_protocol_fee.rs b/cli-rust/src/subcmd/init_protocol_fee.rs index 2207640..74c5830 100644 --- a/cli-rust/src/subcmd/init_protocol_fee.rs +++ b/cli-rust/src/subcmd/init_protocol_fee.rs @@ -2,7 +2,7 @@ use clap::Args; use solana_program::{pubkey::Pubkey, system_program}; use solana_sdk::transaction::Transaction; use unstake::{state::PROTOCOL_FEE_SEED, ID}; -use unstake_interface::{init_protocol_fee_ix, InitProtocolFeeIxArgs, InitProtocolFeeKeys}; +use unstake_interface::{init_protocol_fee_ix, InitProtocolFeeKeys}; use crate::tx_utils::send_or_sim_tx; @@ -19,14 +19,11 @@ impl SubcmdExec for InitProtocolFeeArgs { let protocol_fee_account = Pubkey::find_program_address(&[PROTOCOL_FEE_SEED], &ID); - let ix = init_protocol_fee_ix( - InitProtocolFeeKeys { - payer: payer.pubkey(), - protocol_fee_account: protocol_fee_account.0, - system_program: system_program::id(), - }, - InitProtocolFeeIxArgs {}, - ) + let ix = init_protocol_fee_ix(InitProtocolFeeKeys { + payer: payer.pubkey(), + protocol_fee_account: protocol_fee_account.0, + system_program: system_program::id(), + }) .unwrap(); let blockhash = client.get_latest_blockhash().unwrap(); diff --git a/cli-rust/src/subcmd/reclaim_all.rs b/cli-rust/src/subcmd/reclaim_all.rs index bc83763..fa969c8 100644 --- a/cli-rust/src/subcmd/reclaim_all.rs +++ b/cli-rust/src/subcmd/reclaim_all.rs @@ -4,9 +4,7 @@ use clap::Args; use solana_program::{pubkey::Pubkey, sysvar}; use unstake::ID; -use unstake_interface::{ - reclaim_stake_account_ix, ReclaimStakeAccountIxArgs, ReclaimStakeAccountKeys, -}; +use unstake_interface::{reclaim_stake_account_ix, ReclaimStakeAccountKeys}; use crate::{ tx_utils::{batch_ixs, chunk_array, send_or_sim_tx}, @@ -64,18 +62,15 @@ impl SubcmdExec for ReclaimAllArgs { &[&pool_account.to_bytes(), &stake_account.to_bytes()], &ID, ); - let ix = reclaim_stake_account_ix( - ReclaimStakeAccountKeys { - pool_account, - pool_sol_reserves: pool_sol_reserves.0, - clock: sysvar::clock::id(), - stake_account: *stake_account, - stake_account_record_account: stake_account_record_account.0, - stake_history: sysvar::stake_history::id(), - stake_program: solana_stake_program::id(), - }, - ReclaimStakeAccountIxArgs {}, - ) + let ix = reclaim_stake_account_ix(ReclaimStakeAccountKeys { + pool_account, + pool_sol_reserves: pool_sol_reserves.0, + clock: sysvar::clock::id(), + stake_account: *stake_account, + stake_account_record_account: stake_account_record_account.0, + stake_history: sysvar::stake_history::id(), + stake_program: solana_stake_program::id(), + }) .unwrap(); reclaim_ixs.push(ix); } diff --git a/cli-rust/src/subcmd/reclaim_stake_account.rs b/cli-rust/src/subcmd/reclaim_stake_account.rs index ac0b727..268f8e5 100644 --- a/cli-rust/src/subcmd/reclaim_stake_account.rs +++ b/cli-rust/src/subcmd/reclaim_stake_account.rs @@ -5,9 +5,7 @@ use clap::Args; use solana_program::{message::Message, pubkey::Pubkey, sysvar}; use solana_sdk::transaction::Transaction; use unstake::ID; -use unstake_interface::{ - reclaim_stake_account_ix, ReclaimStakeAccountIxArgs, ReclaimStakeAccountKeys, -}; +use unstake_interface::{reclaim_stake_account_ix, ReclaimStakeAccountKeys}; use crate::tx_utils::send_or_sim_tx; @@ -38,18 +36,15 @@ impl SubcmdExec for ReclaimStakeAccountArgs { &ID, ); - let ix = reclaim_stake_account_ix( - ReclaimStakeAccountKeys { - pool_account, - pool_sol_reserves: pool_sol_reserves.0, - clock: sysvar::clock::id(), - stake_account, - stake_account_record_account: stake_account_record_account.0, - stake_history: sysvar::stake_history::id(), - stake_program: solana_stake_program::id(), - }, - ReclaimStakeAccountIxArgs {}, - ) + let ix = reclaim_stake_account_ix(ReclaimStakeAccountKeys { + pool_account, + pool_sol_reserves: pool_sol_reserves.0, + clock: sysvar::clock::id(), + stake_account, + stake_account_record_account: stake_account_record_account.0, + stake_history: sysvar::stake_history::id(), + stake_program: solana_stake_program::id(), + }) .unwrap(); let payer_pk = payer.pubkey(); diff --git a/cli-rust/src/subcmd/set_fee_authority.rs b/cli-rust/src/subcmd/set_fee_authority.rs index 93d4d55..0401618 100644 --- a/cli-rust/src/subcmd/set_fee_authority.rs +++ b/cli-rust/src/subcmd/set_fee_authority.rs @@ -3,7 +3,7 @@ use std::str::FromStr; use clap::Args; use solana_program::{message::Message, pubkey::Pubkey}; use solana_sdk::{signature::read_keypair_file, signer::Signer, transaction::Transaction}; -use unstake_interface::{set_fee_authority_ix, SetFeeAuthorityIxArgs, SetFeeAuthorityKeys}; +use unstake_interface::{set_fee_authority_ix, SetFeeAuthorityKeys}; use crate::tx_utils::send_or_sim_tx; @@ -39,14 +39,11 @@ impl SubcmdExec for SetFeeAuthorityArgs { signers.push(Box::new(fee_authority_keypair)); } - let ix = set_fee_authority_ix( - SetFeeAuthorityKeys { - pool_account, - fee_authority, - new_fee_authority, - }, - SetFeeAuthorityIxArgs {}, - ) + let ix = set_fee_authority_ix(SetFeeAuthorityKeys { + pool_account, + fee_authority, + new_fee_authority, + }) .unwrap(); let msg = Message::new(&[ix], Some(&payer_pk)); diff --git a/programs/unstake/Cargo.toml b/programs/unstake/Cargo.toml index 132fe50..4adeba8 100644 --- a/programs/unstake/Cargo.toml +++ b/programs/unstake/Cargo.toml @@ -17,16 +17,13 @@ local-testing = [] default = [] -[dev-dependencies] -proptest = "1.0" - -[dependencies.spl-math] -version = "0.1.0" -features = ["no-entrypoint"] - [dependencies] -anchor-lang = { version = "0.28.0", features = ["init-if-needed"] } -anchor-spl = { version = "0.28.0", features = ["metadata", "stake", "token"] } -mpl-token-metadata = { version = "^1.13", features = ["no-entrypoint"] } -serde = { version = "1.0.171", features = ["derive"] } -spl-associated-token-account = "^1.1" # required for anchor-spl token +anchor-lang = { workspace = true, features = ["init-if-needed"] } +anchor-spl = { workspace = true, features = ["metadata", "stake", "token"] } +mpl-token-metadata = { workspace = true, features = ["no-entrypoint"] } +serde = { workspace = true, features = ["derive"] } +spl-associated-token-account = { workspace = true, features = ["no-entrypoint"] } # required for anchor-spl token +spl-math = { workspace = true, features = ["no-entrypoint"] } + +[dev-dependencies] +proptest = { workspace = true } diff --git a/unstake_interface/Cargo.toml b/unstake_interface/Cargo.toml index 691a1db..2021eae 100644 --- a/unstake_interface/Cargo.toml +++ b/unstake_interface/Cargo.toml @@ -1,15 +1,23 @@ [package] name = "unstake_interface" -version = "1.0.0" +version = "2.0.0" edition = "2021" -[dependencies] -borsh = "^0.9.1" -solana-program = "~1.14" -thiserror = "^1.0" -num-derive = "^0.3" -num-traits = "^0.2" +[dependencies.borsh] +workspace = true + +[dependencies.solana-program] +workspace = true [dependencies.serde] optional = true -version = "^1.0" +workspace = true + +[dependencies.thiserror] +workspace = true + +[dependencies.num-derive] +workspace = true + +[dependencies.num-traits] +workspace = true diff --git a/unstake_interface/README.md b/unstake_interface/README.md new file mode 100644 index 0000000..fa68370 --- /dev/null +++ b/unstake_interface/README.md @@ -0,0 +1,20 @@ +# unstake_interface + +Generated with `solores v0.5.0` + +## Generate + +After `anchor build`, run in workspace root: + +```sh +solores \ + --solana-program-vers "workspace=true" \ + --borsh-vers "workspace=true" \ + --thiserror-vers "workspace=true" \ + --num-derive-vers "workspace=true" \ + --num-traits-vers "workspace=true" \ + --serde-vers "workspace=true" \ + target/idl/unstake.json +``` + +And then manually update the program ID in `lib.rs` to the desired program ID. diff --git a/unstake_interface/src/accounts.rs b/unstake_interface/src/accounts.rs index 7d3513b..b406461 100644 --- a/unstake_interface/src/accounts.rs +++ b/unstake_interface/src/accounts.rs @@ -2,27 +2,114 @@ use crate::*; use borsh::{BorshDeserialize, BorshSerialize}; use solana_program::pubkey::Pubkey; pub const FEE_ACCOUNT_DISCM: [u8; 8] = [24, 55, 150, 250, 168, 27, 101, 178]; -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct Fee { pub fee: FeeEnum, } +#[derive(Clone, Debug, PartialEq)] +pub struct FeeAccount(pub Fee); +impl FeeAccount { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + use std::io::Read; + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != FEE_ACCOUNT_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + FEE_ACCOUNT_DISCM, maybe_discm + ), + )); + } + Ok(Self(Fee::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&FEE_ACCOUNT_DISCM)?; + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) + } +} pub const FLASH_LOAN_FEE_ACCOUNT_DISCM: [u8; 8] = [211, 113, 211, 138, 191, 108, 64, 160]; -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct FlashLoanFee { pub fee_ratio: Rational, } +#[derive(Clone, Debug, PartialEq)] +pub struct FlashLoanFeeAccount(pub FlashLoanFee); +impl FlashLoanFeeAccount { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + use std::io::Read; + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != FLASH_LOAN_FEE_ACCOUNT_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + FLASH_LOAN_FEE_ACCOUNT_DISCM, maybe_discm + ), + )); + } + Ok(Self(FlashLoanFee::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&FLASH_LOAN_FEE_ACCOUNT_DISCM)?; + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) + } +} pub const POOL_ACCOUNT_DISCM: [u8; 8] = [241, 154, 109, 4, 17, 177, 109, 188]; -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct Pool { pub fee_authority: Pubkey, pub lp_mint: Pubkey, pub incoming_stake: u64, } +#[derive(Clone, Debug, PartialEq)] +pub struct PoolAccount(pub Pool); +impl PoolAccount { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + use std::io::Read; + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != POOL_ACCOUNT_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + POOL_ACCOUNT_DISCM, maybe_discm + ), + )); + } + Ok(Self(Pool::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&POOL_ACCOUNT_DISCM)?; + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) + } +} pub const PROTOCOL_FEE_ACCOUNT_DISCM: [u8; 8] = [121, 127, 98, 139, 72, 110, 44, 118]; -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct ProtocolFee { pub destination: Pubkey, @@ -30,9 +117,67 @@ pub struct ProtocolFee { pub fee_ratio: Rational, pub referrer_fee_ratio: Rational, } +#[derive(Clone, Debug, PartialEq)] +pub struct ProtocolFeeAccount(pub ProtocolFee); +impl ProtocolFeeAccount { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + use std::io::Read; + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != PROTOCOL_FEE_ACCOUNT_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + PROTOCOL_FEE_ACCOUNT_DISCM, maybe_discm + ), + )); + } + Ok(Self(ProtocolFee::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&PROTOCOL_FEE_ACCOUNT_DISCM)?; + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) + } +} pub const STAKE_ACCOUNT_RECORD_ACCOUNT_DISCM: [u8; 8] = [144, 205, 183, 241, 3, 250, 208, 215]; -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct StakeAccountRecord { pub lamports_at_creation: u64, } +#[derive(Clone, Debug, PartialEq)] +pub struct StakeAccountRecordAccount(pub StakeAccountRecord); +impl StakeAccountRecordAccount { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + use std::io::Read; + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != STAKE_ACCOUNT_RECORD_ACCOUNT_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + STAKE_ACCOUNT_RECORD_ACCOUNT_DISCM, maybe_discm + ), + )); + } + Ok(Self(StakeAccountRecord::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&STAKE_ACCOUNT_RECORD_ACCOUNT_DISCM)?; + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) + } +} diff --git a/unstake_interface/src/errors.rs b/unstake_interface/src/errors.rs index 27ff7e7..340464c 100644 --- a/unstake_interface/src/errors.rs +++ b/unstake_interface/src/errors.rs @@ -1,47 +1,49 @@ use solana_program::{ - decode_error::DecodeError, msg, program_error::{PrintProgramError, ProgramError}, + decode_error::DecodeError, + msg, + program_error::{PrintProgramError, ProgramError}, }; use thiserror::Error; #[derive(Clone, Copy, Debug, Eq, Error, num_derive::FromPrimitive, PartialEq)] pub enum UnstakeError { #[error("The provided LP token account is invalid")] - InvalidLpTokenAccount = 6000u32, + InvalidLpTokenAccount = 6000, #[error("Could not find PDA bump")] - PdaBumpNotCached = 6001u32, + PdaBumpNotCached = 6001, #[error( "The provided fee authority does not have the authority over the provided pool account" )] - InvalidFeeAuthority = 6002u32, + InvalidFeeAuthority = 6002, #[error( "The Authorized of the given stake account is None (possibly an uninitialized stake account was given)" )] - StakeAccountAuthorizedNotRetrievable = 6003u32, + StakeAccountAuthorizedNotRetrievable = 6003, #[error( "The Lockup of the given stake account is None (possibly an uninitialized stake account was given)" )] - StakeAccountLockupNotRetrievable = 6004u32, + StakeAccountLockupNotRetrievable = 6004, #[error("The provided stake account is locked up")] - StakeAccountLockupInForce = 6005u32, + StakeAccountLockupInForce = 6005, #[error("The provided description of fee violates the invariants")] - InvalidFee = 6006u32, + InvalidFee = 6006, #[error("Internal Error")] - InternalError = 6007u32, + InternalError = 6007, #[error("Not enough liquidity to service this unstake")] - NotEnoughLiquidity = 6008u32, + NotEnoughLiquidity = 6008, #[error("Liquidity to add too little")] - LiquidityToAddTooLittle = 6009u32, + LiquidityToAddTooLittle = 6009, #[error("Destination token account is not a wrapped SOL account")] - DestinationNotWSol = 6010u32, + DestinationNotWSol = 6010, #[error("Wrong protocol fee destination account")] - WrongProtocolFeeDestination = 6011u32, + WrongProtocolFeeDestination = 6011, #[error( "The provided protocol fee authority does not have the authority over the protocol fee account" )] - InvalidProtocolFeeAuthority = 6012u32, + InvalidProtocolFeeAuthority = 6012, #[error("Invalid instructions sysvar")] - InvalidInstructionsSysvar = 6013u32, + InvalidInstructionsSysvar = 6013, #[error("No succeeding repay flash loan instruction found")] - NoSucceedingRepayFlashLoan = 6014u32, + NoSucceedingRepayFlashLoan = 6014, } impl From for ProgramError { fn from(e: UnstakeError) -> Self { @@ -56,9 +58,12 @@ impl DecodeError for UnstakeError { impl PrintProgramError for UnstakeError { fn print(&self) where - E: 'static + std::error::Error + DecodeError + PrintProgramError + E: 'static + + std::error::Error + + DecodeError + + PrintProgramError + num_traits::FromPrimitive, { - msg!(& self.to_string()); + msg!(&self.to_string()); } } diff --git a/unstake_interface/src/instructions.rs b/unstake_interface/src/instructions.rs index 91cdd23..59153dc 100644 --- a/unstake_interface/src/instructions.rs +++ b/unstake_interface/src/instructions.rs @@ -5,23 +5,133 @@ use solana_program::{ entrypoint::ProgramResult, instruction::{AccountMeta, Instruction}, program::{invoke, invoke_signed}, + program_error::ProgramError, pubkey::Pubkey, }; -pub const INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN: usize = 3usize; -#[derive(Copy, Clone, Debug)] -pub struct InitProtocolFeeAccounts<'me, 'a0: 'me, 'a1: 'me, 'a2: 'me> { - pub payer: &'me AccountInfo<'a0>, - pub protocol_fee_account: &'me AccountInfo<'a1>, - pub system_program: &'me AccountInfo<'a2>, +use std::io::Read; +#[derive(Clone, Debug, PartialEq)] +pub enum UnstakeProgramIx { + InitProtocolFee, + SetProtocolFee(SetProtocolFeeIxArgs), + CreatePool(CreatePoolIxArgs), + AddLiquidity(AddLiquidityIxArgs), + RemoveLiquidity(RemoveLiquidityIxArgs), + SetFee(SetFeeIxArgs), + SetFeeAuthority, + SetLpTokenMetadata(SetLpTokenMetadataIxArgs), + DeactivateStakeAccount, + ReclaimStakeAccount, + Unstake, + UnstakeWsol, + SetFlashLoanFee(SetFlashLoanFeeIxArgs), + TakeFlashLoan(TakeFlashLoanIxArgs), + RepayFlashLoan, +} +impl UnstakeProgramIx { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + match maybe_discm { + INIT_PROTOCOL_FEE_IX_DISCM => Ok(Self::InitProtocolFee), + SET_PROTOCOL_FEE_IX_DISCM => Ok(Self::SetProtocolFee( + SetProtocolFeeIxArgs::deserialize(&mut reader)?, + )), + CREATE_POOL_IX_DISCM => Ok(Self::CreatePool(CreatePoolIxArgs::deserialize( + &mut reader, + )?)), + ADD_LIQUIDITY_IX_DISCM => Ok(Self::AddLiquidity(AddLiquidityIxArgs::deserialize( + &mut reader, + )?)), + REMOVE_LIQUIDITY_IX_DISCM => Ok(Self::RemoveLiquidity( + RemoveLiquidityIxArgs::deserialize(&mut reader)?, + )), + SET_FEE_IX_DISCM => Ok(Self::SetFee(SetFeeIxArgs::deserialize(&mut reader)?)), + SET_FEE_AUTHORITY_IX_DISCM => Ok(Self::SetFeeAuthority), + SET_LP_TOKEN_METADATA_IX_DISCM => Ok(Self::SetLpTokenMetadata( + SetLpTokenMetadataIxArgs::deserialize(&mut reader)?, + )), + DEACTIVATE_STAKE_ACCOUNT_IX_DISCM => Ok(Self::DeactivateStakeAccount), + RECLAIM_STAKE_ACCOUNT_IX_DISCM => Ok(Self::ReclaimStakeAccount), + UNSTAKE_IX_DISCM => Ok(Self::Unstake), + UNSTAKE_WSOL_IX_DISCM => Ok(Self::UnstakeWsol), + SET_FLASH_LOAN_FEE_IX_DISCM => Ok(Self::SetFlashLoanFee( + SetFlashLoanFeeIxArgs::deserialize(&mut reader)?, + )), + TAKE_FLASH_LOAN_IX_DISCM => Ok(Self::TakeFlashLoan(TakeFlashLoanIxArgs::deserialize( + &mut reader, + )?)), + REPAY_FLASH_LOAN_IX_DISCM => Ok(Self::RepayFlashLoan), + _ => Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!("discm {:?} not found", maybe_discm), + )), + } + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + match self { + Self::InitProtocolFee => writer.write_all(&INIT_PROTOCOL_FEE_IX_DISCM), + Self::SetProtocolFee(args) => { + writer.write_all(&SET_PROTOCOL_FEE_IX_DISCM)?; + args.serialize(&mut writer) + } + Self::CreatePool(args) => { + writer.write_all(&CREATE_POOL_IX_DISCM)?; + args.serialize(&mut writer) + } + Self::AddLiquidity(args) => { + writer.write_all(&ADD_LIQUIDITY_IX_DISCM)?; + args.serialize(&mut writer) + } + Self::RemoveLiquidity(args) => { + writer.write_all(&REMOVE_LIQUIDITY_IX_DISCM)?; + args.serialize(&mut writer) + } + Self::SetFee(args) => { + writer.write_all(&SET_FEE_IX_DISCM)?; + args.serialize(&mut writer) + } + Self::SetFeeAuthority => writer.write_all(&SET_FEE_AUTHORITY_IX_DISCM), + Self::SetLpTokenMetadata(args) => { + writer.write_all(&SET_LP_TOKEN_METADATA_IX_DISCM)?; + args.serialize(&mut writer) + } + Self::DeactivateStakeAccount => writer.write_all(&DEACTIVATE_STAKE_ACCOUNT_IX_DISCM), + Self::ReclaimStakeAccount => writer.write_all(&RECLAIM_STAKE_ACCOUNT_IX_DISCM), + Self::Unstake => writer.write_all(&UNSTAKE_IX_DISCM), + Self::UnstakeWsol => writer.write_all(&UNSTAKE_WSOL_IX_DISCM), + Self::SetFlashLoanFee(args) => { + writer.write_all(&SET_FLASH_LOAN_FEE_IX_DISCM)?; + args.serialize(&mut writer) + } + Self::TakeFlashLoan(args) => { + writer.write_all(&TAKE_FLASH_LOAN_IX_DISCM)?; + args.serialize(&mut writer) + } + Self::RepayFlashLoan => writer.write_all(&REPAY_FLASH_LOAN_IX_DISCM), + } + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) + } } +pub const INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN: usize = 3; #[derive(Copy, Clone, Debug)] +pub struct InitProtocolFeeAccounts<'me, 'info> { + pub payer: &'me AccountInfo<'info>, + pub protocol_fee_account: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct InitProtocolFeeKeys { pub payer: Pubkey, pub protocol_fee_account: Pubkey, pub system_program: Pubkey, } -impl<'me> From<&InitProtocolFeeAccounts<'me, '_, '_, '_>> for InitProtocolFeeKeys { - fn from(accounts: &InitProtocolFeeAccounts<'me, '_, '_, '_>) -> Self { +impl From> for InitProtocolFeeKeys { + fn from(accounts: InitProtocolFeeAccounts) -> Self { Self { payer: *accounts.payer.key, protocol_fee_account: *accounts.protocol_fee_account.key, @@ -29,19 +139,40 @@ impl<'me> From<&InitProtocolFeeAccounts<'me, '_, '_, '_>> for InitProtocolFeeKey } } } -impl From<&InitProtocolFeeKeys> for [AccountMeta; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] { - fn from(keys: &InitProtocolFeeKeys) -> Self { +impl From for [AccountMeta; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] { + fn from(keys: InitProtocolFeeKeys) -> Self { [ - AccountMeta::new(keys.payer, true), - AccountMeta::new(keys.protocol_fee_account, false), - AccountMeta::new_readonly(keys.system_program, false), + AccountMeta { + pubkey: keys.payer, + is_signer: true, + is_writable: true, + }, + AccountMeta { + pubkey: keys.protocol_fee_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&InitProtocolFeeAccounts<'_, 'a, 'a, 'a>> - for [AccountInfo<'a>; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] +impl From<[Pubkey; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN]> for InitProtocolFeeKeys { + fn from(pubkeys: [Pubkey; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN]) -> Self { + Self { + payer: pubkeys[0], + protocol_fee_account: pubkeys[1], + system_program: pubkeys[2], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] { - fn from(accounts: &InitProtocolFeeAccounts<'_, 'a, 'a, 'a>) -> Self { + fn from(accounts: InitProtocolFeeAccounts<'_, 'info>) -> Self { [ accounts.payer.clone(), accounts.protocol_fee_account.clone(), @@ -49,108 +180,204 @@ impl<'a> From<&InitProtocolFeeAccounts<'_, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] -#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] -pub struct InitProtocolFeeIxArgs {} -#[derive(Copy, Clone, Debug)] -pub struct InitProtocolFeeIxData<'me>(pub &'me InitProtocolFeeIxArgs); -pub const INIT_PROTOCOL_FEE_IX_DISCM: [u8; 8] = [225, 155, 167, 170, 29, 145, 165, 90]; -impl<'me> From<&'me InitProtocolFeeIxArgs> for InitProtocolFeeIxData<'me> { - fn from(args: &'me InitProtocolFeeIxArgs) -> Self { - Self(args) +impl<'me, 'info> From<&'me [AccountInfo<'info>; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN]> + for InitProtocolFeeAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN]) -> Self { + Self { + payer: &arr[0], + protocol_fee_account: &arr[1], + system_program: &arr[2], + } } } -impl BorshSerialize for InitProtocolFeeIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { - writer.write_all(&INIT_PROTOCOL_FEE_IX_DISCM)?; - self.0.serialize(writer) +pub const INIT_PROTOCOL_FEE_IX_DISCM: [u8; 8] = [225, 155, 167, 170, 29, 145, 165, 90]; +#[derive(Clone, Debug, PartialEq)] +pub struct InitProtocolFeeIxData; +impl InitProtocolFeeIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != INIT_PROTOCOL_FEE_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + INIT_PROTOCOL_FEE_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&INIT_PROTOCOL_FEE_IX_DISCM) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } -pub fn init_protocol_fee_ix, A: Into>( +pub fn init_protocol_fee_ix>( accounts: K, - args: A, ) -> std::io::Result { let keys: InitProtocolFeeKeys = accounts.into(); - let metas: [AccountMeta; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = (&keys).into(); - let args_full: InitProtocolFeeIxArgs = args.into(); - let data: InitProtocolFeeIxData = (&args_full).into(); + let metas: [AccountMeta; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = keys.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), - data: data.try_to_vec()?, + data: InitProtocolFeeIxData.try_to_vec()?, }) } -pub fn init_protocol_fee_invoke<'a, A: Into>( - accounts: &InitProtocolFeeAccounts<'_, 'a, 'a, 'a>, - args: A, +pub fn init_protocol_fee_invoke<'info>( + accounts: InitProtocolFeeAccounts<'_, 'info>, ) -> ProgramResult { - let ix = init_protocol_fee_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = init_protocol_fee_ix(accounts)?; + let account_info: [AccountInfo<'info>; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn init_protocol_fee_invoke_signed<'a, A: Into>( - accounts: &InitProtocolFeeAccounts<'_, 'a, 'a, 'a>, - args: A, +pub fn init_protocol_fee_invoke_signed<'info>( + accounts: InitProtocolFeeAccounts<'_, 'info>, seeds: &[&[&[u8]]], ) -> ProgramResult { - let ix = init_protocol_fee_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = init_protocol_fee_ix(accounts)?; + let account_info: [AccountInfo<'info>; INIT_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN: usize = 2usize; -#[derive(Copy, Clone, Debug)] -pub struct SetProtocolFeeAccounts<'me, 'a0: 'me, 'a1: 'me> { - pub authority: &'me AccountInfo<'a0>, - pub protocol_fee_account: &'me AccountInfo<'a1>, +pub fn init_protocol_fee_verify_account_keys( + accounts: InitProtocolFeeAccounts<'_, '_>, + keys: InitProtocolFeeKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.payer.key, keys.payer), + ( + *accounts.protocol_fee_account.key, + keys.protocol_fee_account, + ), + (*accounts.system_program.key, keys.system_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn init_protocol_fee_verify_account_privileges<'me, 'info>( + accounts: InitProtocolFeeAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [accounts.payer, accounts.protocol_fee_account] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.payer] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN: usize = 2; #[derive(Copy, Clone, Debug)] +pub struct SetProtocolFeeAccounts<'me, 'info> { + pub authority: &'me AccountInfo<'info>, + pub protocol_fee_account: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct SetProtocolFeeKeys { pub authority: Pubkey, pub protocol_fee_account: Pubkey, } -impl<'me> From<&SetProtocolFeeAccounts<'me, '_, '_>> for SetProtocolFeeKeys { - fn from(accounts: &SetProtocolFeeAccounts<'me, '_, '_>) -> Self { +impl From> for SetProtocolFeeKeys { + fn from(accounts: SetProtocolFeeAccounts) -> Self { Self { authority: *accounts.authority.key, protocol_fee_account: *accounts.protocol_fee_account.key, } } } -impl From<&SetProtocolFeeKeys> for [AccountMeta; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] { - fn from(keys: &SetProtocolFeeKeys) -> Self { +impl From for [AccountMeta; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] { + fn from(keys: SetProtocolFeeKeys) -> Self { [ - AccountMeta::new_readonly(keys.authority, true), - AccountMeta::new(keys.protocol_fee_account, false), + AccountMeta { + pubkey: keys.authority, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.protocol_fee_account, + is_signer: false, + is_writable: true, + }, ] } } -impl<'a> From<&SetProtocolFeeAccounts<'_, 'a, 'a>> - for [AccountInfo<'a>; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] +impl From<[Pubkey; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN]> for SetProtocolFeeKeys { + fn from(pubkeys: [Pubkey; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN]) -> Self { + Self { + authority: pubkeys[0], + protocol_fee_account: pubkeys[1], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] { - fn from(accounts: &SetProtocolFeeAccounts<'_, 'a, 'a>) -> Self { + fn from(accounts: SetProtocolFeeAccounts<'_, 'info>) -> Self { [ accounts.authority.clone(), accounts.protocol_fee_account.clone(), ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] +impl<'me, 'info> From<&'me [AccountInfo<'info>; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN]> + for SetProtocolFeeAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN]) -> Self { + Self { + authority: &arr[0], + protocol_fee_account: &arr[1], + } + } +} +pub const SET_PROTOCOL_FEE_IX_DISCM: [u8; 8] = [173, 239, 83, 242, 136, 43, 144, 217]; +#[derive(BorshDeserialize, BorshSerialize, Clone, Debug, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct SetProtocolFeeIxArgs { pub protocol_fee: ProtocolFee, } -#[derive(Copy, Clone, Debug)] -pub struct SetProtocolFeeIxData<'me>(pub &'me SetProtocolFeeIxArgs); -pub const SET_PROTOCOL_FEE_IX_DISCM: [u8; 8] = [173, 239, 83, 242, 136, 43, 144, 217]; -impl<'me> From<&'me SetProtocolFeeIxArgs> for SetProtocolFeeIxData<'me> { - fn from(args: &'me SetProtocolFeeIxArgs) -> Self { +#[derive(Clone, Debug, PartialEq)] +pub struct SetProtocolFeeIxData(pub SetProtocolFeeIxArgs); +impl From for SetProtocolFeeIxData { + fn from(args: SetProtocolFeeIxArgs) -> Self { Self(args) } } -impl BorshSerialize for SetProtocolFeeIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { +impl SetProtocolFeeIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != SET_PROTOCOL_FEE_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + SET_PROTOCOL_FEE_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self(SetProtocolFeeIxArgs::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { writer.write_all(&SET_PROTOCOL_FEE_IX_DISCM)?; - self.0.serialize(writer) + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } pub fn set_protocol_fee_ix, A: Into>( @@ -158,57 +385,78 @@ pub fn set_protocol_fee_ix, A: Into std::io::Result { let keys: SetProtocolFeeKeys = accounts.into(); - let metas: [AccountMeta; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = (&keys).into(); + let metas: [AccountMeta; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = keys.into(); let args_full: SetProtocolFeeIxArgs = args.into(); - let data: SetProtocolFeeIxData = (&args_full).into(); + let data: SetProtocolFeeIxData = args_full.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), data: data.try_to_vec()?, }) } -pub fn set_protocol_fee_invoke<'a, A: Into>( - accounts: &SetProtocolFeeAccounts<'_, 'a, 'a>, +pub fn set_protocol_fee_invoke<'info, A: Into>( + accounts: SetProtocolFeeAccounts<'_, 'info>, args: A, ) -> ProgramResult { let ix = set_protocol_fee_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn set_protocol_fee_invoke_signed<'a, A: Into>( - accounts: &SetProtocolFeeAccounts<'_, 'a, 'a>, +pub fn set_protocol_fee_invoke_signed<'info, A: Into>( + accounts: SetProtocolFeeAccounts<'_, 'info>, args: A, seeds: &[&[&[u8]]], ) -> ProgramResult { let ix = set_protocol_fee_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; SET_PROTOCOL_FEE_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const CREATE_POOL_IX_ACCOUNTS_LEN: usize = 9usize; -#[derive(Copy, Clone, Debug)] -pub struct CreatePoolAccounts< - 'me, - 'a0: 'me, - 'a1: 'me, - 'a2: 'me, - 'a3: 'me, - 'a4: 'me, - 'a5: 'me, - 'a6: 'me, - 'a7: 'me, - 'a8: 'me, -> { - pub payer: &'me AccountInfo<'a0>, - pub fee_authority: &'me AccountInfo<'a1>, - pub pool_account: &'me AccountInfo<'a2>, - pub pool_sol_reserves: &'me AccountInfo<'a3>, - pub fee_account: &'me AccountInfo<'a4>, - pub lp_mint: &'me AccountInfo<'a5>, - pub token_program: &'me AccountInfo<'a6>, - pub system_program: &'me AccountInfo<'a7>, - pub rent: &'me AccountInfo<'a8>, +pub fn set_protocol_fee_verify_account_keys( + accounts: SetProtocolFeeAccounts<'_, '_>, + keys: SetProtocolFeeKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.authority.key, keys.authority), + ( + *accounts.protocol_fee_account.key, + keys.protocol_fee_account, + ), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn set_protocol_fee_verify_account_privileges<'me, 'info>( + accounts: SetProtocolFeeAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [accounts.protocol_fee_account] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.authority] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const CREATE_POOL_IX_ACCOUNTS_LEN: usize = 9; #[derive(Copy, Clone, Debug)] +pub struct CreatePoolAccounts<'me, 'info> { + pub payer: &'me AccountInfo<'info>, + pub fee_authority: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub fee_account: &'me AccountInfo<'info>, + pub lp_mint: &'me AccountInfo<'info>, + pub token_program: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, + pub rent: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct CreatePoolKeys { pub payer: Pubkey, pub fee_authority: Pubkey, @@ -220,8 +468,8 @@ pub struct CreatePoolKeys { pub system_program: Pubkey, pub rent: Pubkey, } -impl<'me> From<&CreatePoolAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_>> for CreatePoolKeys { - fn from(accounts: &CreatePoolAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_>) -> Self { +impl From> for CreatePoolKeys { + fn from(accounts: CreatePoolAccounts) -> Self { Self { payer: *accounts.payer.key, fee_authority: *accounts.fee_authority.key, @@ -235,25 +483,76 @@ impl<'me> From<&CreatePoolAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_>> for } } } -impl From<&CreatePoolKeys> for [AccountMeta; CREATE_POOL_IX_ACCOUNTS_LEN] { - fn from(keys: &CreatePoolKeys) -> Self { +impl From for [AccountMeta; CREATE_POOL_IX_ACCOUNTS_LEN] { + fn from(keys: CreatePoolKeys) -> Self { [ - AccountMeta::new(keys.payer, true), - AccountMeta::new_readonly(keys.fee_authority, true), - AccountMeta::new(keys.pool_account, true), - AccountMeta::new_readonly(keys.pool_sol_reserves, false), - AccountMeta::new(keys.fee_account, false), - AccountMeta::new(keys.lp_mint, true), - AccountMeta::new_readonly(keys.token_program, false), - AccountMeta::new_readonly(keys.system_program, false), - AccountMeta::new_readonly(keys.rent, false), + AccountMeta { + pubkey: keys.payer, + is_signer: true, + is_writable: true, + }, + AccountMeta { + pubkey: keys.fee_authority, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: true, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.fee_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.lp_mint, + is_signer: true, + is_writable: true, + }, + AccountMeta { + pubkey: keys.token_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.rent, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&CreatePoolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; CREATE_POOL_IX_ACCOUNTS_LEN] +impl From<[Pubkey; CREATE_POOL_IX_ACCOUNTS_LEN]> for CreatePoolKeys { + fn from(pubkeys: [Pubkey; CREATE_POOL_IX_ACCOUNTS_LEN]) -> Self { + Self { + payer: pubkeys[0], + fee_authority: pubkeys[1], + pool_account: pubkeys[2], + pool_sol_reserves: pubkeys[3], + fee_account: pubkeys[4], + lp_mint: pubkeys[5], + token_program: pubkeys[6], + system_program: pubkeys[7], + rent: pubkeys[8], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; CREATE_POOL_IX_ACCOUNTS_LEN] { - fn from(accounts: &CreatePoolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: CreatePoolAccounts<'_, 'info>) -> Self { [ accounts.payer.clone(), accounts.fee_authority.clone(), @@ -267,23 +566,60 @@ impl<'a> From<&CreatePoolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] +impl<'me, 'info> From<&'me [AccountInfo<'info>; CREATE_POOL_IX_ACCOUNTS_LEN]> + for CreatePoolAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; CREATE_POOL_IX_ACCOUNTS_LEN]) -> Self { + Self { + payer: &arr[0], + fee_authority: &arr[1], + pool_account: &arr[2], + pool_sol_reserves: &arr[3], + fee_account: &arr[4], + lp_mint: &arr[5], + token_program: &arr[6], + system_program: &arr[7], + rent: &arr[8], + } + } +} +pub const CREATE_POOL_IX_DISCM: [u8; 8] = [233, 146, 209, 142, 207, 104, 64, 188]; +#[derive(BorshDeserialize, BorshSerialize, Clone, Debug, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct CreatePoolIxArgs { pub fee: Fee, } -#[derive(Copy, Clone, Debug)] -pub struct CreatePoolIxData<'me>(pub &'me CreatePoolIxArgs); -pub const CREATE_POOL_IX_DISCM: [u8; 8] = [233, 146, 209, 142, 207, 104, 64, 188]; -impl<'me> From<&'me CreatePoolIxArgs> for CreatePoolIxData<'me> { - fn from(args: &'me CreatePoolIxArgs) -> Self { +#[derive(Clone, Debug, PartialEq)] +pub struct CreatePoolIxData(pub CreatePoolIxArgs); +impl From for CreatePoolIxData { + fn from(args: CreatePoolIxArgs) -> Self { Self(args) } } -impl BorshSerialize for CreatePoolIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { +impl CreatePoolIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != CREATE_POOL_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + CREATE_POOL_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self(CreatePoolIxArgs::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { writer.write_all(&CREATE_POOL_IX_DISCM)?; - self.0.serialize(writer) + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } pub fn create_pool_ix, A: Into>( @@ -291,55 +627,91 @@ pub fn create_pool_ix, A: Into>( args: A, ) -> std::io::Result { let keys: CreatePoolKeys = accounts.into(); - let metas: [AccountMeta; CREATE_POOL_IX_ACCOUNTS_LEN] = (&keys).into(); + let metas: [AccountMeta; CREATE_POOL_IX_ACCOUNTS_LEN] = keys.into(); let args_full: CreatePoolIxArgs = args.into(); - let data: CreatePoolIxData = (&args_full).into(); + let data: CreatePoolIxData = args_full.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), data: data.try_to_vec()?, }) } -pub fn create_pool_invoke<'a, A: Into>( - accounts: &CreatePoolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn create_pool_invoke<'info, A: Into>( + accounts: CreatePoolAccounts<'_, 'info>, args: A, ) -> ProgramResult { let ix = create_pool_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; CREATE_POOL_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; CREATE_POOL_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn create_pool_invoke_signed<'a, A: Into>( - accounts: &CreatePoolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn create_pool_invoke_signed<'info, A: Into>( + accounts: CreatePoolAccounts<'_, 'info>, args: A, seeds: &[&[&[u8]]], ) -> ProgramResult { let ix = create_pool_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; CREATE_POOL_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; CREATE_POOL_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const ADD_LIQUIDITY_IX_ACCOUNTS_LEN: usize = 8usize; -#[derive(Copy, Clone, Debug)] -pub struct AddLiquidityAccounts< - 'me, - 'a0: 'me, - 'a1: 'me, - 'a2: 'me, - 'a3: 'me, - 'a4: 'me, - 'a5: 'me, - 'a6: 'me, - 'a7: 'me, -> { - pub from: &'me AccountInfo<'a0>, - pub pool_account: &'me AccountInfo<'a1>, - pub pool_sol_reserves: &'me AccountInfo<'a2>, - pub lp_mint: &'me AccountInfo<'a3>, - pub mint_lp_tokens_to: &'me AccountInfo<'a4>, - pub flash_account: &'me AccountInfo<'a5>, - pub token_program: &'me AccountInfo<'a6>, - pub system_program: &'me AccountInfo<'a7>, +pub fn create_pool_verify_account_keys( + accounts: CreatePoolAccounts<'_, '_>, + keys: CreatePoolKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.payer.key, keys.payer), + (*accounts.fee_authority.key, keys.fee_authority), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.fee_account.key, keys.fee_account), + (*accounts.lp_mint.key, keys.lp_mint), + (*accounts.token_program.key, keys.token_program), + (*accounts.system_program.key, keys.system_program), + (*accounts.rent.key, keys.rent), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn create_pool_verify_account_privileges<'me, 'info>( + accounts: CreatePoolAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [ + accounts.payer, + accounts.pool_account, + accounts.fee_account, + accounts.lp_mint, + ] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [ + accounts.payer, + accounts.fee_authority, + accounts.pool_account, + accounts.lp_mint, + ] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const ADD_LIQUIDITY_IX_ACCOUNTS_LEN: usize = 8; #[derive(Copy, Clone, Debug)] +pub struct AddLiquidityAccounts<'me, 'info> { + pub from: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub lp_mint: &'me AccountInfo<'info>, + pub mint_lp_tokens_to: &'me AccountInfo<'info>, + pub flash_account: &'me AccountInfo<'info>, + pub token_program: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct AddLiquidityKeys { pub from: Pubkey, pub pool_account: Pubkey, @@ -350,8 +722,8 @@ pub struct AddLiquidityKeys { pub token_program: Pubkey, pub system_program: Pubkey, } -impl<'me> From<&AddLiquidityAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_>> for AddLiquidityKeys { - fn from(accounts: &AddLiquidityAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_>) -> Self { +impl From> for AddLiquidityKeys { + fn from(accounts: AddLiquidityAccounts) -> Self { Self { from: *accounts.from.key, pool_account: *accounts.pool_account.key, @@ -364,24 +736,70 @@ impl<'me> From<&AddLiquidityAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_>> for A } } } -impl From<&AddLiquidityKeys> for [AccountMeta; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] { - fn from(keys: &AddLiquidityKeys) -> Self { +impl From for [AccountMeta; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] { + fn from(keys: AddLiquidityKeys) -> Self { [ - AccountMeta::new(keys.from, true), - AccountMeta::new(keys.pool_account, false), - AccountMeta::new(keys.pool_sol_reserves, false), - AccountMeta::new(keys.lp_mint, false), - AccountMeta::new(keys.mint_lp_tokens_to, false), - AccountMeta::new(keys.flash_account, false), - AccountMeta::new_readonly(keys.token_program, false), - AccountMeta::new_readonly(keys.system_program, false), + AccountMeta { + pubkey: keys.from, + is_signer: true, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.lp_mint, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.mint_lp_tokens_to, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.flash_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.token_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&AddLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] +impl From<[Pubkey; ADD_LIQUIDITY_IX_ACCOUNTS_LEN]> for AddLiquidityKeys { + fn from(pubkeys: [Pubkey; ADD_LIQUIDITY_IX_ACCOUNTS_LEN]) -> Self { + Self { + from: pubkeys[0], + pool_account: pubkeys[1], + pool_sol_reserves: pubkeys[2], + lp_mint: pubkeys[3], + mint_lp_tokens_to: pubkeys[4], + flash_account: pubkeys[5], + token_program: pubkeys[6], + system_program: pubkeys[7], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] { - fn from(accounts: &AddLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: AddLiquidityAccounts<'_, 'info>) -> Self { [ accounts.from.clone(), accounts.pool_account.clone(), @@ -394,23 +812,59 @@ impl<'a> From<&AddLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] +impl<'me, 'info> From<&'me [AccountInfo<'info>; ADD_LIQUIDITY_IX_ACCOUNTS_LEN]> + for AddLiquidityAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; ADD_LIQUIDITY_IX_ACCOUNTS_LEN]) -> Self { + Self { + from: &arr[0], + pool_account: &arr[1], + pool_sol_reserves: &arr[2], + lp_mint: &arr[3], + mint_lp_tokens_to: &arr[4], + flash_account: &arr[5], + token_program: &arr[6], + system_program: &arr[7], + } + } +} +pub const ADD_LIQUIDITY_IX_DISCM: [u8; 8] = [181, 157, 89, 67, 143, 182, 52, 72]; +#[derive(BorshDeserialize, BorshSerialize, Clone, Debug, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct AddLiquidityIxArgs { pub amount: u64, } -#[derive(Copy, Clone, Debug)] -pub struct AddLiquidityIxData<'me>(pub &'me AddLiquidityIxArgs); -pub const ADD_LIQUIDITY_IX_DISCM: [u8; 8] = [181, 157, 89, 67, 143, 182, 52, 72]; -impl<'me> From<&'me AddLiquidityIxArgs> for AddLiquidityIxData<'me> { - fn from(args: &'me AddLiquidityIxArgs) -> Self { +#[derive(Clone, Debug, PartialEq)] +pub struct AddLiquidityIxData(pub AddLiquidityIxArgs); +impl From for AddLiquidityIxData { + fn from(args: AddLiquidityIxArgs) -> Self { Self(args) } } -impl BorshSerialize for AddLiquidityIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { +impl AddLiquidityIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != ADD_LIQUIDITY_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + ADD_LIQUIDITY_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self(AddLiquidityIxArgs::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { writer.write_all(&ADD_LIQUIDITY_IX_DISCM)?; - self.0.serialize(writer) + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } pub fn add_liquidity_ix, A: Into>( @@ -418,57 +872,88 @@ pub fn add_liquidity_ix, A: Into>( args: A, ) -> std::io::Result { let keys: AddLiquidityKeys = accounts.into(); - let metas: [AccountMeta; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] = (&keys).into(); + let metas: [AccountMeta; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] = keys.into(); let args_full: AddLiquidityIxArgs = args.into(); - let data: AddLiquidityIxData = (&args_full).into(); + let data: AddLiquidityIxData = args_full.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), data: data.try_to_vec()?, }) } -pub fn add_liquidity_invoke<'a, A: Into>( - accounts: &AddLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn add_liquidity_invoke<'info, A: Into>( + accounts: AddLiquidityAccounts<'_, 'info>, args: A, ) -> ProgramResult { let ix = add_liquidity_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn add_liquidity_invoke_signed<'a, A: Into>( - accounts: &AddLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn add_liquidity_invoke_signed<'info, A: Into>( + accounts: AddLiquidityAccounts<'_, 'info>, args: A, seeds: &[&[&[u8]]], ) -> ProgramResult { let ix = add_liquidity_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; ADD_LIQUIDITY_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN: usize = 9usize; -#[derive(Copy, Clone, Debug)] -pub struct RemoveLiquidityAccounts< - 'me, - 'a0: 'me, - 'a1: 'me, - 'a2: 'me, - 'a3: 'me, - 'a4: 'me, - 'a5: 'me, - 'a6: 'me, - 'a7: 'me, - 'a8: 'me, -> { - pub burn_lp_tokens_from_authority: &'me AccountInfo<'a0>, - pub to: &'me AccountInfo<'a1>, - pub pool_account: &'me AccountInfo<'a2>, - pub pool_sol_reserves: &'me AccountInfo<'a3>, - pub lp_mint: &'me AccountInfo<'a4>, - pub burn_lp_tokens_from: &'me AccountInfo<'a5>, - pub flash_account: &'me AccountInfo<'a6>, - pub token_program: &'me AccountInfo<'a7>, - pub system_program: &'me AccountInfo<'a8>, +pub fn add_liquidity_verify_account_keys( + accounts: AddLiquidityAccounts<'_, '_>, + keys: AddLiquidityKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.from.key, keys.from), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.lp_mint.key, keys.lp_mint), + (*accounts.mint_lp_tokens_to.key, keys.mint_lp_tokens_to), + (*accounts.flash_account.key, keys.flash_account), + (*accounts.token_program.key, keys.token_program), + (*accounts.system_program.key, keys.system_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn add_liquidity_verify_account_privileges<'me, 'info>( + accounts: AddLiquidityAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [ + accounts.from, + accounts.pool_account, + accounts.pool_sol_reserves, + accounts.lp_mint, + accounts.mint_lp_tokens_to, + accounts.flash_account, + ] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.from] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN: usize = 9; #[derive(Copy, Clone, Debug)] +pub struct RemoveLiquidityAccounts<'me, 'info> { + pub burn_lp_tokens_from_authority: &'me AccountInfo<'info>, + pub to: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub lp_mint: &'me AccountInfo<'info>, + pub burn_lp_tokens_from: &'me AccountInfo<'info>, + pub flash_account: &'me AccountInfo<'info>, + pub token_program: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct RemoveLiquidityKeys { pub burn_lp_tokens_from_authority: Pubkey, pub to: Pubkey, @@ -480,10 +965,8 @@ pub struct RemoveLiquidityKeys { pub token_program: Pubkey, pub system_program: Pubkey, } -impl<'me> From<&RemoveLiquidityAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_>> - for RemoveLiquidityKeys -{ - fn from(accounts: &RemoveLiquidityAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_>) -> Self { +impl From> for RemoveLiquidityKeys { + fn from(accounts: RemoveLiquidityAccounts) -> Self { Self { burn_lp_tokens_from_authority: *accounts.burn_lp_tokens_from_authority.key, to: *accounts.to.key, @@ -497,25 +980,76 @@ impl<'me> From<&RemoveLiquidityAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_> } } } -impl From<&RemoveLiquidityKeys> for [AccountMeta; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] { - fn from(keys: &RemoveLiquidityKeys) -> Self { +impl From for [AccountMeta; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] { + fn from(keys: RemoveLiquidityKeys) -> Self { [ - AccountMeta::new_readonly(keys.burn_lp_tokens_from_authority, true), - AccountMeta::new(keys.to, false), - AccountMeta::new(keys.pool_account, false), - AccountMeta::new(keys.pool_sol_reserves, false), - AccountMeta::new(keys.lp_mint, false), - AccountMeta::new(keys.burn_lp_tokens_from, false), - AccountMeta::new(keys.flash_account, false), - AccountMeta::new_readonly(keys.token_program, false), - AccountMeta::new_readonly(keys.system_program, false), + AccountMeta { + pubkey: keys.burn_lp_tokens_from_authority, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.to, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.lp_mint, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.burn_lp_tokens_from, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.flash_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.token_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&RemoveLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] +impl From<[Pubkey; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN]> for RemoveLiquidityKeys { + fn from(pubkeys: [Pubkey; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN]) -> Self { + Self { + burn_lp_tokens_from_authority: pubkeys[0], + to: pubkeys[1], + pool_account: pubkeys[2], + pool_sol_reserves: pubkeys[3], + lp_mint: pubkeys[4], + burn_lp_tokens_from: pubkeys[5], + flash_account: pubkeys[6], + token_program: pubkeys[7], + system_program: pubkeys[8], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] { - fn from(accounts: &RemoveLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: RemoveLiquidityAccounts<'_, 'info>) -> Self { [ accounts.burn_lp_tokens_from_authority.clone(), accounts.to.clone(), @@ -529,23 +1063,60 @@ impl<'a> From<&RemoveLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] +impl<'me, 'info> From<&'me [AccountInfo<'info>; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN]> + for RemoveLiquidityAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN]) -> Self { + Self { + burn_lp_tokens_from_authority: &arr[0], + to: &arr[1], + pool_account: &arr[2], + pool_sol_reserves: &arr[3], + lp_mint: &arr[4], + burn_lp_tokens_from: &arr[5], + flash_account: &arr[6], + token_program: &arr[7], + system_program: &arr[8], + } + } +} +pub const REMOVE_LIQUIDITY_IX_DISCM: [u8; 8] = [80, 85, 209, 72, 24, 206, 177, 108]; +#[derive(BorshDeserialize, BorshSerialize, Clone, Debug, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct RemoveLiquidityIxArgs { pub amount_lp: u64, } -#[derive(Copy, Clone, Debug)] -pub struct RemoveLiquidityIxData<'me>(pub &'me RemoveLiquidityIxArgs); -pub const REMOVE_LIQUIDITY_IX_DISCM: [u8; 8] = [80, 85, 209, 72, 24, 206, 177, 108]; -impl<'me> From<&'me RemoveLiquidityIxArgs> for RemoveLiquidityIxData<'me> { - fn from(args: &'me RemoveLiquidityIxArgs) -> Self { +#[derive(Clone, Debug, PartialEq)] +pub struct RemoveLiquidityIxData(pub RemoveLiquidityIxArgs); +impl From for RemoveLiquidityIxData { + fn from(args: RemoveLiquidityIxArgs) -> Self { Self(args) } } -impl BorshSerialize for RemoveLiquidityIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { +impl RemoveLiquidityIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != REMOVE_LIQUIDITY_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + REMOVE_LIQUIDITY_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self(RemoveLiquidityIxArgs::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { writer.write_all(&REMOVE_LIQUIDITY_IX_DISCM)?; - self.0.serialize(writer) + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } pub fn remove_liquidity_ix, A: Into>( @@ -553,42 +1124,88 @@ pub fn remove_liquidity_ix, A: Into std::io::Result { let keys: RemoveLiquidityKeys = accounts.into(); - let metas: [AccountMeta; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] = (&keys).into(); + let metas: [AccountMeta; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] = keys.into(); let args_full: RemoveLiquidityIxArgs = args.into(); - let data: RemoveLiquidityIxData = (&args_full).into(); + let data: RemoveLiquidityIxData = args_full.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), data: data.try_to_vec()?, }) } -pub fn remove_liquidity_invoke<'a, A: Into>( - accounts: &RemoveLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn remove_liquidity_invoke<'info, A: Into>( + accounts: RemoveLiquidityAccounts<'_, 'info>, args: A, ) -> ProgramResult { let ix = remove_liquidity_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn remove_liquidity_invoke_signed<'a, A: Into>( - accounts: &RemoveLiquidityAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn remove_liquidity_invoke_signed<'info, A: Into>( + accounts: RemoveLiquidityAccounts<'_, 'info>, args: A, seeds: &[&[&[u8]]], ) -> ProgramResult { let ix = remove_liquidity_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; REMOVE_LIQUIDITY_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const SET_FEE_IX_ACCOUNTS_LEN: usize = 5usize; -#[derive(Copy, Clone, Debug)] -pub struct SetFeeAccounts<'me, 'a0: 'me, 'a1: 'me, 'a2: 'me, 'a3: 'me, 'a4: 'me> { - pub fee_authority: &'me AccountInfo<'a0>, - pub pool_account: &'me AccountInfo<'a1>, - pub fee_account: &'me AccountInfo<'a2>, - pub system_program: &'me AccountInfo<'a3>, - pub rent: &'me AccountInfo<'a4>, +pub fn remove_liquidity_verify_account_keys( + accounts: RemoveLiquidityAccounts<'_, '_>, + keys: RemoveLiquidityKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + ( + *accounts.burn_lp_tokens_from_authority.key, + keys.burn_lp_tokens_from_authority, + ), + (*accounts.to.key, keys.to), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.lp_mint.key, keys.lp_mint), + (*accounts.burn_lp_tokens_from.key, keys.burn_lp_tokens_from), + (*accounts.flash_account.key, keys.flash_account), + (*accounts.token_program.key, keys.token_program), + (*accounts.system_program.key, keys.system_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn remove_liquidity_verify_account_privileges<'me, 'info>( + accounts: RemoveLiquidityAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [ + accounts.to, + accounts.pool_account, + accounts.pool_sol_reserves, + accounts.lp_mint, + accounts.burn_lp_tokens_from, + accounts.flash_account, + ] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.burn_lp_tokens_from_authority] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const SET_FEE_IX_ACCOUNTS_LEN: usize = 5; #[derive(Copy, Clone, Debug)] +pub struct SetFeeAccounts<'me, 'info> { + pub fee_authority: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub fee_account: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, + pub rent: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct SetFeeKeys { pub fee_authority: Pubkey, pub pool_account: Pubkey, @@ -596,8 +1213,8 @@ pub struct SetFeeKeys { pub system_program: Pubkey, pub rent: Pubkey, } -impl<'me> From<&SetFeeAccounts<'me, '_, '_, '_, '_, '_>> for SetFeeKeys { - fn from(accounts: &SetFeeAccounts<'me, '_, '_, '_, '_, '_>) -> Self { +impl From> for SetFeeKeys { + fn from(accounts: SetFeeAccounts) -> Self { Self { fee_authority: *accounts.fee_authority.key, pool_account: *accounts.pool_account.key, @@ -607,21 +1224,50 @@ impl<'me> From<&SetFeeAccounts<'me, '_, '_, '_, '_, '_>> for SetFeeKeys { } } } -impl From<&SetFeeKeys> for [AccountMeta; SET_FEE_IX_ACCOUNTS_LEN] { - fn from(keys: &SetFeeKeys) -> Self { +impl From for [AccountMeta; SET_FEE_IX_ACCOUNTS_LEN] { + fn from(keys: SetFeeKeys) -> Self { [ - AccountMeta::new_readonly(keys.fee_authority, true), - AccountMeta::new_readonly(keys.pool_account, false), - AccountMeta::new(keys.fee_account, false), - AccountMeta::new_readonly(keys.system_program, false), - AccountMeta::new_readonly(keys.rent, false), + AccountMeta { + pubkey: keys.fee_authority, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.fee_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.rent, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&SetFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; SET_FEE_IX_ACCOUNTS_LEN] -{ - fn from(accounts: &SetFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>) -> Self { +impl From<[Pubkey; SET_FEE_IX_ACCOUNTS_LEN]> for SetFeeKeys { + fn from(pubkeys: [Pubkey; SET_FEE_IX_ACCOUNTS_LEN]) -> Self { + Self { + fee_authority: pubkeys[0], + pool_account: pubkeys[1], + fee_account: pubkeys[2], + system_program: pubkeys[3], + rent: pubkeys[4], + } + } +} +impl<'info> From> for [AccountInfo<'info>; SET_FEE_IX_ACCOUNTS_LEN] { + fn from(accounts: SetFeeAccounts<'_, 'info>) -> Self { [ accounts.fee_authority.clone(), accounts.pool_account.clone(), @@ -631,23 +1277,56 @@ impl<'a> From<&SetFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] +impl<'me, 'info> From<&'me [AccountInfo<'info>; SET_FEE_IX_ACCOUNTS_LEN]> + for SetFeeAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; SET_FEE_IX_ACCOUNTS_LEN]) -> Self { + Self { + fee_authority: &arr[0], + pool_account: &arr[1], + fee_account: &arr[2], + system_program: &arr[3], + rent: &arr[4], + } + } +} +pub const SET_FEE_IX_DISCM: [u8; 8] = [18, 154, 24, 18, 237, 214, 19, 80]; +#[derive(BorshDeserialize, BorshSerialize, Clone, Debug, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct SetFeeIxArgs { pub fee: Fee, } -#[derive(Copy, Clone, Debug)] -pub struct SetFeeIxData<'me>(pub &'me SetFeeIxArgs); -pub const SET_FEE_IX_DISCM: [u8; 8] = [18, 154, 24, 18, 237, 214, 19, 80]; -impl<'me> From<&'me SetFeeIxArgs> for SetFeeIxData<'me> { - fn from(args: &'me SetFeeIxArgs) -> Self { +#[derive(Clone, Debug, PartialEq)] +pub struct SetFeeIxData(pub SetFeeIxArgs); +impl From for SetFeeIxData { + fn from(args: SetFeeIxArgs) -> Self { Self(args) } } -impl BorshSerialize for SetFeeIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { +impl SetFeeIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != SET_FEE_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + SET_FEE_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self(SetFeeIxArgs::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { writer.write_all(&SET_FEE_IX_DISCM)?; - self.0.serialize(writer) + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } pub fn set_fee_ix, A: Into>( @@ -655,47 +1334,79 @@ pub fn set_fee_ix, A: Into>( args: A, ) -> std::io::Result { let keys: SetFeeKeys = accounts.into(); - let metas: [AccountMeta; SET_FEE_IX_ACCOUNTS_LEN] = (&keys).into(); + let metas: [AccountMeta; SET_FEE_IX_ACCOUNTS_LEN] = keys.into(); let args_full: SetFeeIxArgs = args.into(); - let data: SetFeeIxData = (&args_full).into(); + let data: SetFeeIxData = args_full.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), data: data.try_to_vec()?, }) } -pub fn set_fee_invoke<'a, A: Into>( - accounts: &SetFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>, +pub fn set_fee_invoke<'info, A: Into>( + accounts: SetFeeAccounts<'_, 'info>, args: A, ) -> ProgramResult { let ix = set_fee_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_FEE_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; SET_FEE_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn set_fee_invoke_signed<'a, A: Into>( - accounts: &SetFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>, +pub fn set_fee_invoke_signed<'info, A: Into>( + accounts: SetFeeAccounts<'_, 'info>, args: A, seeds: &[&[&[u8]]], ) -> ProgramResult { let ix = set_fee_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_FEE_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; SET_FEE_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN: usize = 3usize; -#[derive(Copy, Clone, Debug)] -pub struct SetFeeAuthorityAccounts<'me, 'a0: 'me, 'a1: 'me, 'a2: 'me> { - pub fee_authority: &'me AccountInfo<'a0>, - pub pool_account: &'me AccountInfo<'a1>, - pub new_fee_authority: &'me AccountInfo<'a2>, +pub fn set_fee_verify_account_keys( + accounts: SetFeeAccounts<'_, '_>, + keys: SetFeeKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.fee_authority.key, keys.fee_authority), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.fee_account.key, keys.fee_account), + (*accounts.system_program.key, keys.system_program), + (*accounts.rent.key, keys.rent), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn set_fee_verify_account_privileges<'me, 'info>( + accounts: SetFeeAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [accounts.fee_account] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.fee_authority] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN: usize = 3; #[derive(Copy, Clone, Debug)] +pub struct SetFeeAuthorityAccounts<'me, 'info> { + pub fee_authority: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub new_fee_authority: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct SetFeeAuthorityKeys { pub fee_authority: Pubkey, pub pool_account: Pubkey, pub new_fee_authority: Pubkey, } -impl<'me> From<&SetFeeAuthorityAccounts<'me, '_, '_, '_>> for SetFeeAuthorityKeys { - fn from(accounts: &SetFeeAuthorityAccounts<'me, '_, '_, '_>) -> Self { +impl From> for SetFeeAuthorityKeys { + fn from(accounts: SetFeeAuthorityAccounts) -> Self { Self { fee_authority: *accounts.fee_authority.key, pool_account: *accounts.pool_account.key, @@ -703,19 +1414,40 @@ impl<'me> From<&SetFeeAuthorityAccounts<'me, '_, '_, '_>> for SetFeeAuthorityKey } } } -impl From<&SetFeeAuthorityKeys> for [AccountMeta; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] { - fn from(keys: &SetFeeAuthorityKeys) -> Self { +impl From for [AccountMeta; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] { + fn from(keys: SetFeeAuthorityKeys) -> Self { [ - AccountMeta::new_readonly(keys.fee_authority, true), - AccountMeta::new(keys.pool_account, false), - AccountMeta::new_readonly(keys.new_fee_authority, false), + AccountMeta { + pubkey: keys.fee_authority, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.new_fee_authority, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&SetFeeAuthorityAccounts<'_, 'a, 'a, 'a>> - for [AccountInfo<'a>; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] +impl From<[Pubkey; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN]> for SetFeeAuthorityKeys { + fn from(pubkeys: [Pubkey; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN]) -> Self { + Self { + fee_authority: pubkeys[0], + pool_account: pubkeys[1], + new_fee_authority: pubkeys[2], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] { - fn from(accounts: &SetFeeAuthorityAccounts<'_, 'a, 'a, 'a>) -> Self { + fn from(accounts: SetFeeAuthorityAccounts<'_, 'info>) -> Self { [ accounts.fee_authority.clone(), accounts.pool_account.clone(), @@ -723,79 +1455,115 @@ impl<'a> From<&SetFeeAuthorityAccounts<'_, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] -#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] -pub struct SetFeeAuthorityIxArgs {} -#[derive(Copy, Clone, Debug)] -pub struct SetFeeAuthorityIxData<'me>(pub &'me SetFeeAuthorityIxArgs); -pub const SET_FEE_AUTHORITY_IX_DISCM: [u8; 8] = [31, 1, 50, 87, 237, 101, 97, 132]; -impl<'me> From<&'me SetFeeAuthorityIxArgs> for SetFeeAuthorityIxData<'me> { - fn from(args: &'me SetFeeAuthorityIxArgs) -> Self { - Self(args) +impl<'me, 'info> From<&'me [AccountInfo<'info>; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN]> + for SetFeeAuthorityAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN]) -> Self { + Self { + fee_authority: &arr[0], + pool_account: &arr[1], + new_fee_authority: &arr[2], + } } } -impl BorshSerialize for SetFeeAuthorityIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { - writer.write_all(&SET_FEE_AUTHORITY_IX_DISCM)?; - self.0.serialize(writer) +pub const SET_FEE_AUTHORITY_IX_DISCM: [u8; 8] = [31, 1, 50, 87, 237, 101, 97, 132]; +#[derive(Clone, Debug, PartialEq)] +pub struct SetFeeAuthorityIxData; +impl SetFeeAuthorityIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != SET_FEE_AUTHORITY_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + SET_FEE_AUTHORITY_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&SET_FEE_AUTHORITY_IX_DISCM) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } -pub fn set_fee_authority_ix, A: Into>( +pub fn set_fee_authority_ix>( accounts: K, - args: A, ) -> std::io::Result { let keys: SetFeeAuthorityKeys = accounts.into(); - let metas: [AccountMeta; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] = (&keys).into(); - let args_full: SetFeeAuthorityIxArgs = args.into(); - let data: SetFeeAuthorityIxData = (&args_full).into(); + let metas: [AccountMeta; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] = keys.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), - data: data.try_to_vec()?, + data: SetFeeAuthorityIxData.try_to_vec()?, }) } -pub fn set_fee_authority_invoke<'a, A: Into>( - accounts: &SetFeeAuthorityAccounts<'_, 'a, 'a, 'a>, - args: A, +pub fn set_fee_authority_invoke<'info>( + accounts: SetFeeAuthorityAccounts<'_, 'info>, ) -> ProgramResult { - let ix = set_fee_authority_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = set_fee_authority_ix(accounts)?; + let account_info: [AccountInfo<'info>; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn set_fee_authority_invoke_signed<'a, A: Into>( - accounts: &SetFeeAuthorityAccounts<'_, 'a, 'a, 'a>, - args: A, +pub fn set_fee_authority_invoke_signed<'info>( + accounts: SetFeeAuthorityAccounts<'_, 'info>, seeds: &[&[&[u8]]], ) -> ProgramResult { - let ix = set_fee_authority_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = set_fee_authority_ix(accounts)?; + let account_info: [AccountInfo<'info>; SET_FEE_AUTHORITY_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN: usize = 9usize; -#[derive(Copy, Clone, Debug)] -pub struct SetLpTokenMetadataAccounts< - 'me, - 'a0: 'me, - 'a1: 'me, - 'a2: 'me, - 'a3: 'me, - 'a4: 'me, - 'a5: 'me, - 'a6: 'me, - 'a7: 'me, - 'a8: 'me, -> { - pub payer: &'me AccountInfo<'a0>, - pub fee_authority: &'me AccountInfo<'a1>, - pub pool_account: &'me AccountInfo<'a2>, - pub pool_sol_reserves: &'me AccountInfo<'a3>, - pub lp_mint: &'me AccountInfo<'a4>, - pub metadata: &'me AccountInfo<'a5>, - pub metadata_program: &'me AccountInfo<'a6>, - pub system_program: &'me AccountInfo<'a7>, - pub rent: &'me AccountInfo<'a8>, +pub fn set_fee_authority_verify_account_keys( + accounts: SetFeeAuthorityAccounts<'_, '_>, + keys: SetFeeAuthorityKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.fee_authority.key, keys.fee_authority), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.new_fee_authority.key, keys.new_fee_authority), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn set_fee_authority_verify_account_privileges<'me, 'info>( + accounts: SetFeeAuthorityAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [accounts.pool_account] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.fee_authority] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN: usize = 9; #[derive(Copy, Clone, Debug)] +pub struct SetLpTokenMetadataAccounts<'me, 'info> { + pub payer: &'me AccountInfo<'info>, + pub fee_authority: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub lp_mint: &'me AccountInfo<'info>, + pub metadata: &'me AccountInfo<'info>, + pub metadata_program: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, + pub rent: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct SetLpTokenMetadataKeys { pub payer: Pubkey, pub fee_authority: Pubkey, @@ -807,12 +1575,8 @@ pub struct SetLpTokenMetadataKeys { pub system_program: Pubkey, pub rent: Pubkey, } -impl<'me> From<&SetLpTokenMetadataAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_>> - for SetLpTokenMetadataKeys -{ - fn from( - accounts: &SetLpTokenMetadataAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_>, - ) -> Self { +impl From> for SetLpTokenMetadataKeys { + fn from(accounts: SetLpTokenMetadataAccounts) -> Self { Self { payer: *accounts.payer.key, fee_authority: *accounts.fee_authority.key, @@ -826,25 +1590,76 @@ impl<'me> From<&SetLpTokenMetadataAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, } } } -impl From<&SetLpTokenMetadataKeys> for [AccountMeta; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] { - fn from(keys: &SetLpTokenMetadataKeys) -> Self { +impl From for [AccountMeta; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] { + fn from(keys: SetLpTokenMetadataKeys) -> Self { [ - AccountMeta::new(keys.payer, true), - AccountMeta::new_readonly(keys.fee_authority, true), - AccountMeta::new_readonly(keys.pool_account, false), - AccountMeta::new_readonly(keys.pool_sol_reserves, false), - AccountMeta::new_readonly(keys.lp_mint, false), - AccountMeta::new(keys.metadata, false), - AccountMeta::new_readonly(keys.metadata_program, false), - AccountMeta::new_readonly(keys.system_program, false), - AccountMeta::new_readonly(keys.rent, false), + AccountMeta { + pubkey: keys.payer, + is_signer: true, + is_writable: true, + }, + AccountMeta { + pubkey: keys.fee_authority, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.lp_mint, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.metadata, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.metadata_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.rent, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&SetLpTokenMetadataAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] +impl From<[Pubkey; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN]> for SetLpTokenMetadataKeys { + fn from(pubkeys: [Pubkey; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN]) -> Self { + Self { + payer: pubkeys[0], + fee_authority: pubkeys[1], + pool_account: pubkeys[2], + pool_sol_reserves: pubkeys[3], + lp_mint: pubkeys[4], + metadata: pubkeys[5], + metadata_program: pubkeys[6], + system_program: pubkeys[7], + rent: pubkeys[8], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] { - fn from(accounts: &SetLpTokenMetadataAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: SetLpTokenMetadataAccounts<'_, 'info>) -> Self { [ accounts.payer.clone(), accounts.fee_authority.clone(), @@ -858,23 +1673,60 @@ impl<'a> From<&SetLpTokenMetadataAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] +impl<'me, 'info> From<&'me [AccountInfo<'info>; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN]> + for SetLpTokenMetadataAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN]) -> Self { + Self { + payer: &arr[0], + fee_authority: &arr[1], + pool_account: &arr[2], + pool_sol_reserves: &arr[3], + lp_mint: &arr[4], + metadata: &arr[5], + metadata_program: &arr[6], + system_program: &arr[7], + rent: &arr[8], + } + } +} +pub const SET_LP_TOKEN_METADATA_IX_DISCM: [u8; 8] = [71, 73, 56, 155, 202, 142, 100, 150]; +#[derive(BorshDeserialize, BorshSerialize, Clone, Debug, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct SetLpTokenMetadataIxArgs { pub data: DataV2LpToken, } -#[derive(Copy, Clone, Debug)] -pub struct SetLpTokenMetadataIxData<'me>(pub &'me SetLpTokenMetadataIxArgs); -pub const SET_LP_TOKEN_METADATA_IX_DISCM: [u8; 8] = [71, 73, 56, 155, 202, 142, 100, 150]; -impl<'me> From<&'me SetLpTokenMetadataIxArgs> for SetLpTokenMetadataIxData<'me> { - fn from(args: &'me SetLpTokenMetadataIxArgs) -> Self { +#[derive(Clone, Debug, PartialEq)] +pub struct SetLpTokenMetadataIxData(pub SetLpTokenMetadataIxArgs); +impl From for SetLpTokenMetadataIxData { + fn from(args: SetLpTokenMetadataIxArgs) -> Self { Self(args) } } -impl BorshSerialize for SetLpTokenMetadataIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { +impl SetLpTokenMetadataIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != SET_LP_TOKEN_METADATA_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + SET_LP_TOKEN_METADATA_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self(SetLpTokenMetadataIxArgs::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { writer.write_all(&SET_LP_TOKEN_METADATA_IX_DISCM)?; - self.0.serialize(writer) + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } pub fn set_lp_token_metadata_ix< @@ -885,42 +1737,78 @@ pub fn set_lp_token_metadata_ix< args: A, ) -> std::io::Result { let keys: SetLpTokenMetadataKeys = accounts.into(); - let metas: [AccountMeta; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] = (&keys).into(); + let metas: [AccountMeta; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] = keys.into(); let args_full: SetLpTokenMetadataIxArgs = args.into(); - let data: SetLpTokenMetadataIxData = (&args_full).into(); + let data: SetLpTokenMetadataIxData = args_full.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), data: data.try_to_vec()?, }) } -pub fn set_lp_token_metadata_invoke<'a, A: Into>( - accounts: &SetLpTokenMetadataAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn set_lp_token_metadata_invoke<'info, A: Into>( + accounts: SetLpTokenMetadataAccounts<'_, 'info>, args: A, ) -> ProgramResult { let ix = set_lp_token_metadata_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn set_lp_token_metadata_invoke_signed<'a, A: Into>( - accounts: &SetLpTokenMetadataAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn set_lp_token_metadata_invoke_signed<'info, A: Into>( + accounts: SetLpTokenMetadataAccounts<'_, 'info>, args: A, seeds: &[&[&[u8]]], ) -> ProgramResult { let ix = set_lp_token_metadata_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; SET_LP_TOKEN_METADATA_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN: usize = 5usize; -#[derive(Copy, Clone, Debug)] -pub struct DeactivateStakeAccountAccounts<'me, 'a0: 'me, 'a1: 'me, 'a2: 'me, 'a3: 'me, 'a4: 'me> { - pub stake_account: &'me AccountInfo<'a0>, - pub pool_account: &'me AccountInfo<'a1>, - pub pool_sol_reserves: &'me AccountInfo<'a2>, - pub clock: &'me AccountInfo<'a3>, - pub stake_program: &'me AccountInfo<'a4>, +pub fn set_lp_token_metadata_verify_account_keys( + accounts: SetLpTokenMetadataAccounts<'_, '_>, + keys: SetLpTokenMetadataKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.payer.key, keys.payer), + (*accounts.fee_authority.key, keys.fee_authority), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.lp_mint.key, keys.lp_mint), + (*accounts.metadata.key, keys.metadata), + (*accounts.metadata_program.key, keys.metadata_program), + (*accounts.system_program.key, keys.system_program), + (*accounts.rent.key, keys.rent), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn set_lp_token_metadata_verify_account_privileges<'me, 'info>( + accounts: SetLpTokenMetadataAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [accounts.payer, accounts.metadata] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.payer, accounts.fee_authority] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN: usize = 5; #[derive(Copy, Clone, Debug)] +pub struct DeactivateStakeAccountAccounts<'me, 'info> { + pub stake_account: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub clock: &'me AccountInfo<'info>, + pub stake_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct DeactivateStakeAccountKeys { pub stake_account: Pubkey, pub pool_account: Pubkey, @@ -928,10 +1816,8 @@ pub struct DeactivateStakeAccountKeys { pub clock: Pubkey, pub stake_program: Pubkey, } -impl<'me> From<&DeactivateStakeAccountAccounts<'me, '_, '_, '_, '_, '_>> - for DeactivateStakeAccountKeys -{ - fn from(accounts: &DeactivateStakeAccountAccounts<'me, '_, '_, '_, '_, '_>) -> Self { +impl From> for DeactivateStakeAccountKeys { + fn from(accounts: DeactivateStakeAccountAccounts) -> Self { Self { stake_account: *accounts.stake_account.key, pool_account: *accounts.pool_account.key, @@ -941,21 +1827,52 @@ impl<'me> From<&DeactivateStakeAccountAccounts<'me, '_, '_, '_, '_, '_>> } } } -impl From<&DeactivateStakeAccountKeys> for [AccountMeta; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] { - fn from(keys: &DeactivateStakeAccountKeys) -> Self { +impl From for [AccountMeta; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] { + fn from(keys: DeactivateStakeAccountKeys) -> Self { [ - AccountMeta::new(keys.stake_account, false), - AccountMeta::new_readonly(keys.pool_account, false), - AccountMeta::new_readonly(keys.pool_sol_reserves, false), - AccountMeta::new_readonly(keys.clock, false), - AccountMeta::new_readonly(keys.stake_program, false), + AccountMeta { + pubkey: keys.stake_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.clock, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&DeactivateStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] +impl From<[Pubkey; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN]> for DeactivateStakeAccountKeys { + fn from(pubkeys: [Pubkey; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN]) -> Self { + Self { + stake_account: pubkeys[0], + pool_account: pubkeys[1], + pool_sol_reserves: pubkeys[2], + clock: pubkeys[3], + stake_program: pubkeys[4], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] { - fn from(accounts: &DeactivateStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: DeactivateStakeAccountAccounts<'_, 'info>) -> Self { [ accounts.stake_account.clone(), accounts.pool_account.clone(), @@ -965,78 +1882,114 @@ impl<'a> From<&DeactivateStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] -#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] -pub struct DeactivateStakeAccountIxArgs {} -#[derive(Copy, Clone, Debug)] -pub struct DeactivateStakeAccountIxData<'me>(pub &'me DeactivateStakeAccountIxArgs); -pub const DEACTIVATE_STAKE_ACCOUNT_IX_DISCM: [u8; 8] = [217, 64, 76, 16, 216, 77, 123, 226]; -impl<'me> From<&'me DeactivateStakeAccountIxArgs> for DeactivateStakeAccountIxData<'me> { - fn from(args: &'me DeactivateStakeAccountIxArgs) -> Self { - Self(args) +impl<'me, 'info> From<&'me [AccountInfo<'info>; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN]> + for DeactivateStakeAccountAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN]) -> Self { + Self { + stake_account: &arr[0], + pool_account: &arr[1], + pool_sol_reserves: &arr[2], + clock: &arr[3], + stake_program: &arr[4], + } } } -impl BorshSerialize for DeactivateStakeAccountIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { - writer.write_all(&DEACTIVATE_STAKE_ACCOUNT_IX_DISCM)?; - self.0.serialize(writer) +pub const DEACTIVATE_STAKE_ACCOUNT_IX_DISCM: [u8; 8] = [217, 64, 76, 16, 216, 77, 123, 226]; +#[derive(Clone, Debug, PartialEq)] +pub struct DeactivateStakeAccountIxData; +impl DeactivateStakeAccountIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != DEACTIVATE_STAKE_ACCOUNT_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + DEACTIVATE_STAKE_ACCOUNT_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&DEACTIVATE_STAKE_ACCOUNT_IX_DISCM) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } -pub fn deactivate_stake_account_ix< - K: Into, - A: Into, ->( +pub fn deactivate_stake_account_ix>( accounts: K, - args: A, ) -> std::io::Result { let keys: DeactivateStakeAccountKeys = accounts.into(); - let metas: [AccountMeta; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = (&keys).into(); - let args_full: DeactivateStakeAccountIxArgs = args.into(); - let data: DeactivateStakeAccountIxData = (&args_full).into(); + let metas: [AccountMeta; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = keys.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), - data: data.try_to_vec()?, + data: DeactivateStakeAccountIxData.try_to_vec()?, }) } -pub fn deactivate_stake_account_invoke<'a, A: Into>( - accounts: &DeactivateStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a>, - args: A, +pub fn deactivate_stake_account_invoke<'info>( + accounts: DeactivateStakeAccountAccounts<'_, 'info>, ) -> ProgramResult { - let ix = deactivate_stake_account_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = deactivate_stake_account_ix(accounts)?; + let account_info: [AccountInfo<'info>; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = + accounts.into(); invoke(&ix, &account_info) } -pub fn deactivate_stake_account_invoke_signed<'a, A: Into>( - accounts: &DeactivateStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a>, - args: A, +pub fn deactivate_stake_account_invoke_signed<'info>( + accounts: DeactivateStakeAccountAccounts<'_, 'info>, seeds: &[&[&[u8]]], ) -> ProgramResult { - let ix = deactivate_stake_account_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = deactivate_stake_account_ix(accounts)?; + let account_info: [AccountInfo<'info>; DEACTIVATE_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = + accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN: usize = 7usize; -#[derive(Copy, Clone, Debug)] -pub struct ReclaimStakeAccountAccounts< - 'me, - 'a0: 'me, - 'a1: 'me, - 'a2: 'me, - 'a3: 'me, - 'a4: 'me, - 'a5: 'me, - 'a6: 'me, -> { - pub stake_account: &'me AccountInfo<'a0>, - pub pool_account: &'me AccountInfo<'a1>, - pub pool_sol_reserves: &'me AccountInfo<'a2>, - pub stake_account_record_account: &'me AccountInfo<'a3>, - pub clock: &'me AccountInfo<'a4>, - pub stake_history: &'me AccountInfo<'a5>, - pub stake_program: &'me AccountInfo<'a6>, +pub fn deactivate_stake_account_verify_account_keys( + accounts: DeactivateStakeAccountAccounts<'_, '_>, + keys: DeactivateStakeAccountKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.stake_account.key, keys.stake_account), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.clock.key, keys.clock), + (*accounts.stake_program.key, keys.stake_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn deactivate_stake_account_verify_account_privileges<'me, 'info>( + accounts: DeactivateStakeAccountAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [accounts.stake_account] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + Ok(()) } +pub const RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN: usize = 7; #[derive(Copy, Clone, Debug)] +pub struct ReclaimStakeAccountAccounts<'me, 'info> { + pub stake_account: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub stake_account_record_account: &'me AccountInfo<'info>, + pub clock: &'me AccountInfo<'info>, + pub stake_history: &'me AccountInfo<'info>, + pub stake_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct ReclaimStakeAccountKeys { pub stake_account: Pubkey, pub pool_account: Pubkey, @@ -1046,10 +1999,8 @@ pub struct ReclaimStakeAccountKeys { pub stake_history: Pubkey, pub stake_program: Pubkey, } -impl<'me> From<&ReclaimStakeAccountAccounts<'me, '_, '_, '_, '_, '_, '_, '_>> - for ReclaimStakeAccountKeys -{ - fn from(accounts: &ReclaimStakeAccountAccounts<'me, '_, '_, '_, '_, '_, '_, '_>) -> Self { +impl From> for ReclaimStakeAccountKeys { + fn from(accounts: ReclaimStakeAccountAccounts) -> Self { Self { stake_account: *accounts.stake_account.key, pool_account: *accounts.pool_account.key, @@ -1061,23 +2012,64 @@ impl<'me> From<&ReclaimStakeAccountAccounts<'me, '_, '_, '_, '_, '_, '_, '_>> } } } -impl From<&ReclaimStakeAccountKeys> for [AccountMeta; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] { - fn from(keys: &ReclaimStakeAccountKeys) -> Self { +impl From for [AccountMeta; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] { + fn from(keys: ReclaimStakeAccountKeys) -> Self { [ - AccountMeta::new(keys.stake_account, false), - AccountMeta::new(keys.pool_account, false), - AccountMeta::new(keys.pool_sol_reserves, false), - AccountMeta::new(keys.stake_account_record_account, false), - AccountMeta::new_readonly(keys.clock, false), - AccountMeta::new_readonly(keys.stake_history, false), - AccountMeta::new_readonly(keys.stake_program, false), + AccountMeta { + pubkey: keys.stake_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.stake_account_record_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.clock, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_history, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&ReclaimStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] +impl From<[Pubkey; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN]> for ReclaimStakeAccountKeys { + fn from(pubkeys: [Pubkey; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN]) -> Self { + Self { + stake_account: pubkeys[0], + pool_account: pubkeys[1], + pool_sol_reserves: pubkeys[2], + stake_account_record_account: pubkeys[3], + clock: pubkeys[4], + stake_history: pubkeys[5], + stake_program: pubkeys[6], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] { - fn from(accounts: &ReclaimStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: ReclaimStakeAccountAccounts<'_, 'info>) -> Self { [ accounts.stake_account.clone(), accounts.pool_account.clone(), @@ -1089,88 +2081,129 @@ impl<'a> From<&ReclaimStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] -#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] -pub struct ReclaimStakeAccountIxArgs {} -#[derive(Copy, Clone, Debug)] -pub struct ReclaimStakeAccountIxData<'me>(pub &'me ReclaimStakeAccountIxArgs); -pub const RECLAIM_STAKE_ACCOUNT_IX_DISCM: [u8; 8] = [47, 127, 90, 221, 10, 160, 183, 117]; -impl<'me> From<&'me ReclaimStakeAccountIxArgs> for ReclaimStakeAccountIxData<'me> { - fn from(args: &'me ReclaimStakeAccountIxArgs) -> Self { - Self(args) +impl<'me, 'info> From<&'me [AccountInfo<'info>; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN]> + for ReclaimStakeAccountAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN]) -> Self { + Self { + stake_account: &arr[0], + pool_account: &arr[1], + pool_sol_reserves: &arr[2], + stake_account_record_account: &arr[3], + clock: &arr[4], + stake_history: &arr[5], + stake_program: &arr[6], + } } } -impl BorshSerialize for ReclaimStakeAccountIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { - writer.write_all(&RECLAIM_STAKE_ACCOUNT_IX_DISCM)?; - self.0.serialize(writer) +pub const RECLAIM_STAKE_ACCOUNT_IX_DISCM: [u8; 8] = [47, 127, 90, 221, 10, 160, 183, 117]; +#[derive(Clone, Debug, PartialEq)] +pub struct ReclaimStakeAccountIxData; +impl ReclaimStakeAccountIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != RECLAIM_STAKE_ACCOUNT_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + RECLAIM_STAKE_ACCOUNT_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&RECLAIM_STAKE_ACCOUNT_IX_DISCM) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } -pub fn reclaim_stake_account_ix< - K: Into, - A: Into, ->( +pub fn reclaim_stake_account_ix>( accounts: K, - args: A, ) -> std::io::Result { let keys: ReclaimStakeAccountKeys = accounts.into(); - let metas: [AccountMeta; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = (&keys).into(); - let args_full: ReclaimStakeAccountIxArgs = args.into(); - let data: ReclaimStakeAccountIxData = (&args_full).into(); + let metas: [AccountMeta; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = keys.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), - data: data.try_to_vec()?, + data: ReclaimStakeAccountIxData.try_to_vec()?, }) } -pub fn reclaim_stake_account_invoke<'a, A: Into>( - accounts: &ReclaimStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - args: A, +pub fn reclaim_stake_account_invoke<'info>( + accounts: ReclaimStakeAccountAccounts<'_, 'info>, ) -> ProgramResult { - let ix = reclaim_stake_account_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = reclaim_stake_account_ix(accounts)?; + let account_info: [AccountInfo<'info>; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn reclaim_stake_account_invoke_signed<'a, A: Into>( - accounts: &ReclaimStakeAccountAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - args: A, +pub fn reclaim_stake_account_invoke_signed<'info>( + accounts: ReclaimStakeAccountAccounts<'_, 'info>, seeds: &[&[&[u8]]], ) -> ProgramResult { - let ix = reclaim_stake_account_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = reclaim_stake_account_ix(accounts)?; + let account_info: [AccountInfo<'info>; RECLAIM_STAKE_ACCOUNT_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const UNSTAKE_IX_ACCOUNTS_LEN: usize = 12usize; -#[derive(Copy, Clone, Debug)] -pub struct UnstakeAccounts< - 'me, - 'a0: 'me, - 'a1: 'me, - 'a2: 'me, - 'a3: 'me, - 'a4: 'me, - 'a5: 'me, - 'a6: 'me, - 'a7: 'me, - 'a8: 'me, - 'a9: 'me, - 'a10: 'me, - 'a11: 'me, -> { - pub unstaker: &'me AccountInfo<'a0>, - pub stake_account: &'me AccountInfo<'a1>, - pub destination: &'me AccountInfo<'a2>, - pub pool_account: &'me AccountInfo<'a3>, - pub pool_sol_reserves: &'me AccountInfo<'a4>, - pub fee_account: &'me AccountInfo<'a5>, - pub stake_account_record_account: &'me AccountInfo<'a6>, - pub protocol_fee_account: &'me AccountInfo<'a7>, - pub protocol_fee_destination: &'me AccountInfo<'a8>, - pub clock: &'me AccountInfo<'a9>, - pub stake_program: &'me AccountInfo<'a10>, - pub system_program: &'me AccountInfo<'a11>, +pub fn reclaim_stake_account_verify_account_keys( + accounts: ReclaimStakeAccountAccounts<'_, '_>, + keys: ReclaimStakeAccountKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.stake_account.key, keys.stake_account), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + ( + *accounts.stake_account_record_account.key, + keys.stake_account_record_account, + ), + (*accounts.clock.key, keys.clock), + (*accounts.stake_history.key, keys.stake_history), + (*accounts.stake_program.key, keys.stake_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn reclaim_stake_account_verify_account_privileges<'me, 'info>( + accounts: ReclaimStakeAccountAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [ + accounts.stake_account, + accounts.pool_account, + accounts.pool_sol_reserves, + accounts.stake_account_record_account, + ] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + Ok(()) } +pub const UNSTAKE_IX_ACCOUNTS_LEN: usize = 12; #[derive(Copy, Clone, Debug)] +pub struct UnstakeAccounts<'me, 'info> { + pub unstaker: &'me AccountInfo<'info>, + pub stake_account: &'me AccountInfo<'info>, + pub destination: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub fee_account: &'me AccountInfo<'info>, + pub stake_account_record_account: &'me AccountInfo<'info>, + pub protocol_fee_account: &'me AccountInfo<'info>, + pub protocol_fee_destination: &'me AccountInfo<'info>, + pub clock: &'me AccountInfo<'info>, + pub stake_program: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct UnstakeKeys { pub unstaker: Pubkey, pub stake_account: Pubkey, @@ -1185,12 +2218,8 @@ pub struct UnstakeKeys { pub stake_program: Pubkey, pub system_program: Pubkey, } -impl<'me> From<&UnstakeAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_>> - for UnstakeKeys -{ - fn from( - accounts: &UnstakeAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_>, - ) -> Self { +impl From> for UnstakeKeys { + fn from(accounts: UnstakeAccounts) -> Self { Self { unstaker: *accounts.unstaker.key, stake_account: *accounts.stake_account.key, @@ -1207,30 +2236,92 @@ impl<'me> From<&UnstakeAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, } } } -impl From<&UnstakeKeys> for [AccountMeta; UNSTAKE_IX_ACCOUNTS_LEN] { - fn from(keys: &UnstakeKeys) -> Self { +impl From for [AccountMeta; UNSTAKE_IX_ACCOUNTS_LEN] { + fn from(keys: UnstakeKeys) -> Self { [ - AccountMeta::new_readonly(keys.unstaker, true), - AccountMeta::new(keys.stake_account, false), - AccountMeta::new(keys.destination, false), - AccountMeta::new(keys.pool_account, false), - AccountMeta::new(keys.pool_sol_reserves, false), - AccountMeta::new_readonly(keys.fee_account, false), - AccountMeta::new(keys.stake_account_record_account, false), - AccountMeta::new_readonly(keys.protocol_fee_account, false), - AccountMeta::new(keys.protocol_fee_destination, false), - AccountMeta::new_readonly(keys.clock, false), - AccountMeta::new_readonly(keys.stake_program, false), - AccountMeta::new_readonly(keys.system_program, false), + AccountMeta { + pubkey: keys.unstaker, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.destination, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.fee_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_account_record_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.protocol_fee_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.protocol_fee_destination, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.clock, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&UnstakeAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; UNSTAKE_IX_ACCOUNTS_LEN] -{ - fn from( - accounts: &UnstakeAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - ) -> Self { +impl From<[Pubkey; UNSTAKE_IX_ACCOUNTS_LEN]> for UnstakeKeys { + fn from(pubkeys: [Pubkey; UNSTAKE_IX_ACCOUNTS_LEN]) -> Self { + Self { + unstaker: pubkeys[0], + stake_account: pubkeys[1], + destination: pubkeys[2], + pool_account: pubkeys[3], + pool_sol_reserves: pubkeys[4], + fee_account: pubkeys[5], + stake_account_record_account: pubkeys[6], + protocol_fee_account: pubkeys[7], + protocol_fee_destination: pubkeys[8], + clock: pubkeys[9], + stake_program: pubkeys[10], + system_program: pubkeys[11], + } + } +} +impl<'info> From> for [AccountInfo<'info>; UNSTAKE_IX_ACCOUNTS_LEN] { + fn from(accounts: UnstakeAccounts<'_, 'info>) -> Self { [ accounts.unstaker.clone(), accounts.stake_account.clone(), @@ -1247,87 +2338,149 @@ impl<'a> From<&UnstakeAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, ' ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] -#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] -pub struct UnstakeIxArgs {} -#[derive(Copy, Clone, Debug)] -pub struct UnstakeIxData<'me>(pub &'me UnstakeIxArgs); -pub const UNSTAKE_IX_DISCM: [u8; 8] = [90, 95, 107, 42, 205, 124, 50, 225]; -impl<'me> From<&'me UnstakeIxArgs> for UnstakeIxData<'me> { - fn from(args: &'me UnstakeIxArgs) -> Self { - Self(args) +impl<'me, 'info> From<&'me [AccountInfo<'info>; UNSTAKE_IX_ACCOUNTS_LEN]> + for UnstakeAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; UNSTAKE_IX_ACCOUNTS_LEN]) -> Self { + Self { + unstaker: &arr[0], + stake_account: &arr[1], + destination: &arr[2], + pool_account: &arr[3], + pool_sol_reserves: &arr[4], + fee_account: &arr[5], + stake_account_record_account: &arr[6], + protocol_fee_account: &arr[7], + protocol_fee_destination: &arr[8], + clock: &arr[9], + stake_program: &arr[10], + system_program: &arr[11], + } } } -impl BorshSerialize for UnstakeIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { - writer.write_all(&UNSTAKE_IX_DISCM)?; - self.0.serialize(writer) +pub const UNSTAKE_IX_DISCM: [u8; 8] = [90, 95, 107, 42, 205, 124, 50, 225]; +#[derive(Clone, Debug, PartialEq)] +pub struct UnstakeIxData; +impl UnstakeIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != UNSTAKE_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + UNSTAKE_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&UNSTAKE_IX_DISCM) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } -pub fn unstake_ix, A: Into>( - accounts: K, - args: A, -) -> std::io::Result { +pub fn unstake_ix>(accounts: K) -> std::io::Result { let keys: UnstakeKeys = accounts.into(); - let metas: [AccountMeta; UNSTAKE_IX_ACCOUNTS_LEN] = (&keys).into(); - let args_full: UnstakeIxArgs = args.into(); - let data: UnstakeIxData = (&args_full).into(); + let metas: [AccountMeta; UNSTAKE_IX_ACCOUNTS_LEN] = keys.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), - data: data.try_to_vec()?, + data: UnstakeIxData.try_to_vec()?, }) } -pub fn unstake_invoke<'a, A: Into>( - accounts: &UnstakeAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - args: A, -) -> ProgramResult { - let ix = unstake_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; UNSTAKE_IX_ACCOUNTS_LEN] = accounts.into(); +pub fn unstake_invoke<'info>(accounts: UnstakeAccounts<'_, 'info>) -> ProgramResult { + let ix = unstake_ix(accounts)?; + let account_info: [AccountInfo<'info>; UNSTAKE_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn unstake_invoke_signed<'a, A: Into>( - accounts: &UnstakeAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - args: A, +pub fn unstake_invoke_signed<'info>( + accounts: UnstakeAccounts<'_, 'info>, seeds: &[&[&[u8]]], ) -> ProgramResult { - let ix = unstake_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; UNSTAKE_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = unstake_ix(accounts)?; + let account_info: [AccountInfo<'info>; UNSTAKE_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const UNSTAKE_WSOL_IX_ACCOUNTS_LEN: usize = 13usize; -#[derive(Copy, Clone, Debug)] -pub struct UnstakeWsolAccounts< - 'me, - 'a0: 'me, - 'a1: 'me, - 'a2: 'me, - 'a3: 'me, - 'a4: 'me, - 'a5: 'me, - 'a6: 'me, - 'a7: 'me, - 'a8: 'me, - 'a9: 'me, - 'a10: 'me, - 'a11: 'me, - 'a12: 'me, -> { - pub unstaker: &'me AccountInfo<'a0>, - pub stake_account: &'me AccountInfo<'a1>, - pub destination: &'me AccountInfo<'a2>, - pub pool_account: &'me AccountInfo<'a3>, - pub pool_sol_reserves: &'me AccountInfo<'a4>, - pub fee_account: &'me AccountInfo<'a5>, - pub stake_account_record_account: &'me AccountInfo<'a6>, - pub protocol_fee_account: &'me AccountInfo<'a7>, - pub protocol_fee_destination: &'me AccountInfo<'a8>, - pub clock: &'me AccountInfo<'a9>, - pub stake_program: &'me AccountInfo<'a10>, - pub system_program: &'me AccountInfo<'a11>, - pub token_program: &'me AccountInfo<'a12>, +pub fn unstake_verify_account_keys( + accounts: UnstakeAccounts<'_, '_>, + keys: UnstakeKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.unstaker.key, keys.unstaker), + (*accounts.stake_account.key, keys.stake_account), + (*accounts.destination.key, keys.destination), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.fee_account.key, keys.fee_account), + ( + *accounts.stake_account_record_account.key, + keys.stake_account_record_account, + ), + ( + *accounts.protocol_fee_account.key, + keys.protocol_fee_account, + ), + ( + *accounts.protocol_fee_destination.key, + keys.protocol_fee_destination, + ), + (*accounts.clock.key, keys.clock), + (*accounts.stake_program.key, keys.stake_program), + (*accounts.system_program.key, keys.system_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn unstake_verify_account_privileges<'me, 'info>( + accounts: UnstakeAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [ + accounts.stake_account, + accounts.destination, + accounts.pool_account, + accounts.pool_sol_reserves, + accounts.stake_account_record_account, + accounts.protocol_fee_destination, + ] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.unstaker] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const UNSTAKE_WSOL_IX_ACCOUNTS_LEN: usize = 13; #[derive(Copy, Clone, Debug)] +pub struct UnstakeWsolAccounts<'me, 'info> { + pub unstaker: &'me AccountInfo<'info>, + pub stake_account: &'me AccountInfo<'info>, + pub destination: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub fee_account: &'me AccountInfo<'info>, + pub stake_account_record_account: &'me AccountInfo<'info>, + pub protocol_fee_account: &'me AccountInfo<'info>, + pub protocol_fee_destination: &'me AccountInfo<'info>, + pub clock: &'me AccountInfo<'info>, + pub stake_program: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, + pub token_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct UnstakeWsolKeys { pub unstaker: Pubkey, pub stake_account: Pubkey, @@ -1343,12 +2496,8 @@ pub struct UnstakeWsolKeys { pub system_program: Pubkey, pub token_program: Pubkey, } -impl<'me> From<&UnstakeWsolAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_>> - for UnstakeWsolKeys -{ - fn from( - accounts: &UnstakeWsolAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_>, - ) -> Self { +impl From> for UnstakeWsolKeys { + fn from(accounts: UnstakeWsolAccounts) -> Self { Self { unstaker: *accounts.unstaker.key, stake_account: *accounts.stake_account.key, @@ -1366,31 +2515,100 @@ impl<'me> From<&UnstakeWsolAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_, '_, '_, } } } -impl From<&UnstakeWsolKeys> for [AccountMeta; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] { - fn from(keys: &UnstakeWsolKeys) -> Self { +impl From for [AccountMeta; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] { + fn from(keys: UnstakeWsolKeys) -> Self { [ - AccountMeta::new_readonly(keys.unstaker, true), - AccountMeta::new(keys.stake_account, false), - AccountMeta::new(keys.destination, false), - AccountMeta::new(keys.pool_account, false), - AccountMeta::new(keys.pool_sol_reserves, false), - AccountMeta::new_readonly(keys.fee_account, false), - AccountMeta::new(keys.stake_account_record_account, false), - AccountMeta::new_readonly(keys.protocol_fee_account, false), - AccountMeta::new(keys.protocol_fee_destination, false), - AccountMeta::new_readonly(keys.clock, false), - AccountMeta::new_readonly(keys.stake_program, false), - AccountMeta::new_readonly(keys.system_program, false), - AccountMeta::new_readonly(keys.token_program, false), + AccountMeta { + pubkey: keys.unstaker, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.destination, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.fee_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_account_record_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.protocol_fee_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.protocol_fee_destination, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.clock, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.stake_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.token_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&UnstakeWsolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] +impl From<[Pubkey; UNSTAKE_WSOL_IX_ACCOUNTS_LEN]> for UnstakeWsolKeys { + fn from(pubkeys: [Pubkey; UNSTAKE_WSOL_IX_ACCOUNTS_LEN]) -> Self { + Self { + unstaker: pubkeys[0], + stake_account: pubkeys[1], + destination: pubkeys[2], + pool_account: pubkeys[3], + pool_sol_reserves: pubkeys[4], + fee_account: pubkeys[5], + stake_account_record_account: pubkeys[6], + protocol_fee_account: pubkeys[7], + protocol_fee_destination: pubkeys[8], + clock: pubkeys[9], + stake_program: pubkeys[10], + system_program: pubkeys[11], + token_program: pubkeys[12], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] { - fn from( - accounts: &UnstakeWsolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - ) -> Self { + fn from(accounts: UnstakeWsolAccounts<'_, 'info>) -> Self { [ accounts.unstaker.clone(), accounts.stake_account.clone(), @@ -1408,64 +2626,143 @@ impl<'a> From<&UnstakeWsolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, ' ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] -#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] -pub struct UnstakeWsolIxArgs {} -#[derive(Copy, Clone, Debug)] -pub struct UnstakeWsolIxData<'me>(pub &'me UnstakeWsolIxArgs); -pub const UNSTAKE_WSOL_IX_DISCM: [u8; 8] = [125, 93, 190, 135, 89, 174, 142, 149]; -impl<'me> From<&'me UnstakeWsolIxArgs> for UnstakeWsolIxData<'me> { - fn from(args: &'me UnstakeWsolIxArgs) -> Self { - Self(args) +impl<'me, 'info> From<&'me [AccountInfo<'info>; UNSTAKE_WSOL_IX_ACCOUNTS_LEN]> + for UnstakeWsolAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; UNSTAKE_WSOL_IX_ACCOUNTS_LEN]) -> Self { + Self { + unstaker: &arr[0], + stake_account: &arr[1], + destination: &arr[2], + pool_account: &arr[3], + pool_sol_reserves: &arr[4], + fee_account: &arr[5], + stake_account_record_account: &arr[6], + protocol_fee_account: &arr[7], + protocol_fee_destination: &arr[8], + clock: &arr[9], + stake_program: &arr[10], + system_program: &arr[11], + token_program: &arr[12], + } } } -impl BorshSerialize for UnstakeWsolIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { - writer.write_all(&UNSTAKE_WSOL_IX_DISCM)?; - self.0.serialize(writer) +pub const UNSTAKE_WSOL_IX_DISCM: [u8; 8] = [125, 93, 190, 135, 89, 174, 142, 149]; +#[derive(Clone, Debug, PartialEq)] +pub struct UnstakeWsolIxData; +impl UnstakeWsolIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != UNSTAKE_WSOL_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + UNSTAKE_WSOL_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&UNSTAKE_WSOL_IX_DISCM) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } -pub fn unstake_wsol_ix, A: Into>( - accounts: K, - args: A, -) -> std::io::Result { +pub fn unstake_wsol_ix>(accounts: K) -> std::io::Result { let keys: UnstakeWsolKeys = accounts.into(); - let metas: [AccountMeta; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] = (&keys).into(); - let args_full: UnstakeWsolIxArgs = args.into(); - let data: UnstakeWsolIxData = (&args_full).into(); + let metas: [AccountMeta; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] = keys.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), - data: data.try_to_vec()?, + data: UnstakeWsolIxData.try_to_vec()?, }) } -pub fn unstake_wsol_invoke<'a, A: Into>( - accounts: &UnstakeWsolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - args: A, -) -> ProgramResult { - let ix = unstake_wsol_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] = accounts.into(); +pub fn unstake_wsol_invoke<'info>(accounts: UnstakeWsolAccounts<'_, 'info>) -> ProgramResult { + let ix = unstake_wsol_ix(accounts)?; + let account_info: [AccountInfo<'info>; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn unstake_wsol_invoke_signed<'a, A: Into>( - accounts: &UnstakeWsolAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - args: A, +pub fn unstake_wsol_invoke_signed<'info>( + accounts: UnstakeWsolAccounts<'_, 'info>, seeds: &[&[&[u8]]], ) -> ProgramResult { - let ix = unstake_wsol_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = unstake_wsol_ix(accounts)?; + let account_info: [AccountInfo<'info>; UNSTAKE_WSOL_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN: usize = 5usize; -#[derive(Copy, Clone, Debug)] -pub struct SetFlashLoanFeeAccounts<'me, 'a0: 'me, 'a1: 'me, 'a2: 'me, 'a3: 'me, 'a4: 'me> { - pub payer: &'me AccountInfo<'a0>, - pub fee_authority: &'me AccountInfo<'a1>, - pub pool_account: &'me AccountInfo<'a2>, - pub flash_loan_fee_account: &'me AccountInfo<'a3>, - pub system_program: &'me AccountInfo<'a4>, +pub fn unstake_wsol_verify_account_keys( + accounts: UnstakeWsolAccounts<'_, '_>, + keys: UnstakeWsolKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.unstaker.key, keys.unstaker), + (*accounts.stake_account.key, keys.stake_account), + (*accounts.destination.key, keys.destination), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.fee_account.key, keys.fee_account), + ( + *accounts.stake_account_record_account.key, + keys.stake_account_record_account, + ), + ( + *accounts.protocol_fee_account.key, + keys.protocol_fee_account, + ), + ( + *accounts.protocol_fee_destination.key, + keys.protocol_fee_destination, + ), + (*accounts.clock.key, keys.clock), + (*accounts.stake_program.key, keys.stake_program), + (*accounts.system_program.key, keys.system_program), + (*accounts.token_program.key, keys.token_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn unstake_wsol_verify_account_privileges<'me, 'info>( + accounts: UnstakeWsolAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [ + accounts.stake_account, + accounts.destination, + accounts.pool_account, + accounts.pool_sol_reserves, + accounts.stake_account_record_account, + accounts.protocol_fee_destination, + ] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.unstaker] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN: usize = 5; #[derive(Copy, Clone, Debug)] +pub struct SetFlashLoanFeeAccounts<'me, 'info> { + pub payer: &'me AccountInfo<'info>, + pub fee_authority: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub flash_loan_fee_account: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct SetFlashLoanFeeKeys { pub payer: Pubkey, pub fee_authority: Pubkey, @@ -1473,8 +2770,8 @@ pub struct SetFlashLoanFeeKeys { pub flash_loan_fee_account: Pubkey, pub system_program: Pubkey, } -impl<'me> From<&SetFlashLoanFeeAccounts<'me, '_, '_, '_, '_, '_>> for SetFlashLoanFeeKeys { - fn from(accounts: &SetFlashLoanFeeAccounts<'me, '_, '_, '_, '_, '_>) -> Self { +impl From> for SetFlashLoanFeeKeys { + fn from(accounts: SetFlashLoanFeeAccounts) -> Self { Self { payer: *accounts.payer.key, fee_authority: *accounts.fee_authority.key, @@ -1484,21 +2781,52 @@ impl<'me> From<&SetFlashLoanFeeAccounts<'me, '_, '_, '_, '_, '_>> for SetFlashLo } } } -impl From<&SetFlashLoanFeeKeys> for [AccountMeta; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] { - fn from(keys: &SetFlashLoanFeeKeys) -> Self { +impl From for [AccountMeta; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] { + fn from(keys: SetFlashLoanFeeKeys) -> Self { [ - AccountMeta::new(keys.payer, true), - AccountMeta::new_readonly(keys.fee_authority, true), - AccountMeta::new_readonly(keys.pool_account, false), - AccountMeta::new(keys.flash_loan_fee_account, false), - AccountMeta::new_readonly(keys.system_program, false), + AccountMeta { + pubkey: keys.payer, + is_signer: true, + is_writable: true, + }, + AccountMeta { + pubkey: keys.fee_authority, + is_signer: true, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.flash_loan_fee_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&SetFlashLoanFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] +impl From<[Pubkey; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN]> for SetFlashLoanFeeKeys { + fn from(pubkeys: [Pubkey; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN]) -> Self { + Self { + payer: pubkeys[0], + fee_authority: pubkeys[1], + pool_account: pubkeys[2], + flash_loan_fee_account: pubkeys[3], + system_program: pubkeys[4], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] { - fn from(accounts: &SetFlashLoanFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: SetFlashLoanFeeAccounts<'_, 'info>) -> Self { [ accounts.payer.clone(), accounts.fee_authority.clone(), @@ -1508,23 +2836,56 @@ impl<'a> From<&SetFlashLoanFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] +impl<'me, 'info> From<&'me [AccountInfo<'info>; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN]> + for SetFlashLoanFeeAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN]) -> Self { + Self { + payer: &arr[0], + fee_authority: &arr[1], + pool_account: &arr[2], + flash_loan_fee_account: &arr[3], + system_program: &arr[4], + } + } +} +pub const SET_FLASH_LOAN_FEE_IX_DISCM: [u8; 8] = [21, 27, 137, 29, 226, 149, 221, 100]; +#[derive(BorshDeserialize, BorshSerialize, Clone, Debug, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct SetFlashLoanFeeIxArgs { pub flash_loan_fee: FlashLoanFee, } -#[derive(Copy, Clone, Debug)] -pub struct SetFlashLoanFeeIxData<'me>(pub &'me SetFlashLoanFeeIxArgs); -pub const SET_FLASH_LOAN_FEE_IX_DISCM: [u8; 8] = [21, 27, 137, 29, 226, 149, 221, 100]; -impl<'me> From<&'me SetFlashLoanFeeIxArgs> for SetFlashLoanFeeIxData<'me> { - fn from(args: &'me SetFlashLoanFeeIxArgs) -> Self { +#[derive(Clone, Debug, PartialEq)] +pub struct SetFlashLoanFeeIxData(pub SetFlashLoanFeeIxArgs); +impl From for SetFlashLoanFeeIxData { + fn from(args: SetFlashLoanFeeIxArgs) -> Self { Self(args) } } -impl BorshSerialize for SetFlashLoanFeeIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { +impl SetFlashLoanFeeIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != SET_FLASH_LOAN_FEE_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + SET_FLASH_LOAN_FEE_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self(SetFlashLoanFeeIxArgs::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { writer.write_all(&SET_FLASH_LOAN_FEE_IX_DISCM)?; - self.0.serialize(writer) + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } pub fn set_flash_loan_fee_ix, A: Into>( @@ -1532,43 +2893,78 @@ pub fn set_flash_loan_fee_ix, A: Into std::io::Result { let keys: SetFlashLoanFeeKeys = accounts.into(); - let metas: [AccountMeta; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] = (&keys).into(); + let metas: [AccountMeta; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] = keys.into(); let args_full: SetFlashLoanFeeIxArgs = args.into(); - let data: SetFlashLoanFeeIxData = (&args_full).into(); + let data: SetFlashLoanFeeIxData = args_full.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), data: data.try_to_vec()?, }) } -pub fn set_flash_loan_fee_invoke<'a, A: Into>( - accounts: &SetFlashLoanFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>, +pub fn set_flash_loan_fee_invoke<'info, A: Into>( + accounts: SetFlashLoanFeeAccounts<'_, 'info>, args: A, ) -> ProgramResult { let ix = set_flash_loan_fee_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn set_flash_loan_fee_invoke_signed<'a, A: Into>( - accounts: &SetFlashLoanFeeAccounts<'_, 'a, 'a, 'a, 'a, 'a>, +pub fn set_flash_loan_fee_invoke_signed<'info, A: Into>( + accounts: SetFlashLoanFeeAccounts<'_, 'info>, args: A, seeds: &[&[&[u8]]], ) -> ProgramResult { let ix = set_flash_loan_fee_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; SET_FLASH_LOAN_FEE_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN: usize = 6usize; -#[derive(Copy, Clone, Debug)] -pub struct TakeFlashLoanAccounts<'me, 'a0: 'me, 'a1: 'me, 'a2: 'me, 'a3: 'me, 'a4: 'me, 'a5: 'me> { - pub receiver: &'me AccountInfo<'a0>, - pub pool_account: &'me AccountInfo<'a1>, - pub pool_sol_reserves: &'me AccountInfo<'a2>, - pub flash_account: &'me AccountInfo<'a3>, - pub system_program: &'me AccountInfo<'a4>, - pub instructions: &'me AccountInfo<'a5>, +pub fn set_flash_loan_fee_verify_account_keys( + accounts: SetFlashLoanFeeAccounts<'_, '_>, + keys: SetFlashLoanFeeKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.payer.key, keys.payer), + (*accounts.fee_authority.key, keys.fee_authority), + (*accounts.pool_account.key, keys.pool_account), + ( + *accounts.flash_loan_fee_account.key, + keys.flash_loan_fee_account, + ), + (*accounts.system_program.key, keys.system_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn set_flash_loan_fee_verify_account_privileges<'me, 'info>( + accounts: SetFlashLoanFeeAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [accounts.payer, accounts.flash_loan_fee_account] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.payer, accounts.fee_authority] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) } +pub const TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN: usize = 6; #[derive(Copy, Clone, Debug)] +pub struct TakeFlashLoanAccounts<'me, 'info> { + pub receiver: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub flash_account: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, + pub instructions: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct TakeFlashLoanKeys { pub receiver: Pubkey, pub pool_account: Pubkey, @@ -1577,8 +2973,8 @@ pub struct TakeFlashLoanKeys { pub system_program: Pubkey, pub instructions: Pubkey, } -impl<'me> From<&TakeFlashLoanAccounts<'me, '_, '_, '_, '_, '_, '_>> for TakeFlashLoanKeys { - fn from(accounts: &TakeFlashLoanAccounts<'me, '_, '_, '_, '_, '_, '_>) -> Self { +impl From> for TakeFlashLoanKeys { + fn from(accounts: TakeFlashLoanAccounts) -> Self { Self { receiver: *accounts.receiver.key, pool_account: *accounts.pool_account.key, @@ -1589,22 +2985,58 @@ impl<'me> From<&TakeFlashLoanAccounts<'me, '_, '_, '_, '_, '_, '_>> for TakeFlas } } } -impl From<&TakeFlashLoanKeys> for [AccountMeta; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] { - fn from(keys: &TakeFlashLoanKeys) -> Self { +impl From for [AccountMeta; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] { + fn from(keys: TakeFlashLoanKeys) -> Self { [ - AccountMeta::new(keys.receiver, false), - AccountMeta::new_readonly(keys.pool_account, false), - AccountMeta::new(keys.pool_sol_reserves, false), - AccountMeta::new(keys.flash_account, false), - AccountMeta::new_readonly(keys.system_program, false), - AccountMeta::new_readonly(keys.instructions, false), + AccountMeta { + pubkey: keys.receiver, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.flash_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.instructions, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&TakeFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] +impl From<[Pubkey; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN]> for TakeFlashLoanKeys { + fn from(pubkeys: [Pubkey; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN]) -> Self { + Self { + receiver: pubkeys[0], + pool_account: pubkeys[1], + pool_sol_reserves: pubkeys[2], + flash_account: pubkeys[3], + system_program: pubkeys[4], + instructions: pubkeys[5], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] { - fn from(accounts: &TakeFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: TakeFlashLoanAccounts<'_, 'info>) -> Self { [ accounts.receiver.clone(), accounts.pool_account.clone(), @@ -1615,23 +3047,57 @@ impl<'a> From<&TakeFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] +impl<'me, 'info> From<&'me [AccountInfo<'info>; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN]> + for TakeFlashLoanAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN]) -> Self { + Self { + receiver: &arr[0], + pool_account: &arr[1], + pool_sol_reserves: &arr[2], + flash_account: &arr[3], + system_program: &arr[4], + instructions: &arr[5], + } + } +} +pub const TAKE_FLASH_LOAN_IX_DISCM: [u8; 8] = [64, 124, 6, 57, 151, 155, 26, 195]; +#[derive(BorshDeserialize, BorshSerialize, Clone, Debug, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct TakeFlashLoanIxArgs { pub lamports: u64, } -#[derive(Copy, Clone, Debug)] -pub struct TakeFlashLoanIxData<'me>(pub &'me TakeFlashLoanIxArgs); -pub const TAKE_FLASH_LOAN_IX_DISCM: [u8; 8] = [64, 124, 6, 57, 151, 155, 26, 195]; -impl<'me> From<&'me TakeFlashLoanIxArgs> for TakeFlashLoanIxData<'me> { - fn from(args: &'me TakeFlashLoanIxArgs) -> Self { +#[derive(Clone, Debug, PartialEq)] +pub struct TakeFlashLoanIxData(pub TakeFlashLoanIxArgs); +impl From for TakeFlashLoanIxData { + fn from(args: TakeFlashLoanIxArgs) -> Self { Self(args) } } -impl BorshSerialize for TakeFlashLoanIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { +impl TakeFlashLoanIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != TAKE_FLASH_LOAN_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + TAKE_FLASH_LOAN_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self(TakeFlashLoanIxArgs::deserialize(&mut reader)?)) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { writer.write_all(&TAKE_FLASH_LOAN_IX_DISCM)?; - self.0.serialize(writer) + self.0.serialize(&mut writer) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } pub fn take_flash_loan_ix, A: Into>( @@ -1639,55 +3105,77 @@ pub fn take_flash_loan_ix, A: Into std::io::Result { let keys: TakeFlashLoanKeys = accounts.into(); - let metas: [AccountMeta; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] = (&keys).into(); + let metas: [AccountMeta; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] = keys.into(); let args_full: TakeFlashLoanIxArgs = args.into(); - let data: TakeFlashLoanIxData = (&args_full).into(); + let data: TakeFlashLoanIxData = args_full.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), data: data.try_to_vec()?, }) } -pub fn take_flash_loan_invoke<'a, A: Into>( - accounts: &TakeFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn take_flash_loan_invoke<'info, A: Into>( + accounts: TakeFlashLoanAccounts<'_, 'info>, args: A, ) -> ProgramResult { let ix = take_flash_loan_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn take_flash_loan_invoke_signed<'a, A: Into>( - accounts: &TakeFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a>, +pub fn take_flash_loan_invoke_signed<'info, A: Into>( + accounts: TakeFlashLoanAccounts<'_, 'info>, args: A, seeds: &[&[&[u8]]], ) -> ProgramResult { let ix = take_flash_loan_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] = accounts.into(); + let account_info: [AccountInfo<'info>; TAKE_FLASH_LOAN_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } -pub const REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN: usize = 8usize; -#[derive(Copy, Clone, Debug)] -pub struct RepayFlashLoanAccounts< - 'me, - 'a0: 'me, - 'a1: 'me, - 'a2: 'me, - 'a3: 'me, - 'a4: 'me, - 'a5: 'me, - 'a6: 'me, - 'a7: 'me, -> { - pub repayer: &'me AccountInfo<'a0>, - pub pool_account: &'me AccountInfo<'a1>, - pub pool_sol_reserves: &'me AccountInfo<'a2>, - pub flash_account: &'me AccountInfo<'a3>, - pub flash_loan_fee_account: &'me AccountInfo<'a4>, - pub protocol_fee_account: &'me AccountInfo<'a5>, - pub protocol_fee_destination: &'me AccountInfo<'a6>, - pub system_program: &'me AccountInfo<'a7>, +pub fn take_flash_loan_verify_account_keys( + accounts: TakeFlashLoanAccounts<'_, '_>, + keys: TakeFlashLoanKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.receiver.key, keys.receiver), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.flash_account.key, keys.flash_account), + (*accounts.system_program.key, keys.system_program), + (*accounts.instructions.key, keys.instructions), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn take_flash_loan_verify_account_privileges<'me, 'info>( + accounts: TakeFlashLoanAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [ + accounts.receiver, + accounts.pool_sol_reserves, + accounts.flash_account, + ] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + Ok(()) } +pub const REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN: usize = 8; #[derive(Copy, Clone, Debug)] +pub struct RepayFlashLoanAccounts<'me, 'info> { + pub repayer: &'me AccountInfo<'info>, + pub pool_account: &'me AccountInfo<'info>, + pub pool_sol_reserves: &'me AccountInfo<'info>, + pub flash_account: &'me AccountInfo<'info>, + pub flash_loan_fee_account: &'me AccountInfo<'info>, + pub protocol_fee_account: &'me AccountInfo<'info>, + pub protocol_fee_destination: &'me AccountInfo<'info>, + pub system_program: &'me AccountInfo<'info>, +} +#[derive(Copy, Clone, Debug, PartialEq)] pub struct RepayFlashLoanKeys { pub repayer: Pubkey, pub pool_account: Pubkey, @@ -1698,10 +3186,8 @@ pub struct RepayFlashLoanKeys { pub protocol_fee_destination: Pubkey, pub system_program: Pubkey, } -impl<'me> From<&RepayFlashLoanAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_>> - for RepayFlashLoanKeys -{ - fn from(accounts: &RepayFlashLoanAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_>) -> Self { +impl From> for RepayFlashLoanKeys { + fn from(accounts: RepayFlashLoanAccounts) -> Self { Self { repayer: *accounts.repayer.key, pool_account: *accounts.pool_account.key, @@ -1714,24 +3200,70 @@ impl<'me> From<&RepayFlashLoanAccounts<'me, '_, '_, '_, '_, '_, '_, '_, '_>> } } } -impl From<&RepayFlashLoanKeys> for [AccountMeta; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] { - fn from(keys: &RepayFlashLoanKeys) -> Self { +impl From for [AccountMeta; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] { + fn from(keys: RepayFlashLoanKeys) -> Self { [ - AccountMeta::new(keys.repayer, true), - AccountMeta::new_readonly(keys.pool_account, false), - AccountMeta::new(keys.pool_sol_reserves, false), - AccountMeta::new(keys.flash_account, false), - AccountMeta::new_readonly(keys.flash_loan_fee_account, false), - AccountMeta::new_readonly(keys.protocol_fee_account, false), - AccountMeta::new(keys.protocol_fee_destination, false), - AccountMeta::new_readonly(keys.system_program, false), + AccountMeta { + pubkey: keys.repayer, + is_signer: true, + is_writable: true, + }, + AccountMeta { + pubkey: keys.pool_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.pool_sol_reserves, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.flash_account, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.flash_loan_fee_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.protocol_fee_account, + is_signer: false, + is_writable: false, + }, + AccountMeta { + pubkey: keys.protocol_fee_destination, + is_signer: false, + is_writable: true, + }, + AccountMeta { + pubkey: keys.system_program, + is_signer: false, + is_writable: false, + }, ] } } -impl<'a> From<&RepayFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> - for [AccountInfo<'a>; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] +impl From<[Pubkey; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN]> for RepayFlashLoanKeys { + fn from(pubkeys: [Pubkey; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN]) -> Self { + Self { + repayer: pubkeys[0], + pool_account: pubkeys[1], + pool_sol_reserves: pubkeys[2], + flash_account: pubkeys[3], + flash_loan_fee_account: pubkeys[4], + protocol_fee_account: pubkeys[5], + protocol_fee_destination: pubkeys[6], + system_program: pubkeys[7], + } + } +} +impl<'info> From> + for [AccountInfo<'info>; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] { - fn from(accounts: &RepayFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>) -> Self { + fn from(accounts: RepayFlashLoanAccounts<'_, 'info>) -> Self { [ accounts.repayer.clone(), accounts.pool_account.clone(), @@ -1744,51 +3276,122 @@ impl<'a> From<&RepayFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>> ] } } -#[derive(BorshDeserialize, BorshSerialize, Clone, Debug)] -#[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] -pub struct RepayFlashLoanIxArgs {} -#[derive(Copy, Clone, Debug)] -pub struct RepayFlashLoanIxData<'me>(pub &'me RepayFlashLoanIxArgs); -pub const REPAY_FLASH_LOAN_IX_DISCM: [u8; 8] = [119, 239, 18, 45, 194, 107, 31, 238]; -impl<'me> From<&'me RepayFlashLoanIxArgs> for RepayFlashLoanIxData<'me> { - fn from(args: &'me RepayFlashLoanIxArgs) -> Self { - Self(args) +impl<'me, 'info> From<&'me [AccountInfo<'info>; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN]> + for RepayFlashLoanAccounts<'me, 'info> +{ + fn from(arr: &'me [AccountInfo<'info>; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN]) -> Self { + Self { + repayer: &arr[0], + pool_account: &arr[1], + pool_sol_reserves: &arr[2], + flash_account: &arr[3], + flash_loan_fee_account: &arr[4], + protocol_fee_account: &arr[5], + protocol_fee_destination: &arr[6], + system_program: &arr[7], + } } } -impl BorshSerialize for RepayFlashLoanIxData<'_> { - fn serialize(&self, writer: &mut W) -> std::io::Result<()> { - writer.write_all(&REPAY_FLASH_LOAN_IX_DISCM)?; - self.0.serialize(writer) +pub const REPAY_FLASH_LOAN_IX_DISCM: [u8; 8] = [119, 239, 18, 45, 194, 107, 31, 238]; +#[derive(Clone, Debug, PartialEq)] +pub struct RepayFlashLoanIxData; +impl RepayFlashLoanIxData { + pub fn deserialize(buf: &[u8]) -> std::io::Result { + let mut reader = buf; + let mut maybe_discm = [0u8; 8]; + reader.read_exact(&mut maybe_discm)?; + if maybe_discm != REPAY_FLASH_LOAN_IX_DISCM { + return Err(std::io::Error::new( + std::io::ErrorKind::Other, + format!( + "discm does not match. Expected: {:?}. Received: {:?}", + REPAY_FLASH_LOAN_IX_DISCM, maybe_discm + ), + )); + } + Ok(Self) + } + pub fn serialize(&self, mut writer: W) -> std::io::Result<()> { + writer.write_all(&REPAY_FLASH_LOAN_IX_DISCM) + } + pub fn try_to_vec(&self) -> std::io::Result> { + let mut data = Vec::new(); + self.serialize(&mut data)?; + Ok(data) } } -pub fn repay_flash_loan_ix, A: Into>( +pub fn repay_flash_loan_ix>( accounts: K, - args: A, ) -> std::io::Result { let keys: RepayFlashLoanKeys = accounts.into(); - let metas: [AccountMeta; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] = (&keys).into(); - let args_full: RepayFlashLoanIxArgs = args.into(); - let data: RepayFlashLoanIxData = (&args_full).into(); + let metas: [AccountMeta; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] = keys.into(); Ok(Instruction { program_id: crate::ID, accounts: Vec::from(metas), - data: data.try_to_vec()?, + data: RepayFlashLoanIxData.try_to_vec()?, }) } -pub fn repay_flash_loan_invoke<'a, A: Into>( - accounts: &RepayFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - args: A, +pub fn repay_flash_loan_invoke<'info>( + accounts: RepayFlashLoanAccounts<'_, 'info>, ) -> ProgramResult { - let ix = repay_flash_loan_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = repay_flash_loan_ix(accounts)?; + let account_info: [AccountInfo<'info>; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] = accounts.into(); invoke(&ix, &account_info) } -pub fn repay_flash_loan_invoke_signed<'a, A: Into>( - accounts: &RepayFlashLoanAccounts<'_, 'a, 'a, 'a, 'a, 'a, 'a, 'a, 'a>, - args: A, +pub fn repay_flash_loan_invoke_signed<'info>( + accounts: RepayFlashLoanAccounts<'_, 'info>, seeds: &[&[&[u8]]], ) -> ProgramResult { - let ix = repay_flash_loan_ix(accounts, args)?; - let account_info: [AccountInfo<'a>; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] = accounts.into(); + let ix = repay_flash_loan_ix(accounts)?; + let account_info: [AccountInfo<'info>; REPAY_FLASH_LOAN_IX_ACCOUNTS_LEN] = accounts.into(); invoke_signed(&ix, &account_info, seeds) } +pub fn repay_flash_loan_verify_account_keys( + accounts: RepayFlashLoanAccounts<'_, '_>, + keys: RepayFlashLoanKeys, +) -> Result<(), (Pubkey, Pubkey)> { + for (actual, expected) in [ + (*accounts.repayer.key, keys.repayer), + (*accounts.pool_account.key, keys.pool_account), + (*accounts.pool_sol_reserves.key, keys.pool_sol_reserves), + (*accounts.flash_account.key, keys.flash_account), + ( + *accounts.flash_loan_fee_account.key, + keys.flash_loan_fee_account, + ), + ( + *accounts.protocol_fee_account.key, + keys.protocol_fee_account, + ), + ( + *accounts.protocol_fee_destination.key, + keys.protocol_fee_destination, + ), + (*accounts.system_program.key, keys.system_program), + ] { + if actual != expected { + return Err((actual, expected)); + } + } + Ok(()) +} +pub fn repay_flash_loan_verify_account_privileges<'me, 'info>( + accounts: RepayFlashLoanAccounts<'me, 'info>, +) -> Result<(), (&'me AccountInfo<'info>, ProgramError)> { + for should_be_writable in [ + accounts.repayer, + accounts.pool_sol_reserves, + accounts.flash_account, + accounts.protocol_fee_destination, + ] { + if !should_be_writable.is_writable { + return Err((should_be_writable, ProgramError::InvalidAccountData)); + } + } + for should_be_signer in [accounts.repayer] { + if !should_be_signer.is_signer { + return Err((should_be_signer, ProgramError::MissingRequiredSignature)); + } + } + Ok(()) +} diff --git a/unstake_interface/src/lib.rs b/unstake_interface/src/lib.rs index af53a70..11fc0e1 100644 --- a/unstake_interface/src/lib.rs +++ b/unstake_interface/src/lib.rs @@ -1,7 +1,9 @@ solana_program::declare_id!("unpXTU2Ndrc7WWNyEhQWe4udTzSibLPi25SXv2xbCHQ"); pub mod accounts; pub use accounts::*; -pub mod instructions; -pub use instructions::*; pub mod typedefs; pub use typedefs::*; +pub mod instructions; +pub use instructions::*; +pub mod errors; +pub use errors::*; diff --git a/unstake_interface/src/typedefs.rs b/unstake_interface/src/typedefs.rs index f3594ea..a17e300 100644 --- a/unstake_interface/src/typedefs.rs +++ b/unstake_interface/src/typedefs.rs @@ -1,24 +1,24 @@ use borsh::{BorshDeserialize, BorshSerialize}; -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct DataV2LpToken { pub name: String, pub symbol: String, pub uri: String, } -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct Rational { pub num: u64, pub denom: u64, } -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub struct LiquidityLinearParams { pub max_liq_remaining: Rational, pub zero_liq_remaining: Rational, } -#[derive(Clone, Debug, BorshDeserialize, BorshSerialize)] +#[derive(Clone, Debug, BorshDeserialize, BorshSerialize, PartialEq)] #[cfg_attr(feature = "serde", derive(serde::Serialize, serde::Deserialize))] pub enum FeeEnum { Flat { ratio: Rational }, From 0670667cb540708b90af6e04d925df87b8222b9b Mon Sep 17 00:00:00 2001 From: billythedummy Date: Fri, 22 Dec 2023 19:34:01 +0800 Subject: [PATCH 2/4] move math to unstake-lib --- Cargo.lock | 9 + Cargo.toml | 3 +- unstake-lib/Cargo.toml | 13 ++ unstake-lib/README.md | 7 + unstake-lib/src/fee.rs | 324 ++++++++++++++++++++++++++++++++++++ unstake-lib/src/lib.rs | 5 + unstake-lib/src/rational.rs | 66 ++++++++ 7 files changed, 426 insertions(+), 1 deletion(-) create mode 100644 unstake-lib/Cargo.toml create mode 100644 unstake-lib/README.md create mode 100644 unstake-lib/src/fee.rs create mode 100644 unstake-lib/src/lib.rs create mode 100644 unstake-lib/src/rational.rs diff --git a/Cargo.lock b/Cargo.lock index d8294da..71c4a4d 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -4422,6 +4422,15 @@ dependencies = [ "unstake_interface", ] +[[package]] +name = "unstake-lib" +version = "0.1.0" +dependencies = [ + "proptest", + "spl-math", + "unstake_interface", +] + [[package]] name = "unstake_interface" version = "2.0.0" diff --git a/Cargo.toml b/Cargo.toml index 2ce4173..23249ea 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,5 +1,5 @@ [workspace] -members = ["programs/*", "cli-rust", "unstake_interface"] +members = ["programs/*", "cli-rust", "unstake-lib", "unstake_interface"] [workspace.dependencies] anchor-lang = "0.28.0" @@ -32,4 +32,5 @@ solana-stake-program = "^1" # workspace members unstake = { path = "./programs/unstake" } +unstake-lib = { path = "./unstake-lib" } unstake_interface = { path = "./unstake_interface" } diff --git a/unstake-lib/Cargo.toml b/unstake-lib/Cargo.toml new file mode 100644 index 0000000..e419054 --- /dev/null +++ b/unstake-lib/Cargo.toml @@ -0,0 +1,13 @@ +[package] +name = "unstake-lib" +version = "0.1.0" +edition = "2021" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +spl-math = { workspace = true, features = ["no-entrypoint"] } +unstake_interface = { workspace = true } + +[dev-dependencies] +proptest = { workspace = true } diff --git a/unstake-lib/README.md b/unstake-lib/README.md new file mode 100644 index 0000000..4b0e0fe --- /dev/null +++ b/unstake-lib/README.md @@ -0,0 +1,7 @@ +# unstake-lib + +Rust client library suitable for use onchain and offchain. + +## Notes + +- There is currently a lot of duplicate types across `unstake_interface`, `unstake` program and `unstake-lib`. We will make them all use the types in `unstake_interface` once we get rid of anchor. diff --git a/unstake-lib/src/fee.rs b/unstake-lib/src/fee.rs new file mode 100644 index 0000000..74ed404 --- /dev/null +++ b/unstake-lib/src/fee.rs @@ -0,0 +1,324 @@ +use spl_math::precise_number::PreciseNumber; +use unstake_interface::{FeeEnum, LiquidityLinearParams, Rational}; + +use crate::RationalQty; + +#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] +pub struct PoolBalance { + pub pool_incoming_stake: u64, + pub sol_reserves_lamports: u64, +} + +#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] +pub struct ApplyFeeArgs { + pub pool_balance: PoolBalance, + pub stake_account_lamports: u64, +} + +#[derive(Clone, Copy, Debug, PartialEq, Eq, PartialOrd, Ord, Hash)] +pub struct ReverseFeeArgs { + pub pool_balance: PoolBalance, + pub lamports_after_fee: u64, +} + +pub trait UnstakeFeeCalc { + fn is_valid(&self) -> bool; + + /// Returns the number of lamports to deduct from + /// `stake_account_lamports` as the fee charged. + /// + /// Returns None if any calculation errors occurred + fn apply(&self, args: ApplyFeeArgs) -> Option; + + /// Returns a possible `stake_account_lamports` value that was fed + /// into `self.apply()` s.t. + /// `stake_account_lamports - self.apply(stake_account_lamports) = lamports_after_fee` + /// + /// Returns None if any calculation errors occured + fn pseudo_reverse(&self, args: ReverseFeeArgs) -> Option; +} + +impl UnstakeFeeCalc for &T { + fn is_valid(&self) -> bool { + (*self).is_valid() + } + + fn apply(&self, args: ApplyFeeArgs) -> Option { + (*self).apply(args) + } + + fn pseudo_reverse(&self, args: ReverseFeeArgs) -> Option { + (*self).pseudo_reverse(args) + } +} + +impl UnstakeFeeCalc for FeeEnum { + fn is_valid(&self) -> bool { + match self { + Self::Flat { ratio } => is_flat_fee_valid(ratio), + Self::LiquidityLinear { params } => is_liq_linear_fee_valid(params), + } + } + + fn apply(&self, args: ApplyFeeArgs) -> Option { + let fee_ratio = match self { + FeeEnum::Flat { ratio } => ratio.to_precise_number()?, + FeeEnum::LiquidityLinear { params } => params.to_fee_ratio(args)?, + }; + + PreciseNumber::new(args.stake_account_lamports as u128)? + .checked_mul(&fee_ratio)? + .ceiling()? + .to_imprecise() + .and_then(|v| u64::try_from(v).ok()) + } + + fn pseudo_reverse(&self, args: ReverseFeeArgs) -> Option { + let fee_ratio = match self { + FeeEnum::Flat { ratio } => ratio.to_precise_number()?, + FeeEnum::LiquidityLinear { params } => params.pseudo_reverse_to_fee_ratio(args)?, + }; + let invert_by = PreciseNumber::new(1)?.checked_sub(&fee_ratio)?; + + PreciseNumber::new(args.lamports_after_fee as u128)? + .checked_div(&invert_by)? + .ceiling()? + .to_imprecise() + .and_then(|v| u64::try_from(v).ok()) + } +} + +pub fn is_flat_fee_valid(ratio: &Rational) -> bool { + ratio.is_valid() && ratio.is_lte_one() +} + +pub fn is_liq_linear_fee_valid(params: &LiquidityLinearParams) -> bool { + if !params.zero_liq_remaining.is_valid() + || !params.zero_liq_remaining.is_lte_one() + || !params.max_liq_remaining.is_valid() + || !params.max_liq_remaining.is_lte_one() + { + return false; + } + let zero: u128 = params.zero_liq_remaining.num as u128 * params.max_liq_remaining.denom as u128; + let max: u128 = params.max_liq_remaining.num as u128 * params.zero_liq_remaining.denom as u128; + max <= zero +} + +#[derive(Clone, Debug, PartialEq)] +pub struct LiqLinearParams { + /// y-intercept of the fee ratio line + pub max_liq_fee: PreciseNumber, + pub slope_num: PreciseNumber, + pub slope_denom: PreciseNumber, +} + +pub trait LiqLinearFeeRatio { + fn liq_linear_params(&self, pool_balance: PoolBalance) -> Option; + + fn to_fee_ratio(&self, args: ApplyFeeArgs) -> Option; + + fn pseudo_reverse_to_fee_ratio(&self, args: ReverseFeeArgs) -> Option; +} + +impl LiqLinearFeeRatio for &T { + fn liq_linear_params(&self, pool_balance: PoolBalance) -> Option { + (*self).liq_linear_params(pool_balance) + } + + fn to_fee_ratio(&self, args: ApplyFeeArgs) -> Option { + (*self).to_fee_ratio(args) + } + + fn pseudo_reverse_to_fee_ratio(&self, args: ReverseFeeArgs) -> Option { + (*self).pseudo_reverse_to_fee_ratio(args) + } +} + +impl LiqLinearFeeRatio for LiquidityLinearParams { + fn liq_linear_params( + &self, + PoolBalance { + pool_incoming_stake, + sol_reserves_lamports, + }: PoolBalance, + ) -> Option { + let zero_liq_fee = self.zero_liq_remaining.to_precise_number()?; + let max_liq_fee = self.max_liq_remaining.to_precise_number()?; + let owned_lamports = + (pool_incoming_stake as u128).checked_add(sol_reserves_lamports as u128)?; + + let slope_num = zero_liq_fee.checked_sub(&max_liq_fee)?; + let slope_denom = PreciseNumber::new(owned_lamports)?; + Some(LiqLinearParams { + max_liq_fee, + slope_num, + slope_denom, + }) + } + + fn to_fee_ratio( + &self, + ApplyFeeArgs { + pool_balance, + stake_account_lamports, + }: ApplyFeeArgs, + ) -> Option { + // linear interpolation from max_liq_remaining to zero_liq_remaining where y-intercept at max_liq_remaining + // x-axis is liquidity consumed in lamports + // y-axis is fee ratio (e.g. 0.01 is 1% fees) + // let I = pool_incoming_stake, S = stake_account_lamports, + // + // fee ratio + // zero_liq_remaining -^------------/ + // | /| + // | / | + // charge y here-|---------/ | + // | /| | + // | / | | + // | / | | + // | / | | + // | / | | + // | / | | + // | / | | + // | /| | | + // |/ | | | + // c (max_liq_remaining)-| | | | + // | | | | + // --------------|--|------|--|-------------------> liquidity consumed + // I I+(1-y)S total amount of lamports in pool + // + // m = slope, c = y-intercept at max_liq_remaining + // new liquidity consumed after unstake = I + (1 - y)S + // y = m(I + (1 - y)S) + c + // y = mI + mS - mSy + c + // y(1 + mS) = m(I + S) + c + // y = (m(I + S) + c) / (1 + mS) + // + // since m <<< 1, use 1/m where possible to preserve precision + // y = m(I + S + c/m) / m(1/m + S) + // y = (I + S + c/m) / (1/m + S) + // TODO: check overflow conditions due to large numbers + // + // note: fee_ratio can go >zero_liq_remaining + // if I + (1 - y)S > pool_owned_lamports + + let LiqLinearParams { + max_liq_fee, + slope_num, + slope_denom, + } = self.liq_linear_params(pool_balance)?; + + let incoming_plus_stake = (pool_balance.pool_incoming_stake as u128) + .checked_add(stake_account_lamports as u128)?; + let num = slope_denom + .checked_mul(&max_liq_fee)? + .checked_div(&slope_num)? + .checked_add(&PreciseNumber::new(incoming_plus_stake)?)?; + let denom = slope_denom + .checked_div(&slope_num)? + .checked_add(&PreciseNumber::new(stake_account_lamports as u128)?)?; + num.checked_div(&denom) + } + + fn pseudo_reverse_to_fee_ratio( + &self, + ReverseFeeArgs { + pool_balance, + lamports_after_fee, + }: ReverseFeeArgs, + ) -> Option { + // From above: + // let z = lamports_after_fee = (1 - y)S + // y = m(I + (1 - y)S) + c + // y = m(I + z) + c + + let LiqLinearParams { + max_liq_fee, + slope_num, + slope_denom, + } = self.liq_linear_params(pool_balance)?; + + let incoming_plus_after_fee = + (pool_balance.pool_incoming_stake as u128).checked_add(lamports_after_fee as u128)?; + + let num = slope_num.checked_mul(&PreciseNumber::new(incoming_plus_after_fee)?)?; + num.checked_div(&slope_denom) + .and_then(|x| x.checked_add(&max_liq_fee)) + } +} + +#[cfg(test)] +mod tests { + use proptest::prelude::*; + + use super::*; + + prop_compose! { + fn pool_balances() + (pool_incoming_stake in any::()) + (sol_reserves_lamports in 0..=(u64::MAX - pool_incoming_stake), pool_incoming_stake in Just(pool_incoming_stake)) -> PoolBalance { + PoolBalance { pool_incoming_stake, sol_reserves_lamports } + } + } + + prop_compose! { + fn valid_ratio_lte_one() + (denom in 1..=u64::MAX) + (num in 0..=denom, denom in Just(denom)) -> Rational { + Rational { num, denom } + } + } + + prop_compose! { + fn flat_fees() + (ratio in valid_ratio_lte_one()) -> FeeEnum { + FeeEnum::Flat { ratio } + } + } + + prop_compose! { + fn liq_linear_fees() + (r1 in valid_ratio_lte_one(), r2 in valid_ratio_lte_one()) -> FeeEnum { + let c1: u128 = r1.num as u128 * r2.denom as u128; + let c2: u128 = r2.num as u128 * r1.denom as u128; + if c1 >= c2 { + FeeEnum::LiquidityLinear { params: LiquidityLinearParams { max_liq_remaining: r2, zero_liq_remaining: r1 } } + } else { + FeeEnum::LiquidityLinear { params: LiquidityLinearParams { max_liq_remaining: r1, zero_liq_remaining: r2 } } + } + } + } + + proptest! { + #[test] + fn flat_fee_pseudo_reverse_round_trip(pool_balance in pool_balances(), fee in flat_fees(), stake_account_lamports: u64) { + let fee_lamports = fee.apply(ApplyFeeArgs { + pool_balance, + stake_account_lamports, + }).unwrap(); + let lamports_after_fee = stake_account_lamports - fee_lamports; + if lamports_after_fee > 0 { + let reversed = fee.pseudo_reverse(ReverseFeeArgs { pool_balance, lamports_after_fee }).unwrap(); + let reversed_fee = fee.apply(ApplyFeeArgs { pool_balance, stake_account_lamports: reversed }).unwrap(); + prop_assert_eq!(lamports_after_fee, reversed - reversed_fee); + } + } + } + + proptest! { + #[test] + fn liq_linear_pseudo_reverse_round_trip(pool_balance in pool_balances(), fee in liq_linear_fees(), stake_account_lamports: u64) { + let fee_lamports = fee.apply(ApplyFeeArgs { + pool_balance, + stake_account_lamports, + }).unwrap(); + let lamports_after_fee = stake_account_lamports - fee_lamports; + if lamports_after_fee > 0 { + let reversed = fee.pseudo_reverse(ReverseFeeArgs { pool_balance, lamports_after_fee }).unwrap(); + let reversed_fee = fee.apply(ApplyFeeArgs { pool_balance, stake_account_lamports: reversed }).unwrap(); + prop_assert_eq!(lamports_after_fee, reversed - reversed_fee); + } + } + } +} diff --git a/unstake-lib/src/lib.rs b/unstake-lib/src/lib.rs new file mode 100644 index 0000000..f7f1a36 --- /dev/null +++ b/unstake-lib/src/lib.rs @@ -0,0 +1,5 @@ +mod fee; +mod rational; + +pub use fee::*; +pub use rational::*; diff --git a/unstake-lib/src/rational.rs b/unstake-lib/src/rational.rs new file mode 100644 index 0000000..d39932e --- /dev/null +++ b/unstake-lib/src/rational.rs @@ -0,0 +1,66 @@ +use spl_math::precise_number::PreciseNumber; +use unstake_interface::Rational; + +pub trait RationalQty { + fn is_valid(&self) -> bool; + + fn to_precise_number(&self) -> Option; + + fn is_lte_one(&self) -> bool; + + fn floor_mul(&self, value: u64) -> Option; + + fn ceil_mul(&self, value: u64) -> Option; +} + +impl RationalQty for &T { + fn is_valid(&self) -> bool { + (*self).is_valid() + } + + fn to_precise_number(&self) -> Option { + (*self).to_precise_number() + } + + fn is_lte_one(&self) -> bool { + (*self).is_lte_one() + } + + fn floor_mul(&self, value: u64) -> Option { + (*self).floor_mul(value) + } + + fn ceil_mul(&self, value: u64) -> Option { + (*self).ceil_mul(value) + } +} + +impl RationalQty for Rational { + fn is_valid(&self) -> bool { + self.denom != 0 + } + + fn to_precise_number(&self) -> Option { + PreciseNumber::new(self.num as u128)?.checked_div(&PreciseNumber::new(self.denom as u128)?) + } + + fn is_lte_one(&self) -> bool { + self.num <= self.denom + } + + fn floor_mul(&self, value: u64) -> Option { + u128::from(value) + .checked_mul(self.num.into()) + .and_then(|product| product.checked_div(self.denom.into())) + .and_then(|result| result.try_into().ok()) + } + + fn ceil_mul(&self, value: u64) -> Option { + u128::from(value) + .checked_mul(self.num.into()) + .and_then(|product| product.checked_add(self.denom.into())) + .and_then(|rounded_up| rounded_up.checked_sub(1)) + .and_then(|rounded_up_sub_one| rounded_up_sub_one.checked_div(self.denom.into())) + .and_then(|result| result.try_into().ok()) + } +} From 069f941d275cc41b2ec0e3c2dbfae78cae63d786 Mon Sep 17 00:00:00 2001 From: billythedummy Date: Sat, 23 Dec 2023 15:06:43 +0800 Subject: [PATCH 3/4] dedup --- Cargo.lock | 2 + Cargo.toml | 7 +- programs/unstake/Cargo.toml | 23 ++- programs/unstake/src/rational.rs | 34 +---- programs/unstake/src/state/fee.rs | 152 +++++-------------- programs/unstake/src/state/flash_loan_fee.rs | 7 +- programs/unstake/src/state/protocol_fee.rs | 18 +-- 7 files changed, 79 insertions(+), 164 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index 71c4a4d..820e270 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -4394,6 +4394,8 @@ dependencies = [ "serde", "spl-associated-token-account", "spl-math", + "unstake-lib", + "unstake_interface", ] [[package]] diff --git a/Cargo.toml b/Cargo.toml index 23249ea..d34d635 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,4 +1,5 @@ [workspace] +resolver = "2" members = ["programs/*", "cli-rust", "unstake-lib", "unstake_interface"] [workspace.dependencies] @@ -7,9 +8,9 @@ anchor-spl = "0.28.0" base64 = "0.21.2" clap = { version = "^4.0" } clap2 = { package = "clap", version = "^2.0" } -borsh = "^0.9" # lock to 0.9.3 for prod +borsh = ">=0.9" derive_more = "^0.99" -mpl-token-metadata = "^1.13" +mpl-token-metadata = "^1" num-derive = ">=0.1" num-traits = ">=0.1" proptest = "^1" @@ -21,7 +22,7 @@ spl-math = ">=0.1" thiserror = "^1" # solana deps -# Lock to 1.14 for prod, but ^1 for flexibility for library users +# Lock to 1.14.20 for prod, but ^1 for flexibility for library users solana-account-decoder = "^1" solana-clap-utils = "^1" solana-cli-config = "^1" diff --git a/programs/unstake/Cargo.toml b/programs/unstake/Cargo.toml index 4adeba8..99f0ff7 100644 --- a/programs/unstake/Cargo.toml +++ b/programs/unstake/Cargo.toml @@ -17,13 +17,22 @@ local-testing = [] default = [] +# Can't use workspace dependencies here because: +# We're using solana-program 1.14 that can't handle workspace dependencies. +# But if we try to upgrade to 1.16, anchor 0.28.0 depends on mpl-token-metadata ^1, +# which has borsh compat issues. +# Fuck this. I'm not gonna waste any more time upgrading anchor or trying to fix this. +# We will remove anchor from all our programs at some point. + [dependencies] -anchor-lang = { workspace = true, features = ["init-if-needed"] } -anchor-spl = { workspace = true, features = ["metadata", "stake", "token"] } -mpl-token-metadata = { workspace = true, features = ["no-entrypoint"] } -serde = { workspace = true, features = ["derive"] } -spl-associated-token-account = { workspace = true, features = ["no-entrypoint"] } # required for anchor-spl token -spl-math = { workspace = true, features = ["no-entrypoint"] } +anchor-lang = { version = "0.28.0", features = ["init-if-needed"] } +anchor-spl = { version = "0.28.0", features = ["metadata", "stake", "token"] } +mpl-token-metadata = { version = "^1", features = ["no-entrypoint"] } +serde = { version = "^1", features = ["derive"] } +spl-associated-token-account = { version = "^1.1", features = ["no-entrypoint"] } # required for anchor-spl token +spl-math = { version = ">=0.1", features = ["no-entrypoint"] } +unstake_interface = { path = "../../unstake_interface" } +unstake-lib = { path = "../../unstake-lib" } [dev-dependencies] -proptest = { workspace = true } +proptest = "^1" diff --git a/programs/unstake/src/rational.rs b/programs/unstake/src/rational.rs index 59bf64b..35e0e7a 100644 --- a/programs/unstake/src/rational.rs +++ b/programs/unstake/src/rational.rs @@ -1,8 +1,7 @@ use anchor_lang::prelude::*; use serde::Deserialize; -use spl_math::precise_number::PreciseNumber; -use std::{convert::TryInto, fmt}; +use std::fmt; /// A ratio. Denom should not = 0 #[derive(Debug, PartialEq, Clone, Copy, AnchorSerialize, AnchorDeserialize, Deserialize)] @@ -11,33 +10,10 @@ pub struct Rational { pub denom: u64, } -impl Rational { - pub fn validate(&self) -> bool { - self.denom != 0 - } - - pub fn into_precise_number(self) -> Option { - PreciseNumber::new(self.num as u128)?.checked_div(&PreciseNumber::new(self.denom as u128)?) - } - - pub fn is_lte_one(&self) -> bool { - self.num <= self.denom - } - - pub fn floor_mul(&self, value: u64) -> Option { - u128::from(value) - .checked_mul(self.num.into()) - .and_then(|product| product.checked_div(self.denom.into())) - .and_then(|result| result.try_into().ok()) - } - - pub fn ceil_mul(&self, value: u64) -> Option { - u128::from(value) - .checked_mul(self.num.into()) - .and_then(|product| product.checked_add(self.denom.into())) - .and_then(|rounded_up| rounded_up.checked_sub(1)) - .and_then(|rounded_up_sub_one| rounded_up_sub_one.checked_div(self.denom.into())) - .and_then(|result| result.try_into().ok()) +// TODO: remove once we unify the types and remove anchor +impl From for unstake_interface::Rational { + fn from(Rational { num, denom }: Rational) -> Self { + Self { num, denom } } } diff --git a/programs/unstake/src/state/fee.rs b/programs/unstake/src/state/fee.rs index ef1bccb..aa29343 100644 --- a/programs/unstake/src/state/fee.rs +++ b/programs/unstake/src/state/fee.rs @@ -1,8 +1,7 @@ use anchor_lang::prelude::*; use serde::Deserialize; -use spl_math::precise_number::PreciseNumber; -use std::convert::TryFrom; use std::fmt; +use unstake_lib::{ApplyFeeArgs, PoolBalance, UnstakeFeeCalc}; use crate::{errors::UnstakeError, rational::Rational}; @@ -18,7 +17,11 @@ pub struct Fee { impl Fee { pub fn validate(&self) -> Result<()> { - self.fee.validate() + let f: unstake_interface::FeeEnum = self.fee.into(); + match f.is_valid() { + true => Ok(()), + false => Err(UnstakeError::InvalidFee.into()), + } } /// Applies the contained fee model to the unstake parameters @@ -29,11 +32,14 @@ impl Fee { sol_reserves_lamports: u64, stake_account_lamports: u64, ) -> Option { - self.fee.apply( - pool_incoming_stake, - sol_reserves_lamports, + let f: unstake_interface::FeeEnum = self.fee.into(); + f.apply(ApplyFeeArgs { + pool_balance: PoolBalance { + pool_incoming_stake, + sol_reserves_lamports, + }, stake_account_lamports, - ) + }) } } @@ -63,6 +69,20 @@ pub enum FeeEnum { LiquidityLinear { params: LiquidityLinearParams }, } +// TODO: remove once we unify the types and remove anchor +impl From for unstake_interface::FeeEnum { + fn from(value: FeeEnum) -> Self { + match value { + FeeEnum::Flat { ratio } => Self::Flat { + ratio: ratio.into(), + }, + FeeEnum::LiquidityLinear { params } => Self::LiquidityLinear { + params: params.into(), + }, + } + } +} + #[derive(Debug, Clone, Copy, AnchorDeserialize, AnchorSerialize, Deserialize)] pub struct LiquidityLinearParams { /// The fee applied to a swap that leaves @@ -74,114 +94,18 @@ pub struct LiquidityLinearParams { pub zero_liq_remaining: Rational, } -impl FeeEnum { - pub fn validate(&self) -> Result<()> { - match self { - FeeEnum::Flat { ratio } => { - if !ratio.validate() || !ratio.is_lte_one() { - return Err(UnstakeError::InvalidFee.into()); - } - } - FeeEnum::LiquidityLinear { params } => { - if !params.zero_liq_remaining.validate() - || !params.zero_liq_remaining.is_lte_one() - || !params.max_liq_remaining.validate() - || !params.max_liq_remaining.is_lte_one() - { - return Err(UnstakeError::InvalidFee.into()); - } - let zero_liq_fee = params - .zero_liq_remaining - .into_precise_number() - .ok_or(UnstakeError::InternalError)?; - let max_liq_fee = params - .max_liq_remaining - .into_precise_number() - .ok_or(UnstakeError::InternalError)?; - if max_liq_fee.greater_than(&zero_liq_fee) { - return Err(UnstakeError::InvalidFee.into()); - } - } +// TODO: remove once we unify the types and remove anchor +impl From for unstake_interface::LiquidityLinearParams { + fn from( + LiquidityLinearParams { + max_liq_remaining, + zero_liq_remaining, + }: LiquidityLinearParams, + ) -> Self { + Self { + max_liq_remaining: max_liq_remaining.into(), + zero_liq_remaining: zero_liq_remaining.into(), } - - Ok(()) - } - - /// Applies swap fee to given swap amount and pool's liquidity - pub fn apply( - &self, - pool_incoming_stake: u64, - sol_reserves_lamports: u64, - stake_account_lamports: u64, - ) -> Option { - let fee_ratio = match self { - FeeEnum::Flat { ratio } => ratio.into_precise_number()?, - FeeEnum::LiquidityLinear { params } => { - // linear interpolation from max_liq_remaining to zero_liq_remaining where y-intercept at max_liq_remaining - // x-axis is liquidity consumed in lamports - // y-axis is fee ratio (e.g. 0.01 is 1% fees) - // let I = pool_incoming_stake, S = stake_account_lamports, - // - // fee ratio - // zero_liq_remaining -^------------/ - // | /| - // | / | - // charge y here-|---------/ | - // | /| | - // | / | | - // | / | | - // | / | | - // | / | | - // | / | | - // | / | | - // | /| | | - // |/ | | | - // c (max_liq_remaining)-| | | | - // | | | | - // --------------|--|------|--|-------------------> liquidity consumed - // I I+(1-y)S total amount of lamports in pool - // - // m = slope, c = y-intercept at max_liq_remaining - // new liquidity consumed after unstake = I + (1 - y)S - // y = m(I + (1 - y)S) + c - // y = mI + mS - mSy + c - // y(1 + mS) = m(I + S) + c - // y = (m(I + S) + c) / (1 + mS) - // - // since m <<< 1, use 1/m where possible to preserve precision - // y = m(I + S + c/m) / m(1/m + S) - // y = (I + S + c/m) / (1/m + S) - // TODO: check overflow conditions due to large numbers - // - // note: fee_ratio can go >zero_liq_remaining - // if I + (1 - y)S > pool_owned_lamports - - let zero_liq_fee = params.zero_liq_remaining.into_precise_number()?; - let max_liq_fee = params.max_liq_remaining.into_precise_number()?; - let owned_lamports = - (pool_incoming_stake as u128).checked_add(sol_reserves_lamports as u128)?; - - let slope_num = zero_liq_fee.checked_sub(&max_liq_fee)?; - let slope_denom = PreciseNumber::new(owned_lamports)?; - - let incoming_plus_stake = - (pool_incoming_stake as u128).checked_add(stake_account_lamports as u128)?; - let num = slope_denom - .checked_mul(&max_liq_fee)? - .checked_div(&slope_num)? - .checked_add(&PreciseNumber::new(incoming_plus_stake)?)?; - let denom = slope_denom - .checked_div(&slope_num)? - .checked_add(&PreciseNumber::new(stake_account_lamports as u128)?)?; - num.checked_div(&denom)? - } - }; - - PreciseNumber::new(stake_account_lamports as u128)? - .checked_mul(&fee_ratio)? - .ceiling()? - .to_imprecise() - .and_then(|v| u64::try_from(v).ok()) } } diff --git a/programs/unstake/src/state/flash_loan_fee.rs b/programs/unstake/src/state/flash_loan_fee.rs index 56bafd0..fa57a7c 100644 --- a/programs/unstake/src/state/flash_loan_fee.rs +++ b/programs/unstake/src/state/flash_loan_fee.rs @@ -1,4 +1,5 @@ use anchor_lang::prelude::*; +use unstake_lib::RationalQty; use crate::{errors::UnstakeError, rational::Rational}; @@ -13,13 +14,15 @@ pub struct FlashLoanFee { impl FlashLoanFee { pub fn validate(&self) -> Result<()> { - match self.fee_ratio.validate() { + let f: unstake_interface::Rational = self.fee_ratio.into(); + match f.is_valid() { true => Ok(()), false => Err(UnstakeError::InvalidFee.into()), } } pub fn apply(&self, flash_loan_amount: u64) -> Option { - self.fee_ratio.ceil_mul(flash_loan_amount) + let f: unstake_interface::Rational = self.fee_ratio.into(); + f.ceil_mul(flash_loan_amount) } } diff --git a/programs/unstake/src/state/protocol_fee.rs b/programs/unstake/src/state/protocol_fee.rs index 9dc353d..282bfd8 100644 --- a/programs/unstake/src/state/protocol_fee.rs +++ b/programs/unstake/src/state/protocol_fee.rs @@ -1,6 +1,7 @@ use anchor_lang::prelude::Pubkey; use anchor_lang::prelude::*; +use unstake_lib::RationalQty; use crate::{errors::UnstakeError, rational::Rational}; @@ -31,7 +32,7 @@ mod default_destination { #[cfg(feature = "local-testing")] declare_id!("6h64tjnsZDcvEta2uZvf2CqoPLf2Q8h79ES74ghjNk8D"); - // Left Curve's wSOL token account + // initial protocol fee wSOL token account #[cfg(not(feature = "local-testing"))] declare_id!("GnRGTBrFuEwb85Zs4zeZWUzQYfTwmPxCPYmQQodDzYUK"); } @@ -43,7 +44,7 @@ mod default_authority { #[cfg(feature = "local-testing")] declare_id!("Cp4BZrED56eBpv5c6zdJmoCiKMrYDURjAWU8KeQhYjM8"); - // LEFT CURVE DAO's unstake program upgrade authority + // initial unstake program upgrade authority #[cfg(not(feature = "local-testing"))] declare_id!("4e3CRid3ugjAFRjSnmbbLie1CaeU41CBYhk4saKQgwBB"); } @@ -63,11 +64,8 @@ impl Default for ProtocolFee { impl ProtocolFee { pub fn validate(&self) -> Result<()> { - if !self.fee_ratio.validate() - || !self.fee_ratio.is_lte_one() - || !self.referrer_fee_ratio.validate() - || !self.referrer_fee_ratio.is_lte_one() - { + let f: unstake_interface::Rational = self.fee_ratio.into(); + if !f.is_valid() || !f.is_lte_one() || !f.is_valid() || !f.is_lte_one() { return Err(UnstakeError::InvalidFee.into()); } @@ -82,7 +80,8 @@ impl ProtocolFee { /// Invariants: /// - return <= `fee_lamports` pub fn apply(&self, fee_lamports: u64) -> Option { - self.fee_ratio.floor_mul(fee_lamports) + let f: unstake_interface::Rational = self.fee_ratio.into(); + f.floor_mul(fee_lamports) } /// Applies the referrer fee on a given protocol fee amount @@ -93,6 +92,7 @@ impl ProtocolFee { /// Invariants: /// - return <= `protocol_fee_lamports` pub fn apply_referrer_fee(&self, protocol_fee_lamports: u64) -> Option { - self.referrer_fee_ratio.floor_mul(protocol_fee_lamports) + let f: unstake_interface::Rational = self.referrer_fee_ratio.into(); + f.floor_mul(protocol_fee_lamports) } } From 3651f1699c5ce73695a549d327e55aa57f5e7a79 Mon Sep 17 00:00:00 2001 From: billythedummy Date: Wed, 20 Nov 2024 16:57:11 +0800 Subject: [PATCH 4/4] loosen solana deps --- Cargo.toml | 16 ++++++++-------- programs/unstake/Cargo.toml | 16 ++++++++-------- 2 files changed, 16 insertions(+), 16 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index d34d635..6ce2b0e 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -22,14 +22,14 @@ spl-math = ">=0.1" thiserror = "^1" # solana deps -# Lock to 1.14.20 for prod, but ^1 for flexibility for library users -solana-account-decoder = "^1" -solana-clap-utils = "^1" -solana-cli-config = "^1" -solana-client = "^1" -solana-sdk = "^1" -solana-program = "^1" -solana-stake-program = "^1" +# Lock to 1.14.20 for prod, but >=1 for flexibility for library users +solana-account-decoder = ">=1" +solana-clap-utils = ">=1" +solana-cli-config = ">=1" +solana-client = ">=1" +solana-sdk = ">=1" +solana-program = ">=1" +solana-stake-program = ">=1" # workspace members unstake = { path = "./programs/unstake" } diff --git a/programs/unstake/Cargo.toml b/programs/unstake/Cargo.toml index 99f0ff7..c56c397 100644 --- a/programs/unstake/Cargo.toml +++ b/programs/unstake/Cargo.toml @@ -25,14 +25,14 @@ default = [] # We will remove anchor from all our programs at some point. [dependencies] -anchor-lang = { version = "0.28.0", features = ["init-if-needed"] } -anchor-spl = { version = "0.28.0", features = ["metadata", "stake", "token"] } -mpl-token-metadata = { version = "^1", features = ["no-entrypoint"] } -serde = { version = "^1", features = ["derive"] } -spl-associated-token-account = { version = "^1.1", features = ["no-entrypoint"] } # required for anchor-spl token -spl-math = { version = ">=0.1", features = ["no-entrypoint"] } -unstake_interface = { path = "../../unstake_interface" } -unstake-lib = { path = "../../unstake-lib" } +anchor-lang = { workspace = true, features = ["init-if-needed"] } +anchor-spl = { workspace = true, features = ["metadata", "stake", "token"] } +mpl-token-metadata = { workspace = true, features = ["no-entrypoint"] } +serde = { workspace = true, features = ["derive"] } +spl-associated-token-account = { workspace = true, features = ["no-entrypoint"] } # required for anchor-spl token +spl-math = { workspace = true, features = ["no-entrypoint"] } +unstake_interface = { workspace = true } +unstake-lib = { workspace = true } [dev-dependencies] proptest = "^1"