Skip to content

Commit

Permalink
dex: add DexTester harness prop
Browse files Browse the repository at this point in the history
Co-authored-by: Lucas Meier <lucas@cronokirby.com>
  • Loading branch information
2 people authored and conorsch committed Dec 18, 2024
1 parent c9b5db8 commit c08b2ef
Show file tree
Hide file tree
Showing 6 changed files with 374 additions and 36 deletions.
25 changes: 16 additions & 9 deletions crates/bin/pd/src/network/generate.rs
Original file line number Diff line number Diff line change
Expand Up @@ -428,8 +428,8 @@ pub fn network_generate(
/// Represents initial allocations to the testnet.
#[derive(Debug, Deserialize)]
pub struct NetworkAllocation {
#[serde(deserialize_with = "string_u64")]
pub amount: u64,
#[serde(deserialize_with = "string_u128")]
pub amount: u128,
pub denom: String,
pub address: String,
}
Expand Down Expand Up @@ -670,36 +670,43 @@ impl TryFrom<NetworkAllocation> for shielded_pool_genesis::Allocation {
}
}

fn string_u64<'de, D>(deserializer: D) -> Result<u64, D::Error>
fn string_u128<'de, D>(deserializer: D) -> Result<u128, D::Error>
where
D: de::Deserializer<'de>,
{
struct U64StringVisitor;
struct U128StringVisitor;

impl<'de> de::Visitor<'de> for U64StringVisitor {
type Value = u64;
impl<'de> de::Visitor<'de> for U128StringVisitor {
type Value = u128;

fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result {
formatter.write_str("a string containing a u64 with optional underscores")
formatter.write_str("a string containing a u128 with optional underscores")
}

fn visit_str<E>(self, v: &str) -> Result<Self::Value, E>
where
E: de::Error,
{
let r = v.replace('_', "");
r.parse::<u64>().map_err(E::custom)
r.parse::<u128>().map_err(E::custom)
}

fn visit_u64<E>(self, v: u64) -> Result<Self::Value, E>
where
E: de::Error,
{
Ok(v as u128)
}

fn visit_u128<E>(self, v: u128) -> std::prelude::v1::Result<Self::Value, E>
where
E: de::Error,
{
Ok(v)
}
}

deserializer.deserialize_any(U64StringVisitor)
deserializer.deserialize_any(U128StringVisitor)
}

#[cfg(test)]
Expand Down
4 changes: 4 additions & 0 deletions crates/core/component/dex/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,10 @@ name = "penumbra-dex"
version = {workspace = true}
edition = {workspace = true}

[[test]]
name = "penumbra-dex-integration-tests"
path = "tests/integration/mod.rs"

[features]
component = [
"cnidarium-component",
Expand Down
65 changes: 38 additions & 27 deletions crates/core/component/dex/src/component/router/route_and_fill.rs
Original file line number Diff line number Diff line change
Expand Up @@ -44,11 +44,21 @@ pub trait HandleBatchSwaps: StateWrite + Sized {
// executions up to the specified `execution_budget` parameter.
let execution_circuit_breaker = ExecutionCircuitBreaker::new(execution_budget);

// We clamp the deltas to the maximum input for batch swaps.
let clamped_delta_1 = delta_1.min(MAX_RESERVE_AMOUNT.into());
let clamped_delta_2 = delta_2.min(MAX_RESERVE_AMOUNT.into());

tracing::debug!(
?clamped_delta_1,
?clamped_delta_2,
"clamped deltas to maximum amount"
);

let swap_execution_1_for_2 = self
.route_and_fill(
trading_pair.asset_1(),
trading_pair.asset_2(),
delta_1,
clamped_delta_1,
params.clone(),
execution_circuit_breaker.clone(),
)
Expand All @@ -58,7 +68,7 @@ pub trait HandleBatchSwaps: StateWrite + Sized {
.route_and_fill(
trading_pair.asset_2(),
trading_pair.asset_1(),
delta_2,
clamped_delta_2,
params.clone(),
execution_circuit_breaker,
)
Expand All @@ -67,6 +77,7 @@ pub trait HandleBatchSwaps: StateWrite + Sized {
let (lambda_2, unfilled_1) = match &swap_execution_1_for_2 {
Some(swap_execution) => (
swap_execution.output.amount,
// The unfilled amount of asset 1 is the trade input minus the amount consumed, plus the excess.
delta_1 - swap_execution.input.amount,
),
None => (0u64.into(), delta_1),
Expand Down Expand Up @@ -148,9 +159,10 @@ pub trait RouteAndFill: StateWrite + Sized {
where
Self: 'static,
{
tracing::debug!(?input, ?asset_1, ?asset_2, "starting route_and_fill");
tracing::debug!(?input, ?asset_1, ?asset_2, "prepare to route and fill");

if input == Amount::zero() {
tracing::debug!("no input, short-circuit exit");
return Ok(None);
}

Expand All @@ -162,13 +174,12 @@ pub trait RouteAndFill: StateWrite + Sized {
// An ordered list of execution traces that were used to fill the trade.
let mut traces: Vec<Vec<Value>> = Vec::new();

let max_delta_1: Amount = MAX_RESERVE_AMOUNT.into();

// Termination conditions:
// 1. We have no more `delta_1` remaining
// 2. A path can no longer be found
// 3. We have reached the `RoutingParams` specified price limit
// 4. The execution circuit breaker has been triggered based on the number of path searches and executions
// 5. An unrecoverable error occurred during the execution of the route.
loop {
// Check if we have exceeded the execution circuit breaker limits.
if execution_circuit_breaker.exceeded_limits() {
Expand Down Expand Up @@ -196,21 +207,20 @@ pub trait RouteAndFill: StateWrite + Sized {
break;
}

// We split off the entire batch swap into smaller chunks to avoid causing
// a series of overflow in the DEX.
// We prepare the input for this execution round, which is the remaining unfilled amount of asset 1.
let delta_1 = Value {
amount: total_unfilled_1.min(max_delta_1),
asset_id: asset_1,
amount: total_unfilled_1,
};

tracing::debug!(?path, delta_1 = ?delta_1.amount, "found path, filling up to spill price");
tracing::debug!(?path, ?delta_1, "found path, filling up to spill price");

let execution = Arc::get_mut(self)
let execution_result = Arc::get_mut(self)
.expect("expected state to have no other refs")
.fill_route(delta_1, &path, spill_price)
.await;

let execution = match execution {
let swap_execution = match execution_result {
Ok(execution) => execution,
Err(FillError::ExecutionOverflow(position_id)) => {
// We have encountered an overflow during the execution of the route.
Expand All @@ -233,24 +243,25 @@ pub trait RouteAndFill: StateWrite + Sized {

// Immediately track the execution in the state.
(total_output_2, total_unfilled_1) = {
let lambda_2 = execution.output;
let unfilled_1 = Value {
amount: total_unfilled_1
.checked_sub(&execution.input.amount)
.expect("unable to subtract unfilled input from total input"),
asset_id: asset_1,
};
tracing::debug!(input = ?delta_1.amount, output = ?lambda_2.amount, unfilled = ?unfilled_1.amount, "filled along best path");
// The exact amount of asset 1 that was consumed in this execution round.
let consumed_input = swap_execution.input;
// The output of this execution round is the amount of asset 2 that was filled.
let produced_output = swap_execution.output;

tracing::debug!(consumed_input = ?consumed_input.amount, output = ?produced_output.amount, "filled along best path");

assert_eq!(lambda_2.asset_id, asset_2);
assert_eq!(unfilled_1.asset_id, asset_1);
// Sanity check that the input and output assets are correct.
assert_eq!(produced_output.asset_id, asset_2);
assert_eq!(consumed_input.asset_id, asset_1);

// Append the traces from this execution to the outer traces.
traces.append(&mut execution.traces.clone());
traces.append(&mut swap_execution.traces.clone());

(
total_output_2 + lambda_2.amount,
total_unfilled_1 - delta_1.amount + unfilled_1.amount,
// The total output of asset 2 is the sum of all outputs.
total_output_2 + produced_output.amount,
// The total unfilled amount of asset 1 is the remaining unfilled amount minus the amount consumed.
total_unfilled_1 - consumed_input.amount,
)
};

Expand All @@ -260,7 +271,7 @@ pub trait RouteAndFill: StateWrite + Sized {
}

// Ensure that we've actually executed, or else bail out.
let Some(accurate_max_price) = execution.max_price() else {
let Some(accurate_max_price) = swap_execution.max_price() else {
tracing::debug!("no traces in execution, exiting route_and_fill");
break;
};
Expand All @@ -278,15 +289,15 @@ pub trait RouteAndFill: StateWrite + Sized {
}
}

// If we didn't execute against any position at all, there
// are no execution records to return.
// If we didn't execute against any position at all, there are no execution records to return.
if traces.is_empty() {
return Ok(None);
} else {
Ok(Some(SwapExecution {
traces,
input: Value {
asset_id: asset_1,
// The total amount of asset 1 that was actually consumed across rounds.
amount: input - total_unfilled_1,
},
output: Value {
Expand Down
2 changes: 2 additions & 0 deletions crates/core/component/dex/src/swap_execution.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,9 @@ use serde::{Deserialize, Serialize};
#[serde(try_from = "pb::SwapExecution", into = "pb::SwapExecution")]
pub struct SwapExecution {
pub traces: Vec<Vec<Value>>,
/// The input value that was consumed.
pub input: Value,
/// The output value that was produced.
pub output: Value,
}

Expand Down
Loading

0 comments on commit c08b2ef

Please sign in to comment.