Merge pull request 'pushed the file' (#1) from zkdream-patch-1 into main
Reviewed-on: zkdream/zkdream#1
This commit is contained in:
commit
6f2d695739
199
file.rs
Normal file
199
file.rs
Normal file
|
@ -0,0 +1,199 @@
|
|||
#![no_std]
|
||||
use ark_ec::CurveGroup;
|
||||
use ark_ff::{fields::PrimeField, UniformRand};
|
||||
use ark_serialize::{CanonicalSerialize, SerializationError};
|
||||
use ark_std::{marker::PhantomData, rand::Rng, vec::Vec};
|
||||
use sha3::{
|
||||
digest::{ExtendableOutput, Update, XofReader},
|
||||
Shake128,
|
||||
};
|
||||
pub type Commitment<C> = C;
|
||||
|
||||
pub struct Ciphertext<C: CurveGroup> {
|
||||
c1: C::Affine,
|
||||
c2: C::Affine,
|
||||
}
|
||||
|
||||
#[derive(Debug)]
|
||||
pub enum Error {
|
||||
SerializationError,
|
||||
}
|
||||
|
||||
impl<C: CurveGroup> Ciphertext<C> {
|
||||
fn serialize_compressed(&self) -> Result<(Vec<u8>, Vec<u8>), SerializationError> {
|
||||
let mut c1_bytes = Vec::new();
|
||||
let mut c2_bytes = Vec::new();
|
||||
|
||||
self.c1.serialize_compressed(&mut c1_bytes)?;
|
||||
self.c2.serialize_compressed(&mut c2_bytes)?;
|
||||
|
||||
Ok((c1_bytes, c2_bytes))
|
||||
}
|
||||
}
|
||||
|
||||
pub struct PoK<C: CurveGroup> {
|
||||
pub t: C,
|
||||
pub a: C,
|
||||
pub z: C::ScalarField,
|
||||
}
|
||||
|
||||
#[derive(Clone, Debug)]
|
||||
pub struct Params<C: CurveGroup> {
|
||||
pub g: C,
|
||||
pub h: C,
|
||||
}
|
||||
|
||||
pub struct ElGamalSigmaProtocol<C> {
|
||||
_c: PhantomData<C>,
|
||||
}
|
||||
|
||||
impl<C: CurveGroup> ElGamalSigmaProtocol<C> {
|
||||
pub fn prove<R: Rng + Sized>(
|
||||
s: C::ScalarField,
|
||||
params: Params<C>,
|
||||
mut rng: R,
|
||||
) -> (Commitment<C>, Ciphertext<C>, PoK<C>) {
|
||||
let r = C::ScalarField::rand(&mut rng);
|
||||
let c1 = params.g * r;
|
||||
let c2 = params.h * (s * r);
|
||||
|
||||
let ct: Ciphertext<C> = Ciphertext {
|
||||
c1: c1.into(),
|
||||
c2: c2.into(),
|
||||
};
|
||||
let c: Commitment<C> = params.g * s + params.h * s;
|
||||
|
||||
let k = C::ScalarField::rand(&mut rng);
|
||||
let t = params.g * k;
|
||||
let a = params.h * k;
|
||||
|
||||
let mut t_bytes = Vec::new();
|
||||
let mut a_bytes = Vec::new();
|
||||
t.serialize_compressed(&mut t_bytes)
|
||||
.expect("group element should exist");
|
||||
a.serialize_compressed(&mut a_bytes)
|
||||
.expect("group element should exist");
|
||||
|
||||
let mut inputs = Vec::new();
|
||||
inputs.push(t_bytes);
|
||||
inputs.push(a_bytes);
|
||||
let (c1_bytes, c2_bytes) = ct
|
||||
.serialize_compressed()
|
||||
.expect("group elements should exist");
|
||||
inputs.push(c1_bytes);
|
||||
inputs.push(c2_bytes);
|
||||
|
||||
let challenge: C::ScalarField =
|
||||
C::ScalarField::from_be_bytes_mod_order(&shake128(inputs.as_ref()));
|
||||
let z = k + challenge * s;
|
||||
(c, ct, PoK { t, a, z })
|
||||
}
|
||||
pub fn verify(
|
||||
commitment: Commitment<C>,
|
||||
ciphertext: Ciphertext<C>,
|
||||
proof: PoK<C>,
|
||||
params: Params<C>,
|
||||
) -> bool {
|
||||
let mut t_bytes = Vec::new();
|
||||
let mut a_bytes = Vec::new();
|
||||
proof
|
||||
.t
|
||||
.serialize_compressed(&mut t_bytes)
|
||||
.expect("group element should exist");
|
||||
proof
|
||||
.a
|
||||
.serialize_compressed(&mut a_bytes)
|
||||
.expect("group element should exist");
|
||||
|
||||
let mut inputs = Vec::new();
|
||||
inputs.push(t_bytes);
|
||||
inputs.push(a_bytes);
|
||||
let (c1_bytes, c2_bytes) = ciphertext
|
||||
.serialize_compressed()
|
||||
.expect("group element should exist");
|
||||
inputs.push(c1_bytes);
|
||||
inputs.push(c2_bytes);
|
||||
|
||||
let challenge: C::ScalarField =
|
||||
C::ScalarField::from_be_bytes_mod_order(&shake128(inputs.as_ref()));
|
||||
|
||||
let zg = params.g * proof.z;
|
||||
let zh = params.h * proof.z;
|
||||
|
||||
zg + zh == proof.t + proof.a + commitment * challenge
|
||||
}
|
||||
}
|
||||
|
||||
fn shake128(input: &[Vec<u8>]) -> [u8; 32] {
|
||||
let mut h = Shake128::default();
|
||||
for item in input.iter() {
|
||||
h.update(item);
|
||||
}
|
||||
let mut o = [0u8; 32];
|
||||
h.finalize_xof().read(&mut o);
|
||||
o
|
||||
}
|
||||
|
||||
#[cfg(test)]
|
||||
mod test {
|
||||
|
||||
use super::*;
|
||||
use ark_ec::Group;
|
||||
use ark_ed_on_bls12_381::EdwardsProjective as JubJub;
|
||||
use ark_std::{ops::Mul, test_rng};
|
||||
|
||||
#[test]
|
||||
pub fn prove_and_verify() {
|
||||
let mut rng = test_rng();
|
||||
let x = <JubJub as Group>::ScalarField::rand(&mut rng);
|
||||
let g: JubJub = JubJub::generator().into();
|
||||
let h: JubJub = g.mul(x).into();
|
||||
|
||||
let params = Params { g, h };
|
||||
|
||||
let (commitment, ciphertext, proof) =
|
||||
ElGamalSigmaProtocol::prove(x, params.clone(), test_rng());
|
||||
let result = ElGamalSigmaProtocol::verify(commitment, ciphertext, proof, params);
|
||||
assert_eq!(result, true);
|
||||
}
|
||||
|
||||
#[test]
|
||||
pub fn verify_fails_with_invalid_proof() {
|
||||
let mut rng = test_rng();
|
||||
let x = <JubJub as Group>::ScalarField::rand(&mut rng);
|
||||
let g: JubJub = JubJub::generator().into();
|
||||
let h: JubJub = g.mul(x).into();
|
||||
|
||||
let j = <JubJub as Group>::ScalarField::rand(&mut rng);
|
||||
let bad_proof = PoK {
|
||||
t: g.mul(j).into(),
|
||||
a: g.mul(j).into(),
|
||||
z: j,
|
||||
};
|
||||
|
||||
let params = Params { g, h };
|
||||
|
||||
let (commitment, ciphertext, _proof) =
|
||||
ElGamalSigmaProtocol::prove(x, params.clone(), test_rng());
|
||||
let result = ElGamalSigmaProtocol::verify(commitment, ciphertext, bad_proof, params);
|
||||
assert_eq!(result, false);
|
||||
}
|
||||
|
||||
#[test]
|
||||
pub fn verify_fails_with_invalid_commitment() {
|
||||
let mut rng = test_rng();
|
||||
let x = <JubJub as Group>::ScalarField::rand(&mut rng);
|
||||
let g: JubJub = JubJub::generator().into();
|
||||
let h: JubJub = g.mul(x).into();
|
||||
|
||||
let j = <JubJub as Group>::ScalarField::rand(&mut rng);
|
||||
let bad_commitment = g.mul(j).into();
|
||||
|
||||
let params = Params { g, h };
|
||||
|
||||
let (_commitment, ciphertext, proof) =
|
||||
ElGamalSigmaProtocol::prove(x, params.clone(), test_rng());
|
||||
let result = ElGamalSigmaProtocol::verify(bad_commitment, ciphertext, proof, params);
|
||||
assert_eq!(result, false);
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue