2023-01-30 12:09:47 -08:00
|
|
|
use super::*;
|
2023-10-11 06:18:23 -07:00
|
|
|
use mango_v4::accounts_ix::{Serum3OrderType, Serum3SelfTradeBehavior, Serum3Side};
|
|
|
|
|
|
|
|
async fn deposit_cu_datapoint(
|
|
|
|
solana: &SolanaCookie,
|
|
|
|
account: Pubkey,
|
|
|
|
owner: TestKeypair,
|
|
|
|
token_account: Pubkey,
|
|
|
|
) -> u64 {
|
|
|
|
let result = send_tx_get_metadata(
|
|
|
|
solana,
|
|
|
|
TokenDepositInstruction {
|
|
|
|
amount: 10,
|
|
|
|
reduce_only: false,
|
|
|
|
account,
|
|
|
|
owner,
|
|
|
|
token_account,
|
|
|
|
token_authority: owner,
|
|
|
|
bank_index: 0,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
result.metadata.unwrap().compute_units_consumed
|
|
|
|
}
|
2022-03-17 06:17:28 -07:00
|
|
|
|
|
|
|
// Try to reach compute limits in health checks by having many different tokens in an account
|
|
|
|
#[tokio::test]
|
2022-05-18 08:16:14 -07:00
|
|
|
async fn test_health_compute_tokens() -> Result<(), TransportError> {
|
2022-03-21 02:45:55 -07:00
|
|
|
let context = TestContext::new().await;
|
2022-03-17 06:17:28 -07:00
|
|
|
let solana = &context.solana.clone();
|
|
|
|
|
2022-09-07 03:39:21 -07:00
|
|
|
let admin = TestKeypair::new();
|
|
|
|
let owner = context.users[0].key;
|
|
|
|
let payer = context.users[1].key;
|
2023-08-18 23:20:40 -07:00
|
|
|
let mints = &context.mints[0..8];
|
2022-03-17 06:17:28 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// SETUP: Create a group and an account
|
|
|
|
//
|
|
|
|
|
2022-08-26 06:59:47 -07:00
|
|
|
let GroupWithTokens { group, .. } = GroupWithTokensConfig {
|
2022-03-20 23:49:51 -07:00
|
|
|
admin,
|
|
|
|
payer,
|
2022-09-12 06:25:50 -07:00
|
|
|
mints: mints.to_vec(),
|
|
|
|
..GroupWithTokensConfig::default()
|
2022-03-20 23:49:51 -07:00
|
|
|
}
|
|
|
|
.create(solana)
|
|
|
|
.await;
|
2022-03-17 06:17:28 -07:00
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
let account =
|
|
|
|
create_funded_account(&solana, group, owner, 0, &context.users[1], &[], 1000, 0).await;
|
2022-03-17 06:17:28 -07:00
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
let mut cu_measurements = vec![];
|
|
|
|
for token_account in &context.users[0].token_accounts[..mints.len()] {
|
|
|
|
cu_measurements.push(deposit_cu_datapoint(solana, account, owner, *token_account).await);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i, pair) in cu_measurements.windows(2).enumerate() {
|
|
|
|
println!(
|
|
|
|
"after adding token {}: {} (+{})",
|
|
|
|
i,
|
|
|
|
pair[1],
|
|
|
|
pair[1] - pair[0]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
let avg_cu_increase = cu_measurements.windows(2).map(|p| p[1] - p[0]).sum::<u64>()
|
|
|
|
/ (cu_measurements.len() - 1) as u64;
|
|
|
|
println!("average cu increase: {avg_cu_increase}");
|
2023-11-20 13:00:57 -08:00
|
|
|
assert!(avg_cu_increase < 3230);
|
2022-03-17 06:17:28 -07:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2023-11-14 05:52:04 -08:00
|
|
|
#[tokio::test]
|
|
|
|
async fn test_health_compute_tokens_during_maint_weight_shift() -> Result<(), TransportError> {
|
|
|
|
let context = TestContext::new().await;
|
|
|
|
let solana = &context.solana.clone();
|
|
|
|
|
|
|
|
let admin = TestKeypair::new();
|
|
|
|
let owner = context.users[0].key;
|
|
|
|
let payer = context.users[1].key;
|
|
|
|
let mints = &context.mints[0..8];
|
|
|
|
|
|
|
|
//
|
|
|
|
// SETUP: Create a group and an account
|
|
|
|
//
|
|
|
|
|
|
|
|
let GroupWithTokens { group, .. } = GroupWithTokensConfig {
|
|
|
|
admin,
|
|
|
|
payer,
|
|
|
|
mints: mints.to_vec(),
|
|
|
|
..GroupWithTokensConfig::default()
|
|
|
|
}
|
|
|
|
.create(solana)
|
|
|
|
.await;
|
|
|
|
|
|
|
|
let account =
|
|
|
|
create_funded_account(&solana, group, owner, 0, &context.users[1], &[], 1000, 0).await;
|
|
|
|
|
|
|
|
let now = solana.clock_timestamp().await;
|
|
|
|
for mint in mints {
|
|
|
|
send_tx(
|
|
|
|
solana,
|
|
|
|
TokenEdit {
|
|
|
|
group,
|
|
|
|
admin,
|
|
|
|
mint: mint.pubkey,
|
|
|
|
options: mango_v4::instruction::TokenEdit {
|
|
|
|
maint_weight_shift_start_opt: Some(now - 1000),
|
|
|
|
maint_weight_shift_end_opt: Some(now + 1000),
|
|
|
|
maint_weight_shift_asset_target_opt: Some(0.1),
|
|
|
|
maint_weight_shift_liab_target_opt: Some(1.1),
|
|
|
|
..token_edit_instruction_default()
|
|
|
|
},
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut cu_measurements = vec![];
|
|
|
|
for token_account in &context.users[0].token_accounts[..mints.len()] {
|
|
|
|
cu_measurements.push(deposit_cu_datapoint(solana, account, owner, *token_account).await);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i, pair) in cu_measurements.windows(2).enumerate() {
|
|
|
|
println!(
|
|
|
|
"after adding token {}: {} (+{})",
|
|
|
|
i,
|
|
|
|
pair[1],
|
|
|
|
pair[1] - pair[0]
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
let avg_cu_increase = cu_measurements.windows(2).map(|p| p[1] - p[0]).sum::<u64>()
|
|
|
|
/ (cu_measurements.len() - 1) as u64;
|
|
|
|
println!("average cu increase: {avg_cu_increase}");
|
|
|
|
assert!(avg_cu_increase < 4200);
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2022-03-17 06:17:28 -07:00
|
|
|
// Try to reach compute limits in health checks by having many serum markets in an account
|
|
|
|
#[tokio::test]
|
2022-05-18 08:16:14 -07:00
|
|
|
async fn test_health_compute_serum() -> Result<(), TransportError> {
|
2022-11-29 00:47:03 -08:00
|
|
|
let mut test_builder = TestContextBuilder::new();
|
2023-10-11 06:19:24 -07:00
|
|
|
test_builder.test().set_compute_max_units(130_000);
|
2022-11-29 00:47:03 -08:00
|
|
|
let context = test_builder.start_default().await;
|
2022-03-17 06:17:28 -07:00
|
|
|
let solana = &context.solana.clone();
|
|
|
|
|
2022-09-07 03:39:21 -07:00
|
|
|
let admin = TestKeypair::new();
|
|
|
|
let owner = context.users[0].key;
|
|
|
|
let payer = context.users[1].key;
|
2023-08-21 07:26:49 -07:00
|
|
|
let mints = &context.mints[0..5];
|
2023-10-11 06:18:23 -07:00
|
|
|
let token_account = context.users[0].token_accounts[0];
|
2022-03-17 06:17:28 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// SETUP: Create a group and an account
|
|
|
|
//
|
|
|
|
|
2022-07-04 04:13:11 -07:00
|
|
|
let mango_setup::GroupWithTokens { group, tokens, .. } = mango_setup::GroupWithTokensConfig {
|
2022-03-20 23:49:51 -07:00
|
|
|
admin,
|
|
|
|
payer,
|
2022-09-12 06:25:50 -07:00
|
|
|
mints: mints.to_vec(),
|
|
|
|
..GroupWithTokensConfig::default()
|
2022-03-20 23:49:51 -07:00
|
|
|
}
|
|
|
|
.create(solana)
|
|
|
|
.await;
|
2022-03-17 06:17:28 -07:00
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
// Also activate all token positions
|
|
|
|
let account = create_funded_account(
|
|
|
|
&solana,
|
|
|
|
group,
|
|
|
|
owner,
|
|
|
|
0,
|
|
|
|
&context.users[1],
|
|
|
|
&mints,
|
|
|
|
10000,
|
|
|
|
0,
|
|
|
|
)
|
|
|
|
.await;
|
|
|
|
|
|
|
|
// Allow 8 tokens + 8 serum
|
|
|
|
send_tx(
|
2022-03-17 06:17:28 -07:00
|
|
|
solana,
|
2023-10-11 06:18:23 -07:00
|
|
|
AccountExpandInstruction {
|
2022-03-17 06:17:28 -07:00
|
|
|
account_num: 0,
|
2023-10-11 06:18:23 -07:00
|
|
|
token_count: 8,
|
|
|
|
serum3_count: 8,
|
|
|
|
perp_count: 0,
|
|
|
|
perp_oo_count: 0,
|
|
|
|
token_conditional_swap_count: 0,
|
2022-03-17 06:17:28 -07:00
|
|
|
group,
|
|
|
|
owner,
|
|
|
|
payer,
|
2023-08-18 23:20:40 -07:00
|
|
|
..Default::default()
|
2022-03-17 06:17:28 -07:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
2023-10-11 06:18:23 -07:00
|
|
|
.unwrap();
|
2022-03-17 06:17:28 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// SETUP: Create serum markets and register them
|
|
|
|
//
|
2022-03-30 03:24:07 -07:00
|
|
|
let quote_token = &tokens[0];
|
2022-03-17 06:17:28 -07:00
|
|
|
let mut serum_market_cookies = vec![];
|
2022-03-30 03:24:07 -07:00
|
|
|
for token in tokens[1..].iter() {
|
|
|
|
serum_market_cookies.push((
|
|
|
|
token,
|
|
|
|
context
|
|
|
|
.serum
|
|
|
|
.list_spot_market(&token.mint, "e_token.mint)
|
|
|
|
.await,
|
|
|
|
));
|
2022-03-17 06:17:28 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut serum_markets = vec![];
|
2022-03-30 03:24:07 -07:00
|
|
|
for (base_token, spot) in serum_market_cookies {
|
2022-03-17 06:17:28 -07:00
|
|
|
serum_markets.push(
|
|
|
|
send_tx(
|
|
|
|
solana,
|
2022-03-18 05:42:20 -07:00
|
|
|
Serum3RegisterMarketInstruction {
|
2022-03-17 06:17:28 -07:00
|
|
|
group,
|
|
|
|
admin,
|
|
|
|
serum_program: context.serum.program_id,
|
2022-03-30 03:24:07 -07:00
|
|
|
serum_market_external: spot.market,
|
|
|
|
market_index: spot.coin_mint.index as u16,
|
|
|
|
base_bank: base_token.bank,
|
|
|
|
quote_bank: quote_token.bank,
|
2022-03-17 06:17:28 -07:00
|
|
|
payer,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.unwrap()
|
|
|
|
.serum_market,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
let mut cu_measurements = vec![];
|
|
|
|
|
|
|
|
// Get the baseline cost of a deposit without an active serum3 oo
|
|
|
|
cu_measurements.push(deposit_cu_datapoint(solana, account, owner, token_account).await);
|
|
|
|
|
2022-03-17 06:17:28 -07:00
|
|
|
//
|
|
|
|
// TEST: Create open orders and trigger a Deposit to check health
|
|
|
|
//
|
2023-10-11 06:18:23 -07:00
|
|
|
for &serum_market in serum_markets.iter() {
|
2022-03-17 06:17:28 -07:00
|
|
|
send_tx(
|
|
|
|
solana,
|
2022-03-18 05:42:20 -07:00
|
|
|
Serum3CreateOpenOrdersInstruction {
|
2022-03-17 06:17:28 -07:00
|
|
|
account,
|
|
|
|
serum_market,
|
|
|
|
owner,
|
|
|
|
payer,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
// Place a bid and ask to make the health computation use more compute
|
2022-03-17 06:17:28 -07:00
|
|
|
send_tx(
|
|
|
|
solana,
|
2023-10-11 06:18:23 -07:00
|
|
|
Serum3PlaceOrderInstruction {
|
|
|
|
side: Serum3Side::Bid,
|
|
|
|
limit_price: 1,
|
|
|
|
max_base_qty: 1,
|
|
|
|
max_native_quote_qty_including_fees: 1,
|
|
|
|
self_trade_behavior: Serum3SelfTradeBehavior::AbortTransaction,
|
|
|
|
order_type: Serum3OrderType::Limit,
|
|
|
|
client_order_id: 0,
|
|
|
|
limit: 10,
|
2022-03-17 06:17:28 -07:00
|
|
|
account,
|
2022-09-30 04:21:01 -07:00
|
|
|
owner,
|
2023-10-11 06:18:23 -07:00
|
|
|
serum_market,
|
2022-03-17 06:17:28 -07:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.unwrap();
|
2023-10-11 06:18:23 -07:00
|
|
|
send_tx(
|
|
|
|
solana,
|
|
|
|
Serum3PlaceOrderInstruction {
|
|
|
|
side: Serum3Side::Ask,
|
|
|
|
limit_price: 2,
|
|
|
|
max_base_qty: 1,
|
|
|
|
max_native_quote_qty_including_fees: 1,
|
|
|
|
self_trade_behavior: Serum3SelfTradeBehavior::AbortTransaction,
|
|
|
|
order_type: Serum3OrderType::Limit,
|
|
|
|
client_order_id: 1,
|
|
|
|
limit: 10,
|
|
|
|
account,
|
|
|
|
owner,
|
|
|
|
serum_market,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
// Simple deposit: it's cost is what we use as reference for health compute cost
|
|
|
|
cu_measurements.push(deposit_cu_datapoint(solana, account, owner, token_account).await);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i, pair) in cu_measurements.windows(2).enumerate() {
|
|
|
|
println!(
|
|
|
|
"after adding market {}: {} (+{})",
|
|
|
|
i,
|
|
|
|
pair[1],
|
|
|
|
pair[1] - pair[0]
|
|
|
|
);
|
2022-03-17 06:17:28 -07:00
|
|
|
}
|
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
let avg_cu_increase = cu_measurements.windows(2).map(|p| p[1] - p[0]).sum::<u64>()
|
|
|
|
/ (cu_measurements.len() - 1) as u64;
|
|
|
|
println!("average cu increase: {avg_cu_increase}");
|
2023-10-11 06:19:24 -07:00
|
|
|
assert!(avg_cu_increase < 7000);
|
2022-05-25 02:27:58 -07:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to reach compute limits in health checks by having many perp markets in an account
|
|
|
|
#[tokio::test]
|
|
|
|
async fn test_health_compute_perp() -> Result<(), TransportError> {
|
2023-02-05 23:33:08 -08:00
|
|
|
let mut test_builder = TestContextBuilder::new();
|
|
|
|
test_builder.test().set_compute_max_units(90_000);
|
|
|
|
let context = test_builder.start_default().await;
|
2022-05-25 02:27:58 -07:00
|
|
|
let solana = &context.solana.clone();
|
|
|
|
|
2022-09-07 03:39:21 -07:00
|
|
|
let admin = TestKeypair::new();
|
|
|
|
let owner = context.users[0].key;
|
|
|
|
let payer = context.users[1].key;
|
2023-08-21 07:26:49 -07:00
|
|
|
let mints = &context.mints[0..5];
|
2023-10-11 06:18:23 -07:00
|
|
|
let token_account = context.users[0].token_accounts[0];
|
2022-05-25 02:27:58 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// SETUP: Create a group and an account
|
|
|
|
//
|
|
|
|
|
2022-07-04 04:13:11 -07:00
|
|
|
let mango_setup::GroupWithTokens { group, tokens, .. } = mango_setup::GroupWithTokensConfig {
|
2022-05-25 02:27:58 -07:00
|
|
|
admin,
|
|
|
|
payer,
|
2022-09-12 06:25:50 -07:00
|
|
|
mints: mints.to_vec(),
|
|
|
|
..GroupWithTokensConfig::default()
|
2022-05-25 02:27:58 -07:00
|
|
|
}
|
|
|
|
.create(solana)
|
|
|
|
.await;
|
|
|
|
|
2022-08-26 06:59:47 -07:00
|
|
|
let account = create_funded_account(
|
|
|
|
&solana,
|
|
|
|
group,
|
|
|
|
owner,
|
|
|
|
0,
|
|
|
|
&context.users[1],
|
|
|
|
&mints[..1],
|
|
|
|
1000,
|
|
|
|
0,
|
2022-05-25 02:27:58 -07:00
|
|
|
)
|
2022-08-26 06:59:47 -07:00
|
|
|
.await;
|
2022-05-25 02:27:58 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// SETUP: Create perp markets
|
|
|
|
//
|
|
|
|
let mut perp_markets = vec![];
|
|
|
|
for (perp_market_index, token) in tokens[1..].iter().enumerate() {
|
2022-11-08 06:27:56 -08:00
|
|
|
let mango_v4::accounts::PerpCreateMarket { perp_market, .. } = send_tx(
|
2022-05-25 02:27:58 -07:00
|
|
|
solana,
|
|
|
|
PerpCreateMarketInstruction {
|
|
|
|
group,
|
|
|
|
admin,
|
|
|
|
payer,
|
|
|
|
perp_market_index: perp_market_index as PerpMarketIndex,
|
|
|
|
quote_lot_size: 10,
|
|
|
|
base_lot_size: 100,
|
2023-01-16 07:49:09 -08:00
|
|
|
maint_base_asset_weight: 0.975,
|
|
|
|
init_base_asset_weight: 0.95,
|
|
|
|
maint_base_liab_weight: 1.025,
|
|
|
|
init_base_liab_weight: 1.05,
|
2023-02-02 00:00:37 -08:00
|
|
|
base_liquidation_fee: 0.012,
|
2022-05-25 02:27:58 -07:00
|
|
|
maker_fee: 0.0002,
|
|
|
|
taker_fee: 0.000,
|
2022-11-30 04:20:19 -08:00
|
|
|
settle_pnl_limit_factor: 0.2,
|
2022-12-02 03:24:11 -08:00
|
|
|
settle_pnl_limit_window_size_ts: 24 * 60 * 60,
|
2022-09-09 01:03:49 -07:00
|
|
|
..PerpCreateMarketInstruction::with_new_book_and_queue(&solana, &token).await
|
2022-05-25 02:27:58 -07:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
2022-11-08 06:27:56 -08:00
|
|
|
perp_markets.push(perp_market);
|
2022-05-25 02:27:58 -07:00
|
|
|
}
|
|
|
|
|
2022-05-25 10:29:53 -07:00
|
|
|
let price_lots = {
|
2022-11-08 06:27:56 -08:00
|
|
|
let perp_market = solana.get_account::<PerpMarket>(perp_markets[0]).await;
|
2022-05-25 10:29:53 -07:00
|
|
|
perp_market.native_price_to_lot(I80F48::from(1))
|
|
|
|
};
|
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
let mut cu_measurements = vec![];
|
|
|
|
|
|
|
|
// Get the baseline cost of a deposit without an active serum3 oo
|
|
|
|
cu_measurements.push(deposit_cu_datapoint(solana, account, owner, token_account).await);
|
|
|
|
|
2022-05-25 02:27:58 -07:00
|
|
|
//
|
|
|
|
// TEST: Create a perp order for each market
|
|
|
|
//
|
2022-11-08 06:27:56 -08:00
|
|
|
for (i, &perp_market) in perp_markets.iter().enumerate() {
|
2022-05-25 02:27:58 -07:00
|
|
|
println!("adding market {}", i);
|
|
|
|
send_tx(
|
|
|
|
solana,
|
|
|
|
PerpPlaceOrderInstruction {
|
|
|
|
account,
|
|
|
|
perp_market,
|
|
|
|
owner,
|
|
|
|
side: Side::Bid,
|
2022-05-25 10:29:53 -07:00
|
|
|
price_lots,
|
2022-05-25 02:27:58 -07:00
|
|
|
max_base_lots: 1,
|
2023-05-15 01:40:41 -07:00
|
|
|
..PerpPlaceOrderInstruction::default()
|
2022-05-25 02:27:58 -07:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.await
|
|
|
|
.unwrap();
|
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
cu_measurements.push(deposit_cu_datapoint(solana, account, owner, token_account).await);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i, pair) in cu_measurements.windows(2).enumerate() {
|
|
|
|
println!(
|
|
|
|
"after adding perp market {}: {} (+{})",
|
|
|
|
i,
|
|
|
|
pair[1],
|
|
|
|
pair[1] - pair[0]
|
|
|
|
);
|
2022-05-25 02:27:58 -07:00
|
|
|
}
|
|
|
|
|
2023-10-11 06:18:23 -07:00
|
|
|
let avg_cu_increase = cu_measurements.windows(2).map(|p| p[1] - p[0]).sum::<u64>()
|
|
|
|
/ (cu_measurements.len() - 1) as u64;
|
|
|
|
println!("average cu increase: {avg_cu_increase}");
|
|
|
|
assert!(avg_cu_increase < 4800);
|
2022-03-17 06:17:28 -07:00
|
|
|
|
|
|
|
Ok(())
|
|
|
|
}
|