use anyhow::{Context, Result};
use ark_ff::ToConstraintField;
use ark_groth16::{
r1cs_to_qap::LibsnarkReduction, Groth16, PreparedVerifyingKey, Proof, ProvingKey,
};
use ark_r1cs_std::prelude::*;
use ark_relations::r1cs::{ConstraintSynthesizer, ConstraintSystemRef};
use ark_serialize::{CanonicalDeserialize, CanonicalSerialize};
use ark_snark::SNARK;
use decaf377::Bls12_377;
use decaf377::{Fq, Fr};
use decaf377_fmd as fmd;
use decaf377_ka as ka;
use penumbra_sdk_fee::Fee;
use penumbra_sdk_proto::{core::component::dex::v1 as pb, DomainType};
use penumbra_sdk_tct as tct;
use penumbra_sdk_tct::r1cs::StateCommitmentVar;
use penumbra_sdk_asset::{
asset,
balance::{self, commitment::BalanceCommitmentVar, BalanceVar},
Value,
};
use penumbra_sdk_keys::{keys::Diversifier, Address};
use penumbra_sdk_shielded_pool::Rseed;
use crate::{
swap::{SwapPlaintext, SwapPlaintextVar},
TradingPair,
};
use penumbra_sdk_proof_params::{DummyWitness, GROTH16_PROOF_LENGTH_BYTES};
#[derive(Clone, Debug)]
pub struct SwapProofPublic {
pub balance_commitment: balance::Commitment,
pub swap_commitment: tct::StateCommitment,
pub fee_commitment: balance::Commitment,
}
#[derive(Clone, Debug)]
pub struct SwapProofPrivate {
pub fee_blinding: Fr,
pub swap_plaintext: SwapPlaintext,
}
#[cfg(test)]
fn check_satisfaction(public: &SwapProofPublic, private: &SwapProofPrivate) -> Result<()> {
use penumbra_sdk_asset::Balance;
let swap_commitment = private.swap_plaintext.swap_commitment();
if swap_commitment != public.swap_commitment {
anyhow::bail!("swap commitment did not match public input");
}
let fee_balance = -Balance::from(private.swap_plaintext.claim_fee.0);
let fee_commitment = fee_balance.commit(private.fee_blinding);
if fee_commitment != public.fee_commitment {
anyhow::bail!("fee commitment did not match public input");
}
let balance_1 = -Balance::from(private.swap_plaintext.delta_1_value());
let balance_2 = -Balance::from(private.swap_plaintext.delta_2_value());
let transparent_blinding = Fr::from(0u64);
let balance_1_commit = balance_1.commit(transparent_blinding);
let balance_2_commit = balance_2.commit(transparent_blinding);
let transparent_balance_commitment = balance_1_commit + balance_2_commit;
let total_balance_commitment = transparent_balance_commitment + fee_commitment;
if total_balance_commitment != public.balance_commitment {
anyhow::bail!("balance commitment did not match public input");
}
Ok(())
}
#[cfg(test)]
fn check_circuit_satisfaction(public: SwapProofPublic, private: SwapProofPrivate) -> Result<()> {
use ark_relations::r1cs::{self, ConstraintSystem};
let cs: ConstraintSystemRef<_> = ConstraintSystem::new_ref();
let circuit = SwapCircuit { public, private };
cs.set_optimization_goal(r1cs::OptimizationGoal::Constraints);
circuit
.generate_constraints(cs.clone())
.expect("can generate constraints from circuit");
cs.finalize();
if !cs.is_satisfied()? {
anyhow::bail!("constraints are not satisfied");
}
Ok(())
}
pub struct SwapCircuit {
public: SwapProofPublic,
private: SwapProofPrivate,
}
impl ConstraintSynthesizer<Fq> for SwapCircuit {
fn generate_constraints(self, cs: ConstraintSystemRef<Fq>) -> ark_relations::r1cs::Result<()> {
let swap_plaintext_var =
SwapPlaintextVar::new_witness(cs.clone(), || Ok(self.private.swap_plaintext.clone()))?;
let fee_blinding_var =
UInt8::new_witness_vec(cs.clone(), &self.private.fee_blinding.to_bytes())?;
let claimed_balance_commitment =
BalanceCommitmentVar::new_input(cs.clone(), || Ok(self.public.balance_commitment))?;
let claimed_swap_commitment =
StateCommitmentVar::new_input(cs.clone(), || Ok(self.public.swap_commitment))?;
let claimed_fee_commitment =
BalanceCommitmentVar::new_input(cs, || Ok(self.public.fee_commitment))?;
let swap_commitment = swap_plaintext_var.commit()?;
claimed_swap_commitment.enforce_equal(&swap_commitment)?;
let fee_balance = BalanceVar::from_negative_value_var(swap_plaintext_var.claim_fee.clone());
let fee_commitment = fee_balance.commit(fee_blinding_var)?;
claimed_fee_commitment.enforce_equal(&fee_commitment)?;
let transparent_blinding_var = UInt8::constant_vec(&[0u8; 32]);
let balance_1 = BalanceVar::from_negative_value_var(swap_plaintext_var.delta_1_value());
let balance_2 = BalanceVar::from_negative_value_var(swap_plaintext_var.delta_2_value());
let balance_1_commit = balance_1.commit(transparent_blinding_var.clone())?;
let balance_2_commit = balance_2.commit(transparent_blinding_var)?;
let transparent_balance_commitment = balance_1_commit + balance_2_commit;
let total_balance_commitment = transparent_balance_commitment + fee_commitment;
claimed_balance_commitment.enforce_equal(&total_balance_commitment)?;
Ok(())
}
}
impl DummyWitness for SwapCircuit {
fn with_dummy_witness() -> Self {
let a = asset::Cache::with_known_assets()
.get_unit("upenumbra")
.expect("upenumbra asset exists");
let b = asset::Cache::with_known_assets()
.get_unit("nala")
.expect("nala asset exists");
let trading_pair = TradingPair::new(a.id(), b.id());
let diversifier_bytes = [1u8; 16];
let pk_d_bytes = decaf377::Element::GENERATOR.vartime_compress().0;
let clue_key_bytes = [1; 32];
let diversifier = Diversifier(diversifier_bytes);
let address = Address::from_components(
diversifier,
ka::Public(pk_d_bytes),
fmd::ClueKey(clue_key_bytes),
)
.expect("generated 1 address");
let swap_plaintext = SwapPlaintext {
trading_pair,
delta_1_i: 100000u64.into(),
delta_2_i: 1u64.into(),
claim_fee: Fee(Value {
amount: 3u64.into(),
asset_id: asset::Cache::with_known_assets()
.get_unit("upenumbra")
.expect("upenumbra asset exists")
.id(),
}),
claim_address: address,
rseed: Rseed([1u8; 32]),
};
Self {
private: SwapProofPrivate {
swap_plaintext: swap_plaintext.clone(),
fee_blinding: Fr::from(1u64),
},
public: SwapProofPublic {
swap_commitment: swap_plaintext.swap_commitment(),
fee_commitment: balance::Commitment(decaf377::Element::GENERATOR),
balance_commitment: balance::Commitment(decaf377::Element::GENERATOR),
},
}
}
}
#[derive(Clone, Debug)]
pub struct SwapProof([u8; GROTH16_PROOF_LENGTH_BYTES]);
impl SwapProof {
#![allow(clippy::too_many_arguments)]
pub fn prove(
blinding_r: Fq,
blinding_s: Fq,
pk: &ProvingKey<Bls12_377>,
public: SwapProofPublic,
private: SwapProofPrivate,
) -> anyhow::Result<Self> {
let circuit = SwapCircuit { public, private };
let proof = Groth16::<Bls12_377, LibsnarkReduction>::create_proof_with_reduction(
circuit, pk, blinding_r, blinding_s,
)
.map_err(|err| anyhow::anyhow!(err))?;
let mut proof_bytes = [0u8; GROTH16_PROOF_LENGTH_BYTES];
Proof::serialize_compressed(&proof, &mut proof_bytes[..]).expect("can serialize Proof");
Ok(Self(proof_bytes))
}
#[tracing::instrument(skip(self, vk))]
pub fn verify(
&self,
vk: &PreparedVerifyingKey<Bls12_377>,
public: SwapProofPublic,
) -> anyhow::Result<()> {
let proof =
Proof::deserialize_compressed_unchecked(&self.0[..]).map_err(|e| anyhow::anyhow!(e))?;
let mut public_inputs = Vec::new();
public_inputs.extend(
public
.balance_commitment
.0
.to_field_elements()
.context("balance_commitment should be a Bls12-377 field member")?,
);
public_inputs.extend(
public
.swap_commitment
.0
.to_field_elements()
.context("swap_commitment should be a Bls12-377 field member")?,
);
public_inputs.extend(
public
.fee_commitment
.0
.to_field_elements()
.context("fee_commitment should be a Bls12-377 field member")?,
);
tracing::trace!(?public_inputs);
let start = std::time::Instant::now();
let proof_result = Groth16::<Bls12_377, LibsnarkReduction>::verify_with_processed_vk(
vk,
public_inputs.as_slice(),
&proof,
)
.map_err(|err| anyhow::anyhow!(err))?;
tracing::debug!(?proof_result, elapsed = ?start.elapsed());
proof_result
.then_some(())
.ok_or_else(|| anyhow::anyhow!("a swap proof did not verify"))
}
}
impl DomainType for SwapProof {
type Proto = pb::ZkSwapProof;
}
impl From<SwapProof> for pb::ZkSwapProof {
fn from(proof: SwapProof) -> Self {
pb::ZkSwapProof {
inner: proof.0.to_vec(),
}
}
}
impl TryFrom<pb::ZkSwapProof> for SwapProof {
type Error = anyhow::Error;
fn try_from(proto: pb::ZkSwapProof) -> Result<Self, Self::Error> {
Ok(SwapProof(proto.inner[..].try_into()?))
}
}
#[cfg(test)]
mod tests {
use super::*;
use penumbra_sdk_asset::{Balance, Value};
use penumbra_sdk_keys::keys::{Bip44Path, SeedPhrase, SpendKey};
use penumbra_sdk_num::Amount;
use proptest::prelude::*;
fn fr_strategy() -> BoxedStrategy<Fr> {
any::<[u8; 32]>()
.prop_map(|bytes| Fr::from_le_bytes_mod_order(&bytes[..]))
.boxed()
}
prop_compose! {
fn arb_valid_swap_statement()(fee_blinding in fr_strategy(), address_index in any::<u32>(), value1_amount in any::<u64>(), seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>()) -> (SwapProofPublic, SwapProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_trader = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_trader = sk_trader.full_viewing_key();
let ivk_trader = fvk_trader.incoming();
let (claim_address, _dtk_d) = ivk_trader.payment_address(address_index.into());
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 delta_1_i = Amount::from(value1_amount);
let delta_2_i = Amount::from(0u64);
let fee = Fee::default();
let rseed = Rseed(rseed_randomness);
let swap_plaintext = SwapPlaintext {
trading_pair,
delta_1_i,
delta_2_i,
claim_fee: fee,
claim_address,
rseed,
};
let fee_commitment = swap_plaintext.claim_fee.commit(fee_blinding);
let swap_commitment = swap_plaintext.swap_commitment();
let value_1 = Value {
amount: swap_plaintext.delta_1_i,
asset_id: swap_plaintext.trading_pair.asset_1(),
};
let value_2 = Value {
amount: swap_plaintext.delta_2_i,
asset_id: swap_plaintext.trading_pair.asset_2(),
};
let value_fee = Value {
amount: swap_plaintext.claim_fee.amount(),
asset_id: swap_plaintext.claim_fee.asset_id(),
};
let mut balance = Balance::default();
balance -= value_1;
balance -= value_2;
balance -= value_fee;
let balance_commitment = balance.commit(fee_blinding);
let public = SwapProofPublic { balance_commitment, swap_commitment, fee_commitment };
let private = SwapProofPrivate { fee_blinding, swap_plaintext };
(public, private)
}
}
proptest! {
#[test]
fn swap_proof_happy_path((public, private) in arb_valid_swap_statement()) {
assert!(check_satisfaction(&public, &private).is_ok());
assert!(check_circuit_satisfaction(public, private).is_ok());
}
}
prop_compose! {
fn arb_invalid_swap_statement_fee_commitment()(fee_blinding in fr_strategy(), invalid_fee_blinding in fr_strategy(), address_index in any::<u32>(), value1_amount in any::<u64>(), seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>()) -> (SwapProofPublic, SwapProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_trader = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_trader = sk_trader.full_viewing_key();
let ivk_trader = fvk_trader.incoming();
let (claim_address, _dtk_d) = ivk_trader.payment_address(address_index.into());
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 delta_1_i = Amount::from(value1_amount);
let delta_2_i = Amount::from(0u64);
let fee = Fee::default();
let rseed = Rseed(rseed_randomness);
let swap_plaintext = SwapPlaintext {
trading_pair,
delta_1_i,
delta_2_i,
claim_fee: fee,
claim_address,
rseed,
};
let swap_commitment = swap_plaintext.swap_commitment();
let value_1 = Value {
amount: swap_plaintext.delta_1_i,
asset_id: swap_plaintext.trading_pair.asset_1(),
};
let value_2 = Value {
amount: swap_plaintext.delta_2_i,
asset_id: swap_plaintext.trading_pair.asset_2(),
};
let value_fee = Value {
amount: swap_plaintext.claim_fee.amount(),
asset_id: swap_plaintext.claim_fee.asset_id(),
};
let mut balance = Balance::default();
balance -= value_1;
balance -= value_2;
balance -= value_fee;
let balance_commitment = balance.commit(fee_blinding);
let invalid_fee_commitment = swap_plaintext.claim_fee.commit(invalid_fee_blinding);
let public = SwapProofPublic { balance_commitment, swap_commitment, fee_commitment: invalid_fee_commitment };
let private = SwapProofPrivate { fee_blinding, swap_plaintext };
(public, private)
}
}
proptest! {
#[test]
fn swap_proof_invalid_fee_commitment((public, private) in arb_invalid_swap_statement_fee_commitment()) {
assert!(check_satisfaction(&public, &private).is_err());
assert!(check_circuit_satisfaction(public, private).is_err());
}
}
}