Skip to content

Commit c62ad70

Browse files
swimrickyswimivan
andauthored
Testing/Docs for Add ix (#3)
* Document defi ix, add helper method for tests for setting up accounts for deposits * Create README.md * decimal: now using uint crate under the hood * Document defi ix, add helper method for tests for setting up accounts for deposits * Add rustfmt.toml, run rustfmt on all existing files * Comment out broken part for now * reverted merge conflict resolution back to original decimal.rs from master Co-authored-by: z0ji <zoji@maelstrom.finance>
1 parent 93cf19c commit c62ad70

13 files changed

+757
-366
lines changed

README.md

+9
Original file line numberDiff line numberDiff line change
@@ -1 +1,10 @@
11
# pool
2+
3+
## Developer Notes
4+
- don't use `msg!()` in test-bpf tests. use `println!` and `cargo test-bpf -- --show-output`
5+
- run `cargo fmt` to use `rustfmt`
6+
- configurations are in `rustfmt.toml`
7+
### Running Tests
8+
```bash
9+
$ cd swim/pool
10+
$ cargo test-bpf -- --show-output

rustfmt.toml

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
max_width = 120

src/amp_factor.rs

+16-22
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,6 @@
1-
use borsh::{BorshDeserialize, BorshSerialize, BorshSchema};
1+
use crate::{decimal::DecimalError, decimal::DecimalU64, error::PoolError};
2+
use borsh::{BorshDeserialize, BorshSchema, BorshSerialize};
23
use solana_program::clock::UnixTimestamp;
3-
use crate::{
4-
error::PoolError,
5-
decimal::DecimalU64,
6-
decimal::DecimalError,
7-
};
84

95
pub type TimestampT = UnixTimestamp;
106
pub type ValueT = DecimalU64;
@@ -71,7 +67,7 @@ impl AmpFactor {
7167
// a 10x over a day (not +10, but potentially much more))
7268

7369
let value_diff = self.target_value - self.initial_value;
74-
let time_since_initial = DecimalU64::new((current_ts - self.initial_ts) as u64,0).unwrap();
70+
let time_since_initial = DecimalU64::new((current_ts - self.initial_ts) as u64, 0).unwrap();
7571
let total_adjustment_time = DecimalU64::new((self.target_ts - self.initial_ts) as u64, 0).unwrap();
7672
let delta = value_diff * (time_since_initial / total_adjustment_time);
7773

@@ -94,10 +90,8 @@ impl AmpFactor {
9490
}
9591

9692
let initial_value = self.get(current_ts);
97-
if (initial_value < target_value
98-
&& initial_value * MAX_RELATIVE_ADJUSTMENT.unwrap() < target_value)
99-
|| (initial_value > target_value
100-
&& initial_value > target_value * MAX_RELATIVE_ADJUSTMENT.unwrap())
93+
if (initial_value < target_value && initial_value * MAX_RELATIVE_ADJUSTMENT.unwrap() < target_value)
94+
|| (initial_value > target_value && initial_value > target_value * MAX_RELATIVE_ADJUSTMENT.unwrap())
10195
{
10296
return Err(PoolError::InvalidAmpFactorValue);
10397
}
@@ -134,31 +128,31 @@ mod tests {
134128

135129
#[test]
136130
fn valid_set_target() {
137-
let mut amp = AmpFactor::new(new_u64(10000,0)).unwrap();
131+
let mut amp = AmpFactor::new(new_u64(10000, 0)).unwrap();
138132
assert_eq!(amp.get(1), 10000);
139-
133+
140134
amp.set_target(20000, new_u64(20000, 0), 106400).unwrap();
141-
135+
142136
assert_eq!(amp.get(20000), 10000);
143-
assert_eq!(amp.get(30000), new_u64(11157407407407407407,15));
144-
assert_eq!(amp.get(50000), new_u64(13472222222222222222,15));
145-
assert_eq!(amp.get(70000), new_u64(15787037037037037037,15));
146-
assert_eq!(amp.get(90000), new_u64(18101851851851851851,15));
137+
assert_eq!(amp.get(30000), new_u64(11157407407407407407, 15));
138+
assert_eq!(amp.get(50000), new_u64(13472222222222222222, 15));
139+
assert_eq!(amp.get(70000), new_u64(15787037037037037037, 15));
140+
assert_eq!(amp.get(90000), new_u64(18101851851851851851, 15));
147141
assert_eq!(amp.get(106400), 20000);
148142
}
149143

150144
#[test]
151145
#[should_panic]
152146
fn invalid_set_target() {
153147
//Target value set to 20x initial value
154-
let mut amp = AmpFactor::new(new_u64(1000,0)).unwrap();
155-
amp.set_target(20000, new_u64(20000,0), 106400).unwrap();
148+
let mut amp = AmpFactor::new(new_u64(1000, 0)).unwrap();
149+
amp.set_target(20000, new_u64(20000, 0), 106400).unwrap();
156150
}
157151

158152
#[test]
159153
#[should_panic]
160154
fn invalid_adjustment_window() {
161-
let mut amp = AmpFactor::new(new_u64(10000,0)).unwrap();
162-
amp.set_target(20000, new_u64(20000,0), 50000).unwrap();
155+
let mut amp = AmpFactor::new(new_u64(10000, 0)).unwrap();
156+
amp.set_target(20000, new_u64(20000, 0), 50000).unwrap();
163157
}
164158
}

src/entrypoint.rs

+1-3
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,5 @@
11
use crate::{error::to_error_msg, processor::Processor};
2-
use solana_program::{
3-
account_info::AccountInfo, entrypoint, entrypoint::ProgramResult, msg, pubkey::Pubkey,
4-
};
2+
use solana_program::{account_info::AccountInfo, entrypoint, entrypoint::ProgramResult, msg, pubkey::Pubkey};
53

64
pub const TOKEN_COUNT: usize = 4; //TODO find a proper way to set/configure this
75

src/error.rs

+1-3
Original file line numberDiff line numberDiff line change
@@ -61,9 +61,7 @@ impl From<PoolError> for ProgramError {
6161

6262
pub fn to_error_msg(error: &ProgramError) -> String {
6363
match error {
64-
ProgramError::Custom(ec) if *ec < OFFSET as u32 => {
65-
TokenError::from_u32(*ec).unwrap().to_string()
66-
}
64+
ProgramError::Custom(ec) if *ec < OFFSET as u32 => TokenError::from_u32(*ec).unwrap().to_string(),
6765
ProgramError::Custom(ec) => PoolError::from_u32(*ec).unwrap().to_string(),
6866
e => e.to_string(),
6967
}

src/instruction.rs

+90-26
Original file line numberDiff line numberDiff line change
@@ -1,30 +1,35 @@
1-
use borsh::{BorshDeserialize, BorshSerialize};
2-
use solana_program::{clock::UnixTimestamp, instruction::{AccountMeta, Instruction}, program_error::ProgramError, pubkey::Pubkey};
31
use crate::decimal::DecimalU64;
2+
use borsh::{BorshDeserialize, BorshSerialize};
3+
use solana_program::{
4+
clock::UnixTimestamp,
5+
instruction::{AccountMeta, Instruction},
6+
program_error::ProgramError,
7+
pubkey::Pubkey,
8+
};
49

510
type AmountT = u64;
611
type DecT = DecimalU64;
712

813
#[derive(BorshSerialize, BorshDeserialize, Debug)]
914
pub enum PoolInstruction<const TOKEN_COUNT: usize> {
10-
/// Initializes a new pool
15+
/// Initializes a new pool
1116
///
1217
/// Accounts expected by this instruction:
1318
/// 0. `[w]` The pool state account to initalize
14-
/// 1. `[]` LP Token Mint. Must be empty, owned by authority
19+
/// 1. `[]` LP Token Mint. Must be empty, owned by authority
1520
/// authority isn't passed in but programatically derived
1621
/// 2. ..2 + TOKEN_COUNT `[]` Token mint accounts
1722
/// 3. ..2 + (2 * TOKEN_COUNT) `[]` Token accounts. Must be empty
1823
/// 4. ..3 + (2 * TOKEN_COUNT) `[]` Governance account
19-
/// 5. ..4 + (2 * TOKEN_COUNT) `[]` Governance Fee account.
24+
/// 5. ..4 + (2 * TOKEN_COUNT) `[]` Governance Fee account.
2025
Init {
2126
nonce: u8,
2227
amp_factor: DecT,
2328
lp_fee: DecT,
2429
governance_fee: DecT,
2530
},
26-
DeFiInstruction(DeFiInstruction::<TOKEN_COUNT>),
27-
GovernanceInstruction(GovernanceInstruction::<TOKEN_COUNT>)
31+
DeFiInstruction(DeFiInstruction<TOKEN_COUNT>),
32+
GovernanceInstruction(GovernanceInstruction<TOKEN_COUNT>),
2833
}
2934

3035
/// Creates an `Init` instruction
@@ -46,37 +51,43 @@ pub fn create_init_ix<const TOKEN_COUNT: usize>(
4651
AccountMeta::new_readonly(*lp_mint, false),
4752
];
4853
for i in 0..TOKEN_COUNT {
49-
accounts.push(
50-
AccountMeta::new_readonly(token_mints[i], false)
51-
);
52-
};
54+
accounts.push(AccountMeta::new_readonly(token_mints[i], false));
55+
}
5356
for i in 0..TOKEN_COUNT {
54-
accounts.push(
55-
AccountMeta::new_readonly(token_accounts[i], false)
56-
);
57-
};
58-
accounts.push(
59-
AccountMeta::new_readonly(*governance_account, false)
60-
);
61-
accounts.push(
62-
AccountMeta::new_readonly(*governance_fee_account, false)
63-
);
57+
accounts.push(AccountMeta::new_readonly(token_accounts[i], false));
58+
}
59+
accounts.push(AccountMeta::new_readonly(*governance_account, false));
60+
accounts.push(AccountMeta::new_readonly(*governance_fee_account, false));
6461
let data = PoolInstruction::<TOKEN_COUNT>::Init {
6562
nonce,
6663
amp_factor,
6764
lp_fee,
68-
governance_fee
69-
}.try_to_vec()?;
65+
governance_fee,
66+
}
67+
.try_to_vec()?;
7068

7169
Ok(Instruction {
7270
program_id: *program_id,
7371
accounts,
74-
data
72+
data,
7573
})
7674
}
7775

7876
#[derive(BorshSerialize, BorshDeserialize, Debug)]
7977
pub enum DeFiInstruction<const TOKEN_COUNT: usize> {
78+
/// Adds/Deposits the specified input_amounts and mints
79+
/// at least `minimum_mint_amount` LP tokens
80+
///
81+
/// Accounts expected by this instruction:
82+
/// 0. `[w]` The pool state account
83+
/// 1. `[]` pool authority
84+
/// 2. ..2 + TOKEN_COUNT `[]` pool's token accounts
85+
/// 3. ..3 + TOKEN_COUNT `[w]` LP Token Mint
86+
/// 4. ..4 + TOKEN_COUNT `[]` governance_fee_account
87+
/// 5. ..5 + TOKEN_COUNT `[s]` user transfer authority account
88+
/// 6. ..6 + TOKEN_COUNT `[w]` user token accounts
89+
/// 7. ..6 + (2 * TOKEN_COUNT) `[]` SPL token program account
90+
/// 8. ..7 + (2 * TOKEN_COUNT) `[w]` user LP token account
8091
Add {
8192
input_amounts: [AmountT; TOKEN_COUNT],
8293
minimum_mint_amount: AmountT,
@@ -103,7 +114,60 @@ pub enum DeFiInstruction<const TOKEN_COUNT: usize> {
103114
RemoveExactOutput {
104115
maximum_burn_amount: AmountT,
105116
exact_output_amounts: [AmountT; TOKEN_COUNT],
117+
},
118+
}
119+
120+
/// Creates an `Add` DefiInstruction
121+
pub fn create_add_ix<const TOKEN_COUNT: usize>(
122+
program_id: &Pubkey,
123+
pool: &Pubkey,
124+
authority: &Pubkey,
125+
pool_token_accounts: [Pubkey; TOKEN_COUNT],
126+
lp_mint: &Pubkey,
127+
governance_fee_account: &Pubkey,
128+
user_transfer_authority: &Pubkey,
129+
user_token_accounts: [Pubkey; TOKEN_COUNT],
130+
token_program_account: &Pubkey,
131+
user_lp_token_account: &Pubkey,
132+
input_amounts: [AmountT; TOKEN_COUNT],
133+
minimum_mint_amount: AmountT,
134+
) -> Result<Instruction, ProgramError> {
135+
let mut accounts = vec![
136+
AccountMeta::new_readonly(*pool, false),
137+
AccountMeta::new_readonly(*authority, false),
138+
];
139+
for i in 0..TOKEN_COUNT {
140+
accounts.push(AccountMeta::new(pool_token_accounts[i], false));
141+
}
142+
accounts.push(AccountMeta::new(*lp_mint, false));
143+
accounts.push(AccountMeta::new(*governance_fee_account, false));
144+
145+
// used from SPL binary-oracle-pair
146+
accounts.push(AccountMeta::new_readonly(
147+
*user_transfer_authority,
148+
authority != user_transfer_authority,
149+
));
150+
for i in 0..TOKEN_COUNT {
151+
accounts.push(AccountMeta::new(user_token_accounts[i], false));
106152
}
153+
accounts.push(AccountMeta::new_readonly(*token_program_account, false));
154+
accounts.push(AccountMeta::new(*user_lp_token_account, false));
155+
156+
let d = DeFiInstruction::<TOKEN_COUNT>::Add {
157+
input_amounts,
158+
minimum_mint_amount,
159+
};
160+
let data = PoolInstruction::<TOKEN_COUNT>::DeFiInstruction(d).try_to_vec()?;
161+
// let data = PoolInstruction::<TOKEN_COUNT>::DeFiInstruction::DeFiInstruction::<TOKEN_COUNT>::Add {
162+
// input_amounts,
163+
// minimum_mint_amount,
164+
// }.try_to_vec()?;
165+
166+
Ok(Instruction {
167+
program_id: *program_id,
168+
accounts,
169+
data,
170+
})
107171
}
108172

109173
#[derive(BorshSerialize, BorshDeserialize, Debug)]
@@ -126,5 +190,5 @@ pub enum GovernanceInstruction<const TOKEN_COUNT: usize> {
126190
},
127191
SetPaused {
128192
paused: bool,
129-
}
130-
}
193+
},
194+
}

0 commit comments

Comments
 (0)