use group::{Curve, ff::PrimeField};
use bellman::{Circuit, ConstraintSystem, SynthesisError};
use masp_primitives::{
constants,
sapling::{PaymentAddress, ProofGenerationKey, SAPLING_COMMITMENT_TREE_DEPTH, ValueCommitment},
};
use super::ecc;
use super::pedersen_hash;
use crate::circuit::gadgets;
use crate::constants::{
NOTE_COMMITMENT_RANDOMNESS_GENERATOR, NULLIFIER_POSITION_GENERATOR,
PROOF_GENERATION_KEY_GENERATOR, SPENDING_KEY_GENERATOR, VALUE_COMMITMENT_RANDOMNESS_GENERATOR,
};
use bellman::gadgets::{Assignment, blake2s, boolean, multipack, num};
use group::ff::Field;
use itertools::multizip;
pub const TREE_DEPTH: usize = SAPLING_COMMITMENT_TREE_DEPTH;
pub struct Spend {
pub value_commitment: Option<ValueCommitment>,
pub proof_generation_key: Option<ProofGenerationKey>,
pub payment_address: Option<PaymentAddress>,
pub commitment_randomness: Option<jubjub::Fr>,
pub ar: Option<jubjub::Fr>,
pub auth_path: Vec<Option<(bls12_381::Scalar, bool)>>,
pub anchor: Option<bls12_381::Scalar>,
}
pub struct Output {
pub value_commitment: Option<ValueCommitment>,
pub asset_identifier: Vec<Option<bool>>,
pub payment_address: Option<PaymentAddress>,
pub commitment_randomness: Option<jubjub::Fr>,
pub esk: Option<jubjub::Fr>,
}
pub fn expose_value_commitment<CS>(
mut cs: CS,
value_commitment: Option<ValueCommitment>,
) -> Result<(Vec<boolean::Boolean>, Vec<boolean::Boolean>), SynthesisError>
where
CS: ConstraintSystem<bls12_381::Scalar>,
{
let asset_generator = ecc::EdwardsPoint::witness(
cs.namespace(|| "asset_generator"),
value_commitment.as_ref().map(|vc| vc.asset_generator),
)?;
let asset_generator_bits = asset_generator.repr(cs.namespace(|| "unpack asset_generator"))?;
let asset_generator =
asset_generator.double(cs.namespace(|| "asset_generator first doubling"))?;
let asset_generator =
asset_generator.double(cs.namespace(|| "asset_generator second doubling"))?;
let asset_generator =
asset_generator.double(cs.namespace(|| "asset_generator third doubling"))?;
asset_generator
.get_u()
.assert_nonzero(cs.namespace(|| "check asset_generator != 0"))?;
let value_bits = boolean::u64_into_boolean_vec_le(
cs.namespace(|| "value"),
value_commitment.as_ref().map(|c| c.value),
)?;
let value = asset_generator.mul(
cs.namespace(|| "compute the value in the exponent"),
&value_bits,
)?;
let rcv = gadgets::field_into_boolean_vec_le(
cs.namespace(|| "rcv"),
value_commitment.as_ref().map(|c| c.randomness),
)?;
let rcv = ecc::fixed_base_multiplication(
cs.namespace(|| "computation of rcv"),
&VALUE_COMMITMENT_RANDOMNESS_GENERATOR,
&rcv,
)?;
let cv = value.add(cs.namespace(|| "computation of cv"), &rcv)?;
cv.inputize(cs.namespace(|| "commitment point"))?;
Ok((asset_generator_bits, value_bits))
}
impl Circuit<bls12_381::Scalar> for Spend {
fn synthesize<CS: ConstraintSystem<bls12_381::Scalar>>(
self,
cs: &mut CS,
) -> Result<(), SynthesisError> {
let ak = ecc::EdwardsPoint::witness(
cs.namespace(|| "ak"),
self.proof_generation_key.as_ref().map(|k| k.ak.into()),
)?;
ak.assert_not_small_order(cs.namespace(|| "ak not small order"))?;
{
let ar = gadgets::field_into_boolean_vec_le(cs.namespace(|| "ar"), self.ar)?;
let ar = ecc::fixed_base_multiplication(
cs.namespace(|| "computation of randomization for the signing key"),
&SPENDING_KEY_GENERATOR,
&ar,
)?;
let rk = ak.add(cs.namespace(|| "computation of rk"), &ar)?;
rk.inputize(cs.namespace(|| "rk"))?;
}
let nk;
{
let nsk = gadgets::field_into_boolean_vec_le(
cs.namespace(|| "nsk"),
self.proof_generation_key.as_ref().map(|k| k.nsk),
)?;
nk = ecc::fixed_base_multiplication(
cs.namespace(|| "computation of nk"),
&PROOF_GENERATION_KEY_GENERATOR,
&nsk,
)?;
}
let mut ivk_preimage = vec![];
ivk_preimage.extend(ak.repr(cs.namespace(|| "representation of ak"))?);
let mut nf_preimage = vec![];
{
let repr_nk = nk.repr(cs.namespace(|| "representation of nk"))?;
ivk_preimage.extend(repr_nk.iter().cloned());
nf_preimage.extend(repr_nk);
}
assert_eq!(ivk_preimage.len(), 512);
assert_eq!(nf_preimage.len(), 256);
let mut ivk = blake2s::blake2s(
cs.namespace(|| "computation of ivk"),
&ivk_preimage,
constants::CRH_IVK_PERSONALIZATION,
)?;
ivk.truncate(jubjub::Fr::CAPACITY as usize);
let g_d = {
ecc::EdwardsPoint::witness(
cs.namespace(|| "witness g_d"),
self.payment_address
.as_ref()
.and_then(|a| a.g_d().map(jubjub::ExtendedPoint::from)),
)?
};
g_d.assert_not_small_order(cs.namespace(|| "g_d not small order"))?;
let pk_d = g_d.mul(cs.namespace(|| "compute pk_d"), &ivk)?;
let mut note_contents = vec![];
let mut value_num = num::Num::zero();
{
let (asset_generator_bits, value_bits) = expose_value_commitment(
cs.namespace(|| "value commitment"),
self.value_commitment,
)?;
let mut coeff = bls12_381::Scalar::ONE;
for bit in &value_bits {
value_num = value_num.add_bool_with_coeff(CS::one(), bit, coeff);
coeff = coeff.double();
}
note_contents.extend(asset_generator_bits);
note_contents.extend(value_bits);
}
note_contents.extend(g_d.repr(cs.namespace(|| "representation of g_d"))?);
note_contents.extend(pk_d.repr(cs.namespace(|| "representation of pk_d"))?);
assert_eq!(
note_contents.len(),
256 + 64 + 256 + 256 );
let mut cm = pedersen_hash::pedersen_hash(
cs.namespace(|| "note content hash"),
pedersen_hash::Personalization::NoteCommitment,
¬e_contents,
)?;
{
let rcm = gadgets::field_into_boolean_vec_le(
cs.namespace(|| "rcm"),
self.commitment_randomness,
)?;
let rcm = ecc::fixed_base_multiplication(
cs.namespace(|| "computation of commitment randomness"),
&NOTE_COMMITMENT_RANDOMNESS_GENERATOR,
&rcm,
)?;
cm = cm.add(cs.namespace(|| "randomization of note commitment"), &rcm)?;
}
let mut position_bits = vec![];
let mut cur = cm.get_u().clone();
for (i, e) in self.auth_path.into_iter().enumerate() {
let cs = &mut cs.namespace(|| format!("merkle tree hash {}", i));
let cur_is_right = boolean::Boolean::from(boolean::AllocatedBit::alloc(
cs.namespace(|| "position bit"),
e.map(|e| e.1),
)?);
position_bits.push(cur_is_right.clone());
let path_element =
num::AllocatedNum::alloc(cs.namespace(|| "path element"), || Ok(e.get()?.0))?;
let (ul, ur) = num::AllocatedNum::conditionally_reverse(
cs.namespace(|| "conditional reversal of preimage"),
&cur,
&path_element,
&cur_is_right,
)?;
let mut preimage = vec![];
preimage.extend(ul.to_bits_le(cs.namespace(|| "ul into bits"))?);
preimage.extend(ur.to_bits_le(cs.namespace(|| "ur into bits"))?);
cur = pedersen_hash::pedersen_hash(
cs.namespace(|| "computation of pedersen hash"),
pedersen_hash::Personalization::MerkleTree(i),
&preimage,
)?
.get_u()
.clone(); }
{
let real_anchor_value = self.anchor;
let rt = num::AllocatedNum::alloc(cs.namespace(|| "conditional anchor"), || {
Ok(*real_anchor_value.get()?)
})?;
cs.enforce(
|| "conditionally enforce correct root",
|lc| lc + cur.get_variable() - rt.get_variable(),
|lc| lc + &value_num.lc(bls12_381::Scalar::ONE),
|lc| lc,
);
rt.inputize(cs.namespace(|| "anchor"))?;
}
let mut rho = cm;
{
let position = ecc::fixed_base_multiplication(
cs.namespace(|| "g^position"),
&NULLIFIER_POSITION_GENERATOR,
&position_bits,
)?;
rho = rho.add(cs.namespace(|| "faerie gold prevention"), &position)?;
}
nf_preimage.extend(rho.repr(cs.namespace(|| "representation of rho"))?);
assert_eq!(nf_preimage.len(), 512);
let nf = blake2s::blake2s(
cs.namespace(|| "nf computation"),
&nf_preimage,
constants::PRF_NF_PERSONALIZATION,
)?;
multipack::pack_into_inputs(cs.namespace(|| "pack nullifier"), &nf)
}
}
impl Circuit<bls12_381::Scalar> for Output {
fn synthesize<CS: ConstraintSystem<bls12_381::Scalar>>(
self,
cs: &mut CS,
) -> Result<(), SynthesisError> {
let mut note_contents = vec![];
let mut asset_generator_preimage = Vec::with_capacity(256);
assert_eq!(256, self.asset_identifier.len());
for (i, bit) in self.asset_identifier.iter().enumerate() {
let cs = &mut cs.namespace(|| format!("witness asset type bit {}", i));
let asset_identifier_preimage_bit = boolean::Boolean::from(
boolean::AllocatedBit::alloc(cs.namespace(|| "asset type bit"), *bit)?,
);
asset_generator_preimage.push(asset_identifier_preimage_bit.clone());
}
assert_eq!(256, asset_generator_preimage.len());
let asset_generator_image = blake2s::blake2s(
cs.namespace(|| "value base computation"),
&asset_generator_preimage,
constants::VALUE_COMMITMENT_GENERATOR_PERSONALIZATION,
)?;
let (asset_generator_bits, value_bits) =
expose_value_commitment(cs.namespace(|| "value commitment"), self.value_commitment)?;
assert_eq!(256, asset_generator_bits.len());
assert_eq!(256, asset_generator_image.len());
for (i, asset_generator_bit, asset_generator_image_bit) in
multizip((0..256, &asset_generator_bits, &asset_generator_image))
{
boolean::Boolean::enforce_equal(
cs.namespace(|| format!("integrity of asset generator bit {}", i)),
asset_generator_bit,
asset_generator_image_bit,
)?;
}
note_contents.extend(asset_generator_bits);
note_contents.extend(value_bits);
{
let g_d = ecc::EdwardsPoint::witness(
cs.namespace(|| "witness g_d"),
self.payment_address
.as_ref()
.and_then(|a| a.g_d().map(jubjub::ExtendedPoint::from)),
)?;
g_d.assert_not_small_order(cs.namespace(|| "g_d not small order"))?;
note_contents.extend(g_d.repr(cs.namespace(|| "representation of g_d"))?);
let esk = gadgets::field_into_boolean_vec_le(cs.namespace(|| "esk"), self.esk)?;
let epk = g_d.mul(cs.namespace(|| "epk computation"), &esk)?;
epk.inputize(cs.namespace(|| "epk"))?;
}
{
let pk_d = self
.payment_address
.as_ref()
.map(|e| jubjub::ExtendedPoint::from(*e.pk_d()).to_affine());
let v_contents = boolean::field_into_boolean_vec_le(
cs.namespace(|| "pk_d bits of v"),
pk_d.map(|e| e.get_v()),
)?;
let sign_bit = boolean::Boolean::from(boolean::AllocatedBit::alloc(
cs.namespace(|| "pk_d bit of u"),
pk_d.map(|e| e.get_u().is_odd().into()),
)?);
note_contents.extend(v_contents);
note_contents.push(sign_bit);
}
assert_eq!(
note_contents.len(),
256 + 64 + 256 + 256 );
let mut cm = pedersen_hash::pedersen_hash(
cs.namespace(|| "note content hash"),
pedersen_hash::Personalization::NoteCommitment,
¬e_contents,
)?;
{
let rcm = gadgets::field_into_boolean_vec_le(
cs.namespace(|| "rcm"),
self.commitment_randomness,
)?;
let rcm = ecc::fixed_base_multiplication(
cs.namespace(|| "computation of commitment randomness"),
&NOTE_COMMITMENT_RANDOMNESS_GENERATOR,
&rcm,
)?;
cm = cm.add(cs.namespace(|| "randomization of note commitment"), &rcm)?;
}
cm.get_u().inputize(cs.namespace(|| "commitment"))?;
Ok(())
}
}
#[test]
fn test_input_circuit_with_bls12_381() {
use bellman::gadgets::test::*;
use group::{Group, ff::Field, ff::PrimeFieldBits};
use masp_primitives::{
asset_type::AssetType,
sapling::{Diversifier, Note, ProofGenerationKey, Rseed, pedersen_hash},
};
use rand_core::{RngCore, SeedableRng};
use rand_xorshift::XorShiftRng;
let mut rng = XorShiftRng::from_seed([
0x58, 0x62, 0xbe, 0x3d, 0x76, 0x3d, 0x31, 0x8d, 0x17, 0xdb, 0x37, 0x32, 0x54, 0x06, 0xbc,
0xe5,
]);
let tree_depth = 32;
for i in 0..30u32 {
let asset_type = if i < 10 {
AssetType::new(b"default")
} else {
AssetType::new(i.to_string().as_bytes())
}
.unwrap();
let mut value_commitment =
asset_type.value_commitment(rng.next_u64(), jubjub::Fr::random(&mut rng));
if i >= 20 {
value_commitment.asset_generator = -value_commitment.asset_generator;
}
let proof_generation_key = ProofGenerationKey {
ak: jubjub::SubgroupPoint::random(&mut rng),
nsk: jubjub::Fr::random(&mut rng),
};
let viewing_key = proof_generation_key.to_viewing_key();
let payment_address;
loop {
let diversifier = {
let mut d = [0; 11];
rng.fill_bytes(&mut d);
Diversifier(d)
};
if let Some(p) = viewing_key.to_payment_address(diversifier) {
payment_address = p;
break;
}
}
let g_d = payment_address.diversifier().g_d().unwrap();
let commitment_randomness = jubjub::Fr::random(&mut rng);
let auth_path =
vec![Some((bls12_381::Scalar::random(&mut rng), rng.next_u32() % 2 != 0)); tree_depth];
let ar = jubjub::Fr::random(&mut rng);
{
let rk = jubjub::ExtendedPoint::from(viewing_key.rk(ar)).to_affine();
let expected_value_commitment =
jubjub::ExtendedPoint::from(value_commitment.commitment()).to_affine();
let note = Note {
asset_type,
value: value_commitment.value,
g_d,
pk_d: *payment_address.pk_d(),
rseed: Rseed::BeforeZip212(commitment_randomness),
};
let mut position = 0u64;
let cmu = note.cmu();
let mut cur = cmu;
for (i, val) in auth_path.clone().into_iter().enumerate() {
let (uncle, b) = val.unwrap();
let mut lhs = cur;
let mut rhs = uncle;
if b {
::std::mem::swap(&mut lhs, &mut rhs);
}
let lhs = lhs.to_le_bits();
let rhs = rhs.to_le_bits();
cur = jubjub::ExtendedPoint::from(pedersen_hash::pedersen_hash(
pedersen_hash::Personalization::MerkleTree(i),
lhs.iter()
.by_vals()
.take(bls12_381::Scalar::NUM_BITS as usize)
.chain(
rhs.iter()
.by_vals()
.take(bls12_381::Scalar::NUM_BITS as usize),
),
))
.to_affine()
.get_u();
if b {
position |= 1 << i;
}
}
let expected_nf = note.nf(&viewing_key.nk, position);
let expected_nf = multipack::bytes_to_bits_le(&expected_nf.0);
let expected_nf = multipack::compute_multipacking(&expected_nf);
assert_eq!(expected_nf.len(), 2);
let mut cs = TestConstraintSystem::new();
let instance = Spend {
value_commitment: Some(value_commitment.clone()),
proof_generation_key: Some(proof_generation_key.clone()),
payment_address: Some(payment_address),
commitment_randomness: Some(commitment_randomness),
ar: Some(ar),
auth_path: auth_path.clone(),
anchor: Some(cur),
};
instance.synthesize(&mut cs).unwrap();
if i < 20 {
assert!(cs.is_satisfied());
} else {
assert!(!cs.is_satisfied());
}
assert_eq!(cs.num_constraints(), 100637);
assert_eq!(
cs.hash(),
"34e4a634c80e4e4c6250e63b7855532e60b36d1371d4d7b1163218b69f09eb3d"
);
if i < 20 {
assert_eq!(cs.get("randomization of note commitment/u3/num"), cmu);
} else {
assert_ne!(cs.get("randomization of note commitment/u3/num"), cmu);
}
assert_eq!(cs.num_inputs(), 8);
assert_eq!(cs.get_input(0, "ONE"), bls12_381::Scalar::ONE);
assert_eq!(cs.get_input(1, "rk/u/input variable"), rk.get_u());
assert_eq!(cs.get_input(2, "rk/v/input variable"), rk.get_v());
assert_eq!(
cs.get_input(3, "value commitment/commitment point/u/input variable"),
expected_value_commitment.get_u()
);
assert_eq!(
cs.get_input(4, "value commitment/commitment point/v/input variable"),
expected_value_commitment.get_v()
);
assert_eq!(cs.get_input(5, "anchor/input variable"), cur);
if i < 20 {
assert_eq!(cs.get_input(6, "pack nullifier/input 0"), expected_nf[0]);
assert_eq!(cs.get_input(7, "pack nullifier/input 1"), expected_nf[1]);
} else {
assert_ne!(cs.get_input(6, "pack nullifier/input 0"), expected_nf[0]);
}
}
}
}
#[test]
fn test_input_circuit_with_bls12_381_external_test_vectors() {
use bellman::gadgets::test::*;
use group::{Group, ff::Field, ff::PrimeField, ff::PrimeFieldBits};
use masp_primitives::{
asset_type::AssetType,
sapling::pedersen_hash,
sapling::{Diversifier, Note, ProofGenerationKey, Rseed},
};
use rand_core::{RngCore, SeedableRng};
use rand_xorshift::XorShiftRng;
let mut rng = XorShiftRng::from_seed([
0x59, 0x62, 0xbe, 0x3d, 0x76, 0x3d, 0x31, 0x8d, 0x17, 0xdb, 0x37, 0x32, 0x54, 0x06, 0xbc,
0xe5,
]);
let tree_depth = 32;
let expected_commitment_us = [
"15274760159508878651789682992925045402656388195689586056903525226511870631006",
"17926082480702379779301751040578316677060182517930108360303758506447415843229",
"47560733217722603616763811825500591868568811326811130069535870262273364981945",
"3800891689291852208719409763066191375952446148569504124915840587177301316887",
"42605451358726896269346670960800907068736580931770467343442333651979812783507",
"2186124196248736405363923904916329765421958395459957351012037099196644523519",
"1141914194379178008776608799121446552214386159445356778422457950073807217391",
"4723282540978794624483635488138659467675602905263923545920612233258386488162",
"9817985978230076566482131380463677459892992710371329861360645363311468893053",
"27618789340710350120647137095252986938132361388195675764406370494688910938013",
];
let expected_commitment_vs = [
"34821791232396287888199995100305255761362584209078006239735148846881442279277",
"25119990066174545608121950753413857831099772082356729649061420500567639159355",
"37379068700729686079521798425830021519833420633231595656391703260880647751299",
"41866535334944468208261223722134220321702695454463459117958311496151517396608",
"22815243378235771837066051140494563507512924813701395974049305004556621752999",
"32580943391199462206001867000285792160642911175912464838584939697793150575579",
"19466322163466228937035549603042240330689838936758470332197790607062875140040",
"37409705443279116387495124812424670311932220465698221026006921521796611194301",
"4817145647901840172966045688653436033808505237142136464043537162611284452519",
"33112537425917174283144333017659536059363113223507009786626165162100944911092",
];
let asset_type = AssetType::from_identifier(
b"sO\x0e\xc5os\x1e\x02\xccs~ki=\xb5+\x82\x1fonL\xd7\xfe<vCS\xf2cf\x9f\xbe",
)
.unwrap();
for i in 0..10 {
let value_commitment = asset_type.value_commitment(i, jubjub::Fr::from(1000 * (i + 1)));
let proof_generation_key = ProofGenerationKey {
ak: jubjub::SubgroupPoint::random(&mut rng),
nsk: jubjub::Fr::random(&mut rng),
};
let viewing_key = proof_generation_key.to_viewing_key();
let payment_address;
loop {
let diversifier = {
let mut d = [0; 11];
rng.fill_bytes(&mut d);
Diversifier(d)
};
if let Some(p) = viewing_key.to_payment_address(diversifier) {
payment_address = p;
break;
}
}
let g_d = payment_address.diversifier().g_d().unwrap();
let commitment_randomness = jubjub::Fr::random(&mut rng);
let auth_path =
vec![Some((bls12_381::Scalar::random(&mut rng), rng.next_u32() % 2 != 0)); tree_depth];
let ar = jubjub::Fr::random(&mut rng);
{
let rk = jubjub::ExtendedPoint::from(viewing_key.rk(ar)).to_affine();
let expected_value_commitment =
jubjub::ExtendedPoint::from(value_commitment.commitment()).to_affine();
assert_eq!(
expected_value_commitment.get_u(),
bls12_381::Scalar::from_str_vartime(expected_commitment_us[i as usize]).unwrap()
);
assert_eq!(
expected_value_commitment.get_v(),
bls12_381::Scalar::from_str_vartime(expected_commitment_vs[i as usize]).unwrap()
);
let note = Note {
asset_type,
value: value_commitment.value,
g_d,
pk_d: *payment_address.pk_d(),
rseed: Rseed::BeforeZip212(commitment_randomness),
};
let mut position = 0u64;
let cmu = note.cmu();
let mut cur = cmu;
for (i, val) in auth_path.clone().into_iter().enumerate() {
let (uncle, b) = val.unwrap();
let mut lhs = cur;
let mut rhs = uncle;
if b {
::std::mem::swap(&mut lhs, &mut rhs);
}
let lhs = lhs.to_le_bits();
let rhs = rhs.to_le_bits();
cur = jubjub::ExtendedPoint::from(pedersen_hash::pedersen_hash(
pedersen_hash::Personalization::MerkleTree(i),
lhs.iter()
.by_vals()
.take(bls12_381::Scalar::NUM_BITS as usize)
.chain(
rhs.iter()
.by_vals()
.take(bls12_381::Scalar::NUM_BITS as usize),
),
))
.to_affine()
.get_u();
if b {
position |= 1 << i;
}
}
let expected_nf = note.nf(&viewing_key.nk, position);
let expected_nf = multipack::bytes_to_bits_le(&expected_nf.0);
let expected_nf = multipack::compute_multipacking(&expected_nf);
assert_eq!(expected_nf.len(), 2);
let mut cs = TestConstraintSystem::new();
let instance = Spend {
value_commitment: Some(value_commitment.clone()),
proof_generation_key: Some(proof_generation_key.clone()),
payment_address: Some(payment_address),
commitment_randomness: Some(commitment_randomness),
ar: Some(ar),
auth_path: auth_path.clone(),
anchor: Some(cur),
};
instance.synthesize(&mut cs).unwrap();
assert!(cs.is_satisfied());
assert_eq!(cs.num_constraints(), 100637);
assert_eq!(
cs.hash(),
"34e4a634c80e4e4c6250e63b7855532e60b36d1371d4d7b1163218b69f09eb3d"
);
assert_eq!(cs.get("randomization of note commitment/u3/num"), cmu);
assert_eq!(cs.num_inputs(), 8);
assert_eq!(cs.get_input(0, "ONE"), bls12_381::Scalar::ONE);
assert_eq!(cs.get_input(1, "rk/u/input variable"), rk.get_u());
assert_eq!(cs.get_input(2, "rk/v/input variable"), rk.get_v());
assert_eq!(
cs.get_input(3, "value commitment/commitment point/u/input variable"),
expected_value_commitment.get_u()
);
assert_eq!(
cs.get_input(4, "value commitment/commitment point/v/input variable"),
expected_value_commitment.get_v()
);
assert_eq!(cs.get_input(5, "anchor/input variable"), cur);
assert_eq!(cs.get_input(6, "pack nullifier/input 0"), expected_nf[0]);
assert_eq!(cs.get_input(7, "pack nullifier/input 1"), expected_nf[1]);
}
}
}
#[test]
fn test_output_circuit_with_bls12_381() {
use bellman::gadgets::test::*;
use group::{Group, ff::Field};
use masp_primitives::{
asset_type::AssetType,
sapling::{Diversifier, ProofGenerationKey, Rseed},
};
use rand_core::{RngCore, SeedableRng};
use rand_xorshift::XorShiftRng;
let mut rng = XorShiftRng::from_seed([
0x58, 0x62, 0xbe, 0x3d, 0x76, 0x3d, 0x31, 0x8d, 0x17, 0xdb, 0x37, 0x32, 0x54, 0x06, 0xbc,
0xe5,
]);
for i in 0..30 {
let asset_type = if i < 10 {
AssetType::new(b"default")
} else {
AssetType::new(i.to_string().as_bytes())
}
.unwrap();
let mut value_commitment =
asset_type.value_commitment(rng.next_u64(), jubjub::Fr::random(&mut rng));
if i >= 20 {
value_commitment.asset_generator = -value_commitment.asset_generator;
}
let nsk = jubjub::Fr::random(&mut rng);
let ak = jubjub::SubgroupPoint::random(&mut rng);
let proof_generation_key = ProofGenerationKey { ak, nsk };
let viewing_key = proof_generation_key.to_viewing_key();
let payment_address;
loop {
let diversifier = {
let mut d = [0; 11];
rng.fill_bytes(&mut d);
Diversifier(d)
};
if let Some(p) = viewing_key.to_payment_address(diversifier) {
payment_address = p;
break;
}
}
let commitment_randomness = jubjub::Fr::random(&mut rng);
let esk = jubjub::Fr::random(&mut rng);
{
let mut cs = TestConstraintSystem::new();
let instance = Output {
value_commitment: Some(value_commitment.clone()),
payment_address: Some(payment_address),
commitment_randomness: Some(commitment_randomness),
esk: Some(esk),
asset_identifier: asset_type.identifier_bits(),
};
instance.synthesize(&mut cs).unwrap();
if i < 20 {
assert!(cs.is_satisfied());
} else {
assert!(!cs.is_satisfied());
}
assert_eq!(
cs.hash(),
"93e445d7858e98c7138558df341f020aedfe75893535025587d64731e244276a"
);
let expected_cmu = payment_address
.create_note(
asset_type,
value_commitment.value,
Rseed::BeforeZip212(commitment_randomness),
)
.expect("should be valid")
.cmu();
let expected_value_commitment =
jubjub::ExtendedPoint::from(value_commitment.commitment()).to_affine();
let expected_epk =
jubjub::ExtendedPoint::from(payment_address.g_d().expect("should be valid") * esk)
.to_affine();
assert_eq!(cs.num_inputs(), 6);
assert_eq!(cs.get_input(0, "ONE"), bls12_381::Scalar::ONE);
assert_eq!(
cs.get_input(1, "value commitment/commitment point/u/input variable"),
expected_value_commitment.get_u()
);
assert_eq!(
cs.get_input(2, "value commitment/commitment point/v/input variable"),
expected_value_commitment.get_v()
);
assert_eq!(
cs.get_input(3, "epk/u/input variable"),
expected_epk.get_u()
);
assert_eq!(
cs.get_input(4, "epk/v/input variable"),
expected_epk.get_v()
);
if i < 20 {
assert_eq!(cs.get_input(5, "commitment/input variable"), expected_cmu);
}
}
}
}