use anyhow::{anyhow, Result};
use ark_ff::ToConstraintField;
use ark_r1cs_std::{
prelude::{AllocVar, EqGadget},
select::CondSelectGadget,
};
use ark_relations::r1cs::{ConstraintSystemRef, SynthesisError};
use decaf377::{r1cs::FqVar, Fq};
use penumbra_sdk_proto::{penumbra::core::component::dex::v1 as pb, DomainType};
use penumbra_sdk_tct::Position;
use serde::{Deserialize, Serialize};
use penumbra_sdk_num::fixpoint::{bit_constrain, U128x128, U128x128Var};
use penumbra_sdk_num::{Amount, AmountVar};
use crate::TradingPairVar;
use super::TradingPair;
#[derive(Clone, Debug, Copy, PartialEq, Eq, Serialize, Deserialize)]
#[serde(try_from = "pb::BatchSwapOutputData", into = "pb::BatchSwapOutputData")]
pub struct BatchSwapOutputData {
pub delta_1: Amount,
pub delta_2: Amount,
pub lambda_1: Amount,
pub lambda_2: Amount,
pub unfilled_1: Amount,
pub unfilled_2: Amount,
pub height: u64,
pub trading_pair: TradingPair,
pub sct_position_prefix: Position,
}
impl BatchSwapOutputData {
pub fn pro_rata_outputs(&self, (delta_1_i, delta_2_i): (Amount, Amount)) -> (Amount, Amount) {
let delta_1_i = U128x128::from(delta_1_i);
let delta_2_i = U128x128::from(delta_2_i);
let delta_1 = U128x128::from(self.delta_1);
let delta_2 = U128x128::from(self.delta_2);
let lambda_1 = U128x128::from(self.lambda_1);
let lambda_2 = U128x128::from(self.lambda_2);
let unfilled_1 = U128x128::from(self.unfilled_1);
let unfilled_2 = U128x128::from(self.unfilled_2);
let pro_rata_input_1 = (delta_1_i / delta_1).unwrap_or_default();
let pro_rata_input_2 = (delta_2_i / delta_2).unwrap_or_default();
let lambda_2_i = (pro_rata_input_1 * lambda_2).unwrap_or_default()
+ (pro_rata_input_2 * unfilled_2).unwrap_or_default();
let lambda_1_i = (pro_rata_input_1 * unfilled_1).unwrap_or_default()
+ (pro_rata_input_2 * lambda_1).unwrap_or_default();
(
lambda_1_i
.unwrap_or_default()
.round_down()
.try_into()
.expect("rounded amount is integral"),
lambda_2_i
.unwrap_or_default()
.round_down()
.try_into()
.expect("rounded amount is integral"),
)
}
}
impl ToConstraintField<Fq> for BatchSwapOutputData {
fn to_field_elements(&self) -> Option<Vec<Fq>> {
let mut public_inputs = Vec::new();
let delta_1 = U128x128::from(self.delta_1);
public_inputs.extend(
delta_1
.to_field_elements()
.expect("delta_1 is a Bls12-377 field member"),
);
public_inputs.extend(
U128x128::from(self.delta_2)
.to_field_elements()
.expect("U128x128 types are Bls12-377 field members"),
);
public_inputs.extend(
U128x128::from(self.lambda_1)
.to_field_elements()
.expect("U128x128 types are Bls12-377 field members"),
);
public_inputs.extend(
U128x128::from(self.lambda_2)
.to_field_elements()
.expect("U128x128 types are Bls12-377 field members"),
);
public_inputs.extend(
U128x128::from(self.unfilled_1)
.to_field_elements()
.expect("U128x128 types are Bls12-377 field members"),
);
public_inputs.extend(
U128x128::from(self.unfilled_2)
.to_field_elements()
.expect("U128x128 types are Bls12-377 field members"),
);
public_inputs.extend(
self.trading_pair
.to_field_elements()
.expect("trading_pair is a Bls12-377 field member"),
);
public_inputs.extend(
Fq::from(self.sct_position_prefix.epoch())
.to_field_elements()
.expect("Position types are Bls12-377 field members"),
);
public_inputs.extend(
Fq::from(self.sct_position_prefix.block())
.to_field_elements()
.expect("Position types are Bls12-377 field members"),
);
Some(public_inputs)
}
}
pub struct BatchSwapOutputDataVar {
pub delta_1: U128x128Var,
pub delta_2: U128x128Var,
pub lambda_1: U128x128Var,
pub lambda_2: U128x128Var,
pub unfilled_1: U128x128Var,
pub unfilled_2: U128x128Var,
pub trading_pair: TradingPairVar,
pub epoch: FqVar,
pub block_within_epoch: FqVar,
}
impl AllocVar<BatchSwapOutputData, Fq> for BatchSwapOutputDataVar {
fn new_variable<T: std::borrow::Borrow<BatchSwapOutputData>>(
cs: impl Into<ark_relations::r1cs::Namespace<Fq>>,
f: impl FnOnce() -> Result<T, SynthesisError>,
mode: ark_r1cs_std::prelude::AllocationMode,
) -> Result<Self, SynthesisError> {
let ns = cs.into();
let cs = ns.cs();
let output_data = *(f()?.borrow());
let delta_1_fixpoint: U128x128 = output_data.delta_1.into();
let delta_1 = U128x128Var::new_variable(cs.clone(), || Ok(delta_1_fixpoint), mode)?;
let delta_2_fixpoint: U128x128 = output_data.delta_2.into();
let delta_2 = U128x128Var::new_variable(cs.clone(), || Ok(delta_2_fixpoint), mode)?;
let lambda_1_fixpoint: U128x128 = output_data.lambda_1.into();
let lambda_1 = U128x128Var::new_variable(cs.clone(), || Ok(lambda_1_fixpoint), mode)?;
let lambda_2_fixpoint: U128x128 = output_data.lambda_2.into();
let lambda_2 = U128x128Var::new_variable(cs.clone(), || Ok(lambda_2_fixpoint), mode)?;
let unfilled_1_fixpoint: U128x128 = output_data.unfilled_1.into();
let unfilled_1 = U128x128Var::new_variable(cs.clone(), || Ok(unfilled_1_fixpoint), mode)?;
let unfilled_2_fixpoint: U128x128 = output_data.unfilled_2.into();
let unfilled_2 = U128x128Var::new_variable(cs.clone(), || Ok(unfilled_2_fixpoint), mode)?;
let trading_pair = TradingPairVar::new_variable_unchecked(
cs.clone(),
|| Ok(output_data.trading_pair),
mode,
)?;
let epoch = FqVar::new_variable(
cs.clone(),
|| Ok(Fq::from(output_data.sct_position_prefix.epoch())),
mode,
)?;
bit_constrain(epoch.clone(), 16)?;
let block_within_epoch = FqVar::new_variable(
cs.clone(),
|| Ok(Fq::from(output_data.sct_position_prefix.block())),
mode,
)?;
bit_constrain(block_within_epoch.clone(), 16)?;
Ok(Self {
delta_1,
delta_2,
lambda_1,
lambda_2,
unfilled_1,
unfilled_2,
trading_pair,
epoch,
block_within_epoch,
})
}
}
impl DomainType for BatchSwapOutputData {
type Proto = pb::BatchSwapOutputData;
}
impl From<BatchSwapOutputData> for pb::BatchSwapOutputData {
fn from(s: BatchSwapOutputData) -> Self {
#[allow(deprecated)]
pb::BatchSwapOutputData {
delta_1: Some(s.delta_1.into()),
delta_2: Some(s.delta_2.into()),
lambda_1: Some(s.lambda_1.into()),
lambda_2: Some(s.lambda_2.into()),
unfilled_1: Some(s.unfilled_1.into()),
unfilled_2: Some(s.unfilled_2.into()),
height: s.height,
trading_pair: Some(s.trading_pair.into()),
sct_position_prefix: s.sct_position_prefix.into(),
epoch_starting_height: Default::default(),
}
}
}
impl BatchSwapOutputDataVar {
pub fn pro_rata_outputs(
&self,
delta_1_i: AmountVar,
delta_2_i: AmountVar,
cs: ConstraintSystemRef<Fq>,
) -> Result<(AmountVar, AmountVar), SynthesisError> {
let delta_1_i = U128x128Var::from_amount_var(delta_1_i)?;
let delta_2_i = U128x128Var::from_amount_var(delta_2_i)?;
let zero = U128x128Var::zero();
let one = U128x128Var::new_constant(cs.clone(), U128x128::from(1u64))?;
let delta_1_is_zero = self.delta_1.is_eq(&zero)?;
let divisor_1 = U128x128Var::conditionally_select(&delta_1_is_zero, &one, &self.delta_1)?;
let division_result_1 = delta_1_i.checked_div(&divisor_1, cs.clone())?;
let pro_rata_input_1 =
U128x128Var::conditionally_select(&delta_1_is_zero, &zero, &division_result_1)?;
let delta_2_is_zero = self.delta_2.is_eq(&zero)?;
let divisor_2 = U128x128Var::conditionally_select(&delta_2_is_zero, &one, &self.delta_2)?;
let division_result_2 = delta_2_i.checked_div(&divisor_2, cs)?;
let pro_rata_input_2 =
U128x128Var::conditionally_select(&delta_2_is_zero, &zero, &division_result_2)?;
let addition_term2_1 = pro_rata_input_1.clone().checked_mul(&self.lambda_2)?;
let addition_term2_2 = pro_rata_input_2.clone().checked_mul(&self.unfilled_2)?;
let lambda_2_i = addition_term2_1.checked_add(&addition_term2_2)?;
let addition_term1_1 = pro_rata_input_1.checked_mul(&self.unfilled_1)?;
let addition_term1_2 = pro_rata_input_2.checked_mul(&self.lambda_1)?;
let lambda_1_i = addition_term1_1.checked_add(&addition_term1_2)?;
let lambda_1_i_rounded = lambda_1_i.round_down();
let lambda_2_i_rounded = lambda_2_i.round_down();
Ok((lambda_1_i_rounded.into(), lambda_2_i_rounded.into()))
}
}
impl From<BatchSwapOutputData> for pb::BatchSwapOutputDataResponse {
fn from(s: BatchSwapOutputData) -> Self {
pb::BatchSwapOutputDataResponse {
data: Some(s.into()),
}
}
}
impl TryFrom<pb::BatchSwapOutputData> for BatchSwapOutputData {
type Error = anyhow::Error;
fn try_from(s: pb::BatchSwapOutputData) -> Result<Self, Self::Error> {
let sct_position_prefix = {
let prefix = Position::from(s.sct_position_prefix);
anyhow::ensure!(
prefix.commitment() == 0,
"sct_position_prefix.commitment() != 0"
);
prefix
};
Ok(Self {
delta_1: s
.delta_1
.ok_or_else(|| anyhow!("Missing delta_1"))?
.try_into()?,
delta_2: s
.delta_2
.ok_or_else(|| anyhow!("Missing delta_2"))?
.try_into()?,
lambda_1: s
.lambda_1
.ok_or_else(|| anyhow!("Missing lambda_1"))?
.try_into()?,
lambda_2: s
.lambda_2
.ok_or_else(|| anyhow!("Missing lambda_2"))?
.try_into()?,
unfilled_1: s
.unfilled_1
.ok_or_else(|| anyhow!("Missing unfilled_1"))?
.try_into()?,
unfilled_2: s
.unfilled_2
.ok_or_else(|| anyhow!("Missing unfilled_2"))?
.try_into()?,
height: s.height,
trading_pair: s
.trading_pair
.ok_or_else(|| anyhow!("Missing trading_pair"))?
.try_into()?,
sct_position_prefix,
})
}
}
impl TryFrom<pb::BatchSwapOutputDataResponse> for BatchSwapOutputData {
type Error = anyhow::Error;
fn try_from(value: pb::BatchSwapOutputDataResponse) -> Result<Self, Self::Error> {
value
.data
.ok_or_else(|| anyhow::anyhow!("empty BatchSwapOutputDataResponse message"))?
.try_into()
}
}
#[cfg(test)]
mod tests {
use ark_groth16::{r1cs_to_qap::LibsnarkReduction, Groth16};
use ark_relations::r1cs::ConstraintSynthesizer;
use ark_snark::SNARK;
use decaf377::Bls12_377;
use penumbra_sdk_asset::asset;
use penumbra_sdk_proof_params::{generate_test_parameters, DummyWitness};
use rand_core::OsRng;
use super::*;
#[test]
fn pasiphae_inflation_bug() {
let bsod: BatchSwapOutputData = serde_json::from_str(
r#"
{
"delta1": {
"lo": "31730032"
},
"delta2": {},
"unfilled1": {},
"lambda2": {
"lo": "28766268"
},
"lambda1": {},
"unfilled2": {},
"height": "2185",
"tradingPair": {
"asset1": {
"inner": "HW2Eq3UZVSBttoUwUi/MUtE7rr2UU7/UH500byp7OAc="
},
"asset2": {
"inner": "KeqcLzNx9qSH5+lcJHBB9KNW+YPrBk5dKzvPMiypahA="
}
}
}"#,
)
.unwrap();
let (delta_1_i, delta_2_i) = (Amount::from(31730032u64), Amount::from(0u64));
let (lambda_1_i, lambda_2_i) = bsod.pro_rata_outputs((delta_1_i, delta_2_i));
assert_eq!(lambda_1_i, Amount::from(0u64));
assert_eq!(lambda_2_i, Amount::from(28766268u64));
}
struct ProRataOutputCircuit {
delta_1_i: Amount,
delta_2_i: Amount,
lambda_1_i: Amount,
lambda_2_i: Amount,
pub bsod: BatchSwapOutputData,
}
impl ConstraintSynthesizer<Fq> for ProRataOutputCircuit {
fn generate_constraints(
self,
cs: ConstraintSystemRef<Fq>,
) -> ark_relations::r1cs::Result<()> {
let delta_1_i_var = AmountVar::new_witness(cs.clone(), || Ok(self.delta_1_i))?;
let delta_2_i_var = AmountVar::new_witness(cs.clone(), || Ok(self.delta_2_i))?;
let lambda_1_i_var = AmountVar::new_witness(cs.clone(), || Ok(self.lambda_1_i))?;
let lambda_2_i_var = AmountVar::new_witness(cs.clone(), || Ok(self.lambda_2_i))?;
let bsod_var = BatchSwapOutputDataVar::new_input(cs.clone(), || Ok(self.bsod))?;
let (calculated_lambda_1_i_var, calculated_lambda_2_i_var) =
bsod_var.pro_rata_outputs(delta_1_i_var, delta_2_i_var, cs.clone())?;
calculated_lambda_1_i_var.enforce_equal(&lambda_1_i_var)?;
calculated_lambda_2_i_var.enforce_equal(&lambda_2_i_var)?;
Ok(())
}
}
impl DummyWitness for ProRataOutputCircuit {
fn with_dummy_witness() -> Self {
let trading_pair = TradingPair {
asset_1: asset::Cache::with_known_assets()
.get_unit("upenumbra")
.expect("upenumbra denom should always be known by the asset registry")
.id(),
asset_2: asset::Cache::with_known_assets()
.get_unit("nala")
.expect("nala denom should always be known by the asset registry")
.id(),
};
Self {
delta_1_i: Amount::from(1u32),
delta_2_i: Amount::from(1u32),
lambda_1_i: Amount::from(1u32),
lambda_2_i: Amount::from(1u32),
bsod: BatchSwapOutputData {
delta_1: Amount::from(1u32),
delta_2: Amount::from(1u32),
lambda_1: Amount::from(1u32),
lambda_2: Amount::from(1u32),
unfilled_1: Amount::from(1u32),
unfilled_2: Amount::from(1u32),
height: 0,
trading_pair,
sct_position_prefix: 0u64.into(),
},
}
}
}
#[test]
fn happy_path_bsod_pro_rata() {
let gm = asset::Cache::with_known_assets().get_unit("gm").unwrap();
let gn = asset::Cache::with_known_assets().get_unit("gn").unwrap();
let trading_pair = TradingPair::new(gm.id(), gn.id());
let bsod = BatchSwapOutputData {
delta_1: Amount::from(200u64),
delta_2: Amount::from(300u64),
lambda_1: Amount::from(150u64),
lambda_2: Amount::from(125u64),
unfilled_1: Amount::from(23u64),
unfilled_2: Amount::from(50u64),
height: 0u64,
trading_pair,
sct_position_prefix: 0u64.into(),
};
let delta_1_i = Amount::from(100u64);
let delta_2_i = Amount::from(200u64);
let (lambda_1_i, lambda_2_i) = bsod.pro_rata_outputs((delta_1_i, delta_2_i));
let circuit = ProRataOutputCircuit {
delta_1_i,
delta_2_i,
lambda_1_i,
lambda_2_i,
bsod,
};
let mut rng = OsRng;
let (pk, vk) = generate_test_parameters::<ProRataOutputCircuit>(&mut rng);
let proof = Groth16::<Bls12_377, LibsnarkReduction>::prove(&pk, circuit, &mut rng)
.expect("should be able to form proof");
let proof_result = Groth16::<Bls12_377, LibsnarkReduction>::verify(
&vk,
&bsod.to_field_elements().unwrap(),
&proof,
)
.expect("should be able to verify proof");
assert!(proof_result);
}
}