penumbra_sdk_dex/swap/
plan.rsuse anyhow::{anyhow, Context, Result};
use ark_ff::Zero;
use decaf377::{Fq, Fr};
use penumbra_sdk_asset::{balance, Balance, Value};
use penumbra_sdk_keys::FullViewingKey;
use penumbra_sdk_proto::{penumbra::core::component::dex::v1 as pb, DomainType};
use rand_core::{CryptoRng, RngCore};
use serde::{Deserialize, Serialize};
use crate::swap::proof::{SwapProofPrivate, SwapProofPublic};
use super::{action as swap, proof::SwapProof, Swap, SwapPlaintext};
#[derive(Clone, Debug, Deserialize, Serialize)]
#[serde(try_from = "pb::SwapPlan", into = "pb::SwapPlan")]
pub struct SwapPlan {
pub swap_plaintext: SwapPlaintext,
pub fee_blinding: Fr,
pub proof_blinding_r: Fq,
pub proof_blinding_s: Fq,
}
impl SwapPlan {
pub fn new<R: CryptoRng + RngCore>(rng: &mut R, swap_plaintext: SwapPlaintext) -> SwapPlan {
let fee_blinding = Fr::rand(rng);
SwapPlan {
fee_blinding,
swap_plaintext,
proof_blinding_r: Fq::rand(rng),
proof_blinding_s: Fq::rand(rng),
}
}
pub fn swap(&self, fvk: &FullViewingKey) -> Swap {
Swap {
body: self.swap_body(fvk),
proof: self.swap_proof(),
}
}
pub fn swap_body(&self, fvk: &FullViewingKey) -> swap::Body {
swap::Body {
trading_pair: self.swap_plaintext.trading_pair,
delta_1_i: self.swap_plaintext.delta_1_i,
delta_2_i: self.swap_plaintext.delta_2_i,
fee_commitment: self.fee_commitment(),
payload: self.swap_plaintext.encrypt(fvk.outgoing()),
}
}
pub fn swap_proof(&self) -> SwapProof {
use penumbra_sdk_proof_params::SWAP_PROOF_PROVING_KEY;
let balance_commitment =
self.transparent_balance().commit(Fr::zero()) + self.fee_commitment();
SwapProof::prove(
self.proof_blinding_r,
self.proof_blinding_s,
&SWAP_PROOF_PROVING_KEY,
SwapProofPublic {
balance_commitment,
swap_commitment: self.swap_plaintext.swap_commitment(),
fee_commitment: self.fee_commitment(),
},
SwapProofPrivate {
fee_blinding: self.fee_blinding,
swap_plaintext: self.swap_plaintext.clone(),
},
)
.expect("can generate ZKSwapProof")
}
pub fn fee_commitment(&self) -> balance::Commitment {
self.swap_plaintext.claim_fee.commit(self.fee_blinding)
}
pub fn transparent_balance(&self) -> Balance {
let value_1 = Value {
amount: self.swap_plaintext.delta_1_i,
asset_id: self.swap_plaintext.trading_pair.asset_1(),
};
let value_2 = Value {
amount: self.swap_plaintext.delta_2_i,
asset_id: self.swap_plaintext.trading_pair.asset_2(),
};
let mut balance = Balance::default();
balance -= value_1;
balance -= value_2;
balance
}
pub fn balance(&self) -> Balance {
let value_fee = Value {
amount: self.swap_plaintext.claim_fee.amount(),
asset_id: self.swap_plaintext.claim_fee.asset_id(),
};
let mut balance = self.transparent_balance();
balance -= value_fee;
balance
}
}
impl DomainType for SwapPlan {
type Proto = pb::SwapPlan;
}
impl From<SwapPlan> for pb::SwapPlan {
fn from(msg: SwapPlan) -> Self {
Self {
swap_plaintext: Some(msg.swap_plaintext.into()),
fee_blinding: msg.fee_blinding.to_bytes().to_vec(),
proof_blinding_r: msg.proof_blinding_r.to_bytes().to_vec(),
proof_blinding_s: msg.proof_blinding_s.to_bytes().to_vec(),
}
}
}
impl TryFrom<pb::SwapPlan> for SwapPlan {
type Error = anyhow::Error;
fn try_from(msg: pb::SwapPlan) -> Result<Self, Self::Error> {
let proof_blinding_r_bytes: [u8; 32] = msg
.proof_blinding_r
.try_into()
.map_err(|_| anyhow::anyhow!("malformed r in `SwapPlan`"))?;
let proof_blinding_s_bytes: [u8; 32] = msg
.proof_blinding_s
.try_into()
.map_err(|_| anyhow::anyhow!("malformed s in `SwapPlan`"))?;
let fee_blinding_bytes: [u8; 32] = msg.fee_blinding[..]
.try_into()
.map_err(|_| anyhow!("expected 32 byte fee blinding"))?;
Ok(Self {
fee_blinding: Fr::from_bytes_checked(&fee_blinding_bytes)
.expect("fee blinding malformed"),
swap_plaintext: msg
.swap_plaintext
.ok_or_else(|| anyhow!("missing swap_plaintext"))?
.try_into()
.context("swap plaintext malformed")?,
proof_blinding_r: Fq::from_bytes_checked(&proof_blinding_r_bytes)
.expect("proof_blinding_r malformed"),
proof_blinding_s: Fq::from_bytes_checked(&proof_blinding_s_bytes)
.expect("proof_blinding_r malformed"),
})
}
}