use ark_ec::{CurveGroup, Group}; use ark_std::{rand::Rng, UniformRand}; use std::marker::PhantomData; use crate::utils::vec::{vec_add, vec_scalar_mul}; use crate::transcript::Transcript; use ark_crypto_primitives::sponge::Absorb; #[derive(Debug, Clone, Eq, PartialEq)] pub struct Proof { R: C, u: Vec, r_u: C::ScalarField, } #[derive(Debug, Clone, Eq, PartialEq)] pub struct Params { h: C, pub generators: Vec, } #[derive(Debug, Clone, Eq, PartialEq)] pub struct Pedersen where ::ScalarField: Absorb, { _c: PhantomData, } impl Pedersen where ::ScalarField: Absorb, { pub fn new_params(rng: &mut R, max: usize) -> Params { let generators: Vec = std::iter::repeat_with(|| C::Affine::rand(rng)) .take(max.next_power_of_two()) .collect(); let params: Params = Params:: { h: C::rand(rng), generators, }; params } pub fn commit(params: &Params, v: &Vec, r: &C::ScalarField) -> C { // h⋅r + params.h.mul(r) + C::msm(¶ms.generators[..v.len()], v).unwrap() } pub fn prove( params: &Params, transcript: &mut impl Transcript, cm: &C, v: &Vec, r: &C::ScalarField, ) -> Proof { transcript.absorb_point(cm); let r1 = transcript.get_challenge(); let d = transcript.get_challenges(v.len()); // R = h⋅r_1 + let R: C = params.h.mul(r1) + C::msm(¶ms.generators[..d.len()], &d).unwrap(); transcript.absorb_point(&R); let e = transcript.get_challenge(); // u = d + v⋅e let u = vec_add(&vec_scalar_mul(v, &e), &d); // r_u = e⋅r + r_1 let r_u = e * r + r1; Proof:: { R, u, r_u } } pub fn verify( params: &Params, transcript: &mut impl Transcript, cm: C, proof: Proof, ) -> bool { transcript.absorb_point(&cm); transcript.get_challenge(); // r_1 transcript.get_challenges(proof.u.len()); // d transcript.absorb_point(&proof.R); let e = transcript.get_challenge(); // check that: R + cm == h⋅r_u + let lhs = proof.R + cm.mul(e); let rhs = params.h.mul(proof.r_u) + C::msm(¶ms.generators[..proof.u.len()], &proof.u).unwrap(); if lhs != rhs { return false; } true } } #[cfg(test)] mod tests { use super::*; use crate::transcript::poseidon::{tests::poseidon_test_config, PoseidonTranscript}; use ark_pallas::{Fr, Projective}; #[test] fn test_pedersen_vector() { let mut rng = ark_std::test_rng(); const n: usize = 10; // setup params let params = Pedersen::::new_params(&mut rng, n); let poseidon_config = poseidon_test_config::(); // init Prover's transcript let mut transcript_p = PoseidonTranscript::::new(&poseidon_config); // init Verifier's transcript let mut transcript_v = PoseidonTranscript::::new(&poseidon_config); let v: Vec = vec![Fr::rand(&mut rng); n]; let r: Fr = Fr::rand(&mut rng); let cm = Pedersen::::commit(¶ms, &v, &r); let proof = Pedersen::::prove(¶ms, &mut transcript_p, &cm, &v, &r); let v = Pedersen::::verify(¶ms, &mut transcript_v, cm, proof); assert!(v); } }