2022-03-20 02:11:36 -07:00
|
|
|
use anchor_lang::prelude::*;
|
|
|
|
|
2023-02-14 23:42:07 -08:00
|
|
|
use crate::accounts_ix::*;
|
2022-06-08 04:43:12 -07:00
|
|
|
use crate::accounts_zerocopy::*;
|
2022-03-30 01:00:52 -07:00
|
|
|
use crate::error::*;
|
2024-03-04 06:49:14 -08:00
|
|
|
use crate::health::*;
|
2023-02-14 23:42:07 -08:00
|
|
|
use crate::state::*;
|
2024-03-04 06:49:14 -08:00
|
|
|
use crate::util::clock_now;
|
2022-03-21 12:29:28 -07:00
|
|
|
|
2022-03-22 03:19:12 -07:00
|
|
|
// TODO
|
|
|
|
#[allow(clippy::too_many_arguments)]
|
2023-01-25 00:03:35 -08:00
|
|
|
pub fn perp_place_order(
|
|
|
|
ctx: Context<PerpPlaceOrder>,
|
|
|
|
mut order: Order,
|
|
|
|
limit: u8,
|
|
|
|
) -> Result<Option<u128>> {
|
2022-11-08 06:27:56 -08:00
|
|
|
require_gte!(order.max_base_lots, 0);
|
|
|
|
require_gte!(order.max_quote_lots, 0);
|
2022-04-01 06:47:12 -07:00
|
|
|
|
2024-03-04 06:49:14 -08:00
|
|
|
let (now_ts, now_slot) = clock_now();
|
2022-11-10 05:41:20 -08:00
|
|
|
let oracle_price;
|
|
|
|
|
|
|
|
// Update funding if possible.
|
|
|
|
//
|
|
|
|
// Doing this automatically here makes it impossible for attackers to add orders to the orderbook
|
|
|
|
// before triggering the funding computation.
|
|
|
|
{
|
|
|
|
let mut perp_market = ctx.accounts.perp_market.load_mut()?;
|
2022-12-07 12:03:28 -08:00
|
|
|
let book = Orderbook {
|
|
|
|
bids: ctx.accounts.bids.load_mut()?,
|
|
|
|
asks: ctx.accounts.asks.load_mut()?,
|
|
|
|
};
|
2022-11-10 05:41:20 -08:00
|
|
|
|
2024-01-04 09:29:54 -08:00
|
|
|
let oracle_ref = &AccountInfoRef::borrow(ctx.accounts.oracle.as_ref())?;
|
2023-08-07 07:15:45 -07:00
|
|
|
let oracle_state = perp_market.oracle_state(
|
2024-01-04 09:29:54 -08:00
|
|
|
&OracleAccountInfos::from_reader(oracle_ref),
|
2022-11-10 06:47:11 -08:00
|
|
|
None, // staleness checked in health
|
|
|
|
)?;
|
2023-08-07 07:15:45 -07:00
|
|
|
oracle_price = oracle_state.price;
|
2022-11-10 05:41:20 -08:00
|
|
|
|
2023-08-07 07:15:45 -07:00
|
|
|
perp_market.update_funding_and_stable_price(&book, &oracle_state, now_ts)?;
|
2022-11-10 05:41:20 -08:00
|
|
|
}
|
|
|
|
|
2022-12-29 02:48:46 -08:00
|
|
|
let mut account = ctx.accounts.account.load_full_mut()?;
|
2023-01-18 04:19:10 -08:00
|
|
|
// account constraint #1
|
2022-07-25 07:07:53 -07:00
|
|
|
require!(
|
|
|
|
account.fixed.is_owner_or_delegate(ctx.accounts.owner.key()),
|
|
|
|
MangoError::SomeError
|
|
|
|
);
|
|
|
|
|
|
|
|
let account_pk = ctx.accounts.account.key();
|
2022-03-21 12:29:28 -07:00
|
|
|
|
2022-09-29 05:13:28 -07:00
|
|
|
let (perp_market_index, settle_token_index) = {
|
|
|
|
let perp_market = ctx.accounts.perp_market.load()?;
|
|
|
|
(
|
|
|
|
perp_market.perp_market_index,
|
|
|
|
perp_market.settle_token_index,
|
|
|
|
)
|
|
|
|
};
|
2022-09-09 01:50:09 -07:00
|
|
|
|
|
|
|
//
|
|
|
|
// Create the perp position if needed
|
|
|
|
//
|
2022-09-29 05:13:28 -07:00
|
|
|
account.ensure_perp_position(perp_market_index, settle_token_index)?;
|
2022-05-25 10:29:53 -07:00
|
|
|
|
2022-08-24 07:07:22 -07:00
|
|
|
//
|
|
|
|
// Pre-health computation, _after_ perp position is created
|
|
|
|
//
|
|
|
|
let pre_health_opt = if !account.fixed.is_in_health_region() {
|
2024-03-04 06:49:14 -08:00
|
|
|
let retriever = new_fixed_order_account_retriever_with_optional_banks(
|
|
|
|
ctx.remaining_accounts,
|
|
|
|
&account.borrow(),
|
|
|
|
now_slot,
|
|
|
|
)?;
|
|
|
|
let health_cache = new_health_cache_skipping_missing_banks_and_bad_oracles(
|
|
|
|
&account.borrow(),
|
|
|
|
&retriever,
|
|
|
|
now_ts,
|
|
|
|
)
|
|
|
|
.context("pre init health")?;
|
|
|
|
|
|
|
|
// The settle token banks/oracles must be passed and be valid
|
|
|
|
health_cache.token_info_index(settle_token_index)?;
|
|
|
|
|
2023-02-10 00:00:36 -08:00
|
|
|
let pre_init_health = account.check_health_pre(&health_cache)?;
|
|
|
|
Some((health_cache, pre_init_health))
|
2022-08-24 07:07:22 -07:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
|
|
|
|
let mut perp_market = ctx.accounts.perp_market.load_mut()?;
|
2022-12-07 12:03:28 -08:00
|
|
|
let mut book = Orderbook {
|
|
|
|
bids: ctx.accounts.bids.load_mut()?,
|
|
|
|
asks: ctx.accounts.asks.load_mut()?,
|
|
|
|
};
|
2022-08-24 07:07:22 -07:00
|
|
|
|
|
|
|
let mut event_queue = ctx.accounts.event_queue.load_mut()?;
|
2023-02-27 07:36:27 -08:00
|
|
|
let group = ctx.accounts.group.load()?;
|
2022-08-24 07:07:22 -07:00
|
|
|
|
2022-11-10 05:41:20 -08:00
|
|
|
let now_ts: u64 = Clock::get()?.unix_timestamp.try_into().unwrap();
|
2023-02-27 07:36:27 -08:00
|
|
|
account
|
|
|
|
.fixed
|
|
|
|
.expire_buyback_fees(now_ts, group.buyback_fees_expiry_interval);
|
2022-08-24 07:07:22 -07:00
|
|
|
|
2023-01-04 00:24:40 -08:00
|
|
|
let pp = account.perp_position(perp_market_index)?;
|
|
|
|
let effective_pos = pp.effective_base_position_lots();
|
|
|
|
let max_base_lots = if order.reduce_only || perp_market.is_reduce_only() {
|
2023-03-03 00:53:27 -08:00
|
|
|
reduce_only_max_base_lots(pp, &order, perp_market.is_reduce_only())
|
2023-01-04 00:24:40 -08:00
|
|
|
} else {
|
|
|
|
order.max_base_lots
|
|
|
|
};
|
|
|
|
if perp_market.is_reduce_only() {
|
|
|
|
require!(
|
|
|
|
order.reduce_only || max_base_lots == order.max_base_lots,
|
|
|
|
MangoError::MarketInReduceOnlyMode
|
|
|
|
)
|
|
|
|
};
|
|
|
|
order.max_base_lots = max_base_lots;
|
2022-08-24 07:07:22 -07:00
|
|
|
|
2023-01-25 00:03:35 -08:00
|
|
|
let order_id_opt = book.new_order(
|
2022-11-08 06:27:56 -08:00
|
|
|
order,
|
2022-08-24 07:07:22 -07:00
|
|
|
&mut perp_market,
|
|
|
|
&mut event_queue,
|
|
|
|
oracle_price,
|
|
|
|
&mut account.borrow_mut(),
|
|
|
|
&account_pk,
|
|
|
|
now_ts,
|
|
|
|
limit,
|
|
|
|
)?;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Health check
|
|
|
|
//
|
2023-02-10 00:00:36 -08:00
|
|
|
if let Some((mut health_cache, pre_init_health)) = pre_health_opt {
|
2022-09-09 01:50:09 -07:00
|
|
|
let perp_position = account.perp_position(perp_market_index)?;
|
2022-08-24 07:07:22 -07:00
|
|
|
health_cache.recompute_perp_info(perp_position, &perp_market)?;
|
2023-02-10 00:00:36 -08:00
|
|
|
account.check_health_post(&health_cache, pre_init_health)?;
|
2022-08-22 02:02:01 -07:00
|
|
|
}
|
2022-03-20 02:11:36 -07:00
|
|
|
|
2023-01-25 00:03:35 -08:00
|
|
|
Ok(order_id_opt)
|
2022-03-20 02:11:36 -07:00
|
|
|
}
|
2023-03-03 00:53:27 -08:00
|
|
|
|
|
|
|
fn reduce_only_max_base_lots(pp: &PerpPosition, order: &Order, market_reduce_only: bool) -> i64 {
|
|
|
|
let effective_pos = pp.effective_base_position_lots();
|
|
|
|
msg!(
|
|
|
|
"reduce only: current effective position: {} lots",
|
|
|
|
effective_pos
|
|
|
|
);
|
|
|
|
let allowed_base_lots = if (order.side == Side::Bid && effective_pos >= 0)
|
|
|
|
|| (order.side == Side::Ask && effective_pos <= 0)
|
|
|
|
{
|
|
|
|
msg!("reduce only: cannot increase magnitude of effective position");
|
|
|
|
0
|
|
|
|
} else if market_reduce_only {
|
|
|
|
// If the market is in reduce-only mode, we are stricter and pretend
|
|
|
|
// all open orders that go into the same direction as the new order
|
|
|
|
// execute.
|
|
|
|
if order.side == Side::Bid {
|
|
|
|
msg!(
|
|
|
|
"reduce only: effective base position incl open bids is {} lots",
|
|
|
|
effective_pos + pp.bids_base_lots
|
|
|
|
);
|
|
|
|
(effective_pos + pp.bids_base_lots).min(0).abs()
|
|
|
|
} else {
|
|
|
|
msg!(
|
|
|
|
"reduce only: effective base position incl open asks is {} lots",
|
|
|
|
effective_pos - pp.asks_base_lots
|
|
|
|
);
|
|
|
|
(effective_pos - pp.asks_base_lots).max(0)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
effective_pos.abs()
|
|
|
|
};
|
|
|
|
msg!(
|
|
|
|
"reduce only: max allowed {:?}: {} base lots",
|
|
|
|
order.side,
|
|
|
|
allowed_base_lots
|
|
|
|
);
|
|
|
|
allowed_base_lots.min(order.max_base_lots)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_perp_reduce_only() {
|
|
|
|
let test_cases = vec![
|
|
|
|
("null", true, 0, (0, 0), (Side::Bid, 0), 0),
|
|
|
|
("ok bid", true, -5, (0, 0), (Side::Bid, 1), 1),
|
|
|
|
("limited bid", true, -5, (0, 0), (Side::Bid, 10), 5),
|
|
|
|
("limited bid2", true, -5, (1, 10), (Side::Bid, 10), 4),
|
|
|
|
("limited bid3", false, -5, (1, 10), (Side::Bid, 10), 5),
|
|
|
|
("no bid", true, 5, (0, 0), (Side::Bid, 1), 0),
|
|
|
|
("ok ask", true, 5, (0, 0), (Side::Ask, 1), 1),
|
|
|
|
("limited ask", true, 5, (0, 0), (Side::Ask, 10), 5),
|
|
|
|
("limited ask2", true, 5, (10, 1), (Side::Ask, 10), 4),
|
|
|
|
("limited ask3", false, 5, (10, 1), (Side::Ask, 10), 5),
|
|
|
|
("no ask", true, -5, (0, 0), (Side::Ask, 1), 0),
|
|
|
|
];
|
|
|
|
|
|
|
|
for (
|
|
|
|
name,
|
|
|
|
market_reduce_only,
|
|
|
|
base_lots,
|
|
|
|
(open_bids, open_asks),
|
|
|
|
(side, amount),
|
|
|
|
expected,
|
|
|
|
) in test_cases
|
|
|
|
{
|
|
|
|
println!("test: {name}");
|
|
|
|
|
|
|
|
let pp = PerpPosition {
|
|
|
|
base_position_lots: base_lots,
|
|
|
|
bids_base_lots: open_bids,
|
|
|
|
asks_base_lots: open_asks,
|
|
|
|
..PerpPosition::default()
|
|
|
|
};
|
|
|
|
let order = Order {
|
|
|
|
side,
|
|
|
|
max_base_lots: amount,
|
|
|
|
max_quote_lots: 0,
|
|
|
|
client_order_id: 0,
|
|
|
|
reduce_only: true,
|
|
|
|
time_in_force: 0,
|
2023-05-15 01:40:41 -07:00
|
|
|
self_trade_behavior: SelfTradeBehavior::DecrementTake,
|
|
|
|
params: OrderParams::Market {},
|
2023-03-03 00:53:27 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
let result = reduce_only_max_base_lots(&pp, &order, market_reduce_only);
|
|
|
|
assert_eq!(result, expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|