penumbra_sdk_dex/component/circuit_breaker/
value.rsuse anyhow::{anyhow, Result};
use cnidarium::{StateRead, StateWrite};
use penumbra_sdk_asset::{asset, Value};
use penumbra_sdk_num::Amount;
use penumbra_sdk_proto::{DomainType, StateReadProto, StateWriteProto};
use tonic::async_trait;
use tracing::instrument;
use crate::{event, state_key};
#[async_trait]
pub trait ValueCircuitBreakerRead: StateRead {
async fn get_dex_vcb_for_asset(&self, id: &asset::Id) -> Result<Option<Amount>> {
Ok(self.get(&state_key::value_balance(&id)).await?)
}
}
impl<T: StateRead + ?Sized> ValueCircuitBreakerRead for T {}
#[async_trait]
pub(crate) trait ValueCircuitBreaker: StateWrite {
#[instrument(skip(self))]
async fn dex_vcb_credit(&mut self, value: Value) -> Result<()> {
if value.amount == Amount::zero() {
return Ok(());
}
let prev_balance: Amount = self
.get_dex_vcb_for_asset(&value.asset_id)
.await?
.unwrap_or_default();
let new_balance = prev_balance
.checked_add(&value.amount)
.ok_or_else(|| anyhow!("overflowed balance while crediting value circuit breaker (prev balance={prev_balance:?}, credit={value:?}"))?;
tracing::debug!(?prev_balance, ?new_balance, "crediting the dex VCB");
self.put(state_key::value_balance(&value.asset_id), new_balance);
self.record_proto(
event::EventValueCircuitBreakerCredit {
asset_id: value.asset_id,
previous_balance: prev_balance,
new_balance,
}
.to_proto(),
);
Ok(())
}
#[instrument(skip(self))]
async fn dex_vcb_debit(&mut self, value: Value) -> Result<()> {
if value.amount == Amount::zero() {
return Ok(());
}
let prev_balance: Amount = self
.get_dex_vcb_for_asset(&value.asset_id)
.await?
.unwrap_or_default();
let new_balance = prev_balance
.checked_sub(&value.amount)
.ok_or_else(|| anyhow!("underflowed balance while debiting value circuit breaker (prev balance={prev_balance:?}, debit={value:?}"))?;
tracing::debug!(?prev_balance, ?new_balance, "crediting the dex VCB");
self.put(state_key::value_balance(&value.asset_id), new_balance);
self.record_proto(
event::EventValueCircuitBreakerDebit {
asset_id: value.asset_id,
previous_balance: prev_balance,
new_balance,
}
.to_proto(),
);
Ok(())
}
}
impl<T: StateWrite + ?Sized> ValueCircuitBreaker for T {}
#[cfg(test)]
mod tests {
use std::sync::Arc;
use crate::component::position_manager::price_index::PositionByPriceIndex;
use crate::component::router::HandleBatchSwaps as _;
use crate::component::{InternalDexWrite, StateReadExt as _, SwapDataRead, SwapDataWrite};
use crate::lp::plan::PositionWithdrawPlan;
use crate::{
component::{router::create_buy, tests::TempStorageExt},
state_key, DirectedUnitPair,
};
use crate::{BatchSwapOutputData, PositionOpen};
use cnidarium::{ArcStateDeltaExt as _, StateDelta, TempStorage};
use cnidarium_component::ActionHandler as _;
use penumbra_sdk_asset::asset;
use penumbra_sdk_num::Amount;
use penumbra_sdk_proto::StateWriteProto as _;
use penumbra_sdk_sct::component::clock::EpochManager as _;
use penumbra_sdk_sct::component::source::SourceContext as _;
use penumbra_sdk_sct::epoch::Epoch;
use super::*;
#[tokio::test]
async fn value_circuit_breaker() -> anyhow::Result<()> {
let _ = tracing_subscriber::fmt::try_init();
let storage = TempStorage::new().await?.apply_minimal_genesis().await?;
let mut state = Arc::new(StateDelta::new(storage.latest_snapshot()));
let mut state_tx = state.try_begin_transaction().unwrap();
let gm = asset::Cache::with_known_assets().get_unit("gm").unwrap();
let gn = asset::Cache::with_known_assets().get_unit("gn").unwrap();
let test_usd = asset::Cache::with_known_assets()
.get_unit("test_usd")
.unwrap();
state_tx.dex_vcb_credit(gm.value(100u64.into())).await?;
state_tx.dex_vcb_credit(gn.value(100u64.into())).await?;
state_tx.dex_vcb_debit(gm.value(100u64.into())).await?;
state_tx.dex_vcb_debit(gn.value(100u64.into())).await?;
assert!(state_tx.dex_vcb_debit(gm.value(1u64.into())).await.is_err());
assert!(state_tx
.dex_vcb_debit(test_usd.value(1u64.into()))
.await
.is_err());
Ok(())
}
#[tokio::test]
async fn position_value_circuit_breaker() -> anyhow::Result<()> {
let _ = tracing_subscriber::fmt::try_init();
let storage = TempStorage::new().await?.apply_minimal_genesis().await?;
let mut state = Arc::new(StateDelta::new(storage.latest_snapshot()));
let mut state_tx = state.try_begin_transaction().unwrap();
let height = 1;
state_tx.put_epoch_by_height(
height,
Epoch {
index: 0,
start_height: 0,
},
);
state_tx.put_block_height(height);
state_tx.apply();
let gm = asset::Cache::with_known_assets().get_unit("gm").unwrap();
let gn = asset::Cache::with_known_assets().get_unit("gn").unwrap();
let pair_1 = DirectedUnitPair::new(gm.clone(), gn.clone());
let one = 1u64.into();
let price1 = one;
let buy_1 = create_buy(pair_1.clone(), 1u64.into(), price1);
let pos_open = PositionOpen {
position: buy_1.clone(),
};
pos_open.check_stateless(()).await?;
pos_open.check_historical(state.clone()).await?;
let mut state_tx = state.try_begin_transaction().unwrap();
state_tx.put_mock_source(1u8);
pos_open.check_and_execute(&mut state_tx).await?;
state_tx.apply();
let mut state_tx = state.try_begin_transaction().unwrap();
state_tx
.set_output_data(
BatchSwapOutputData {
delta_1: 0u64.into(),
delta_2: 1u64.into(),
lambda_1: 0u64.into(),
lambda_2: 0u64.into(),
unfilled_1: 0u64.into(),
unfilled_2: 0u64.into(),
height: 1,
trading_pair: pair_1.into_directed_trading_pair().into(),
sct_position_prefix: Default::default(),
},
None,
None,
)
.await?;
state_tx.put(state_key::value_balance(&gm.id()), Amount::from(0u64));
let pos_withdraw_plan = PositionWithdrawPlan {
position_id: buy_1.id(),
reserves: buy_1.reserves,
sequence: 1,
pair: pair_1.into_directed_trading_pair().into(),
rewards: vec![],
};
let pos_withdraw = pos_withdraw_plan.position_withdraw();
pos_withdraw.check_stateless(()).await?;
pos_withdraw.check_historical(state.clone()).await?;
let mut state_tx = state.try_begin_transaction().unwrap();
state_tx.put_mock_source(1u8);
assert!(pos_withdraw.check_and_execute(&mut state_tx).await.is_err());
state_tx.apply();
Ok(())
}
#[tokio::test]
#[should_panic(expected = "underflowed balance while debiting value circuit breaker")]
async fn batch_swap_circuit_breaker() {
let _ = tracing_subscriber::fmt::try_init();
let storage = TempStorage::new()
.await
.expect("able to create storage")
.apply_minimal_genesis()
.await
.expect("able to apply genesis");
let mut state = Arc::new(StateDelta::new(storage.latest_snapshot()));
let mut state_tx = state.try_begin_transaction().unwrap();
let gm = asset::Cache::with_known_assets().get_unit("gm").unwrap();
let gn = asset::Cache::with_known_assets().get_unit("gn").unwrap();
let pair_1 = DirectedUnitPair::new(gm.clone(), gn.clone());
let one = 1u64.into();
let price1 = one;
let buy_1 = create_buy(pair_1.clone(), 1u64.into(), price1);
let id = buy_1.id();
let position = buy_1;
state_tx
.update_position_by_price_index(&position.id(), &None, &position)
.expect("can update price index");
state_tx.put(state_key::position_by_id(&id), position);
let trading_pair = pair_1.into_directed_trading_pair().into();
let mut swap_flow = state_tx.swap_flow(&trading_pair);
assert!(trading_pair.asset_1() == gm.id());
swap_flow.0 += gm.value(5u32.into()).amount;
swap_flow.1 += 0u32.into();
state_tx
.accumulate_swap_flow(&trading_pair, swap_flow.clone())
.await
.unwrap();
state_tx.apply();
let routing_params = state.routing_params().await.unwrap();
let max_execution = state.get_dex_params().await.unwrap().max_execution_budget;
state
.handle_batch_swaps(trading_pair, swap_flow, 0, routing_params, max_execution)
.await
.expect("unable to process batch swaps");
}
}