-
Notifications
You must be signed in to change notification settings - Fork 154
/
Copy pathdefault_transcript.rs
119 lines (100 loc) · 3.43 KB
/
default_transcript.rs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
use super::is_transcript::IsTranscript;
use core::marker::PhantomData;
use lambdaworks_math::{
field::{element::FieldElement, traits::HasDefaultTranscript},
traits::ByteConversion,
};
use rand_chacha::{rand_core::SeedableRng, ChaCha20Rng};
use sha3::{Digest, Keccak256};
pub struct DefaultTranscript<F: HasDefaultTranscript> {
hasher: Keccak256,
phantom: PhantomData<F>,
}
impl<F> DefaultTranscript<F>
where
F: HasDefaultTranscript,
FieldElement<F>: ByteConversion,
{
pub fn new(data: &[u8]) -> Self {
let mut res = Self {
hasher: Keccak256::new(),
phantom: PhantomData,
};
res.append_bytes(data);
res
}
pub fn sample(&mut self) -> [u8; 32] {
let mut result_hash = [0_u8; 32];
result_hash.copy_from_slice(&self.hasher.finalize_reset());
result_hash.reverse();
self.hasher.update(result_hash);
result_hash
}
}
impl<F> Default for DefaultTranscript<F>
where
F: HasDefaultTranscript,
FieldElement<F>: ByteConversion,
{
fn default() -> Self {
Self::new(&[])
}
}
impl<F> IsTranscript<F> for DefaultTranscript<F>
where
F: HasDefaultTranscript,
FieldElement<F>: ByteConversion,
{
fn append_bytes(&mut self, new_bytes: &[u8]) {
self.hasher.update(new_bytes);
}
fn append_field_element(&mut self, element: &FieldElement<F>) {
self.append_bytes(&element.to_bytes_be());
}
fn state(&self) -> [u8; 32] {
self.hasher.clone().finalize().into()
}
fn sample_field_element(&mut self) -> FieldElement<F> {
let mut rng = <ChaCha20Rng as SeedableRng>::from_seed(self.sample());
F::get_random_field_element_from_rng(&mut rng)
}
fn sample_u64(&mut self, upper_bound: u64) -> u64 {
u64::from_be_bytes(self.state()[..8].try_into().unwrap()) % upper_bound
}
}
#[cfg(test)]
mod tests {
use super::*;
use alloc::vec::Vec;
use lambdaworks_math::elliptic_curve::short_weierstrass::curves::bls12_381::default_types::FrField;
#[test]
fn basic_challenge() {
let mut transcript = DefaultTranscript::<FrField>::default();
let point_a: Vec<u8> = vec![0xFF, 0xAB];
let point_b: Vec<u8> = vec![0xDD, 0x8C, 0x9D];
transcript.append_bytes(&point_a); // point_a
transcript.append_bytes(&point_b); // point_a + point_b
let challenge1 = transcript.sample(); // Hash(point_a + point_b)
assert_eq!(
challenge1,
[
0x0c, 0x2b, 0xd8, 0xcf, 0x2d, 0x71, 0xe0, 0x0a, 0xce, 0xa3, 0xbd, 0x5d, 0xc7, 0x9f,
0x4f, 0x93, 0xed, 0x57, 0x42, 0xd0, 0x23, 0xbd, 0x47, 0xc9, 0x04, 0xc2, 0x67, 0x9d,
0xbc, 0xfa, 0x7c, 0xa7
]
);
let point_c: Vec<u8> = vec![0xFF, 0xAB];
let point_d: Vec<u8> = vec![0xDD, 0x8C, 0x9D];
transcript.append_bytes(&point_c); // Hash(point_a + point_b) + point_c
transcript.append_bytes(&point_d); // Hash(point_a + point_b) + point_c + point_d
let challenge2 = transcript.sample(); // Hash(Hash(point_a + point_b) + point_c + point_d)
assert_eq!(
challenge2,
[
0x81, 0x61, 0x51, 0xc5, 0x7e, 0xcb, 0x45, 0xd5, 0x17, 0x1a, 0x3c, 0x2e, 0x38, 0x04,
0x5d, 0xfb, 0x3a, 0x3d, 0x33, 0x8a, 0x22, 0xaf, 0xf8, 0x60, 0x85, 0xb9, 0x54, 0x3f,
0xf8, 0x32, 0x32, 0xbc
]
);
}
}