(tests) rcode parsing tests

Check if valid and invalid response codes are parsed correctly.
This commit is contained in:
ddidderr 2022-04-04 01:36:51 +02:00
parent 6a60951487
commit b998d47164

View File

@ -74,7 +74,7 @@ impl TryFrom<u8> for DNSOpCode {
/// ///
/// 65,535 /// 65,535
/// 0xFFFF Reserved; can only be allocated by Standards Action. /// 0xFFFF Reserved; can only be allocated by Standards Action.
#[derive(Debug)] #[derive(Debug, PartialEq, Copy, Clone)]
pub enum DNSRCode { pub enum DNSRCode {
NoError = 0, NoError = 0,
FormErr = 1, FormErr = 1,
@ -110,7 +110,7 @@ impl TryFrom<u8> for DNSRCode {
} }
} }
#[derive(Debug, PartialEq)] #[derive(Debug, PartialEq, Clone, Copy)]
pub enum DNSMessageType { pub enum DNSMessageType {
Query, Query,
Response, Response,
@ -125,7 +125,7 @@ impl From<bool> for DNSMessageType {
} }
} }
#[derive(Debug)] #[derive(Debug, Clone)]
pub struct DNSHeader { pub struct DNSHeader {
/// used by the requester to match up replies to outstanding queries /// used by the requester to match up replies to outstanding queries
pub id: u16, pub id: u16,
@ -155,7 +155,7 @@ pub struct DNSHeader {
pub additional_count: u16, pub additional_count: u16,
} }
#[derive(Debug)] #[derive(Debug, Clone)]
pub struct DNSQuery { pub struct DNSQuery {
pub hdr: DNSHeader, pub hdr: DNSHeader,
pub name: String, pub name: String,
@ -219,7 +219,7 @@ impl DNSHeader {
} }
} }
#[derive(Debug, PartialEq)] #[derive(Debug, Clone, PartialEq)]
pub enum DNSParseError { pub enum DNSParseError {
DatagramTooShort, DatagramTooShort,
DNSOpCodeInvalid, DNSOpCodeInvalid,
@ -228,7 +228,9 @@ pub enum DNSParseError {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*; use super::*;
#[test] #[test]
fn parse_dns_header_too_short() { fn parse_dns_header_too_short() {
// minimal header with 1 byte missing // minimal header with 1 byte missing
@ -264,20 +266,19 @@ mod tests {
let mut dns_query = [ let mut dns_query = [
0xff, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xff, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
]; ];
let valid_opcodes = [0, 1, 2, 4, 5]; let valid_opcodes = [
let valid_parsed_opcodes = [ (0, DNSOpCode::Query),
DNSOpCode::Query, (1, DNSOpCode::IQuery),
DNSOpCode::IQuery, (2, DNSOpCode::Status),
DNSOpCode::Status, (4, DNSOpCode::Notify),
DNSOpCode::Notify, (5, DNSOpCode::Update),
DNSOpCode::Update,
]; ];
for (idx, opcode) in valid_opcodes.iter().enumerate() { for (opcode, parsed_opcode) in valid_opcodes {
dns_query[2] = opcode << 3; dns_query[2] = opcode << 3;
let parse_result = DNSHeader::from_udp_datagram(&dns_query); let parse_result = DNSHeader::from_udp_datagram(&dns_query);
assert_eq!( assert_eq!(
parse_result?.opcode, valid_parsed_opcodes[idx], parse_result?.opcode, parsed_opcode,
"query: {:02x?}, opcode: {}", "query: {:02x?}, opcode: {}",
dns_query, opcode dns_query, opcode
); );
@ -304,4 +305,61 @@ mod tests {
assert_eq!(parse_result?.message_type, DNSMessageType::Response); assert_eq!(parse_result?.message_type, DNSMessageType::Response);
Ok(()) Ok(())
} }
#[test]
fn parse_dns_header_response_code_invalid() {
let mut dns_query = [
0xff, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
];
let invalid_rcodes = [11, 12, 13, 14, 15];
for rcode in invalid_rcodes {
dns_query[3] = (rcode as u8) << 0;
let parse_result = DNSHeader::from_udp_datagram(&dns_query);
let parse_result_clone = parse_result.clone();
assert_eq!(
parse_result.err(),
Some(DNSParseError::DNSRCodeInvalid),
"query: {:02x?}, rcode: {}, parsed_rcode: {:?}",
dns_query,
rcode,
parse_result_clone.map(|x| x.response_code).unwrap()
);
}
}
#[test]
fn parse_dns_header_response_code_valid() -> Result<(), DNSParseError> {
let mut dns_query = [
0xff, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
];
// for now we only support RCodes 0 to 10 (inclusive)
let valid_rcodes = [
(0, DNSRCode::NoError),
(1, DNSRCode::FormErr),
(2, DNSRCode::ServFail),
(3, DNSRCode::NXDomain),
(4, DNSRCode::NotImp),
(5, DNSRCode::Refused),
(6, DNSRCode::YXDomain),
(7, DNSRCode::YXRRSet),
(8, DNSRCode::NXRRSet),
(9, DNSRCode::NotAuth),
(10, DNSRCode::NotZone),
];
for (rcode, parsed_rcode) in valid_rcodes {
dns_query[3] = rcode << 0;
let parse_result = DNSHeader::from_udp_datagram(&dns_query);
assert_eq!(
parse_result?.response_code, parsed_rcode,
"query: {:02x?}, rcode: {}, parsed_rcode: {:?}",
dns_query, rcode, parsed_rcode
);
}
Ok(())
}
} }