use arbitrary::Arbitrary;
use get_size::GetSize;
use itertools::Itertools;
use serde::Deserialize;
use serde::Serialize;
use twenty_first::prelude::*;
use crate::error::ProofStreamError;
use crate::program::Program;
use crate::proof_stream::ProofStream;
#[derive(Debug, Clone, Eq, PartialEq, Serialize, Deserialize, GetSize, BFieldCodec, Arbitrary)]
pub struct Proof(pub Vec<BFieldElement>);
impl Proof {
pub fn padded_height(&self) -> Result<usize, ProofStreamError> {
let proof_stream = ProofStream::try_from(self)?;
let proof_items = proof_stream.items.into_iter();
let log_2_padded_heights = proof_items
.filter_map(|item| item.try_into_log2_padded_height().ok())
.collect_vec();
if log_2_padded_heights.is_empty() {
return Err(ProofStreamError::NoLog2PaddedHeight);
}
if log_2_padded_heights.len() > 1 {
return Err(ProofStreamError::TooManyLog2PaddedHeights);
}
Ok(1 << log_2_padded_heights[0])
}
}
#[derive(
Debug, Clone, Eq, PartialEq, Hash, Serialize, Deserialize, GetSize, BFieldCodec, Arbitrary,
)]
pub struct Claim {
pub program_digest: Digest,
pub input: Vec<BFieldElement>,
pub output: Vec<BFieldElement>,
}
impl Claim {
pub fn new(program_digest: Digest) -> Self {
Self {
program_digest,
input: vec![],
output: vec![],
}
}
#[must_use]
pub fn about_program(program: &Program) -> Self {
Self::new(program.hash::<Tip5>())
}
#[must_use]
pub fn with_input(mut self, input: Vec<BFieldElement>) -> Self {
self.input = input;
self
}
#[must_use]
pub fn with_output(mut self, output: Vec<BFieldElement>) -> Self {
self.output = output;
self
}
}
#[cfg(test)]
mod tests {
use assert2::assert;
use proptest::collection::vec;
use proptest::prelude::*;
use proptest_arbitrary_interop::arb;
use test_strategy::proptest;
use crate::proof_item::ProofItem;
use super::*;
impl Default for Claim {
fn default() -> Self {
Self::new(Digest::default())
}
}
#[proptest]
fn decode_proof(#[strategy(arb())] proof: Proof) {
let encoded = proof.encode();
let decoded = *Proof::decode(&encoded).unwrap();
prop_assert_eq!(proof, decoded);
}
#[proptest]
fn decode_claim(#[strategy(arb())] claim: Claim) {
let encoded = claim.encode();
let decoded = *Claim::decode(&encoded).unwrap();
prop_assert_eq!(claim, decoded);
}
#[proptest(cases = 10)]
fn proof_with_no_log_2_padded_height_gives_err(#[strategy(arb())] root: Digest) {
let mut proof_stream = ProofStream::new();
proof_stream.enqueue(ProofItem::MerkleRoot(root));
let proof: Proof = proof_stream.into();
let maybe_padded_height = proof.padded_height();
assert!(maybe_padded_height.is_err());
}
#[proptest(cases = 10)]
fn proof_with_multiple_log_2_padded_height_gives_err(#[strategy(arb())] root: Digest) {
let mut proof_stream = ProofStream::new();
proof_stream.enqueue(ProofItem::Log2PaddedHeight(8));
proof_stream.enqueue(ProofItem::MerkleRoot(root));
proof_stream.enqueue(ProofItem::Log2PaddedHeight(7));
let proof: Proof = proof_stream.into();
let maybe_padded_height = proof.padded_height();
assert!(maybe_padded_height.is_err());
}
#[proptest]
fn decoding_arbitrary_proof_data_does_not_panic(
#[strategy(vec(arb(), 0..1_000))] proof_data: Vec<BFieldElement>,
) {
let _ = Proof::decode(&proof_data);
}
}