Skip to main content

Staking and Governance

info

This documentation is for version 3 of DeepBook. For documentation on version 2 of DeepBook, see DeepBookV2 docs.

DeepBook's novel approach to governance allows users to update a single pool's three parameters:

  • taker fee rate
  • maker fee rate
  • stake required

Stake required is the amount of DEEP tokens a user must have staked in the pool to take advantage of taker and maker incentives. Each individual DeepBook pool has independent governance, and governance can be conducted every epoch. See Design to learn more about governance.

DeepBook Governance Timeline.png

API

Pool exposes the following endpoints for staking and governance.

Stake

DEEP tokens must be available in the balance_manager for staking. A user's stake becomes active in the following epoch. If the user's active stake is greater than the stake required, the user can get reduced taker fees and can accumulate trading fee rebates during that epoch.

public fun stake<BaseAsset, QuoteAsset>(
self: &mut Pool<BaseAsset, QuoteAsset>,
balance_manager: &mut BalanceManager,
trade_proof: &TradeProof,
amount: u64,
ctx: &TxContext,
) {
assert!(amount > 0, EInvalidStake);
let self = self.load_inner_mut();
let (settled, owed) = self
.state
.process_stake(self.pool_id, balance_manager.id(), amount, ctx);
self
.vault
.settle_balance_manager(settled, owed, balance_manager, trade_proof);
}

Unstake

All of the user's active and inactive stake are removed and added back into the BalanceManager. Any casted votes are removed. Maker rebates for the epoch are forfeited, and any reduced taker fees for the remaining epoch are disabled.

The balance_manager must have enough staked DEEP tokens. The balance_manager data is updated with the unstaked amount. Balance is transferred to the balance_manager immediately.

public fun unstake<BaseAsset, QuoteAsset>(
self: &mut Pool<BaseAsset, QuoteAsset>,
balance_manager: &mut BalanceManager,
trade_proof: &TradeProof,
ctx: &TxContext,
) {
let self = self.load_inner_mut();
let (settled, owed) = self
.state
.process_unstake(self.pool_id, balance_manager.id(), ctx);
self
.vault
.settle_balance_manager(settled, owed, balance_manager, trade_proof);
}

Submit proposal

Users with a non zero active stake can submit proposals. One proposal per user. The user automatically votes for the proposal they submit.

Submit a proposal to change the taker fee, maker fee, and stake required. The balance_manager must have enough staked DEEP tokens to participate. Each balance_manager can only submit one proposal per epoch. If the maximum proposal is reached, the proposal with the lowest vote is removed. If the balance_manager has less voting power than the lowest voted proposal, the proposal is not added.

public fun submit_proposal<BaseAsset, QuoteAsset>(
self: &mut Pool<BaseAsset, QuoteAsset>,
balance_manager: &mut BalanceManager,
trade_proof: &TradeProof,
taker_fee: u64,
maker_fee: u64,
stake_required: u64,
ctx: &TxContext,
) {
let self = self.load_inner_mut();
balance_manager.validate_proof(trade_proof);
self
.state
.process_proposal(
self.pool_id,
balance_manager.id(),
taker_fee,
maker_fee,
stake_required,
ctx,
);
}

Vote

Users with non-zero voting power can vote on a proposal. All voting power is used on a single proposal. If the user has voted on a different proposal during this epoch, then that vote is removed and recasted into the new proposal. The balance_manager must have enough staked DEEP tokens to participate.

public fun vote<BaseAsset, QuoteAsset>(
self: &mut Pool<BaseAsset, QuoteAsset>,
balance_manager: &mut BalanceManager,
trade_proof: &TradeProof,
proposal_id: ID,
ctx: &TxContext,
) {
let self = self.load_inner_mut();
balance_manager.validate_proof(trade_proof);
self
.state
.process_vote(self.pool_id, balance_manager.id(), proposal_id, ctx);
}

Claim rebates

Use claim_rebates to claim the rewards for the balance_manager. The balance_manager must have rewards to claim. The balance_manager data is updated with the claimed rewards.

public fun claim_rebates<BaseAsset, QuoteAsset>(
self: &mut Pool<BaseAsset, QuoteAsset>,
balance_manager: &mut BalanceManager,
trade_proof: &TradeProof,
ctx: &TxContext,
) {
let self = self.load_inner_mut();
let (settled, owed) = self
.state
.process_claim_rebates(self.pool_id, balance_manager.id(), ctx);
self
.vault
.settle_balance_manager(settled, owed, balance_manager, trade_proof);
}