use toxcore::binary_io::*;
use toxcore::crypto_core::*;
use toxcore::onion::packet::{
IpPort,
ONION_MAX_PACKET_SIZE,
ONION_RETURN_1_SIZE,
SIZE_IPPORT
};
use nom::rest;
const ONION_MIN_PAYLOAD_SIZE: usize = (SIZE_IPPORT + MACBYTES) * 2 + PUBLICKEYBYTES;
const ONION_MAX_PAYLOAD_SIZE: usize = ONION_MAX_PACKET_SIZE - (1 + NONCEBYTES + PUBLICKEYBYTES + ONION_RETURN_1_SIZE);
#[derive(Debug, PartialEq, Clone)]
pub struct OnionRequest {
pub nonce: Nonce,
pub ip_port: IpPort,
pub temporary_pk: PublicKey,
pub payload: Vec<u8>
}
impl FromBytes for OnionRequest {
named!(from_bytes<OnionRequest>, do_parse!(
tag!("\x08") >>
nonce: call!(Nonce::from_bytes) >>
ip_port: call!(IpPort::from_bytes) >>
temporary_pk: call!(PublicKey::from_bytes) >>
payload: verify!(
rest,
|payload: &[u8]| payload.len() >= ONION_MIN_PAYLOAD_SIZE && payload.len() <= ONION_MAX_PAYLOAD_SIZE
) >>
(OnionRequest { nonce, ip_port, temporary_pk, payload: payload.to_vec() })
));
}
impl ToBytes for OnionRequest {
fn to_bytes<'a>(&self, buf: (&'a mut [u8], usize)) -> Result<(&'a mut [u8], usize), GenError> {
do_gen!(buf,
gen_cond!(
self.payload.len() < ONION_MIN_PAYLOAD_SIZE || self.payload.len() > ONION_MAX_PAYLOAD_SIZE,
|buf| gen_error(buf, 0)
) >>
gen_be_u8!(0x08) >>
gen_slice!(self.nonce.as_ref()) >>
gen_call!(|buf, ip_port| IpPort::to_bytes(ip_port, buf), &self.ip_port) >>
gen_slice!(self.temporary_pk.as_ref()) >>
gen_slice!(self.payload)
)
}
}
#[cfg(test)]
mod test {
use super::*;
use toxcore::onion::packet::ProtocolType;
encode_decode_test!(
onion_request_encode_decode,
OnionRequest {
nonce: gen_nonce(),
ip_port: IpPort {
protocol: ProtocolType::TCP,
ip_addr: "5.6.7.8".parse().unwrap(),
port: 12345,
},
temporary_pk: gen_keypair().0,
payload: vec![42; ONION_MIN_PAYLOAD_SIZE]
}
);
}