bellperson 0.27.0

zk-SNARK library
Documentation
use std::marker::PhantomData;

use ff::{Field, PrimeField};
use serde::Serialize;
use sha2::{Digest, Sha256};

use pairing::Engine;

const PREFIX: &str = "snarkpack-v1";

#[derive(Debug)]
pub struct Transcript<E: Engine> {
    hasher: Sha256,
    buffer: Vec<u8>,
    _e: PhantomData<E>,
}

/// A challenge derived from the transcript.
#[derive(Debug, Clone)]
pub struct Challenge<E: Engine>(E::Fr);

impl<E: Engine> Copy for Challenge<E> {}

impl<E: Engine> PartialEq for Challenge<E> {
    fn eq(&self, other: &Self) -> bool {
        self.0 == other.0
    }
}

impl<E: Engine> Eq for Challenge<E> {}

impl<E: Engine> std::ops::Deref for Challenge<E> {
    type Target = E::Fr;

    fn deref(&self) -> &Self::Target {
        &self.0
    }
}

impl<E: Engine> Serialize for Challenge<E>
where
    E::Fr: Serialize,
{
    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
    where
        S: serde::Serializer,
    {
        self.0.serialize(serializer)
    }
}

impl<E: Engine> Transcript<E> {
    pub fn new(application_tag: &str) -> Self {
        let mut hasher = sha2::Sha256::new();
        hasher.update(PREFIX);
        hasher.update(application_tag);

        Transcript {
            hasher,
            buffer: Vec::new(),
            _e: Default::default(),
        }
    }

    pub fn write<S: Serialize>(mut self, el: &S) -> Self {
        bincode::serialize_into(&mut self.buffer, el).expect("vec");
        self.hasher.update(&self.buffer);
        self.buffer.clear();
        self
    }

    pub fn into_bytes(self) -> Vec<u8> {
        self.hasher.finalize().to_vec()
    }

    /// Generate a challenge from the transcript.
    pub fn into_challenge(mut self) -> Challenge<E> {
        let repr_bits = <<E as Engine>::Fr as PrimeField>::Repr::default()
            .as_ref()
            .len()
            * 8;
        let shave_bits = repr_bits - E::Fr::NUM_BITS as usize;

        let mut counter_nonce: usize = 0;
        let one = E::Fr::ONE;
        let r = loop {
            counter_nonce += 1;
            self.hasher.update(&counter_nonce.to_be_bytes()[..]);
            let curr_state = self.hasher.clone();
            let digest = curr_state.finalize();

            let mut repr = <<E as Engine>::Fr as PrimeField>::Repr::default();
            repr.as_mut().copy_from_slice(&digest);

            // Mask away the digest bits which exceed the field size.
            *repr.as_mut().last_mut().unwrap() &= 0xff >> shave_bits;

            if let Some(c) = E::Fr::from_repr_vartime(repr) {
                if c == one {
                    continue;
                }
                if c.invert().is_some().into() {
                    break c;
                }
            }
        };

        Challenge(r)
    }
}

#[cfg(test)]
mod test {
    use super::*;
    use blstrs::{Bls12, G1Affine, G2Affine, Scalar as Fr};
    use ff::Field;
    use group::prime::PrimeCurveAffine;
    use pairing::{MillerLoopResult, MultiMillerLoop};

    #[test]
    fn test_transcript() {
        let mut t = Transcript::<Bls12>::new("test");
        let g1 = G1Affine::generator();
        let g2 = G2Affine::generator();
        let gt = <Bls12 as MultiMillerLoop>::multi_miller_loop(&[(&g1, &g2.into())])
            .final_exponentiation();

        t = t.write(&g1).write(&g2).write(&gt).write(&Fr::ONE);

        let c1 = t.into_challenge();

        let t2 = Transcript::new("test")
            .write(&g1)
            .write(&g2)
            .write(&gt)
            .write(&Fr::ONE);

        let c12 = t2.into_challenge();
        assert_eq!(c1, c12);
    }
}