Swaps
This documentation is for version 3 of DeepBook. For documentation on version 2 of DeepBook, see DeepBookV2 docs.
DeepBook provides a swap-like interface commonly seen in automatic market makers (AMMs). Unlike the order functions, you can call swap_exact_amount
without a BalanceManager
. You call it directly with Coin
objects instead. When swapping from base to quote, base_in
must have a positive value while quote_in
must be zero. When swapping from quote to base, quote_in
must be positive and base_in
zero. Some deep_in
amount is required to pay for trading fees. You can overestimate this amount, as the unused DEEP tokens are returned at the end of the call.
You can use the get_amount_out
endpoint to simulate a swap. The function returns the exact amount of DEEP tokens that the swap requires.
API
Following are the endpoints that the Pool
exposes for swaps.
Swap exact base for quote
Swap exact base quantity without needing a balance_manager
. DEEP quantity can be overestimated. Returns three Coin
objects:
BaseAsset
QuoteAsset
DEEP
Some base quantity may be left over, if the input quantity is not divisible by lot size.
You can overestimate the amount of DEEP required. The remaining balance is returned.
public fun swap_exact_base_for_quote<BaseAsset, QuoteAsset>(
self: &mut Pool<BaseAsset, QuoteAsset>,
base_in: Coin<BaseAsset>,
deep_in: Coin<DEEP>,
min_quote_out: u64,
clock: &Clock,
ctx: &mut TxContext,
): (Coin<BaseAsset>, Coin<QuoteAsset>, Coin<DEEP>) {
let quote_in = coin::zero(ctx);
self.swap_exact_quantity(
base_in,
quote_in,
deep_in,
min_quote_out,
clock,
ctx,
)
}
Swap exact quote for base
Swap exact quote quantity without needing a balance_manager
. You can overestimate DEEP quantity. Returns three Coin
objects:
BaseAsset
QuoteAsset
DEEP
Some quote quantity might be left over if the input quantity is not divisible by lot size.
public fun swap_exact_quote_for_base<BaseAsset, QuoteAsset>(
self: &mut Pool<BaseAsset, QuoteAsset>,
quote_in: Coin<QuoteAsset>,
deep_in: Coin<DEEP>,
min_base_out: u64,
clock: &Clock,
ctx: &mut TxContext,
): (Coin<BaseAsset>, Coin<QuoteAsset>, Coin<DEEP>) {
let base_in = coin::zero(ctx);
self.swap_exact_quantity(
base_in,
quote_in,
deep_in,
min_base_out,
clock,
ctx,
)
}
Swap exact quantity
This function is what the previous two functions call with coin::zero()
set for the third coin. Users can call this directly for base → quote or quote → base as long as base or quote have a zero value.
public fun swap_exact_quantity<BaseAsset, QuoteAsset>(
self: &mut Pool<BaseAsset, QuoteAsset>,
base_in: Coin<BaseAsset>,
quote_in: Coin<QuoteAsset>,
deep_in: Coin<DEEP>,
min_out: u64,
clock: &Clock,
ctx: &mut TxContext,
): (Coin<BaseAsset>, Coin<QuoteAsset>, Coin<DEEP>) {
let mut base_quantity = base_in.value();
let quote_quantity = quote_in.value();
assert!((base_quantity > 0) != (quote_quantity > 0), EInvalidQuantityIn);
let pay_with_deep = deep_in.value() > 0;
let is_bid = quote_quantity > 0;
if (is_bid) {
(base_quantity, _, _) = self.get_quantity_out(0, quote_quantity, clock);
};
base_quantity =
base_quantity - base_quantity % self.load_inner().book.lot_size();
if (base_quantity < self.load_inner().book.min_size()) {
return (base_in, quote_in, deep_in)
};
let mut temp_balance_manager = balance_manager::new(ctx);
let trade_proof = temp_balance_manager.generate_proof_as_owner(ctx);
temp_balance_manager.deposit(base_in, ctx);
temp_balance_manager.deposit(quote_in, ctx);
temp_balance_manager.deposit(deep_in, ctx);
self.place_market_order(
&mut temp_balance_manager,
&trade_proof,
0,
constants::self_matching_allowed(),
base_quantity,
is_bid,
pay_with_deep,
clock,
ctx,
);
let base_out = temp_balance_manager.withdraw_all<BaseAsset>(ctx);
let quote_out = temp_balance_manager.withdraw_all<QuoteAsset>(ctx);
let deep_out = temp_balance_manager.withdraw_all<DEEP>(ctx);
if (is_bid) {
assert!(base_out.value() >= min_out, EMinimumQuantityOutNotMet);
} else {
assert!(quote_out.value() >= min_out, EMinimumQuantityOutNotMet);
};
temp_balance_manager.delete();
(base_out, quote_out, deep_out)
}