snarkos_node_router_messages/
puzzle_response.rs

1// Copyright (c) 2019-2025 Provable Inc.
2// This file is part of the snarkOS library.
3
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at:
7
8// http://www.apache.org/licenses/LICENSE-2.0
9
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15
16use 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    /// Returns the message name.
33    #[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}