//! This module defines various traits required by the users of the library to implement. use bellperson::{gadgets::num::AllocatedNum, ConstraintSystem, SynthesisError}; use core::{ fmt::Debug, ops::{Add, AddAssign, Mul, MulAssign, Sub, SubAssign}, }; use ff::{PrimeField, PrimeFieldBits}; use merlin::Transcript; use rug::Integer; /// Represents an element of a group pub trait Group: Clone + Copy + Debug + Eq + Sized + GroupOps + GroupOpsOwned + ScalarMul<::Scalar> + ScalarMulOwned<::Scalar> { /// A type representing an element of the base field of the group type Base: PrimeField + PrimeFieldBits; /// A type representing an element of the scalar field of the group type Scalar: PrimeField + PrimeFieldBits + ChallengeTrait; /// A type representing the compressed version of the group element type CompressedGroupElement: CompressedGroup; /// A type representing preprocessed group element type PreprocessedGroupElement; /// A type that represents a hash function that consumes elements /// from the base field and squeezes out elements of the scalar field type HashFunc: HashFuncTrait; /// A method to compute a multiexponentation fn vartime_multiscalar_mul( scalars: &[Self::Scalar], bases: &[Self::PreprocessedGroupElement], ) -> Self; /// Compresses the group element fn compress(&self) -> Self::CompressedGroupElement; /// Attempts to create a group element from a sequence of bytes, /// failing with a `None` if the supplied bytes do not encode the group element fn from_uniform_bytes(bytes: &[u8]) -> Option; /// Returns the affine coordinates (x, y, infinty) for the point fn to_coordinates(&self) -> (Self::Base, Self::Base, bool); /// Returns the order of the group as a big integer fn get_order() -> Integer; } /// Represents a compressed version of a group element pub trait CompressedGroup: Clone + Copy + Debug + Eq + Sized + Send + Sync + 'static { /// A type that holds the decompressed version of the compressed group element type GroupElement: Group; /// Decompresses the compressed group element fn decompress(&self) -> Option; /// Returns a byte array representing the compressed group element fn as_bytes(&self) -> &[u8]; } /// A helper trait to append different types to the transcript pub trait AppendToTranscriptTrait { /// appends the value to the transcript under the provided label fn append_to_transcript(&self, label: &'static [u8], transcript: &mut Transcript); } /// A helper trait to generate challenges using a transcript object pub trait ChallengeTrait { /// Returns a Scalar representing the challenge using the transcript fn challenge(label: &'static [u8], transcript: &mut Transcript) -> Self; } /// A helper trait that defines the behavior of a hash function that we use as an RO pub trait HashFuncTrait { /// A type representing constants/parameters associated with the hash function type Constants: HashFuncConstantsTrait; /// Initializes the hash function fn new(constants: Self::Constants) -> Self; /// Adds a scalar to the internal state fn absorb(&mut self, e: Base); /// Returns a random challenge by hashing the internal state fn get_challenge(&self) -> Scalar; /// Returns a hash of the internal state fn get_hash(&self) -> Scalar; } /// A helper trait that defines the constants associated with a hash function pub trait HashFuncConstantsTrait { /// produces constants/parameters associated with the hash function fn new() -> Self; } /// A helper trait for types with a group operation. pub trait GroupOps: Add + Sub + AddAssign + SubAssign { } impl GroupOps for T where T: Add + Sub + AddAssign + SubAssign { } /// A helper trait for references with a group operation. pub trait GroupOpsOwned: for<'r> GroupOps<&'r Rhs, Output> {} impl GroupOpsOwned for T where T: for<'r> GroupOps<&'r Rhs, Output> {} /// A helper trait for types implementing group scalar multiplication. pub trait ScalarMul: Mul + MulAssign {} impl ScalarMul for T where T: Mul + MulAssign {} /// A helper trait for references implementing group scalar multiplication. pub trait ScalarMulOwned: for<'r> ScalarMul<&'r Rhs, Output> {} impl ScalarMulOwned for T where T: for<'r> ScalarMul<&'r Rhs, Output> {} /// A helper trait for a step of the incremental computation (i.e., circuit for F) pub trait StepCircuit { /// Sythesize the circuit for a computation step and return variable /// that corresponds to the output of the step z_{i+1} fn synthesize>( &self, cs: &mut CS, z: AllocatedNum, ) -> Result, SynthesisError>; } impl AppendToTranscriptTrait for F { fn append_to_transcript(&self, label: &'static [u8], transcript: &mut Transcript) { transcript.append_message(label, self.to_repr().as_ref()); } } impl AppendToTranscriptTrait for [F] { fn append_to_transcript(&self, label: &'static [u8], transcript: &mut Transcript) { for s in self { s.append_to_transcript(label, transcript); } } }