use base64::prelude::*;
use std::str::FromStr;
use tct::Root;
use anyhow::Result;
use ark_r1cs_std::{
prelude::{EqGadget, FieldVar},
uint8::UInt8,
ToBitsGadget,
};
use ark_serialize::{CanonicalDeserialize, CanonicalSerialize};
use decaf377::{r1cs::FqVar, Bls12_377, Fq, Fr};
use ark_ff::ToConstraintField;
use ark_groth16::{
r1cs_to_qap::LibsnarkReduction, Groth16, PreparedVerifyingKey, Proof, ProvingKey,
};
use ark_r1cs_std::prelude::AllocVar;
use ark_relations::r1cs::{ConstraintSynthesizer, ConstraintSystemRef};
use ark_snark::SNARK;
use decaf377_rdsa::{SpendAuth, VerificationKey};
use penumbra_sdk_proto::{penumbra::core::component::shielded_pool::v1 as pb, DomainType};
use penumbra_sdk_tct as tct;
use penumbra_sdk_tct::r1cs::StateCommitmentVar;
use crate::{note, Note, Rseed};
use penumbra_sdk_asset::{
balance::commitment::BalanceCommitmentVar,
balance::{self, Commitment},
Value,
};
use penumbra_sdk_keys::keys::{
AuthorizationKeyVar, Bip44Path, IncomingViewingKeyVar, NullifierKey, NullifierKeyVar,
RandomizedVerificationKey, SeedPhrase, SpendAuthRandomizerVar, SpendKey,
};
use penumbra_sdk_proof_params::{DummyWitness, VerifyingKeyExt, GROTH16_PROOF_LENGTH_BYTES};
use penumbra_sdk_sct::{Nullifier, NullifierVar};
use tap::Tap;
#[derive(Clone, Debug)]
pub struct SpendProofPublic {
pub anchor: tct::Root,
pub balance_commitment: balance::Commitment,
pub nullifier: Nullifier,
pub rk: VerificationKey<SpendAuth>,
}
#[derive(Clone, Debug)]
pub struct SpendProofPrivate {
pub state_commitment_proof: tct::Proof,
pub note: Note,
pub v_blinding: Fr,
pub spend_auth_randomizer: Fr,
pub ak: VerificationKey<SpendAuth>,
pub nk: NullifierKey,
}
#[cfg(test)]
fn check_satisfaction(public: &SpendProofPublic, private: &SpendProofPrivate) -> Result<()> {
use penumbra_sdk_keys::keys::FullViewingKey;
let note_commitment = private.note.commit();
if note_commitment != private.state_commitment_proof.commitment() {
anyhow::bail!("note commitment did not match state commitment proof");
}
let nullifier = Nullifier::derive(
&private.nk,
private.state_commitment_proof.position(),
¬e_commitment,
);
if nullifier != public.nullifier {
anyhow::bail!("nullifier did not match public input");
}
let amount_u128: u128 = private.note.value().amount.into();
if amount_u128 != 0u128 {
private.state_commitment_proof.verify(public.anchor)?;
}
let rk = private.ak.randomize(&private.spend_auth_randomizer);
if rk != public.rk {
anyhow::bail!("randomized spend auth key did not match public input");
}
let fvk = FullViewingKey::from_components(private.ak, private.nk);
let ivk = fvk.incoming();
let transmission_key = ivk.diversified_public(&private.note.diversified_generator());
if transmission_key != *private.note.transmission_key() {
anyhow::bail!("transmission key did not match note");
}
let balance_commitment = private.note.value().commit(private.v_blinding);
if balance_commitment != public.balance_commitment {
anyhow::bail!("balance commitment did not match public input");
}
if private.note.diversified_generator() == decaf377::Element::default() {
anyhow::bail!("diversified generator is identity");
}
if private.ak.is_identity() {
anyhow::bail!("ak is identity");
}
Ok(())
}
#[cfg(test)]
fn check_circuit_satisfaction(public: SpendProofPublic, private: SpendProofPrivate) -> Result<()> {
use ark_relations::r1cs::{self, ConstraintSystem};
let cs = ConstraintSystem::new_ref();
let circuit = SpendCircuit { 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(())
}
#[derive(Clone, Debug)]
pub struct SpendCircuit {
public: SpendProofPublic,
private: SpendProofPrivate,
}
impl ConstraintSynthesizer<Fq> for SpendCircuit {
fn generate_constraints(self, cs: ConstraintSystemRef<Fq>) -> ark_relations::r1cs::Result<()> {
let note_var = note::NoteVar::new_witness(cs.clone(), || Ok(self.private.note.clone()))?;
let claimed_note_commitment = StateCommitmentVar::new_witness(cs.clone(), || {
Ok(self.private.state_commitment_proof.commitment())
})?;
let position_var = tct::r1cs::PositionVar::new_witness(cs.clone(), || {
Ok(self.private.state_commitment_proof.position())
})?;
let position_bits = position_var.to_bits_le()?;
let merkle_path_var = tct::r1cs::MerkleAuthPathVar::new_witness(cs.clone(), || {
Ok(self.private.state_commitment_proof)
})?;
let v_blinding_arr: [u8; 32] = self.private.v_blinding.to_bytes();
let v_blinding_vars = UInt8::new_witness_vec(cs.clone(), &v_blinding_arr)?;
let spend_auth_randomizer_var = SpendAuthRandomizerVar::new_witness(cs.clone(), || {
Ok(self.private.spend_auth_randomizer)
})?;
let ak_element_var: AuthorizationKeyVar =
AuthorizationKeyVar::new_witness(cs.clone(), || Ok(self.private.ak))?;
let nk_var = NullifierKeyVar::new_witness(cs.clone(), || Ok(self.private.nk))?;
let anchor_var = FqVar::new_input(cs.clone(), || Ok(Fq::from(self.public.anchor)))?;
let claimed_balance_commitment_var =
BalanceCommitmentVar::new_input(cs.clone(), || Ok(self.public.balance_commitment))?;
let claimed_nullifier_var =
NullifierVar::new_input(cs.clone(), || Ok(self.public.nullifier))?;
let rk_var = RandomizedVerificationKey::new_input(cs.clone(), || Ok(self.public.rk))?;
let note_commitment_var = note_var.commit()?;
note_commitment_var.enforce_equal(&claimed_note_commitment)?;
let nullifier_var = NullifierVar::derive(&nk_var, &position_var, &claimed_note_commitment)?;
nullifier_var.enforce_equal(&claimed_nullifier_var)?;
let is_not_dummy = note_var.amount().is_eq(&FqVar::zero())?.not();
merkle_path_var.verify(
cs.clone(),
&is_not_dummy,
&position_bits,
anchor_var,
claimed_note_commitment.inner(),
)?;
let computed_rk_var = ak_element_var.randomize(&spend_auth_randomizer_var)?;
computed_rk_var.enforce_equal(&rk_var)?;
let ivk = IncomingViewingKeyVar::derive(&nk_var, &ak_element_var)?;
let computed_transmission_key =
ivk.diversified_public(¬e_var.diversified_generator())?;
computed_transmission_key.enforce_equal(¬e_var.transmission_key())?;
let balance_commitment = note_var.value().commit(v_blinding_vars)?;
balance_commitment.enforce_equal(&claimed_balance_commitment_var)?;
Ok(())
}
}
impl DummyWitness for SpendCircuit {
fn with_dummy_witness() -> Self {
let seed_phrase = SeedPhrase::from_randomness(&[b'f'; 32]);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (address, _dtk_d) = ivk_sender.payment_address(0u32.into());
let spend_auth_randomizer = Fr::from(1u64);
let rsk = sk_sender.spend_auth_key().randomize(&spend_auth_randomizer);
let nk = *sk_sender.nullifier_key();
let ak = sk_sender.spend_auth_key().into();
let note = Note::from_parts(
address,
Value::from_str("1upenumbra").expect("valid value"),
Rseed([1u8; 32]),
)
.expect("can make a note");
let v_blinding = Fr::from(1u64);
let rk: VerificationKey<SpendAuth> = rsk.into();
let nullifier = Nullifier(Fq::from(1u64));
let mut sct = tct::Tree::new();
let anchor: tct::Root = sct.root();
let note_commitment = note.commit();
sct.insert(tct::Witness::Keep, note_commitment)
.expect("able to insert note commitment into SCT");
let state_commitment_proof = sct
.witness(note_commitment)
.expect("able to witness just-inserted note commitment");
let public = SpendProofPublic {
anchor,
balance_commitment: balance::Commitment(decaf377::Element::GENERATOR),
nullifier,
rk,
};
let private = SpendProofPrivate {
state_commitment_proof,
note,
v_blinding,
spend_auth_randomizer,
ak,
nk,
};
Self { public, private }
}
}
#[derive(Clone, Debug)]
pub struct SpendProof([u8; GROTH16_PROOF_LENGTH_BYTES]);
#[derive(Debug, thiserror::Error)]
pub enum VerificationError {
#[error("error deserializing compressed proof: {0:?}")]
ProofDeserialize(ark_serialize::SerializationError),
#[error("Fq types are Bls12-377 field members")]
Anchor,
#[error("balance commitment is a Bls12-377 field member")]
BalanceCommitment,
#[error("nullifier is a Bls12-377 field member")]
Nullifier,
#[error("could not decompress element points: {0:?}")]
DecompressRk(decaf377::EncodingError),
#[error("randomized spend key is a Bls12-377 field member")]
Rk,
#[error("start position is a Bls12-377 field member")]
StartPosition,
#[error("error verifying proof: {0:?}")]
SynthesisError(ark_relations::r1cs::SynthesisError),
#[error("spend proof did not verify")]
InvalidProof,
}
impl SpendProof {
pub fn prove(
blinding_r: Fq,
blinding_s: Fq,
pk: &ProvingKey<Bls12_377>,
public: SpendProofPublic,
private: SpendProofPrivate,
) -> anyhow::Result<Self> {
let circuit = SpendCircuit { 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(level="debug", skip(self, vk), fields(self = ?BASE64_STANDARD.encode(self.clone().encode_to_vec()), vk = ?vk.debug_id()))]
pub fn verify(
&self,
vk: &PreparedVerifyingKey<Bls12_377>,
SpendProofPublic {
anchor: Root(anchor),
balance_commitment: Commitment(balance_commitment),
nullifier: Nullifier(nullifier),
rk,
}: SpendProofPublic,
) -> Result<(), VerificationError> {
let proof = Proof::deserialize_compressed_unchecked(&self.0[..])
.map_err(VerificationError::ProofDeserialize)?;
let element_rk = decaf377::Encoding(rk.to_bytes())
.vartime_decompress()
.map_err(VerificationError::DecompressRk)?;
macro_rules! to_field_elements {
($fe:expr, $err:expr) => {
$fe.to_field_elements().ok_or($err)?
};
}
use VerificationError::*;
let public_inputs = [
to_field_elements!(Fq::from(anchor), Anchor),
to_field_elements!(balance_commitment, BalanceCommitment),
to_field_elements!(nullifier, Nullifier),
to_field_elements!(element_rk, Rk),
]
.into_iter()
.flatten()
.collect::<Vec<_>>()
.tap(|public_inputs| tracing::trace!(?public_inputs));
let start = std::time::Instant::now();
Groth16::<Bls12_377, LibsnarkReduction>::verify_with_processed_vk(
vk,
public_inputs.as_slice(),
&proof,
)
.map_err(VerificationError::SynthesisError)?
.tap(|proof_result| tracing::debug!(?proof_result, elapsed = ?start.elapsed()))
.then_some(())
.ok_or(VerificationError::InvalidProof)
}
}
impl DomainType for SpendProof {
type Proto = pb::ZkSpendProof;
}
impl From<SpendProof> for pb::ZkSpendProof {
fn from(proof: SpendProof) -> Self {
pb::ZkSpendProof {
inner: proof.0.to_vec(),
}
}
}
impl TryFrom<pb::ZkSpendProof> for SpendProof {
type Error = anyhow::Error;
fn try_from(proto: pb::ZkSpendProof) -> Result<Self, Self::Error> {
Ok(SpendProof(proto.inner[..].try_into()?))
}
}
#[cfg(test)]
mod tests {
use super::*;
use ark_r1cs_std::prelude::Boolean;
use decaf377::{Fq, Fr};
use penumbra_sdk_asset::{asset, Value};
use penumbra_sdk_keys::{
keys::{Bip44Path, SeedPhrase, SpendKey},
Address,
};
use penumbra_sdk_num::Amount;
use penumbra_sdk_proof_params::generate_prepared_test_parameters;
use penumbra_sdk_sct::Nullifier;
use penumbra_sdk_tct::StateCommitment;
use proptest::prelude::*;
use crate::Note;
use decaf377_rdsa::{SpendAuth, VerificationKey};
use penumbra_sdk_tct as tct;
use rand_core::OsRng;
fn fr_strategy() -> BoxedStrategy<Fr> {
any::<[u8; 32]>()
.prop_map(|bytes| Fr::from_le_bytes_mod_order(&bytes[..]))
.boxed()
}
prop_compose! {
fn arb_valid_spend_statement()(v_blinding in fr_strategy(), spend_auth_randomizer in fr_strategy(), asset_id64 in any::<u64>(), address_index in any::<u32>(), amount in any::<u64>(), seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>(), num_commitments in 0..100) -> (SpendProofPublic, SpendProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (sender, _dtk_d) = ivk_sender.payment_address(address_index.into());
let value_to_send = Value {
amount: Amount::from(amount),
asset_id: asset::Id(Fq::from(asset_id64)),
};
let note = Note::from_parts(
sender.clone(),
value_to_send,
Rseed(rseed_randomness),
).expect("should be able to create note");
let note_commitment = note.commit();
let rsk = sk_sender.spend_auth_key().randomize(&spend_auth_randomizer);
let nk = *sk_sender.nullifier_key();
let ak: VerificationKey<SpendAuth> = sk_sender.spend_auth_key().into();
let mut sct = tct::Tree::new();
for i in 0..num_commitments {
let rseed = Rseed([i as u8; 32]);
let dummy_note_commitment = Note::from_parts(sender.clone(), value_to_send, rseed).expect("can create note").commit();
sct.insert(tct::Witness::Keep, dummy_note_commitment).expect("should be able to insert note commitments into the SCT");
}
sct.insert(tct::Witness::Keep, note_commitment).expect("should be able to insert note commitments into the SCT");
let anchor = sct.root();
let state_commitment_proof = sct.witness(note_commitment).expect("can witness note commitment");
let balance_commitment = value_to_send.commit(v_blinding);
let rk: VerificationKey<SpendAuth> = rsk.into();
let nullifier = Nullifier::derive(&nk, state_commitment_proof.position(), ¬e_commitment);
let public = SpendProofPublic {
anchor,
balance_commitment,
nullifier,
rk,
};
let private = SpendProofPrivate {
state_commitment_proof,
note,
v_blinding,
spend_auth_randomizer,
ak,
nk,
};
(public, private)
}
}
proptest! {
#[test]
fn spend_proof_happy_path((public, private) in arb_valid_spend_statement()) {
assert!(check_satisfaction(&public, &private).is_ok());
assert!(check_circuit_satisfaction(public, private).is_ok());
}
}
prop_compose! {
fn arb_invalid_spend_statement_incorrect_anchor()(v_blinding in fr_strategy(), spend_auth_randomizer in fr_strategy(), asset_id64 in any::<u64>(), address_index in any::<u32>(), amount in any::<u64>(), seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>(), num_commitments in 0..100) -> (SpendProofPublic, SpendProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (sender, _dtk_d) = ivk_sender.payment_address(address_index.into());
let value_to_send = Value {
amount: Amount::from(amount),
asset_id: asset::Id(Fq::from(asset_id64)),
};
let note = Note::from_parts(
sender.clone(),
value_to_send,
Rseed(rseed_randomness),
).expect("should be able to create note");
let note_commitment = note.commit();
let rsk = sk_sender.spend_auth_key().randomize(&spend_auth_randomizer);
let nk = *sk_sender.nullifier_key();
let ak: VerificationKey<SpendAuth> = sk_sender.spend_auth_key().into();
let mut sct = tct::Tree::new();
for i in 0..num_commitments {
let rseed = Rseed([i as u8; 32]);
let dummy_note_commitment = Note::from_parts(sender.clone(), value_to_send, rseed).expect("can create note").commit();
sct.insert(tct::Witness::Keep, dummy_note_commitment).expect("should be able to insert note commitments into the SCT");
}
let incorrect_anchor = sct.root();
sct.insert(tct::Witness::Keep, note_commitment).expect("should be able to insert note commitments into the SCT");
let state_commitment_proof = sct.witness(note_commitment).expect("can witness note commitment");
let balance_commitment = value_to_send.commit(v_blinding);
let rk: VerificationKey<SpendAuth> = rsk.into();
let nullifier = Nullifier::derive(&nk, state_commitment_proof.position(), ¬e_commitment);
let public = SpendProofPublic {
anchor: incorrect_anchor,
balance_commitment,
nullifier,
rk,
};
let private = SpendProofPrivate {
state_commitment_proof,
note,
v_blinding,
spend_auth_randomizer,
ak,
nk,
};
(public, private)
}
}
proptest! {
#[test]
fn spend_proof_verification_merkle_path_integrity_failure((public, private) in arb_invalid_spend_statement_incorrect_anchor()) {
assert!(check_satisfaction(&public, &private).is_err());
assert!(check_circuit_satisfaction(public, private).is_err());
}
}
prop_compose! {
fn arb_invalid_spend_statement_diversified_address()(v_blinding in fr_strategy(), spend_auth_randomizer in fr_strategy(), asset_id64 in any::<u64>(), address_index in any::<u32>(), amount in any::<u64>(), seed_phrase_randomness in any::<[u8; 32]>(), incorrect_seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>()) -> (SpendProofPublic, SpendProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (_sender, _dtk_d) = ivk_sender.payment_address(address_index.into());
let value_to_send = Value {
amount: Amount::from(amount),
asset_id: asset::Id(Fq::from(asset_id64)),
};
let wrong_seed_phrase = SeedPhrase::from_randomness(&incorrect_seed_phrase_randomness);
let wrong_sk_sender = SpendKey::from_seed_phrase_bip44(wrong_seed_phrase, &Bip44Path::new(0));
let wrong_fvk_sender = wrong_sk_sender.full_viewing_key();
let wrong_ivk_sender = wrong_fvk_sender.incoming();
let (wrong_sender, _dtk_d) = wrong_ivk_sender.payment_address(address_index.into());
let note = Note::from_parts(
wrong_sender,
value_to_send,
Rseed(rseed_randomness),
).expect("should be able to create note");
let note_commitment = note.commit();
let rsk = sk_sender.spend_auth_key().randomize(&spend_auth_randomizer);
let nk = *sk_sender.nullifier_key();
let ak: VerificationKey<SpendAuth> = sk_sender.spend_auth_key().into();
let mut sct = tct::Tree::new();
sct.insert(tct::Witness::Keep, note_commitment).expect("should be able to insert note commitments into the SCT");
let anchor = sct.root();
let state_commitment_proof = sct.witness(note_commitment).expect("can witness note commitment");
let balance_commitment = value_to_send.commit(v_blinding);
let rk: VerificationKey<SpendAuth> = rsk.into();
let nullifier = Nullifier::derive(&nk, state_commitment_proof.position(), ¬e_commitment);
let public = SpendProofPublic {
anchor,
balance_commitment,
nullifier,
rk,
};
let private = SpendProofPrivate {
state_commitment_proof,
note,
v_blinding,
spend_auth_randomizer,
ak,
nk,
};
(public, private)
}
}
proptest! {
#[test]
fn spend_proof_verification_diversified_address_integrity_failure((public, private) in arb_invalid_spend_statement_diversified_address()) {
assert!(check_satisfaction(&public, &private).is_err());
assert!(check_circuit_satisfaction(public, private).is_err());
}
}
prop_compose! {
fn arb_invalid_spend_statement_nullifier()(v_blinding in fr_strategy(), spend_auth_randomizer in fr_strategy(), asset_id64 in any::<u64>(), address_index in any::<u32>(), amount in any::<u64>(), seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>(), num_commitments in 0..100) -> (SpendProofPublic, SpendProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (sender, _dtk_d) = ivk_sender.payment_address(address_index.into());
let value_to_send = Value {
amount: Amount::from(amount),
asset_id: asset::Id(Fq::from(asset_id64)),
};
let note = Note::from_parts(
sender.clone(),
value_to_send,
Rseed(rseed_randomness),
).expect("should be able to create note");
let note_commitment = note.commit();
let rsk = sk_sender.spend_auth_key().randomize(&spend_auth_randomizer);
let nk = *sk_sender.nullifier_key();
let ak: VerificationKey<SpendAuth> = sk_sender.spend_auth_key().into();
let mut sct = tct::Tree::new();
for i in 0..num_commitments {
let rseed = Rseed([i as u8; 32]);
let dummy_note_commitment = Note::from_parts(sender.clone(), value_to_send, rseed).expect("can create note").commit();
sct.insert(tct::Witness::Keep, dummy_note_commitment).expect("should be able to insert note commitments into the SCT");
}
let rseed = Rseed([num_commitments as u8; 32]);
let dummy_note_commitment = Note::from_parts(sender.clone(), value_to_send, rseed).expect("can create note").commit();
sct.insert(tct::Witness::Keep, dummy_note_commitment).expect("should be able to insert note commitments into the SCT");
let incorrect_position = sct.witness(dummy_note_commitment).expect("can witness note commitment").position();
sct.insert(tct::Witness::Keep, note_commitment).expect("should be able to insert note commitments into the SCT");
let anchor = sct.root();
let state_commitment_proof = sct.witness(note_commitment).expect("can witness note commitment");
let balance_commitment = value_to_send.commit(v_blinding);
let rk: VerificationKey<SpendAuth> = rsk.into();
let incorrect_nf = Nullifier::derive(&nk, incorrect_position, ¬e_commitment);
let public = SpendProofPublic {
anchor,
balance_commitment,
nullifier: incorrect_nf,
rk,
};
let private = SpendProofPrivate {
state_commitment_proof,
note,
v_blinding,
spend_auth_randomizer,
ak,
nk,
};
(public, private)
}
}
proptest! {
#[test]
fn spend_proof_verification_nullifier_integrity_failure((public, private) in arb_invalid_spend_statement_nullifier()) {
assert!(check_satisfaction(&public, &private).is_err());
assert!(check_circuit_satisfaction(public, private).is_err());
}
}
prop_compose! {
fn arb_invalid_spend_statement_v_blinding_factor()(v_blinding in fr_strategy(), incorrect_v_blinding in fr_strategy(), spend_auth_randomizer in fr_strategy(), asset_id64 in any::<u64>(), address_index in any::<u32>(), amount in any::<u64>(), seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>(), num_commitments in 0..100) -> (SpendProofPublic, SpendProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (sender, _dtk_d) = ivk_sender.payment_address(address_index.into());
let value_to_send = Value {
amount: Amount::from(amount),
asset_id: asset::Id(Fq::from(asset_id64)),
};
let note = Note::from_parts(
sender.clone(),
value_to_send,
Rseed(rseed_randomness),
).expect("should be able to create note");
let note_commitment = note.commit();
let rsk = sk_sender.spend_auth_key().randomize(&spend_auth_randomizer);
let nk = *sk_sender.nullifier_key();
let ak: VerificationKey<SpendAuth> = sk_sender.spend_auth_key().into();
let mut sct = tct::Tree::new();
for i in 0..num_commitments {
let rseed = Rseed([i as u8; 32]);
let dummy_note_commitment = Note::from_parts(sender.clone(), value_to_send, rseed).expect("can create note").commit();
sct.insert(tct::Witness::Keep, dummy_note_commitment).expect("should be able to insert note commitments into the SCT");
}
sct.insert(tct::Witness::Keep, note_commitment).expect("should be able to insert note commitments into the SCT");
let anchor = sct.root();
let state_commitment_proof = sct.witness(note_commitment).expect("can witness note commitment");
let balance_commitment = value_to_send.commit(v_blinding);
let rk: VerificationKey<SpendAuth> = rsk.into();
let nullifier = Nullifier::derive(&nk, state_commitment_proof.position(), ¬e_commitment);
let public = SpendProofPublic {
anchor,
balance_commitment,
nullifier,
rk,
};
let private = SpendProofPrivate {
state_commitment_proof,
note,
v_blinding: incorrect_v_blinding,
spend_auth_randomizer,
ak,
nk,
};
(public, private)
}
}
proptest! {
#[test]
fn spend_proof_verification_balance_commitment_integrity_failure((public, private) in arb_invalid_spend_statement_v_blinding_factor()) {
assert!(check_satisfaction(&public, &private).is_err());
assert!(check_circuit_satisfaction(public, private).is_err());
}
}
prop_compose! {
fn arb_invalid_spend_statement_rk_integrity()(v_blinding in fr_strategy(), spend_auth_randomizer in fr_strategy(), asset_id64 in any::<u64>(), address_index in any::<u32>(), amount in any::<u64>(), seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>(), num_commitments in 0..100, incorrect_spend_auth_randomizer in fr_strategy()) -> (SpendProofPublic, SpendProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (sender, _dtk_d) = ivk_sender.payment_address(address_index.into());
let value_to_send = Value {
amount: Amount::from(amount),
asset_id: asset::Id(Fq::from(asset_id64)),
};
let note = Note::from_parts(
sender.clone(),
value_to_send,
Rseed(rseed_randomness),
).expect("should be able to create note");
let note_commitment = note.commit();
let nk = *sk_sender.nullifier_key();
let ak: VerificationKey<SpendAuth> = sk_sender.spend_auth_key().into();
let mut sct = tct::Tree::new();
for i in 0..num_commitments {
let rseed = Rseed([i as u8; 32]);
let dummy_note_commitment = Note::from_parts(sender.clone(), value_to_send, rseed).expect("can create note").commit();
sct.insert(tct::Witness::Keep, dummy_note_commitment).expect("should be able to insert note commitments into the SCT");
}
sct.insert(tct::Witness::Keep, note_commitment).expect("should be able to insert note commitments into the SCT");
let anchor = sct.root();
let state_commitment_proof = sct.witness(note_commitment).expect("can witness note commitment");
let balance_commitment = value_to_send.commit(v_blinding);
let nullifier = Nullifier::derive(&nk, state_commitment_proof.position(), ¬e_commitment);
let incorrect_rsk = sk_sender
.spend_auth_key()
.randomize(&incorrect_spend_auth_randomizer);
let incorrect_rk: VerificationKey<SpendAuth> = incorrect_rsk.into();
let public = SpendProofPublic {
anchor,
balance_commitment,
nullifier,
rk: incorrect_rk,
};
let private = SpendProofPrivate {
state_commitment_proof,
note,
v_blinding,
spend_auth_randomizer,
ak,
nk,
};
(public, private)
}
}
proptest! {
#[test]
fn spend_proof_verification_fails_rk_integrity((public, private) in arb_invalid_spend_statement_rk_integrity()) {
assert!(check_satisfaction(&public, &private).is_err());
assert!(check_circuit_satisfaction(public, private).is_err());
}
}
prop_compose! {
fn arb_valid_dummy_spend_statement()(v_blinding in fr_strategy(), spend_auth_randomizer in fr_strategy(), asset_id64 in any::<u64>(), address_index in any::<u32>(), seed_phrase_randomness in any::<[u8; 32]>(), rseed_randomness in any::<[u8; 32]>()) -> (SpendProofPublic, SpendProofPrivate) {
let seed_phrase = SeedPhrase::from_randomness(&seed_phrase_randomness);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (sender, _dtk_d) = ivk_sender.payment_address(address_index.into());
let value_to_send = Value {
amount: Amount::from(0u64),
asset_id: asset::Id(Fq::from(asset_id64)),
};
let note = Note::from_parts(
sender.clone(),
value_to_send,
Rseed(rseed_randomness),
).expect("should be able to create note");
let note_commitment = note.commit();
let rsk = sk_sender.spend_auth_key().randomize(&spend_auth_randomizer);
let nk = *sk_sender.nullifier_key();
let ak: VerificationKey<SpendAuth> = sk_sender.spend_auth_key().into();
let mut sct = tct::Tree::new();
sct.insert(tct::Witness::Keep, note_commitment).expect("should be able to insert note commitments into the SCT");
let state_commitment_proof = sct.witness(note_commitment).expect("can witness note commitment");
let balance_commitment = value_to_send.commit(v_blinding);
let rk: VerificationKey<SpendAuth> = rsk.into();
let nullifier = Nullifier::derive(&nk, state_commitment_proof.position(), ¬e_commitment);
let invalid_anchor = tct::Tree::new().root();
let public = SpendProofPublic {
anchor: invalid_anchor,
balance_commitment,
nullifier,
rk,
};
let private = SpendProofPrivate {
state_commitment_proof,
note,
v_blinding,
spend_auth_randomizer,
ak,
nk,
};
(public, private)
}
}
proptest! {
#[test]
fn spend_proof_dummy_verification_suceeds((public, private) in arb_valid_dummy_spend_statement()) {
assert!(check_satisfaction(&public, &private).is_ok());
assert!(check_circuit_satisfaction(public, private).is_ok());
}
}
struct MerkleProofCircuit {
state_commitment_proof: tct::Proof,
pub anchor: tct::Root,
pub epoch: Fq,
pub block: Fq,
pub commitment_index: Fq,
}
impl ConstraintSynthesizer<Fq> for MerkleProofCircuit {
fn generate_constraints(
self,
cs: ConstraintSystemRef<Fq>,
) -> ark_relations::r1cs::Result<()> {
let anchor_var = FqVar::new_input(cs.clone(), || Ok(Fq::from(self.anchor)))?;
let epoch_var = FqVar::new_input(cs.clone(), || Ok(self.epoch))?;
let block_var = FqVar::new_input(cs.clone(), || Ok(self.block))?;
let commitment_index_var = FqVar::new_input(cs.clone(), || Ok(self.commitment_index))?;
let merkle_path_var = tct::r1cs::MerkleAuthPathVar::new_witness(cs.clone(), || {
Ok(self.state_commitment_proof.clone())
})?;
let claimed_note_commitment = StateCommitmentVar::new_witness(cs.clone(), || {
Ok(self.state_commitment_proof.commitment())
})?;
let position_var = tct::r1cs::PositionVar::new_witness(cs.clone(), || {
Ok(self.state_commitment_proof.position())
})?;
let position_bits = position_var.to_bits_le()?;
merkle_path_var.verify(
cs,
&Boolean::TRUE,
&position_bits,
anchor_var,
claimed_note_commitment.inner(),
)?;
let computed_epoch = position_var.epoch()?;
let computed_block = position_var.block()?;
let computed_commitment_index = position_var.commitment()?;
computed_epoch.enforce_equal(&epoch_var)?;
computed_block.enforce_equal(&block_var)?;
computed_commitment_index.enforce_equal(&commitment_index_var)?;
Ok(())
}
}
impl DummyWitness for MerkleProofCircuit {
fn with_dummy_witness() -> Self {
let seed_phrase = SeedPhrase::from_randomness(&[b'f'; 32]);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (address, _dtk_d) = ivk_sender.payment_address(0u32.into());
let note = Note::from_parts(
address,
Value::from_str("1upenumbra").expect("valid value"),
Rseed([1u8; 32]),
)
.expect("can make a note");
let mut sct = tct::Tree::new();
let note_commitment = note.commit();
sct.insert(tct::Witness::Keep, note_commitment)
.expect("able to insert note commitment into SCT");
let anchor = sct.root();
let state_commitment_proof = sct
.witness(note_commitment)
.expect("able to witness just-inserted note commitment");
let position = state_commitment_proof.position();
let epoch = Fq::from(position.epoch());
let block = Fq::from(position.block());
let commitment_index = Fq::from(position.commitment());
Self {
state_commitment_proof,
anchor,
epoch,
block,
commitment_index,
}
}
}
fn make_random_note_commitment(address: Address) -> StateCommitment {
let note = Note::from_parts(
address,
Value::from_str("1upenumbra").expect("valid value"),
Rseed([1u8; 32]),
)
.expect("can make a note");
note.commit()
}
#[test]
fn merkle_proof_verification_succeeds() {
let mut rng = OsRng;
let (pk, vk) = generate_prepared_test_parameters::<MerkleProofCircuit>(&mut rng);
let seed_phrase = SeedPhrase::from_randomness(&[b'f'; 32]);
let sk_sender = SpendKey::from_seed_phrase_bip44(seed_phrase, &Bip44Path::new(0));
let fvk_sender = sk_sender.full_viewing_key();
let ivk_sender = fvk_sender.incoming();
let (address, _dtk_d) = ivk_sender.payment_address(0u32.into());
let mut sct = tct::Tree::new();
for _ in 0..5 {
let note_commitment = make_random_note_commitment(address.clone());
sct.insert(tct::Witness::Keep, note_commitment).unwrap();
let anchor = sct.root();
let state_commitment_proof = sct.witness(note_commitment).unwrap();
let position = state_commitment_proof.position();
let epoch = Fq::from(position.epoch());
let block = Fq::from(position.block());
let commitment_index = Fq::from(position.commitment());
let circuit = MerkleProofCircuit {
state_commitment_proof,
anchor,
epoch,
block,
commitment_index,
};
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_with_processed_vk(
&vk,
&[Fq::from(anchor), epoch, block, commitment_index],
&proof,
);
assert!(proof_result.is_ok());
}
sct.end_block().expect("can end block");
for _ in 0..100 {
let note_commitment = make_random_note_commitment(address.clone());
sct.insert(tct::Witness::Forget, note_commitment).unwrap();
}
for _ in 0..5 {
let note_commitment = make_random_note_commitment(address.clone());
sct.insert(tct::Witness::Keep, note_commitment).unwrap();
let anchor = sct.root();
let state_commitment_proof = sct.witness(note_commitment).unwrap();
let position = state_commitment_proof.position();
let epoch = Fq::from(position.epoch());
let block = Fq::from(position.block());
let commitment_index = Fq::from(position.commitment());
let circuit = MerkleProofCircuit {
state_commitment_proof,
anchor,
epoch,
block,
commitment_index,
};
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_with_processed_vk(
&vk,
&[Fq::from(anchor), epoch, block, commitment_index],
&proof,
);
assert!(proof_result.is_ok());
}
sct.end_epoch().expect("can end epoch");
for _ in 0..100 {
let note_commitment = make_random_note_commitment(address.clone());
sct.insert(tct::Witness::Forget, note_commitment).unwrap();
}
for _ in 0..5 {
let note_commitment = make_random_note_commitment(address.clone());
sct.insert(tct::Witness::Keep, note_commitment).unwrap();
let anchor = sct.root();
let state_commitment_proof = sct.witness(note_commitment).unwrap();
let position = state_commitment_proof.position();
let epoch = Fq::from(position.epoch());
let block = Fq::from(position.block());
let commitment_index = Fq::from(position.commitment());
let circuit = MerkleProofCircuit {
state_commitment_proof,
anchor,
epoch,
block,
commitment_index,
};
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_with_processed_vk(
&vk,
&[Fq::from(anchor), epoch, block, commitment_index],
&proof,
);
assert!(proof_result.is_ok());
}
}
}