use crate::{Address, Flags, ICMPExtension, Timeval, WartsSized};
use deku::prelude::*;
#[derive(Debug, PartialEq, DekuRead, DekuWrite)]
#[deku(ctx = "endian: deku::ctx::Endian", endian = "endian", type = "u8")]
pub enum TraceType {
ICMPEcho = 0x01,
UDP = 0x02,
TCP = 0x03,
ICMPEchoParis = 0x04,
UDPParis = 0x05,
TCPAck = 0x06,
}
#[derive(Debug, PartialEq, DekuRead, DekuWrite)]
#[deku(ctx = "endian: deku::ctx::Endian", endian = "endian", type = "u8")]
pub enum TraceGapAction {
None = 0x00,
Stop = 0x01,
LastDitch = 0x02,
}
#[derive(Debug, PartialEq, DekuRead, DekuWrite)]
#[deku(ctx = "endian: deku::ctx::Endian", endian = "endian", type = "u8")]
pub enum TraceStopReason {
None = 0x00,
Completed = 0x01,
Unreach = 0x02,
ICMP = 0x03,
Loop = 0x04,
GapLimit = 0x05,
Error = 0x06,
HopLimit = 0x07,
GSS = 0x08,
Halted = 0x09,
}
#[derive(Debug, PartialEq, DekuRead, DekuWrite)]
#[deku(ctx = "endian: deku::ctx::Endian", endian = "endian")]
pub struct Traceroute {
pub length: u32,
pub flags: Flags,
#[deku(cond = "flags.any()")]
pub param_length: Option<u16>,
#[deku(cond = "flags.get(1)")]
pub list_id: Option<u32>,
#[deku(cond = "flags.get(2)")]
pub cycle_id: Option<u32>,
#[deku(cond = "flags.get(3)")]
pub src_addr_id: Option<u32>,
#[deku(cond = "flags.get(4)")]
pub dst_addr_id: Option<u32>,
#[deku(cond = "flags.get(5)")]
pub start_time: Option<Timeval>,
#[deku(cond = "flags.get(6)")]
pub stop_reason: Option<TraceStopReason>,
#[deku(cond = "flags.get(7)")]
pub stop_data: Option<u8>,
#[deku(cond = "flags.get(8)")]
pub trace_flags: Option<Flags>,
#[deku(cond = "flags.get(9)")]
pub attempts: Option<u8>,
#[deku(cond = "flags.get(10)")]
pub hop_limit: Option<u8>,
#[deku(cond = "flags.get(11)")]
pub trace_type: Option<TraceType>,
#[deku(cond = "flags.get(12)")]
pub probe_size: Option<u16>,
#[deku(cond = "flags.get(13)")]
pub src_port: Option<u16>,
#[deku(cond = "flags.get(14)")]
pub dst_port: Option<u16>,
#[deku(cond = "flags.get(15)")]
pub first_ttl: Option<u8>,
#[deku(cond = "flags.get(16)")]
pub ip_tos: Option<u8>,
#[deku(cond = "flags.get(17)")]
pub timeout_sec: Option<u8>,
#[deku(cond = "flags.get(18)")]
pub allowed_loops: Option<u8>,
#[deku(cond = "flags.get(19)")]
pub hops_probed: Option<u16>,
#[deku(cond = "flags.get(20)")]
pub gap_limit: Option<u8>,
#[deku(cond = "flags.get(21)")]
pub gap_limit_action: Option<TraceGapAction>,
#[deku(cond = "flags.get(22)")]
pub loop_action: Option<u8>,
#[deku(cond = "flags.get(23)")]
pub probes_sent: Option<u16>,
#[deku(cond = "flags.get(24)")]
pub interval_csec: Option<u8>,
#[deku(cond = "flags.get(25)")]
pub confidence_level: Option<u8>,
#[deku(cond = "flags.get(26)")]
pub src_addr: Option<Address>,
#[deku(cond = "flags.get(27)")]
pub dst_addr: Option<Address>,
#[deku(cond = "flags.get(28)")]
pub user_id: Option<u32>,
#[deku(cond = "flags.get(29)")]
pub ip_offset: Option<u16>,
#[deku(cond = "flags.get(30)")]
pub router_addr: Option<Address>,
pub hop_count: u16,
#[deku(count = "hop_count")]
pub hops: Vec<TraceProbe>,
#[deku(assert_eq = "0")]
pub eof: u16,
}
#[derive(Debug, PartialEq, DekuRead, DekuWrite)]
#[deku(ctx = "endian: deku::ctx::Endian", endian = "endian")]
pub struct TraceProbe {
pub flags: Flags,
#[deku(cond = "flags.any()")]
pub param_length: Option<u16>,
#[deku(cond = "flags.get(1)")]
pub addr_id: Option<u32>,
#[deku(cond = "flags.get(2)")]
pub probe_ttl: Option<u8>,
#[deku(cond = "flags.get(3)")]
pub reply_ttl: Option<u8>,
#[deku(cond = "flags.get(4)")]
pub hop_flags: Option<u8>,
#[deku(cond = "flags.get(5)")]
pub probe_id: Option<u8>,
#[deku(cond = "flags.get(6)")]
pub rtt_usec: Option<u32>,
#[deku(cond = "flags.get(7)")]
pub icmp_type: Option<u8>,
#[deku(cond = "flags.get(7)")]
pub icmp_code: Option<u8>,
#[deku(cond = "flags.get(8)")]
pub probe_size: Option<u16>,
#[deku(cond = "flags.get(9)")]
pub reply_size: Option<u16>,
#[deku(cond = "flags.get(10)")]
pub reply_ip_id: Option<u16>,
#[deku(cond = "flags.get(11)")]
pub reply_ip_tos: Option<u8>,
#[deku(cond = "flags.get(12)")]
pub next_hop_mtu: Option<u16>,
#[deku(cond = "flags.get(13)")]
pub quoted_length: Option<u16>,
#[deku(cond = "flags.get(14)")]
pub quoted_ttl: Option<u8>,
#[deku(cond = "flags.get(15)")]
pub reply_tcp_flags: Option<u8>,
#[deku(cond = "flags.get(16)")]
pub quoted_tos: Option<u8>,
#[deku(cond = "flags.get(17)")]
pub icmp_extensions_length: Option<u16>,
#[deku(
cond = "flags.get(17)",
count = "if icmp_extensions_length.unwrap() > 0 { 1 } else { 0 }"
)]
pub icmp_extensions: Vec<ICMPExtension>,
#[deku(cond = "flags.get(18)")]
pub addr: Option<Address>,
#[deku(cond = "flags.get(19)")]
pub tx: Option<Timeval>,
}
impl Traceroute {
pub fn finalize(mut self) -> Self {
let mut flags = Vec::new();
let mut param_length = 0;
push_flag!(flags, param_length, 1, self.list_id);
push_flag!(flags, param_length, 2, self.cycle_id);
push_flag!(flags, param_length, 3, self.src_addr_id);
push_flag!(flags, param_length, 4, self.dst_addr_id);
push_flag!(flags, param_length, 5, self.start_time);
push_flag!(flags, param_length, 6, self.stop_reason);
push_flag!(flags, param_length, 7, self.stop_data);
push_flag!(flags, param_length, 8, self.trace_flags);
push_flag!(flags, param_length, 9, self.attempts);
push_flag!(flags, param_length, 10, self.hop_limit);
push_flag!(flags, param_length, 11, self.trace_type);
push_flag!(flags, param_length, 12, self.probe_size);
push_flag!(flags, param_length, 13, self.src_port);
push_flag!(flags, param_length, 14, self.dst_port);
push_flag!(flags, param_length, 15, self.first_ttl);
push_flag!(flags, param_length, 16, self.ip_tos);
push_flag!(flags, param_length, 17, self.timeout_sec);
push_flag!(flags, param_length, 18, self.allowed_loops);
push_flag!(flags, param_length, 19, self.hops_probed);
push_flag!(flags, param_length, 20, self.gap_limit);
push_flag!(flags, param_length, 21, self.gap_limit_action);
push_flag!(flags, param_length, 22, self.loop_action);
push_flag!(flags, param_length, 23, self.probes_sent);
push_flag!(flags, param_length, 24, self.interval_csec);
push_flag!(flags, param_length, 25, self.confidence_level);
push_flag!(flags, param_length, 26, self.src_addr);
push_flag!(flags, param_length, 27, self.dst_addr);
push_flag!(flags, param_length, 28, self.user_id);
push_flag!(flags, param_length, 29, self.ip_offset);
push_flag!(flags, param_length, 30, self.router_addr);
self.flags = Flags::from(flags);
self.param_length = Some(param_length as u16);
self.hop_count = self.hops.len() as u16;
let hops_size: usize = self.hops.iter().map(|hop| hop.warts_size()).sum();
self.length = (self.flags.warts_size()
+ self.param_length.warts_size()
+ param_length
+ self.hop_count.warts_size()
+ hops_size
+ self.eof.warts_size()) as u32;
self
}
}
impl TraceProbe {
pub fn finalize(mut self) -> Self {
let mut flags = Vec::new();
let mut param_length = 0;
push_flag!(flags, param_length, 1, self.addr_id);
push_flag!(flags, param_length, 2, self.probe_ttl);
push_flag!(flags, param_length, 3, self.reply_ttl);
push_flag!(flags, param_length, 4, self.hop_flags);
push_flag!(flags, param_length, 5, self.probe_id);
push_flag!(flags, param_length, 6, self.rtt_usec);
push_flag!(flags, param_length, 7, self.icmp_type);
push_flag!(flags, param_length, 7, self.icmp_code);
push_flag!(flags, param_length, 8, self.probe_size);
push_flag!(flags, param_length, 9, self.reply_size);
push_flag!(flags, param_length, 10, self.reply_ip_id);
push_flag!(flags, param_length, 11, self.reply_ip_tos);
push_flag!(flags, param_length, 12, self.next_hop_mtu);
push_flag!(flags, param_length, 13, self.quoted_length);
push_flag!(flags, param_length, 14, self.quoted_ttl);
push_flag!(flags, param_length, 15, self.reply_tcp_flags);
push_flag!(flags, param_length, 16, self.quoted_tos);
push_flag!(flags, param_length, 17, self.icmp_extensions_length);
push_flag!(flags, param_length, 18, self.addr);
push_flag!(flags, param_length, 19, self.tx.as_ref());
self.flags = Flags::from(flags);
self.param_length = Some(param_length as u16);
self
}
pub fn rtt_ms(&self) -> Option<f64> {
self.rtt_usec.map(|x| x as f64 / 1000.0)
}
}
impl WartsSized for TraceType {
fn warts_size(&self) -> usize {
1
}
}
impl WartsSized for TraceGapAction {
fn warts_size(&self) -> usize {
1
}
}
impl WartsSized for TraceStopReason {
fn warts_size(&self) -> usize {
1
}
}
impl WartsSized for TraceProbe {
fn warts_size(&self) -> usize {
self.flags.warts_size()
+ self.param_length.warts_size()
+ self.param_length.unwrap() as usize
}
}