diff --git a/src/amm_core/state.cairo b/src/amm_core/state.cairo index 342219d..5d9ae1a 100644 --- a/src/amm_core/state.cairo +++ b/src/amm_core/state.cairo @@ -253,6 +253,10 @@ mod State { // @param balance: New lpool balance in u256 fn set_lpool_balance(lptoken_address: LPTAddress, balance: u256) { let mut state = AMM::unsafe_new_contract_state(); + + let locked = state.pool_locked_capital_.read(lptoken_address); + assert(balance >= locked, 'Cant set lpool < locked'); + state.lpool_balance_.write(lptoken_address, balance) } @@ -330,6 +334,10 @@ mod State { // @param balance: New amount of locked pooled tokens in u256 fn set_pool_locked_capital(lptoken_address: LPTAddress, balance: u256) { let mut state = AMM::unsafe_new_contract_state(); + + let lpool_bal = state.lpool_balance_.read(lptoken_address); + assert(lpool_bal >= balance, 'Cant set locked > lpool'); + state.pool_locked_capital_.write(lptoken_address, balance) } diff --git a/tests/lib.cairo b/tests/lib.cairo index f920127..40eaf9d 100644 --- a/tests/lib.cairo +++ b/tests/lib.cairo @@ -14,3 +14,6 @@ mod trading { mod test_trading_halt; mod test_validate_trade_input; } + +mod test_set_balances; + diff --git a/tests/test_set_balances.cairo b/tests/test_set_balances.cairo new file mode 100644 index 0000000..f77681e --- /dev/null +++ b/tests/test_set_balances.cairo @@ -0,0 +1,151 @@ + +use carmine_protocol::testing::setup::deploy_setup; +use traits::{Into, TryInto}; +use debug::PrintTrait; +use option::OptionTrait; +use carmine_protocol::testing::test_utils::{Stats, StatsTrait}; +use carmine_protocol::amm_core::oracles::pragma::Pragma::PRAGMA_ORACLE_ADDRESS; +use carmine_protocol::amm_core::oracles::pragma::PragmaUtils::{ + PragmaPricesResponse, Checkpoint, AggregationMode +}; +use cubit::f128::types::fixed::{Fixed, FixedTrait}; +use snforge_std::{ + declare, ContractClassTrait, start_prank, stop_prank, start_warp, stop_warp, ContractClass, + start_mock_call, stop_mock_call, start_roll +}; +use carmine_protocol::amm_core::helpers::{FixedHelpersTrait, toU256_balance}; +use carmine_protocol::amm_core::amm::AMM; +use carmine_protocol::amm_interface::{IAMMDispatcher, IAMMDispatcherTrait}; +use carmine_protocol::testing::setup::{Ctx, Dispatchers}; + +use carmine_protocol::tokens::my_token::{MyToken, IMyTokenDispatcher, IMyTokenDispatcherTrait}; + + +#[test] +#[should_panic(expected: ('Cant set lpool < locked',))] +fn test_set_balance() { + let (ctx, dsps) = deploy_setup(); + + let five_tokens: u256 = 5000000000000000000; // with 18 decimals + let five_k_tokens: u256 = 5000000000; // with 6 decimals + + let almost_five_k_tokens: u256 = 4950000000 ; // 4 950 USDC + + start_warp(ctx.amm_address, 1000000000); + start_prank(ctx.amm_address, ctx.admin_address); + start_mock_call( + PRAGMA_ORACLE_ADDRESS.try_into().unwrap(), + 'get_data', + PragmaPricesResponse { + price: 190000000000, + decimals: 8, + last_updated_timestamp: 1000000000 + 60 * 60 * 12, + num_sources_aggregated: 0, + expiration_timestamp: Option::None(()) + } + ); + + + /////////////////////////////////////////////////// + // Open a long trade to lock in capital + /////////////////////////////////////////////////// + + let new_expiry = _add_options_with_longer_expiry(ctx, dsps); + + // Opening 3.3 options -> all of unlocked (before premia) + let five_int = 3300000000000000000; // 1*10**18 + let long_put_premia = dsps + .amm + .trade_open( + 1, + ctx.strike_price, + new_expiry.try_into().unwrap(), + 0, + five_int, + ctx.usdc_address, + ctx.eth_address, + FixedTrait::from_unscaled_felt(99999999999), // Disable this check + 99999999999 // Disable this check + ); + + + /////////////////////////////////////////////////// + // Open a short trade + /////////////////////////////////////////////////// + + start_mock_call( + PRAGMA_ORACLE_ADDRESS.try_into().unwrap(), + 'get_data', + PragmaPricesResponse { + price: 140000000000, + decimals: 8, + last_updated_timestamp: 1000000000 + 60 * 60 * 12, + num_sources_aggregated: 0, + expiration_timestamp: Option::None(()) + } + ); + + let one_int = 1000000000000000000; // 1*10**18 + let short_put_premia = dsps + .amm + .trade_open( + 1, // Put + ctx.strike_price, + ctx.expiry, + 1, // short + one_int, + ctx.usdc_address, + ctx.eth_address, + FixedTrait::from_felt(1), // Disable this check + 99999999999 // Disable this check + ); + +} + +// We need another options that will be traded and pool will lock some caital in them +// adds put options only +fn _add_options_with_longer_expiry(ctx: Ctx, dsps: Dispatchers) -> felt252 { + let another_expiry = 1000000000 + 60 * 60 * 48; + + let mut long_put_data = ArrayTrait::::new(); + long_put_data.append('OptLongPut'); + long_put_data.append('OLP'); + long_put_data.append(ctx.amm_address.into()); + long_put_data.append(ctx.usdc_address.into()); + long_put_data.append(ctx.eth_address.into()); + long_put_data.append(1); // PUT + long_put_data.append(27670116110564327424000); + long_put_data.append(another_expiry); + long_put_data.append(0); // LONG + + let mut short_put_data = ArrayTrait::::new(); + short_put_data.append('OptShortPut'); + short_put_data.append('OSP'); + short_put_data.append(ctx.amm_address.into()); + short_put_data.append(ctx.usdc_address.into()); + short_put_data.append(ctx.eth_address.into()); + short_put_data.append(1); // Put + short_put_data.append(27670116110564327424000); + short_put_data.append(another_expiry); + short_put_data.append(1); // short + + let new_long_put_address = ctx.opt_contract.deploy(@long_put_data).unwrap(); + let new_short_put_address = ctx.opt_contract.deploy(@short_put_data).unwrap(); + + dsps + .amm + .add_option_both_sides( + another_expiry.try_into().unwrap(), + ctx.strike_price, + ctx.usdc_address, + ctx.eth_address, + 1, // type + ctx.put_lpt_address, + new_long_put_address, + new_short_put_address, + FixedTrait::from_unscaled_felt(100) + ); + + another_expiry +} +