snarkos_node_router_messages/
puzzle_response.rs1use super::*;
17
18use snarkvm::{
19 ledger::narwhal::Data,
20 prelude::{FromBytes, ToBytes},
21};
22
23use std::borrow::Cow;
24
25#[derive(Clone, Debug, PartialEq, Eq)]
26pub struct PuzzleResponse<N: Network> {
27 pub epoch_hash: N::BlockHash,
28 pub block_header: Data<Header<N>>,
29}
30
31impl<N: Network> MessageTrait for PuzzleResponse<N> {
32 #[inline]
34 fn name(&self) -> Cow<'static, str> {
35 "PuzzleResponse".into()
36 }
37}
38
39impl<N: Network> ToBytes for PuzzleResponse<N> {
40 fn write_le<W: io::Write>(&self, mut writer: W) -> io::Result<()> {
41 self.epoch_hash.write_le(&mut writer)?;
42 self.block_header.write_le(&mut writer)
43 }
44}
45
46impl<N: Network> FromBytes for PuzzleResponse<N> {
47 fn read_le<R: io::Read>(mut reader: R) -> io::Result<Self> {
48 Ok(Self { epoch_hash: N::BlockHash::read_le(&mut reader)?, block_header: Data::read_le(reader)? })
49 }
50}
51
52#[cfg(test)]
53pub mod prop_tests {
54 use crate::{PuzzleResponse, challenge_response::prop_tests::any_genesis_header};
55 use snarkvm::{
56 console::prelude::{FromBytes, ToBytes},
57 ledger::narwhal::Data,
58 prelude::{Network, Rng, TestRng},
59 };
60
61 use bytes::{Buf, BufMut, BytesMut};
62 use proptest::prelude::{BoxedStrategy, Strategy, any};
63 use test_strategy::proptest;
64
65 type CurrentNetwork = snarkvm::prelude::MainnetV0;
66
67 pub fn any_epoch_hash() -> BoxedStrategy<<CurrentNetwork as Network>::BlockHash> {
68 any::<u64>()
69 .prop_map(|seed| {
70 let mut rng = TestRng::fixed(seed);
71 rng.r#gen()
72 })
73 .boxed()
74 }
75
76 pub fn any_puzzle_response() -> BoxedStrategy<PuzzleResponse<CurrentNetwork>> {
77 (any_epoch_hash(), any_genesis_header())
78 .prop_map(|(epoch_hash, bh)| PuzzleResponse { epoch_hash, block_header: Data::Object(bh) })
79 .boxed()
80 }
81
82 #[proptest]
83 fn puzzle_response_roundtrip(#[strategy(any_puzzle_response())] original: PuzzleResponse<CurrentNetwork>) {
84 let mut buf = BytesMut::default().writer();
85 PuzzleResponse::write_le(&original, &mut buf).unwrap();
86
87 let deserialized: PuzzleResponse<CurrentNetwork> = PuzzleResponse::read_le(buf.into_inner().reader()).unwrap();
88 assert_eq!(original.epoch_hash, deserialized.epoch_hash);
89 assert_eq!(
90 original.block_header.deserialize_blocking().unwrap(),
91 deserialized.block_header.deserialize_blocking().unwrap(),
92 );
93 }
94}