1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
use super::Private;
use crate::Vec;
use ark_crypto_primitives::{crh::CRHGadget, CRH};
use ark_ff::fields::PrimeField;
use ark_r1cs_std::{fields::fp::FpVar, prelude::*};
use ark_relations::r1cs::{Namespace, SynthesisError};
use ark_std::marker::PhantomData;
use core::borrow::Borrow;
#[derive(Clone)]
pub struct PrivateVar<F: PrimeField> {
r: FpVar<F>,
nullifier: FpVar<F>,
}
#[derive(Clone, Default)]
pub struct PublicVar<F: PrimeField> {
f: PhantomData<F>,
}
impl<F: PrimeField> PrivateVar<F> {
pub fn new(r: FpVar<F>, nullifier: FpVar<F>) -> Self {
Self { r, nullifier }
}
}
pub struct BasicLeafGadget<F: PrimeField, H: CRH, HG: CRHGadget<H, F>> {
field: PhantomData<F>,
hasher: PhantomData<H>,
hasher_gadget: PhantomData<HG>,
}
impl<F: PrimeField, H: CRH, HG: CRHGadget<H, F>> BasicLeafGadget<F, H, HG> {
pub fn create_leaf(
private: &PrivateVar<F>,
h: &HG::ParametersVar,
) -> Result<HG::OutputVar, SynthesisError> {
let mut bytes = Vec::new();
bytes.extend(private.r.to_bytes()?);
bytes.extend(private.nullifier.to_bytes()?);
HG::evaluate(h, &bytes)
}
pub fn create_nullifier(
private: &PrivateVar<F>,
h: &HG::ParametersVar,
) -> Result<HG::OutputVar, SynthesisError> {
let mut bytes = Vec::new();
bytes.extend(private.nullifier.to_bytes()?);
bytes.extend(private.nullifier.to_bytes()?);
HG::evaluate(h, &bytes)
}
}
impl<F: PrimeField> AllocVar<Private<F>, F> for PrivateVar<F> {
fn new_variable<T: Borrow<Private<F>>>(
into_ns: impl Into<Namespace<F>>,
f: impl FnOnce() -> Result<T, SynthesisError>,
mode: AllocationMode,
) -> Result<Self, SynthesisError> {
let secrets = f()?.borrow().clone();
let ns = into_ns.into();
let cs = ns.cs();
let r = secrets.r;
let nullifier = secrets.nullifier;
let r_var = FpVar::new_variable(cs.clone(), || Ok(r), mode)?;
let nullifier_var = FpVar::new_variable(cs, || Ok(nullifier), mode)?;
Ok(PrivateVar::new(r_var, nullifier_var))
}
}
#[cfg(test)]
mod test {
use super::*;
use crate::{
leaf::basic::BasicLeaf,
poseidon::{
constraints::{CRHGadget, PoseidonParametersVar},
CRH,
},
};
use ark_bls12_381::Fq;
use ark_relations::r1cs::ConstraintSystem;
use ark_std::test_rng;
use arkworks_utils::utils::common::setup_params_x5_3;
type PoseidonCRH3 = CRH<Fq>;
type PoseidonCRH3Gadget = CRHGadget<Fq>;
type Leaf = BasicLeaf<Fq, PoseidonCRH3>;
type LeafGadget = BasicLeafGadget<Fq, PoseidonCRH3, PoseidonCRH3Gadget>;
#[test]
fn should_crate_basic_leaf_constraints() {
let rng = &mut test_rng();
let curve = arkworks_utils::utils::common::Curve::Bls381;
let cs = ConstraintSystem::<Fq>::new_ref();
let secrets = Private::generate(rng);
let params = setup_params_x5_3(curve);
let leaf = Leaf::create_leaf(&secrets, ¶ms).unwrap();
let secrets_var = PrivateVar::new_witness(cs.clone(), || Ok(&secrets)).unwrap();
let params_var =
PoseidonParametersVar::new_variable(cs, || Ok(¶ms), AllocationMode::Constant)
.unwrap();
let leaf_var = LeafGadget::create_leaf(&secrets_var, ¶ms_var).unwrap();
let leaf_new_var = FpVar::<Fq>::new_witness(leaf_var.cs(), || Ok(leaf)).unwrap();
let res = leaf_var.is_eq(&leaf_new_var).unwrap();
assert!(res.value().unwrap());
assert!(res.cs().is_satisfied().unwrap());
}
}