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 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136
// Copyright (C) 2019-2023 Aleo Systems Inc.
// This file is part of the snarkVM library.
// The snarkVM library is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// The snarkVM library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with the snarkVM library. If not, see <https://www.gnu.org/licenses/>.
use super::*;
impl<N: Network> FromBytes for Request<N> {
/// Reads the request from a buffer.
fn read_le<R: Read>(mut reader: R) -> IoResult<Self> {
// Read the version.
let version = u16::read_le(&mut reader)?;
// Ensure the version is valid.
if version != 0 {
return Err(error("Invalid verifying key version"));
}
// Read the caller.
let caller = FromBytes::read_le(&mut reader)?;
// Read the network ID.
let network_id = FromBytes::read_le(&mut reader)?;
// Read the program ID.
let program_id = FromBytes::read_le(&mut reader)?;
// Read the function name.
let function_name = FromBytes::read_le(&mut reader)?;
// Read the number of inputs.
let inputs_len = u16::read_le(&mut reader)?;
// Read the input IDs.
let input_ids = (0..inputs_len).map(|_| FromBytes::read_le(&mut reader)).collect::<Result<Vec<_>, _>>()?;
// Read the inputs.
let inputs = (0..inputs_len).map(|_| FromBytes::read_le(&mut reader)).collect::<Result<Vec<_>, _>>()?;
// Read the signature.
let signature = FromBytes::read_le(&mut reader)?;
// Read the tag secret key.
let sk_tag = FromBytes::read_le(&mut reader)?;
// Read the transition view key.
let tvk = FromBytes::read_le(&mut reader)?;
// Read the transition secret key.
let tsk = FromBytes::read_le(&mut reader)?;
// Read the transition commitment.
let tcm = FromBytes::read_le(&mut reader)?;
Ok(Self::from((
caller,
network_id,
program_id,
function_name,
input_ids,
inputs,
signature,
sk_tag,
tvk,
tsk,
tcm,
)))
}
}
impl<N: Network> ToBytes for Request<N> {
/// Writes the request to a buffer.
fn write_le<W: Write>(&self, mut writer: W) -> IoResult<()> {
// Write the version.
0u16.write_le(&mut writer)?;
// Write the caller.
self.caller.write_le(&mut writer)?;
// Write the network ID.
self.network_id.write_le(&mut writer)?;
// Write the program ID.
self.program_id.write_le(&mut writer)?;
// Write the function name.
self.function_name.write_le(&mut writer)?;
// Ensure the input IDs and inputs are the same length.
if self.input_ids.len() != self.inputs.len() {
return Err(error("Invalid request: mismatching number of input IDs and inputs"));
}
// Write the number of inputs.
u16::try_from(self.input_ids.len())
.or_halt_with::<N>("Request inputs length exceeds u16")
.write_le(&mut writer)?;
// Write the input IDs.
for input_id in &self.input_ids {
input_id.write_le(&mut writer)?;
}
// Write the inputs.
for input in &self.inputs {
input.write_le(&mut writer)?;
}
// Write the signature.
self.signature.write_le(&mut writer)?;
// Write the tag secret key.
self.sk_tag.write_le(&mut writer)?;
// Write the transition view key.
self.tvk.write_le(&mut writer)?;
// Write the transition secret key.
self.tsk.write_le(&mut writer)?;
// Write the transition commitment.
self.tcm.write_le(&mut writer)
}
}
#[cfg(test)]
mod tests {
use super::*;
use snarkvm_console_network::Testnet3;
type CurrentNetwork = Testnet3;
#[test]
fn test_bytes() {
let mut rng = TestRng::default();
for expected in test_helpers::sample_requests(&mut rng).into_iter() {
// Check the byte representation.
let expected_bytes = expected.to_bytes_le().unwrap();
assert_eq!(expected, Request::read_le(&expected_bytes[..]).unwrap());
assert!(Request::<CurrentNetwork>::read_le(&expected_bytes[1..]).is_err());
}
}
}