Skip to main content

Swaps

info

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)
}