use crate::error::*;
use crate::parser::bgp::attributes::AttributeParser;
use crate::parser::ReadUtils;
use bgp_models::mrt::tabledump::TableDumpMessage;
use bgp_models::network::{Afi, AsnLength, NetworkPrefix};
use byteorder::{ReadBytesExt, BE};
use std::io::Cursor;
use std::net::IpAddr;
pub fn parse_table_dump_message(
sub_type: u16,
data: &[u8],
) -> Result<TableDumpMessage, ParserError> {
let asn_len = AsnLength::Bits16;
let afi = match sub_type {
1 => Afi::Ipv4,
2 => Afi::Ipv6,
_ => {
return Err(ParserError::ParseError(format!(
"Invalid subtype found for TABLE_DUMP (V1) message: {}",
sub_type
)))
}
};
let mut input = Cursor::new(data);
let view_number = input.read_u16::<BE>()?;
let sequence_number = input.read_u16::<BE>()?;
let prefix = match &afi {
Afi::Ipv4 => input.read_ipv4_prefix().map(ipnet::IpNet::V4),
Afi::Ipv6 => input.read_ipv6_prefix().map(ipnet::IpNet::V6),
}?;
let status = input.read_8b()?;
let time = input.read_32b()? as u64;
let peer_address: IpAddr = input.read_address(&afi)?;
let peer_asn = input.read_asn(&asn_len)?;
let attribute_length = input.read_u16::<BE>()? as usize;
let attr_parser = AttributeParser::new(false);
let current_position = input.position() as usize;
let attr_data_slice = &input.into_inner()[current_position..];
assert_eq!(attr_data_slice.len(), attribute_length);
let attributes = attr_parser.parse_attributes(attr_data_slice, &asn_len, None, None, None)?;
Ok(TableDumpMessage {
view_number,
sequence_number,
prefix: NetworkPrefix::new(prefix, 0),
status,
originated_time: time,
peer_address,
peer_asn,
attributes,
})
}