diff options
author | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 19:33:14 +0000 |
---|---|---|
committer | Daniel Baumann <daniel.baumann@progress-linux.org> | 2024-04-07 19:33:14 +0000 |
commit | 36d22d82aa202bb199967e9512281e9a53db42c9 (patch) | |
tree | 105e8c98ddea1c1e4784a60a5a6410fa416be2de /third_party/rust/dns-parser/src | |
parent | Initial commit. (diff) | |
download | firefox-esr-36d22d82aa202bb199967e9512281e9a53db42c9.tar.xz firefox-esr-36d22d82aa202bb199967e9512281e9a53db42c9.zip |
Adding upstream version 115.7.0esr.upstream/115.7.0esr
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
Diffstat (limited to 'third_party/rust/dns-parser/src')
32 files changed, 2466 insertions, 0 deletions
diff --git a/third_party/rust/dns-parser/src/builder.rs b/third_party/rust/dns-parser/src/builder.rs new file mode 100644 index 0000000000..22aff0b962 --- /dev/null +++ b/third_party/rust/dns-parser/src/builder.rs @@ -0,0 +1,132 @@ +use byteorder::{ByteOrder, BigEndian, WriteBytesExt}; + +use {Opcode, ResponseCode, Header, QueryType, QueryClass}; + +/// Allows to build a DNS packet +/// +/// Both query and answer packets may be built with this interface, although, +/// much of functionality is not implemented yet. +#[derive(Debug)] +pub struct Builder { + buf: Vec<u8>, +} + +impl Builder { + /// Creates a new query + /// + /// Initially all sections are empty. You're expected to fill + /// the questions section with `add_question` + pub fn new_query(id: u16, recursion: bool) -> Builder { + let mut buf = Vec::with_capacity(512); + let head = Header { + id: id, + query: true, + opcode: Opcode::StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: recursion, + recursion_available: false, + authenticated_data: false, + checking_disabled: false, + response_code: ResponseCode::NoError, + questions: 0, + answers: 0, + nameservers: 0, + additional: 0, + }; + buf.extend([0u8; 12].iter()); + head.write(&mut buf[..12]); + Builder { buf: buf } + } + /// Adds a question to the packet + /// + /// # Panics + /// + /// * Answers, nameservers or additional section has already been written + /// * There are already 65535 questions in the buffer. + /// * When name is invalid + pub fn add_question(&mut self, qname: &str, prefer_unicast: bool, + qtype: QueryType, qclass: QueryClass) + -> &mut Builder + { + if &self.buf[6..12] != b"\x00\x00\x00\x00\x00\x00" { + panic!("Too late to add a question"); + } + self.write_name(qname); + self.buf.write_u16::<BigEndian>(qtype as u16).unwrap(); + let prefer_unicast: u16 = if prefer_unicast { 0x8000 } else { 0x0000 }; + self.buf.write_u16::<BigEndian>(qclass as u16 | prefer_unicast).unwrap(); + let oldq = BigEndian::read_u16(&self.buf[4..6]); + if oldq == 65535 { + panic!("Too many questions"); + } + BigEndian::write_u16(&mut self.buf[4..6], oldq+1); + self + } + fn write_name(&mut self, name: &str) { + for part in name.split('.') { + assert!(part.len() < 63); + let ln = part.len() as u8; + self.buf.push(ln); + self.buf.extend(part.as_bytes()); + } + self.buf.push(0); + } + /// Returns the final packet + /// + /// When packet is not truncated method returns `Ok(packet)`. If + /// packet is truncated the method returns `Err(packet)`. In both + /// cases the packet is fully valid. + /// + /// In the server implementation you may use + /// `x.build().unwrap_or_else(|x| x)`. + /// + /// In the client implementation it's probably unwise to send truncated + /// packet, as it doesn't make sense. Even panicking may be more + /// appropriate. + // TODO(tailhook) does the truncation make sense for TCP, and how + // to treat it for EDNS0? + pub fn build(mut self) -> Result<Vec<u8>,Vec<u8>> { + // TODO(tailhook) optimize labels + if self.buf.len() > 512 { + Header::set_truncated(&mut self.buf[..12]); + Err(self.buf) + } else { + Ok(self.buf) + } + } +} + +#[cfg(test)] +mod test { + use QueryType as QT; + use QueryClass as QC; + use super::Builder; + + #[test] + fn build_query() { + let mut bld = Builder::new_query(1573, true); + bld.add_question("example.com", false, QT::A, QC::IN); + let result = b"\x06%\x01\x00\x00\x01\x00\x00\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x00\x01"; + assert_eq!(&bld.build().unwrap()[..], &result[..]); + } + + #[test] + fn build_unicast_query() { + let mut bld = Builder::new_query(1573, true); + bld.add_question("example.com", true, QT::A, QC::IN); + let result = b"\x06%\x01\x00\x00\x01\x00\x00\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x80\x01"; + assert_eq!(&bld.build().unwrap()[..], &result[..]); + } + + #[test] + fn build_srv_query() { + let mut bld = Builder::new_query(23513, true); + bld.add_question("_xmpp-server._tcp.gmail.com", false, QT::SRV, QC::IN); + let result = b"[\xd9\x01\x00\x00\x01\x00\x00\x00\x00\x00\x00\ + \x0c_xmpp-server\x04_tcp\x05gmail\x03com\x00\x00!\x00\x01"; + assert_eq!(&bld.build().unwrap()[..], &result[..]); + } +} diff --git a/third_party/rust/dns-parser/src/enums.rs b/third_party/rust/dns-parser/src/enums.rs new file mode 100644 index 0000000000..9f58ddfec1 --- /dev/null +++ b/third_party/rust/dns-parser/src/enums.rs @@ -0,0 +1,299 @@ +use {Error}; +use rdata::Record; +use rdata::*; + +/// The TYPE value according to RFC 1035 +/// +/// All "EXPERIMENTAL" markers here are from the RFC +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub enum Type { + /// a host addresss + A = a::Record::TYPE, + /// an authoritative name server + NS = ns::Record::TYPE, + /// a mail forwarder (Obsolete - use MX) + MF = mf::Record::TYPE, + /// the canonical name for an alias + CNAME = cname::Record::TYPE, + /// marks the start of a zone of authority + SOA = soa::Record::TYPE, + /// a mailbox domain name (EXPERIMENTAL) + MB = mb::Record::TYPE, + /// a mail group member (EXPERIMENTAL) + MG = mg::Record::TYPE, + /// a mail rename domain name (EXPERIMENTAL) + MR = mr::Record::TYPE, + /// a null RR (EXPERIMENTAL) + NULL = null::Record::TYPE, + /// a well known service description + WKS = wks::Record::TYPE, + /// a domain name pointer + PTR = ptr::Record::TYPE, + /// host information + HINFO = hinfo::Record::TYPE, + /// mailbox or mail list information + MINFO = minfo::Record::TYPE, + /// mail exchange + MX = mx::Record::TYPE, + /// text strings + TXT = txt::Record::TYPE, + /// IPv6 host address (RFC 2782) + AAAA = aaaa::Record::TYPE, + /// service record (RFC 2782) + SRV = srv::Record::TYPE, + /// EDNS0 options (RFC 6891) + OPT = opt::Record::TYPE, + /// next secure record (RFC 4034, RFC 6762) + NSEC = nsec::Record::TYPE, +} + +/// The QTYPE value according to RFC 1035 +/// +/// All "EXPERIMENTAL" markers here are from the RFC +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +#[cfg_attr(feature = "with-serde", derive(Serialize, Deserialize))] +pub enum QueryType { + /// a host addresss + A = a::Record::TYPE, + /// an authoritative name server + NS = ns::Record::TYPE, + /// a mail forwarder (Obsolete - use MX) + MF = mf::Record::TYPE, + /// the canonical name for an alias + CNAME = cname::Record::TYPE, + /// marks the start of a zone of authority + SOA = soa::Record::TYPE, + /// a mailbox domain name (EXPERIMENTAL) + MB = mb::Record::TYPE, + /// a mail group member (EXPERIMENTAL) + MG = mg::Record::TYPE, + /// a mail rename domain name (EXPERIMENTAL) + MR = mr::Record::TYPE, + /// a null RR (EXPERIMENTAL) + NULL = null::Record::TYPE, + /// a well known service description + WKS = wks::Record::TYPE, + /// a domain name pointer + PTR = ptr::Record::TYPE, + /// host information + HINFO = hinfo::Record::TYPE, + /// mailbox or mail list information + MINFO = minfo::Record::TYPE, + /// mail exchange + MX = mx::Record::TYPE, + /// text strings + TXT = txt::Record::TYPE, + /// IPv6 host address (RFC 2782) + AAAA = aaaa::Record::TYPE, + /// service record (RFC 2782) + SRV = srv::Record::TYPE, + /// A request for a transfer of an entire zone + AXFR = axfr::Record::TYPE, + /// A request for mailbox-related records (MB, MG or MR) + MAILB = mailb::Record::TYPE, + /// A request for mail agent RRs (Obsolete - see MX) + MAILA = maila::Record::TYPE, + /// A request for all records + All = all::Record::TYPE, +} + + +/// The CLASS value according to RFC 1035 +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub enum Class { + /// the Internet + IN = 1, + /// the CSNET class (Obsolete - used only for examples in some obsolete + /// RFCs) + CS = 2, + /// the CHAOS class + CH = 3, + /// Hesiod [Dyer 87] + HS = 4, +} + +/// The QCLASS value according to RFC 1035 +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub enum QueryClass { + /// the Internet + IN = 1, + /// the CSNET class (Obsolete - used only for examples in some obsolete + /// RFCs) + CS = 2, + /// the CHAOS class + CH = 3, + /// Hesiod [Dyer 87] + HS = 4, + /// Any class + Any = 255, +} + +/// The OPCODE value according to RFC 1035 +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub enum Opcode { + /// Normal query + StandardQuery, + /// Inverse query (query a name by IP) + InverseQuery, + /// Server status request + ServerStatusRequest, + /// Reserved opcode for future use + Reserved(u16), +} + +quick_error! { + /// The RCODE value according to RFC 1035 + #[derive(Debug, PartialEq, Eq, Clone, Copy)] + #[allow(missing_docs)] // names are from spec + pub enum ResponseCode { + NoError + FormatError + ServerFailure + NameError + NotImplemented + Refused + Reserved(code: u8) + } +} + +impl From<u16> for Opcode { + fn from(code: u16) -> Opcode { + use self::Opcode::*; + match code { + 0 => StandardQuery, + 1 => InverseQuery, + 2 => ServerStatusRequest, + x => Reserved(x), + } + } +} +impl Into<u16> for Opcode { + fn into(self) -> u16 { + use self::Opcode::*; + match self { + StandardQuery => 0, + InverseQuery => 1, + ServerStatusRequest => 2, + Reserved(x) => x, + } + } +} + +impl From<u8> for ResponseCode { + fn from(code: u8) -> ResponseCode { + use self::ResponseCode::*; + match code { + 0 => NoError, + 1 => FormatError, + 2 => ServerFailure, + 3 => NameError, + 4 => NotImplemented, + 5 => Refused, + 6...15 => Reserved(code), + x => panic!("Invalid response code {}", x), + } + } +} +impl Into<u8> for ResponseCode { + fn into(self) -> u8 { + use self::ResponseCode::*; + match self { + NoError => 0, + FormatError => 1, + ServerFailure => 2, + NameError => 3, + NotImplemented => 4, + Refused => 5, + Reserved(code) => code, + } + } +} + +impl QueryType { + /// Parse a query type code + pub fn parse(code: u16) -> Result<QueryType, Error> { + use self::QueryType::*; + match code as isize { + a::Record::TYPE => Ok(A), + ns::Record::TYPE => Ok(NS), + mf::Record::TYPE => Ok(MF), + cname::Record::TYPE => Ok(CNAME), + soa::Record::TYPE => Ok(SOA), + mb::Record::TYPE => Ok(MB), + mg::Record::TYPE => Ok(MG), + mr::Record::TYPE => Ok(MR), + null::Record::TYPE => Ok(NULL), + wks::Record::TYPE => Ok(WKS), + ptr::Record::TYPE => Ok(PTR), + hinfo::Record::TYPE => Ok(HINFO), + minfo::Record::TYPE => Ok(MINFO), + mx::Record::TYPE => Ok(MX), + txt::Record::TYPE => Ok(TXT), + aaaa::Record::TYPE => Ok(AAAA), + srv::Record::TYPE => Ok(SRV), + axfr::Record::TYPE => Ok(AXFR), + mailb::Record::TYPE => Ok(MAILB), + maila::Record::TYPE => Ok(MAILA), + all::Record::TYPE => Ok(All), + x => Err(Error::InvalidQueryType(x as u16)), + } + } +} + +impl QueryClass { + /// Parse a query class code + pub fn parse(code: u16) -> Result<QueryClass, Error> { + use self::QueryClass::*; + match code { + 1 => Ok(IN), + 2 => Ok(CS), + 3 => Ok(CH), + 4 => Ok(HS), + 255 => Ok(Any), + x => Err(Error::InvalidQueryClass(x)), + } + } +} + +impl Type { + /// Parse a type code + pub fn parse(code: u16) -> Result<Type, Error> { + use self::Type::*; + match code as isize { + a::Record::TYPE => Ok(A), + ns::Record::TYPE => Ok(NS), + mf::Record::TYPE => Ok(MF), + cname::Record::TYPE => Ok(CNAME), + soa::Record::TYPE => Ok(SOA), + mb::Record::TYPE => Ok(MB), + mg::Record::TYPE => Ok(MG), + mr::Record::TYPE => Ok(MR), + null::Record::TYPE => Ok(NULL), + wks::Record::TYPE => Ok(WKS), + ptr::Record::TYPE => Ok(PTR), + hinfo::Record::TYPE => Ok(HINFO), + minfo::Record::TYPE => Ok(MINFO), + mx::Record::TYPE => Ok(MX), + txt::Record::TYPE => Ok(TXT), + aaaa::Record::TYPE => Ok(AAAA), + srv::Record::TYPE => Ok(SRV), + opt::Record::TYPE => Ok(OPT), + nsec::Record::TYPE => Ok(NSEC), + x => Err(Error::InvalidType(x as u16)), + } + } +} + +impl Class { + /// Parse a class code + pub fn parse(code: u16) -> Result<Class, Error> { + use self::Class::*; + match code { + 1 => Ok(IN), + 2 => Ok(CS), + 3 => Ok(CH), + 4 => Ok(HS), + x => Err(Error::InvalidClass(x)), + } + } +} diff --git a/third_party/rust/dns-parser/src/error.rs b/third_party/rust/dns-parser/src/error.rs new file mode 100644 index 0000000000..ae6a39747d --- /dev/null +++ b/third_party/rust/dns-parser/src/error.rs @@ -0,0 +1,71 @@ +use std::str::Utf8Error; + +quick_error! { + /// Error parsing DNS packet + #[derive(Debug)] + pub enum Error { + /// Invalid compression pointer not pointing backwards + /// when parsing label + BadPointer { + description("invalid compression pointer not pointing backwards \ + when parsing label") + } + /// Packet is smaller than header size + HeaderTooShort { + description("packet is smaller than header size") + } + /// Packet ihas incomplete data + UnexpectedEOF { + description("packet is has incomplete data") + } + /// Wrong (too short or too long) size of RDATA + WrongRdataLength { + description("wrong (too short or too long) size of RDATA") + } + /// Packet has non-zero reserved bits + ReservedBitsAreNonZero { + description("packet has non-zero reserved bits") + } + /// Label in domain name has unknown label format + UnknownLabelFormat { + description("label in domain name has unknown label format") + } + /// Query type code is invalid + InvalidQueryType(code: u16) { + description("query type code is invalid") + display("query type {} is invalid", code) + } + /// Query class code is invalid + InvalidQueryClass(code: u16) { + description("query class code is invalid") + display("query class {} is invalid", code) + } + /// Type code is invalid + InvalidType(code: u16) { + description("type code is invalid") + display("type {} is invalid", code) + } + /// Class code is invalid + InvalidClass(code: u16) { + description("class code is invalid") + display("class {} is invalid", code) + } + /// Invalid characters encountered while reading label + LabelIsNotAscii { + description("invalid characters encountered while reading label") + } + /// Invalid characters encountered while reading TXT + TxtDataIsNotUTF8(error: Utf8Error) { + description("invalid characters encountered while reading TXT") + display("{:?}", error) + } + /// Parser is in the wrong state + WrongState { + description("parser is in the wrong state") + } + /// Additional OPT record found + AdditionalOPT { + description("additional OPT record found") + } + } +} diff --git a/third_party/rust/dns-parser/src/header.rs b/third_party/rust/dns-parser/src/header.rs new file mode 100644 index 0000000000..01d18add99 --- /dev/null +++ b/third_party/rust/dns-parser/src/header.rs @@ -0,0 +1,203 @@ +use byteorder::{BigEndian, ByteOrder}; + +use {Error, ResponseCode, Opcode}; + +mod flag { + pub const QUERY: u16 = 0b1000_0000_0000_0000; + pub const OPCODE_MASK: u16 = 0b0111_1000_0000_0000; + pub const AUTHORITATIVE: u16 = 0b0000_0100_0000_0000; + pub const TRUNCATED: u16 = 0b0000_0010_0000_0000; + pub const RECURSION_DESIRED: u16 = 0b0000_0001_0000_0000; + pub const RECURSION_AVAILABLE: u16 = 0b0000_0000_1000_0000; + pub const AUTHENTICATED_DATA: u16 = 0b0000_0000_0010_0000; + pub const CHECKING_DISABLED: u16 = 0b0000_0000_0001_0000; + pub const RESERVED_MASK: u16 = 0b0000_0000_0100_0000; + pub const RESPONSE_CODE_MASK: u16 = 0b0000_0000_0000_1111; +} + +/// Represents parsed header of the packet +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +#[allow(missing_docs)] // fields are from the spec I think +pub struct Header { + pub id: u16, + pub query: bool, + pub opcode: Opcode, + pub authoritative: bool, + pub truncated: bool, + pub recursion_desired: bool, + pub recursion_available: bool, + pub authenticated_data: bool, + pub checking_disabled: bool, + pub response_code: ResponseCode, + pub questions: u16, + pub answers: u16, + pub nameservers: u16, + pub additional: u16, +} + +impl Header { + /// Parse the header into a header structure + pub fn parse(data: &[u8]) -> Result<Header, Error> { + if data.len() < 12 { + return Err(Error::HeaderTooShort); + } + let flags = BigEndian::read_u16(&data[2..4]); + if flags & flag::RESERVED_MASK != 0 { + return Err(Error::ReservedBitsAreNonZero); + } + let header = Header { + id: BigEndian::read_u16(&data[..2]), + query: flags & flag::QUERY == 0, + opcode: ((flags & flag::OPCODE_MASK) + >> flag::OPCODE_MASK.trailing_zeros()).into(), + authoritative: flags & flag::AUTHORITATIVE != 0, + truncated: flags & flag::TRUNCATED != 0, + recursion_desired: flags & flag::RECURSION_DESIRED != 0, + recursion_available: flags & flag::RECURSION_AVAILABLE != 0, + authenticated_data: flags & flag::AUTHENTICATED_DATA != 0, + checking_disabled: flags & flag::CHECKING_DISABLED != 0, + response_code: From::from((flags&flag::RESPONSE_CODE_MASK) as u8), + questions: BigEndian::read_u16(&data[4..6]), + answers: BigEndian::read_u16(&data[6..8]), + nameservers: BigEndian::read_u16(&data[8..10]), + additional: BigEndian::read_u16(&data[10..12]), + }; + Ok(header) + } + /// Write a header to a buffer slice + /// + /// # Panics + /// + /// When buffer size is not exactly 12 bytes + pub fn write(&self, data: &mut [u8]) { + if data.len() != 12 { + panic!("Header size is exactly 12 bytes"); + } + let mut flags = 0u16; + flags |= Into::<u16>::into(self.opcode) + << flag::OPCODE_MASK.trailing_zeros(); + flags |= Into::<u8>::into(self.response_code) as u16; + if !self.query { flags |= flag::QUERY; } + if self.authoritative { flags |= flag::AUTHORITATIVE; } + if self.recursion_desired { flags |= flag::RECURSION_DESIRED; } + if self.recursion_available { flags |= flag::RECURSION_AVAILABLE; } + if self.truncated { flags |= flag::TRUNCATED; } + BigEndian::write_u16(&mut data[..2], self.id); + BigEndian::write_u16(&mut data[2..4], flags); + BigEndian::write_u16(&mut data[4..6], self.questions); + BigEndian::write_u16(&mut data[6..8], self.answers); + BigEndian::write_u16(&mut data[8..10], self.nameservers); + BigEndian::write_u16(&mut data[10..12], self.additional); + } + /// Set "truncated flag" in the raw data + // shouldn't this method be non-public? + pub fn set_truncated(data: &mut [u8]) { + let oldflags = BigEndian::read_u16(&data[2..4]); + BigEndian::write_u16(&mut data[2..4], oldflags & flag::TRUNCATED); + } + /// Returns a size of the header (always 12 bytes) + pub fn size() -> usize { 12 } +} + + +#[cfg(test)] +mod test { + + use {Header}; + use Opcode::*; + use ResponseCode::NoError; + + #[test] + fn parse_example_query() { + let query = b"\x06%\x01\x00\x00\x01\x00\x00\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x00\x01"; + let header = Header::parse(query).unwrap(); + assert_eq!(header, Header { + id: 1573, + query: true, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: false, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 0, + nameservers: 0, + additional: 0, + }); + } + + #[test] + fn parse_example_response() { + let response = b"\x06%\x81\x80\x00\x01\x00\x01\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x00\x01\ + \xc0\x0c\x00\x01\x00\x01\x00\x00\x04\xf8\ + \x00\x04]\xb8\xd8\""; + let header = Header::parse(response).unwrap(); + assert_eq!(header, Header { + id: 1573, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 1, + nameservers: 0, + additional: 0, + }); + } + + #[test] + fn parse_query_with_ad_set() { + let query = b"\x06%\x01\x20\x00\x01\x00\x00\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x00\x01"; + let header = Header::parse(query).unwrap(); + assert_eq!(header, Header { + id: 1573, + query: true, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: false, + authenticated_data: true, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 0, + nameservers: 0, + additional: 0, + }); + } + + #[test] + fn parse_query_with_cd_set() { + let query = b"\x06%\x01\x10\x00\x01\x00\x00\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x00\x01"; + let header = Header::parse(query).unwrap(); + assert_eq!(header, Header { + id: 1573, + query: true, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: false, + authenticated_data: false, + checking_disabled: true, + response_code: NoError, + questions: 1, + answers: 0, + nameservers: 0, + additional: 0, + }); + } +} diff --git a/third_party/rust/dns-parser/src/lib.rs b/third_party/rust/dns-parser/src/lib.rs new file mode 100644 index 0000000000..e89c005244 --- /dev/null +++ b/third_party/rust/dns-parser/src/lib.rs @@ -0,0 +1,39 @@ +#![recursion_limit="100"] +//! The network-agnostic DNS parser library +//! +//! [Documentation](https://docs.rs/dns-parser) | +//! [Github](https://github.com/tailhook/dns-parser) | +//! [Crate](https://crates.io/crates/dns-parser) +//! +//! Use [`Builder`] to create a new outgoing packet. +//! +//! Use [`Packet::parse`] to parse a packet into a data structure. +//! +//! [`Builder`]: struct.Builder.html +//! [`Packet::parse`]: struct.Packet.html#method.parse +//! +#![warn(missing_docs)] +#![warn(missing_debug_implementations)] + +extern crate byteorder; +#[cfg(test)] #[macro_use] extern crate matches; +#[macro_use(quick_error)] extern crate quick_error; +#[cfg(feature = "with-serde")] #[macro_use] extern crate serde_derive; + +mod enums; +mod structs; +mod name; +mod parser; +mod error; +mod header; +mod builder; + +pub mod rdata; + +pub use enums::{Type, QueryType, Class, QueryClass, ResponseCode, Opcode}; +pub use structs::{Question, ResourceRecord, Packet}; +pub use name::{Name}; +pub use error::{Error}; +pub use header::{Header}; +pub use rdata::{RData}; +pub use builder::{Builder}; diff --git a/third_party/rust/dns-parser/src/name.rs b/third_party/rust/dns-parser/src/name.rs new file mode 100644 index 0000000000..8763259cbe --- /dev/null +++ b/third_party/rust/dns-parser/src/name.rs @@ -0,0 +1,183 @@ +use std::fmt; +use std::fmt::Write; +use std::str::from_utf8; + +// Deprecated since rustc 1.23 +#[allow(unused_imports, deprecated)] +use std::ascii::AsciiExt; + +use byteorder::{BigEndian, ByteOrder}; + +use {Error}; + +/// The DNS name as stored in the original packet +/// +/// This contains just a reference to a slice that contains the data. +/// You may turn this into a string using `.to_string()` +#[derive(Clone, Copy)] +pub struct Name<'a>{ + labels: &'a [u8], + /// This is the original buffer size. The compressed names in original + /// are calculated in this buffer + original: &'a [u8], +} + +impl<'a> Name<'a> { + /// Scan the data to get Name object + /// + /// The `data` should be a part of `original` where name should start. + /// The `original` is the data starting a the start of a packet, so + /// that offsets in compressed name starts from the `original`. + pub fn scan(data: &'a[u8], original: &'a[u8]) -> Result<Name<'a>, Error> { + let mut parse_data = data; + let mut return_pos = None; + let mut pos = 0; + if parse_data.len() <= pos { + return Err(Error::UnexpectedEOF); + } + // By setting the largest_pos to be the original len, a side effect + // is that the pos variable can move forwards in the buffer once. + let mut largest_pos = original.len(); + let mut byte = parse_data[pos]; + while byte != 0 { + if parse_data.len() <= pos { + return Err(Error::UnexpectedEOF); + } + if byte & 0b1100_0000 == 0b1100_0000 { + if parse_data.len() < pos+2 { + return Err(Error::UnexpectedEOF); + } + let off = (BigEndian::read_u16(&parse_data[pos..pos+2]) + & !0b1100_0000_0000_0000) as usize; + if off >= original.len() { + return Err(Error::UnexpectedEOF); + } + // Set value for return_pos which is the pos in the original + // data buffer that should be used to return after validating + // the offsetted labels. + if let None = return_pos { + return_pos = Some(pos); + } + + // Check then set largest_pos to ensure we never go backwards + // in the buffer. + if off >= largest_pos { + return Err(Error::BadPointer); + } + largest_pos = off; + pos = 0; + parse_data = &original[off..]; + } else if byte & 0b1100_0000 == 0 { + let end = pos + byte as usize + 1; + if parse_data.len() < end { + return Err(Error::UnexpectedEOF); + } + if !parse_data[pos+1..end].is_ascii() { + return Err(Error::LabelIsNotAscii); + } + pos = end; + if parse_data.len() <= pos { + return Err(Error::UnexpectedEOF); + } + } else { + return Err(Error::UnknownLabelFormat); + } + byte = parse_data[pos]; + } + if let Some(return_pos) = return_pos { + return Ok(Name {labels: &data[..return_pos+2], original: original}); + } else { + return Ok(Name {labels: &data[..pos+1], original: original }); + } + } + /// Number of bytes serialized name occupies + pub fn byte_len(&self) -> usize { + self.labels.len() + } +} + +impl<'a> fmt::Display for Name<'a> { + fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { + let data = self.labels; + let original = self.original; + let mut pos = 0; + loop { + let byte = data[pos]; + if byte == 0 { + return Ok(()); + } else if byte & 0b1100_0000 == 0b1100_0000 { + let off = (BigEndian::read_u16(&data[pos..pos+2]) + & !0b1100_0000_0000_0000) as usize; + if pos != 0 { + try!(fmt.write_char('.')); + } + return fmt::Display::fmt( + &Name::scan(&original[off..], original).unwrap(), fmt) + } else if byte & 0b1100_0000 == 0 { + if pos != 0 { + try!(fmt.write_char('.')); + } + let end = pos + byte as usize + 1; + try!(fmt.write_str(from_utf8(&data[pos+1..end]).unwrap())); + pos = end; + continue; + } else { + unreachable!(); + } + } + } +} +impl<'a> fmt::Debug for Name<'a> { + fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result { + fmt.debug_tuple("Name") + .field(&format!("{}", self)) + .finish() + } +} + +#[cfg(test)] +mod test { + use Error; + use Name; + + #[test] + fn parse_badpointer_same_offset() { + // A buffer where an offset points to itself, + // which is a bad compression pointer. + let same_offset = vec![192, 2, 192, 2]; + let is_match = matches!(Name::scan(&same_offset, &same_offset), + Err(Error::BadPointer)); + + assert!(is_match); + } + + #[test] + fn parse_badpointer_forward_offset() { + // A buffer where the offsets points back to each other which causes + // infinite recursion if never checked, a bad compression pointer. + let forwards_offset = vec![192, 2, 192, 4, 192, 2]; + let is_match = matches!(Name::scan(&forwards_offset, &forwards_offset), + Err(Error::BadPointer)); + + assert!(is_match); + } + + #[test] + fn nested_names() { + // A buffer where an offset points to itself, a bad compression pointer. + let buf = b"\x02xx\x00\x02yy\xc0\x00\x02zz\xc0\x04"; + + assert_eq!(Name::scan(&buf[..], buf).unwrap().to_string(), + "xx"); + assert_eq!(Name::scan(&buf[..], buf).unwrap().labels, + b"\x02xx\x00"); + assert_eq!(Name::scan(&buf[4..], buf).unwrap().to_string(), + "yy.xx"); + assert_eq!(Name::scan(&buf[4..], buf).unwrap().labels, + b"\x02yy\xc0\x00"); + assert_eq!(Name::scan(&buf[9..], buf).unwrap().to_string(), + "zz.yy.xx"); + assert_eq!(Name::scan(&buf[9..], buf).unwrap().labels, + b"\x02zz\xc0\x04"); + } +} diff --git a/third_party/rust/dns-parser/src/parser.rs b/third_party/rust/dns-parser/src/parser.rs new file mode 100644 index 0000000000..0718798349 --- /dev/null +++ b/third_party/rust/dns-parser/src/parser.rs @@ -0,0 +1,455 @@ +use std::i32; + +use byteorder::{BigEndian, ByteOrder}; + +use {Header, Packet, Error, Question, Name, QueryType, QueryClass}; +use {Type, Class, ResourceRecord, RData}; +use rdata::opt::Record as Opt; + +const OPT_RR_START: [u8; 3] = [0, 0, 41]; + +impl<'a> Packet<'a> { + /// Parse a full DNS Packet and return a structure that has all the + /// data borrowed from the passed buffer. + pub fn parse(data: &[u8]) -> Result<Packet, Error> { + let header = try!(Header::parse(data)); + let mut offset = Header::size(); + let mut questions = Vec::with_capacity(header.questions as usize); + for _ in 0..header.questions { + let name = try!(Name::scan(&data[offset..], data)); + offset += name.byte_len(); + if offset + 4 > data.len() { + return Err(Error::UnexpectedEOF); + } + let qtype = try!(QueryType::parse( + BigEndian::read_u16(&data[offset..offset+2]))); + offset += 2; + + let (prefer_unicast, qclass) = try!(parse_qclass_code( + BigEndian::read_u16(&data[offset..offset+2]))); + offset += 2; + + questions.push(Question { + qname: name, + qtype: qtype, + prefer_unicast: prefer_unicast, + qclass: qclass, + }); + } + let mut answers = Vec::with_capacity(header.answers as usize); + for _ in 0..header.answers { + answers.push(try!(parse_record(data, &mut offset))); + } + let mut nameservers = Vec::with_capacity(header.nameservers as usize); + for _ in 0..header.nameservers { + nameservers.push(try!(parse_record(data, &mut offset))); + } + let mut additional = Vec::with_capacity(header.additional as usize); + let mut opt = None; + for _ in 0..header.additional { + if offset + 3 <= data.len() && data[offset..offset+3] == OPT_RR_START { + if opt.is_none() { + opt = Some(try!(parse_opt_record(data, &mut offset))); + } else { + return Err(Error::AdditionalOPT); + } + } else { + additional.push(try!(parse_record(data, &mut offset))); + } + } + Ok(Packet { + header: header, + questions: questions, + answers: answers, + nameservers: nameservers, + additional: additional, + opt: opt, + }) + } +} + +fn parse_qclass_code(value: u16) -> Result<(bool, QueryClass), Error> { + let prefer_unicast = value & 0x8000 == 0x8000; + let qclass_code = value & 0x7FFF; + + let qclass = try!(QueryClass::parse(qclass_code)); + Ok((prefer_unicast, qclass)) +} + +fn parse_class_code(value: u16) -> Result<(bool, Class), Error> { + let is_unique = value & 0x8000 == 0x8000; + let class_code = value & 0x7FFF; + + let cls = try!(Class::parse(class_code)); + Ok((is_unique, cls)) +} + +// Generic function to parse answer, nameservers, and additional records. +fn parse_record<'a>(data: &'a [u8], offset: &mut usize) -> Result<ResourceRecord<'a>, Error> { + let name = try!(Name::scan(&data[*offset..], data)); + *offset += name.byte_len(); + if *offset + 10 > data.len() { + return Err(Error::UnexpectedEOF); + } + let typ = try!(Type::parse( + BigEndian::read_u16(&data[*offset..*offset+2]))); + *offset += 2; + + let class_code = BigEndian::read_u16(&data[*offset..*offset+2]); + let (multicast_unique, cls) = try!(parse_class_code(class_code)); + *offset += 2; + + let mut ttl = BigEndian::read_u32(&data[*offset..*offset+4]); + if ttl > i32::MAX as u32 { + ttl = 0; + } + *offset += 4; + let rdlen = BigEndian::read_u16(&data[*offset..*offset+2]) as usize; + *offset += 2; + if *offset + rdlen > data.len() { + return Err(Error::UnexpectedEOF); + } + let data = try!(RData::parse(typ, + &data[*offset..*offset+rdlen], data)); + *offset += rdlen; + Ok(ResourceRecord { + name: name, + multicast_unique: multicast_unique, + cls: cls, + ttl: ttl, + data: data, + }) +} + +// Function to parse an RFC 6891 OPT Pseudo RR +fn parse_opt_record<'a>(data: &'a [u8], offset: &mut usize) -> Result<Opt<'a>, Error> { + if *offset + 11 > data.len() { + return Err(Error::UnexpectedEOF); + } + *offset += 1; + let typ = try!(Type::parse( + BigEndian::read_u16(&data[*offset..*offset+2]))); + if typ != Type::OPT { + return Err(Error::InvalidType(typ as u16)); + } + *offset += 2; + let udp = BigEndian::read_u16(&data[*offset..*offset+2]); + *offset += 2; + let extrcode = data[*offset]; + *offset += 1; + let version = data[*offset]; + *offset += 1; + let flags = BigEndian::read_u16(&data[*offset..*offset+2]); + *offset += 2; + let rdlen = BigEndian::read_u16(&data[*offset..*offset+2]) as usize; + *offset += 2; + if *offset + rdlen > data.len() { + return Err(Error::UnexpectedEOF); + } + let data = try!(RData::parse(typ, + &data[*offset..*offset+rdlen], data)); + *offset += rdlen; + + Ok(Opt { + udp: udp, + extrcode: extrcode, + version: version, + flags: flags, + data: data, + }) +} + +#[cfg(test)] +mod test { + + use std::net::Ipv4Addr; + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NoError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + + #[test] + fn parse_example_query() { + let query = b"\x06%\x01\x00\x00\x01\x00\x00\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x00\x01"; + let packet = Packet::parse(query).unwrap(); + assert_eq!(packet.header, Header { + id: 1573, + query: true, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: false, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 0, + nameservers: 0, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "example.com"); + assert_eq!(packet.answers.len(), 0); + } + + #[test] + fn parse_example_response() { + let response = b"\x06%\x81\x80\x00\x01\x00\x01\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x00\x01\ + \xc0\x0c\x00\x01\x00\x01\x00\x00\x04\xf8\ + \x00\x04]\xb8\xd8\""; + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 1573, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 1, + nameservers: 0, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "example.com"); + assert_eq!(packet.answers.len(), 1); + assert_eq!(&packet.answers[0].name.to_string()[..], "example.com"); + assert_eq!(packet.answers[0].multicast_unique, false); + assert_eq!(packet.answers[0].cls, C::IN); + assert_eq!(packet.answers[0].ttl, 1272); + match packet.answers[0].data { + RData::A(addr) => { + assert_eq!(addr.0, Ipv4Addr::new(93, 184, 216, 34)); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } + + #[test] + fn parse_response_with_multicast_unique() { + let response = b"\x06%\x81\x80\x00\x01\x00\x01\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x00\x01\ + \xc0\x0c\x00\x01\x80\x01\x00\x00\x04\xf8\ + \x00\x04]\xb8\xd8\""; + let packet = Packet::parse(response).unwrap(); + + assert_eq!(packet.answers.len(), 1); + assert_eq!(packet.answers[0].multicast_unique, true); + assert_eq!(packet.answers[0].cls, C::IN); + } + + #[test] + fn parse_additional_record_response() { + let response = b"\x4a\xf0\x81\x80\x00\x01\x00\x01\x00\x01\x00\x01\ + \x03www\x05skype\x03com\x00\x00\x01\x00\x01\ + \xc0\x0c\x00\x05\x00\x01\x00\x00\x0e\x10\ + \x00\x1c\x07\x6c\x69\x76\x65\x63\x6d\x73\x0e\x74\ + \x72\x61\x66\x66\x69\x63\x6d\x61\x6e\x61\x67\x65\ + \x72\x03\x6e\x65\x74\x00\ + \xc0\x42\x00\x02\x00\x01\x00\x01\xd5\xd3\x00\x11\ + \x01\x67\x0c\x67\x74\x6c\x64\x2d\x73\x65\x72\x76\x65\x72\x73\ + \xc0\x42\ + \x01\x61\xc0\x55\x00\x01\x00\x01\x00\x00\xa3\x1c\ + \x00\x04\xc0\x05\x06\x1e"; + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 19184, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 1, + nameservers: 1, + additional: 1, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "www.skype.com"); + assert_eq!(packet.answers.len(), 1); + assert_eq!(&packet.answers[0].name.to_string()[..], "www.skype.com"); + assert_eq!(packet.answers[0].cls, C::IN); + assert_eq!(packet.answers[0].ttl, 3600); + match packet.answers[0].data { + RData::CNAME(cname) => { + assert_eq!(&cname.0.to_string()[..], "livecms.trafficmanager.net"); + } + ref x => panic!("Wrong rdata {:?}", x), + } + assert_eq!(packet.nameservers.len(), 1); + assert_eq!(&packet.nameservers[0].name.to_string()[..], "net"); + assert_eq!(packet.nameservers[0].cls, C::IN); + assert_eq!(packet.nameservers[0].ttl, 120275); + match packet.nameservers[0].data { + RData::NS(ns) => { + assert_eq!(&ns.0.to_string()[..], "g.gtld-servers.net"); + } + ref x => panic!("Wrong rdata {:?}", x), + } + assert_eq!(packet.additional.len(), 1); + assert_eq!(&packet.additional[0].name.to_string()[..], "a.gtld-servers.net"); + assert_eq!(packet.additional[0].cls, C::IN); + assert_eq!(packet.additional[0].ttl, 41756); + match packet.additional[0].data { + RData::A(addr) => { + assert_eq!(addr.0, Ipv4Addr::new(192, 5, 6, 30)); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } + + #[test] + fn parse_multiple_answers() { + let response = b"\x9d\xe9\x81\x80\x00\x01\x00\x06\x00\x00\x00\x00\ + \x06google\x03com\x00\x00\x01\x00\x01\xc0\x0c\ + \x00\x01\x00\x01\x00\x00\x00\xef\x00\x04@\xe9\ + \xa4d\xc0\x0c\x00\x01\x00\x01\x00\x00\x00\xef\ + \x00\x04@\xe9\xa4\x8b\xc0\x0c\x00\x01\x00\x01\ + \x00\x00\x00\xef\x00\x04@\xe9\xa4q\xc0\x0c\x00\ + \x01\x00\x01\x00\x00\x00\xef\x00\x04@\xe9\xa4f\ + \xc0\x0c\x00\x01\x00\x01\x00\x00\x00\xef\x00\x04@\ + \xe9\xa4e\xc0\x0c\x00\x01\x00\x01\x00\x00\x00\xef\ + \x00\x04@\xe9\xa4\x8a"; + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 40425, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 6, + nameservers: 0, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "google.com"); + assert_eq!(packet.answers.len(), 6); + let ips = vec![ + Ipv4Addr::new(64, 233, 164, 100), + Ipv4Addr::new(64, 233, 164, 139), + Ipv4Addr::new(64, 233, 164, 113), + Ipv4Addr::new(64, 233, 164, 102), + Ipv4Addr::new(64, 233, 164, 101), + Ipv4Addr::new(64, 233, 164, 138), + ]; + for i in 0..6 { + assert_eq!(&packet.answers[i].name.to_string()[..], "google.com"); + assert_eq!(packet.answers[i].cls, C::IN); + assert_eq!(packet.answers[i].ttl, 239); + match packet.answers[i].data { + RData::A(addr) => { + assert_eq!(addr.0, ips[i]); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } + } + + #[test] + fn parse_srv_query() { + let query = b"[\xd9\x01\x00\x00\x01\x00\x00\x00\x00\x00\x00\ + \x0c_xmpp-server\x04_tcp\x05gmail\x03com\x00\x00!\x00\x01"; + let packet = Packet::parse(query).unwrap(); + assert_eq!(packet.header, Header { + id: 23513, + query: true, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: false, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 0, + nameservers: 0, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::SRV); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(packet.questions[0].prefer_unicast, false); + assert_eq!(&packet.questions[0].qname.to_string()[..], + "_xmpp-server._tcp.gmail.com"); + assert_eq!(packet.answers.len(), 0); + } + + #[test] + fn parse_multicast_prefer_unicast_query() { + let query = b"\x06%\x01\x00\x00\x01\x00\x00\x00\x00\x00\x00\ + \x07example\x03com\x00\x00\x01\x80\x01"; + let packet = Packet::parse(query).unwrap(); + + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(packet.questions[0].prefer_unicast, true); + } + + #[test] + fn parse_example_query_edns() { + let query = b"\x95\xce\x01\x00\x00\x01\x00\x00\x00\x00\x00\x01\ + \x06google\x03com\x00\x00\x01\x00\ + \x01\x00\x00\x29\x10\x00\x00\x00\x00\x00\x00\x00"; + let packet = Packet::parse(query).unwrap(); + assert_eq!(packet.header, Header { + id: 38350, + query: true, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: false, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 0, + nameservers: 0, + additional: 1, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "google.com"); + assert_eq!(packet.answers.len(), 0); + match packet.opt { + Some(opt) => { + assert_eq!(opt.udp, 4096); + assert_eq!(opt.extrcode, 0); + assert_eq!(opt.version, 0); + assert_eq!(opt.flags, 0); + }, + None => panic!("Missing OPT RR") + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/a.rs b/third_party/rust/dns-parser/src/rdata/a.rs new file mode 100644 index 0000000000..f4196d5ac2 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/a.rs @@ -0,0 +1,21 @@ +use std::net::Ipv4Addr; + +use Error; +use byteorder::{BigEndian, ByteOrder}; + +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record(pub Ipv4Addr); + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 1; + + fn parse(rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + if rdata.len() != 4 { + return Err(Error::WrongRdataLength); + } + let address = Ipv4Addr::from(BigEndian::read_u32(rdata)); + let record = Record(address); + Ok(super::RData::A(record)) + } +} diff --git a/third_party/rust/dns-parser/src/rdata/aaaa.rs b/third_party/rust/dns-parser/src/rdata/aaaa.rs new file mode 100644 index 0000000000..e59937a525 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/aaaa.rs @@ -0,0 +1,85 @@ +use std::net::Ipv6Addr; + +use Error; +use byteorder::{BigEndian, ByteOrder}; + +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record(pub Ipv6Addr); + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 28; + + fn parse(rdata: &'a [u8], _record: &'a [u8]) -> super::RDataResult<'a> { + if rdata.len() != 16 { + return Err(Error::WrongRdataLength); + } + let address = Ipv6Addr::new( + BigEndian::read_u16(&rdata[0..2]), + BigEndian::read_u16(&rdata[2..4]), + BigEndian::read_u16(&rdata[4..6]), + BigEndian::read_u16(&rdata[6..8]), + BigEndian::read_u16(&rdata[8..10]), + BigEndian::read_u16(&rdata[10..12]), + BigEndian::read_u16(&rdata[12..14]), + BigEndian::read_u16(&rdata[14..16]), + ); + let record = Record(address); + Ok(super::RData::AAAA(record)) + } +} + +#[cfg(test)] +mod test { + + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NoError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + use super::*; + + #[test] + fn parse_response() { + let response = b"\xa9\xd9\x81\x80\x00\x01\x00\x01\x00\x00\x00\x00\x06\ + google\x03com\x00\x00\x1c\x00\x01\xc0\x0c\x00\x1c\x00\x01\x00\x00\ + \x00\x8b\x00\x10*\x00\x14P@\t\x08\x12\x00\x00\x00\x00\x00\x00 \x0e"; + + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 43481, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 1, + nameservers: 0, + additional: 0, + }); + + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::AAAA); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "google.com"); + assert_eq!(packet.answers.len(), 1); + assert_eq!(&packet.answers[0].name.to_string()[..], "google.com"); + assert_eq!(packet.answers[0].cls, C::IN); + assert_eq!(packet.answers[0].ttl, 139); + match packet.answers[0].data { + RData::AAAA(addr) => { + assert_eq!(addr.0, Ipv6Addr::new( + 0x2A00, 0x1450, 0x4009, 0x812, 0, 0, 0, 0x200e) + ); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/all.rs b/third_party/rust/dns-parser/src/rdata/all.rs new file mode 100644 index 0000000000..106c91b089 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/all.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 255; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/axfr.rs b/third_party/rust/dns-parser/src/rdata/axfr.rs new file mode 100644 index 0000000000..b48dc1ae24 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/axfr.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 252; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/cname.rs b/third_party/rust/dns-parser/src/rdata/cname.rs new file mode 100644 index 0000000000..0dcb46952f --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/cname.rs @@ -0,0 +1,102 @@ +use Name; + +#[derive(Debug, Clone, Copy)] +pub struct Record<'a>(pub Name<'a>); + +impl<'a> ToString for Record<'a> { + #[inline] + fn to_string(&self) -> String { + self.0.to_string() + } +} + +impl<'a> super::Record<'a> for Record<'a> { + + const TYPE: isize = 5; + + fn parse(rdata: &'a [u8], original: &'a [u8]) -> super::RDataResult<'a> { + let name = Name::scan(rdata, original)?; + let record = Record(name); + Ok(super::RData::CNAME(record)) + } +} + +#[cfg(test)] +mod test { + + use std::net::Ipv4Addr; + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NoError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + + #[test] + fn parse_response() { + let response = b"\xfc\x9d\x81\x80\x00\x01\x00\x06\x00\x02\x00\x02\x03\ + cdn\x07sstatic\x03net\x00\x00\x01\x00\x01\xc0\x0c\x00\x05\x00\x01\ + \x00\x00\x00f\x00\x02\xc0\x10\xc0\x10\x00\x01\x00\x01\x00\x00\x00\ + f\x00\x04h\x10g\xcc\xc0\x10\x00\x01\x00\x01\x00\x00\x00f\x00\x04h\ + \x10k\xcc\xc0\x10\x00\x01\x00\x01\x00\x00\x00f\x00\x04h\x10h\xcc\ + \xc0\x10\x00\x01\x00\x01\x00\x00\x00f\x00\x04h\x10j\xcc\xc0\x10\ + \x00\x01\x00\x01\x00\x00\x00f\x00\x04h\x10i\xcc\xc0\x10\x00\x02\ + \x00\x01\x00\x00\x99L\x00\x0b\x08cf-dns02\xc0\x10\xc0\x10\x00\x02\ + \x00\x01\x00\x00\x99L\x00\x0b\x08cf-dns01\xc0\x10\xc0\xa2\x00\x01\ + \x00\x01\x00\x00\x99L\x00\x04\xad\xf5:5\xc0\x8b\x00\x01\x00\x01\x00\ + \x00\x99L\x00\x04\xad\xf5;\x04"; + + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 64669, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 6, + nameservers: 2, + additional: 2, + }); + + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "cdn.sstatic.net"); + assert_eq!(packet.answers.len(), 6); + assert_eq!(&packet.answers[0].name.to_string()[..], "cdn.sstatic.net"); + assert_eq!(packet.answers[0].cls, C::IN); + assert_eq!(packet.answers[0].ttl, 102); + match packet.answers[0].data { + RData::CNAME(cname) => { + assert_eq!(&cname.0.to_string(), "sstatic.net"); + } + ref x => panic!("Wrong rdata {:?}", x), + } + + let ips = vec![ + Ipv4Addr::new(104, 16, 103, 204), + Ipv4Addr::new(104, 16, 107, 204), + Ipv4Addr::new(104, 16, 104, 204), + Ipv4Addr::new(104, 16, 106, 204), + Ipv4Addr::new(104, 16, 105, 204), + ]; + for i in 1..6 { + assert_eq!(&packet.answers[i].name.to_string()[..], "sstatic.net"); + assert_eq!(packet.answers[i].cls, C::IN); + assert_eq!(packet.answers[i].ttl, 102); + match packet.answers[i].data { + RData::A(addr) => { + assert_eq!(addr.0, ips[i-1]); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/hinfo.rs b/third_party/rust/dns-parser/src/rdata/hinfo.rs new file mode 100644 index 0000000000..6ee160bb86 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/hinfo.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 13; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/maila.rs b/third_party/rust/dns-parser/src/rdata/maila.rs new file mode 100644 index 0000000000..ab1166cf31 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/maila.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 254; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/mailb.rs b/third_party/rust/dns-parser/src/rdata/mailb.rs new file mode 100644 index 0000000000..9f043c60c3 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/mailb.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 253; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/mb.rs b/third_party/rust/dns-parser/src/rdata/mb.rs new file mode 100644 index 0000000000..d14e7b4fc5 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/mb.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 7; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/mf.rs b/third_party/rust/dns-parser/src/rdata/mf.rs new file mode 100644 index 0000000000..11c935d9f7 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/mf.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 4; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/mg.rs b/third_party/rust/dns-parser/src/rdata/mg.rs new file mode 100644 index 0000000000..4fce456b7d --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/mg.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 8; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/minfo.rs b/third_party/rust/dns-parser/src/rdata/minfo.rs new file mode 100644 index 0000000000..29b3a459b2 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/minfo.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 14; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/mod.rs b/third_party/rust/dns-parser/src/rdata/mod.rs new file mode 100644 index 0000000000..72df27c232 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/mod.rs @@ -0,0 +1,83 @@ +//! Data types and methods for handling the RData field + +#![allow(missing_docs)] // resource records are pretty self-descriptive + +pub mod a; +pub mod aaaa; +pub mod all; +pub mod axfr; +pub mod cname; +pub mod hinfo; +pub mod maila; +pub mod mailb; +pub mod mb; +pub mod mf; +pub mod mg; +pub mod minfo; +pub mod mr; +pub mod mx; +pub mod ns; +pub mod nsec; +pub mod null; +pub mod opt; +pub mod ptr; +pub mod soa; +pub mod srv; +pub mod txt; +pub mod wks; + +use {Type, Error}; + +pub use self::a::Record as A; +pub use self::aaaa::Record as Aaaa; +pub use self::cname::Record as Cname; +pub use self::mx::Record as Mx; +pub use self::ns::Record as Ns; +pub use self::nsec::Record as Nsec; +pub use self::opt::Record as Opt; +pub use self::ptr::Record as Ptr; +pub use self::soa::Record as Soa; +pub use self::srv::Record as Srv; +pub use self::txt::Record as Txt; + +pub type RDataResult<'a> = Result<RData<'a>, Error>; + +/// The enumeration that represents known types of DNS resource records data +#[derive(Debug)] +pub enum RData<'a> { + A(A), + AAAA(Aaaa), + CNAME(Cname<'a>), + MX(Mx<'a>), + NS(Ns<'a>), + PTR(Ptr<'a>), + SOA(Soa<'a>), + SRV(Srv<'a>), + TXT(Txt<'a>), + /// Anything that can't be parsed yet + Unknown(&'a [u8]), +} + +pub (crate) trait Record<'a> { + const TYPE: isize; + + fn parse(rdata: &'a [u8], original: &'a [u8]) -> RDataResult<'a>; +} + +impl<'a> RData<'a> { + /// Parse an RR data and return RData enumeration + pub fn parse(typ: Type, rdata: &'a [u8], original: &'a [u8]) -> RDataResult<'a> { + match typ { + Type::A => A::parse(rdata, original), + Type::AAAA => Aaaa::parse(rdata, original), + Type::CNAME => Cname::parse(rdata, original), + Type::NS => Ns::parse(rdata, original), + Type::MX => Mx::parse(rdata, original), + Type::PTR => Ptr::parse(rdata, original), + Type::SOA => Soa::parse(rdata, original), + Type::SRV => Srv::parse(rdata, original), + Type::TXT => Txt::parse(rdata, original), + _ => Ok(RData::Unknown(rdata)), + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/mr.rs b/third_party/rust/dns-parser/src/rdata/mr.rs new file mode 100644 index 0000000000..e313372c4a --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/mr.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 9; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/mx.rs b/third_party/rust/dns-parser/src/rdata/mx.rs new file mode 100644 index 0000000000..33c5c41dfc --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/mx.rs @@ -0,0 +1,92 @@ +use {Name, Error}; +use byteorder::{BigEndian, ByteOrder}; + +#[derive(Debug, Clone, Copy)] +pub struct Record<'a> { + pub preference: u16, + pub exchange: Name<'a>, +} + +impl<'a> super::Record<'a> for Record<'a> { + + const TYPE: isize = 15; + + fn parse(rdata: &'a [u8], original: &'a [u8]) -> super::RDataResult<'a> { + if rdata.len() < 3 { + return Err(Error::WrongRdataLength); + } + let record = Record { + preference: BigEndian::read_u16(&rdata[..2]), + exchange: Name::scan(&rdata[2..], original)?, + }; + Ok(super::RData::MX(record)) + } +} + +#[cfg(test)] +mod test { + + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NoError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + use super::*; + + #[test] + fn parse_response() { + let response = b"\xe3\xe8\x81\x80\x00\x01\x00\x05\x00\x00\x00\x00\ + \x05gmail\x03com\x00\x00\x0f\x00\x01\xc0\x0c\x00\x0f\x00\x01\ + \x00\x00\x04|\x00\x1b\x00\x05\rgmail-smtp-in\x01l\x06google\xc0\ + \x12\xc0\x0c\x00\x0f\x00\x01\x00\x00\x04|\x00\t\x00\ + \n\x04alt1\xc0)\xc0\x0c\x00\x0f\x00\x01\x00\x00\x04|\ + \x00\t\x00(\x04alt4\xc0)\xc0\x0c\x00\x0f\x00\x01\x00\ + \x00\x04|\x00\t\x00\x14\x04alt2\xc0)\xc0\x0c\x00\x0f\ + \x00\x01\x00\x00\x04|\x00\t\x00\x1e\x04alt3\xc0)"; + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 58344, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 5, + nameservers: 0, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::MX); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], + "gmail.com"); + assert_eq!(packet.answers.len(), 5); + let items = vec![ + ( 5, "gmail-smtp-in.l.google.com"), + (10, "alt1.gmail-smtp-in.l.google.com"), + (40, "alt4.gmail-smtp-in.l.google.com"), + (20, "alt2.gmail-smtp-in.l.google.com"), + (30, "alt3.gmail-smtp-in.l.google.com"), + ]; + for i in 0..5 { + assert_eq!(&packet.answers[i].name.to_string()[..], + "gmail.com"); + assert_eq!(packet.answers[i].cls, C::IN); + assert_eq!(packet.answers[i].ttl, 1148); + match *&packet.answers[i].data { + RData::MX( Record { preference, exchange }) => { + assert_eq!(preference, items[i].0); + assert_eq!(exchange.to_string(), (items[i].1).to_string()); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/ns.rs b/third_party/rust/dns-parser/src/rdata/ns.rs new file mode 100644 index 0000000000..42a2f29ab1 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/ns.rs @@ -0,0 +1,88 @@ +use Name; + +#[derive(Debug, Clone, Copy)] +pub struct Record<'a>(pub Name<'a>); + +impl<'a> ToString for Record<'a> { + #[inline] + fn to_string(&self) -> String { + self.0.to_string() + } +} + +impl<'a> super::Record<'a> for Record<'a> { + + const TYPE: isize = 2; + + fn parse(rdata: &'a [u8], original: &'a [u8]) -> super::RDataResult<'a> { + let name = Name::scan(rdata, original)?; + let record = Record(name); + Ok(super::RData::NS(record)) + } +} + +#[cfg(test)] +mod test { + + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NoError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + + #[test] + fn parse_response() { + let response = b"\x4a\xf0\x81\x80\x00\x01\x00\x01\x00\x01\x00\x00\ + \x03www\x05skype\x03com\x00\x00\x01\x00\x01\ + \xc0\x0c\x00\x05\x00\x01\x00\x00\x0e\x10\ + \x00\x1c\x07\x6c\x69\x76\x65\x63\x6d\x73\x0e\x74\ + \x72\x61\x66\x66\x69\x63\x6d\x61\x6e\x61\x67\x65\ + \x72\x03\x6e\x65\x74\x00\ + \xc0\x42\x00\x02\x00\x01\x00\x01\xd5\xd3\x00\x11\ + \x01\x67\x0c\x67\x74\x6c\x64\x2d\x73\x65\x72\x76\x65\x72\x73\ + \xc0\x42"; + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 19184, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 1, + nameservers: 1, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "www.skype.com"); + assert_eq!(packet.answers.len(), 1); + assert_eq!(&packet.answers[0].name.to_string()[..], "www.skype.com"); + assert_eq!(packet.answers[0].cls, C::IN); + assert_eq!(packet.answers[0].ttl, 3600); + match packet.answers[0].data { + RData::CNAME(cname) => { + assert_eq!(&cname.0.to_string()[..], "livecms.trafficmanager.net"); + } + ref x => panic!("Wrong rdata {:?}", x), + } + assert_eq!(packet.nameservers.len(), 1); + assert_eq!(&packet.nameservers[0].name.to_string()[..], "net"); + assert_eq!(packet.nameservers[0].cls, C::IN); + assert_eq!(packet.nameservers[0].ttl, 120275); + match packet.nameservers[0].data { + RData::NS(ns) => { + assert_eq!(&ns.0.to_string()[..], "g.gtld-servers.net"); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/nsec.rs b/third_party/rust/dns-parser/src/rdata/nsec.rs new file mode 100644 index 0000000000..354c512273 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/nsec.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 47; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/null.rs b/third_party/rust/dns-parser/src/rdata/null.rs new file mode 100644 index 0000000000..b3bfcd35ee --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/null.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 10; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/opt.rs b/third_party/rust/dns-parser/src/rdata/opt.rs new file mode 100644 index 0000000000..694d393440 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/opt.rs @@ -0,0 +1,18 @@ +/// RFC 6891 OPT RR +#[derive(Debug)] +pub struct Record<'a> { + pub udp: u16, + pub extrcode: u8, + pub version: u8, + pub flags: u16, + pub data: super::RData<'a>, +} + +impl<'a> super::Record<'a> for Record<'a> { + + const TYPE: isize = 41; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/rdata/ptr.rs b/third_party/rust/dns-parser/src/rdata/ptr.rs new file mode 100644 index 0000000000..315c3795a8 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/ptr.rs @@ -0,0 +1,74 @@ +use Name; + +#[derive(Debug, Clone, Copy)] +pub struct Record<'a>(pub Name<'a>); + +impl<'a> ToString for Record<'a> { + #[inline] + fn to_string(&self) -> String { + self.0.to_string() + } +} + +impl<'a> super::Record<'a> for Record<'a> { + + const TYPE: isize = 12; + + fn parse(rdata: &'a [u8], original: &'a [u8]) -> super::RDataResult<'a> { + let name = Name::scan(rdata, original)?; + let record = Record(name); + Ok(super::RData::PTR(record)) + } +} + +#[cfg(test)] +mod test { + + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NoError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + + #[test] + fn parse_response() { + let response = b"\x53\xd6\x81\x80\x00\x01\x00\x01\x00\x00\x00\x00\ + \x0269\x0293\x0275\x0272\x07in-addr\x04arpa\x00\ + \x00\x0c\x00\x01\ + \xc0\x0c\x00\x0c\x00\x01\x00\x01\x51\x80\x00\x1e\ + \x10pool-72-75-93-69\x07verizon\x03net\x00"; + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 21462, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 1, + nameservers: 0, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::PTR); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "69.93.75.72.in-addr.arpa"); + assert_eq!(packet.answers.len(), 1); + assert_eq!(&packet.answers[0].name.to_string()[..], "69.93.75.72.in-addr.arpa"); + assert_eq!(packet.answers[0].cls, C::IN); + assert_eq!(packet.answers[0].ttl, 86400); + match packet.answers[0].data { + RData::PTR(name) => { + assert_eq!(&name.0.to_string()[..], "pool-72-75-93-69.verizon.net"); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/soa.rs b/third_party/rust/dns-parser/src/rdata/soa.rs new file mode 100644 index 0000000000..e6d5f1712d --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/soa.rs @@ -0,0 +1,101 @@ +use {Name, Error}; +use byteorder::{BigEndian, ByteOrder}; + +/// The SOA (Start of Authority) record +#[derive(Debug, Clone, Copy)] +pub struct Record<'a> { + pub primary_ns: Name<'a>, + pub mailbox: Name<'a>, + pub serial: u32, + pub refresh: u32, + pub retry: u32, + pub expire: u32, + pub minimum_ttl: u32, +} + +impl<'a> super::Record<'a> for Record<'a> { + + const TYPE: isize = 6; + + fn parse(rdata: &'a [u8], original: &'a [u8]) -> super::RDataResult<'a> { + let mut pos = 0; + let primary_name_server = try!(Name::scan(rdata, original)); + pos += primary_name_server.byte_len(); + let mailbox = try!(Name::scan(&rdata[pos..], original)); + pos += mailbox.byte_len(); + if rdata[pos..].len() < 20 { + return Err(Error::WrongRdataLength); + } + let record = Record { + primary_ns: primary_name_server, + mailbox: mailbox, + serial: BigEndian::read_u32(&rdata[pos..(pos+4)]), + refresh: BigEndian::read_u32(&rdata[(pos+4)..(pos+8)]), + retry: BigEndian::read_u32(&rdata[(pos+8)..(pos+12)]), + expire: BigEndian::read_u32(&rdata[(pos+12)..(pos+16)]), + minimum_ttl: BigEndian::read_u32(&rdata[(pos+16)..(pos+20)]), + }; + Ok(super::RData::SOA(record)) + } +} + +#[cfg(test)] +mod test { + + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NameError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + + #[test] + fn parse_response() { + let response = b"\x9f\xc5\x85\x83\x00\x01\x00\x00\x00\x01\x00\x00\ + \x0edlkfjkdjdslfkj\x07youtube\x03com\x00\x00\x01\x00\x01\ + \xc0\x1b\x00\x06\x00\x01\x00\x00\x2a\x30\x00\x1e\xc0\x1b\ + \x05admin\xc0\x1b\x77\xed\x2a\x73\x00\x00\x51\x80\x00\x00\ + \x0e\x10\x00\x00\x3a\x80\x00\x00\x2a\x30"; + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 40901, + query: false, + opcode: StandardQuery, + authoritative: true, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NameError, + questions: 1, + answers: 0, + nameservers: 1, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::A); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "dlkfjkdjdslfkj.youtube.com"); + assert_eq!(packet.answers.len(), 0); + + assert_eq!(packet.nameservers.len(), 1); + assert_eq!(&packet.nameservers[0].name.to_string()[..], "youtube.com"); + assert_eq!(packet.nameservers[0].cls, C::IN); + assert_eq!(packet.nameservers[0].multicast_unique, false); + assert_eq!(packet.nameservers[0].ttl, 10800); + match packet.nameservers[0].data { + RData::SOA(ref soa_rec) => { + assert_eq!(&soa_rec.primary_ns.to_string()[..], "youtube.com"); + assert_eq!(&soa_rec.mailbox.to_string()[..], "admin.youtube.com"); + assert_eq!(soa_rec.serial, 2012031603); + assert_eq!(soa_rec.refresh, 20864); + assert_eq!(soa_rec.retry, 3600); + assert_eq!(soa_rec.expire, 14976); + assert_eq!(soa_rec.minimum_ttl, 10800); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/srv.rs b/third_party/rust/dns-parser/src/rdata/srv.rs new file mode 100644 index 0000000000..dbc151d8aa --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/srv.rs @@ -0,0 +1,102 @@ +use {Name, Error}; +use byteorder::{BigEndian, ByteOrder}; + +#[derive(Debug, Clone, Copy)] +pub struct Record<'a> { + pub priority: u16, + pub weight: u16, + pub port: u16, + pub target: Name<'a>, +} + +impl<'a> super::Record<'a> for Record<'a> { + + const TYPE: isize = 33; + + fn parse(rdata: &'a [u8], original: &'a [u8]) -> super::RDataResult<'a> { + if rdata.len() < 7 { + return Err(Error::WrongRdataLength); + } + let record = Record { + priority: BigEndian::read_u16(&rdata[..2]), + weight: BigEndian::read_u16(&rdata[2..4]), + port: BigEndian::read_u16(&rdata[4..6]), + target: Name::scan(&rdata[6..], original)?, + }; + Ok(super::RData::SRV(record)) + } +} + +#[cfg(test)] +mod test { + + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NoError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + use super::*; + + #[test] + fn parse_response() { + let response = b"[\xd9\x81\x80\x00\x01\x00\x05\x00\x00\x00\x00\ + \x0c_xmpp-server\x04_tcp\x05gmail\x03com\x00\x00!\x00\x01\ + \xc0\x0c\x00!\x00\x01\x00\x00\x03\x84\x00 \x00\x05\x00\x00\ + \x14\x95\x0bxmpp-server\x01l\x06google\x03com\x00\xc0\x0c\x00!\ + \x00\x01\x00\x00\x03\x84\x00%\x00\x14\x00\x00\x14\x95\ + \x04alt3\x0bxmpp-server\x01l\x06google\x03com\x00\ + \xc0\x0c\x00!\x00\x01\x00\x00\x03\x84\x00%\x00\x14\x00\x00\ + \x14\x95\x04alt1\x0bxmpp-server\x01l\x06google\x03com\x00\ + \xc0\x0c\x00!\x00\x01\x00\x00\x03\x84\x00%\x00\x14\x00\x00\ + \x14\x95\x04alt2\x0bxmpp-server\x01l\x06google\x03com\x00\ + \xc0\x0c\x00!\x00\x01\x00\x00\x03\x84\x00%\x00\x14\x00\x00\ + \x14\x95\x04alt4\x0bxmpp-server\x01l\x06google\x03com\x00"; + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 23513, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 5, + nameservers: 0, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::SRV); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], + "_xmpp-server._tcp.gmail.com"); + assert_eq!(packet.answers.len(), 5); + let items = vec![ + (5, 0, 5269, "xmpp-server.l.google.com"), + (20, 0, 5269, "alt3.xmpp-server.l.google.com"), + (20, 0, 5269, "alt1.xmpp-server.l.google.com"), + (20, 0, 5269, "alt2.xmpp-server.l.google.com"), + (20, 0, 5269, "alt4.xmpp-server.l.google.com"), + ]; + for i in 0..5 { + assert_eq!(&packet.answers[i].name.to_string()[..], + "_xmpp-server._tcp.gmail.com"); + assert_eq!(packet.answers[i].cls, C::IN); + assert_eq!(packet.answers[i].ttl, 900); + match *&packet.answers[i].data { + RData::SRV(Record { priority, weight, port, target }) => { + assert_eq!(priority, items[i].0); + assert_eq!(weight, items[i].1); + assert_eq!(port, items[i].2); + assert_eq!(target.to_string(), (items[i].3).to_string()); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/txt.rs b/third_party/rust/dns-parser/src/rdata/txt.rs new file mode 100644 index 0000000000..8f5f5fc26d --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/txt.rs @@ -0,0 +1,125 @@ +use Error; + +#[derive(Debug, Clone)] +pub struct Record<'a> { + bytes: &'a [u8], +} + +#[derive(Debug)] +pub struct RecordIter<'a> { + bytes: &'a [u8], +} + +impl<'a> Iterator for RecordIter<'a> { + type Item = &'a [u8]; + fn next(&mut self) -> Option<&'a [u8]> { + if self.bytes.len() >= 1 { + let len = self.bytes[0] as usize; + debug_assert!(self.bytes.len() >= len+1); + let (head, tail) = self.bytes[1..].split_at(len); + self.bytes = tail; + return Some(head); + } + return None; + } +} + +impl<'a> Record<'a> { + + // Returns iterator over text chunks + pub fn iter(&self) -> RecordIter<'a> { + RecordIter { + bytes: self.bytes, + } + } +} + +impl<'a> super::Record<'a> for Record<'a> { + + const TYPE: isize = 16; + + fn parse(rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + // Just a quick check that record is valid + let len = rdata.len(); + if len < 1 { + return Err(Error::WrongRdataLength); + } + let mut pos = 0; + while pos < len { + let rdlen = rdata[pos] as usize; + pos += 1; + if len < rdlen + pos { + return Err(Error::WrongRdataLength); + } + pos += rdlen; + } + Ok(super::RData::TXT(Record { + bytes: rdata, + })) + } +} + +#[cfg(test)] +mod test { + + use std::str::from_utf8; + + use {Packet, Header}; + use Opcode::*; + use ResponseCode::NoError; + use QueryType as QT; + use QueryClass as QC; + use Class as C; + use RData; + + #[test] + fn parse_response_multiple_strings() { + let response = b"\x06%\x81\x80\x00\x01\x00\x01\x00\x00\x00\x00\ + \x08facebook\x03com\x00\x00\x10\x00\x01\ + \xc0\x0c\x00\x10\x00\x01\x00\x01\x51\x3d\x00\x23\ + \x15\x76\x3d\x73\x70\x66\x31\x20\x72\x65\x64\x69\ + \x72\x65\x63\x74\x3d\x5f\x73\x70\x66\x2e\ + \x0c\x66\x61\x63\x65\x62\x6f\x6f\x6b\x2e\x63\x6f\x6d"; + + let packet = Packet::parse(response).unwrap(); + assert_eq!(packet.header, Header { + id: 1573, + query: false, + opcode: StandardQuery, + authoritative: false, + truncated: false, + recursion_desired: true, + recursion_available: true, + authenticated_data: false, + checking_disabled: false, + response_code: NoError, + questions: 1, + answers: 1, + nameservers: 0, + additional: 0, + }); + assert_eq!(packet.questions.len(), 1); + assert_eq!(packet.questions[0].qtype, QT::TXT); + assert_eq!(packet.questions[0].qclass, QC::IN); + assert_eq!(&packet.questions[0].qname.to_string()[..], "facebook.com"); + assert_eq!(packet.answers.len(), 1); + assert_eq!(&packet.answers[0].name.to_string()[..], "facebook.com"); + assert_eq!(packet.answers[0].multicast_unique, false); + assert_eq!(packet.answers[0].cls, C::IN); + assert_eq!(packet.answers[0].ttl, 86333); + match packet.answers[0].data { + RData::TXT(ref text) => { + assert_eq!(text.iter() + .map(|x| from_utf8(x).unwrap()) + .collect::<Vec<_>>() + .concat(), "v=spf1 redirect=_spf.facebook.com"); + + // also assert boundaries are kept + assert_eq!(text.iter().collect::<Vec<_>>(), + ["v=spf1 redirect=_spf.".as_bytes(), + "facebook.com".as_bytes()]); + } + ref x => panic!("Wrong rdata {:?}", x), + } + } +} diff --git a/third_party/rust/dns-parser/src/rdata/wks.rs b/third_party/rust/dns-parser/src/rdata/wks.rs new file mode 100644 index 0000000000..ff6e5f1881 --- /dev/null +++ b/third_party/rust/dns-parser/src/rdata/wks.rs @@ -0,0 +1,11 @@ +#[derive(Debug, PartialEq, Eq, Clone, Copy)] +pub struct Record; + +impl<'a> super::Record<'a> for Record { + + const TYPE: isize = 11; + + fn parse(_rdata: &'a [u8], _original: &'a [u8]) -> super::RDataResult<'a> { + unimplemented!(); + } +} diff --git a/third_party/rust/dns-parser/src/structs.rs b/third_party/rust/dns-parser/src/structs.rs new file mode 100644 index 0000000000..4f60639047 --- /dev/null +++ b/third_party/rust/dns-parser/src/structs.rs @@ -0,0 +1,50 @@ +use {QueryType, QueryClass, Name, Class, Header, RData}; +use rdata::opt; + + +/// Parsed DNS packet +#[derive(Debug)] +#[allow(missing_docs)] // should be covered by spec +pub struct Packet<'a> { + pub header: Header, + pub questions: Vec<Question<'a>>, + pub answers: Vec<ResourceRecord<'a>>, + pub nameservers: Vec<ResourceRecord<'a>>, + pub additional: Vec<ResourceRecord<'a>>, + /// Optional Pseudo-RR + /// When present it is sent as an RR in the additional section. In this RR + /// the `class` and `ttl` fields store max udp packet size and flags + /// respectively. To keep `ResourceRecord` clean we store the OPT record + /// here. + pub opt: Option<opt::Record<'a>>, +} + +/// A parsed chunk of data in the Query section of the packet +#[derive(Debug)] +#[allow(missing_docs)] // should be covered by spec +pub struct Question<'a> { + pub qname: Name<'a>, + /// Whether or not we prefer unicast responses. + /// This is used in multicast DNS. + pub prefer_unicast: bool, + pub qtype: QueryType, + pub qclass: QueryClass, +} + +/// A single DNS record +/// +/// We aim to provide whole range of DNS records available. But as time is +/// limited we have some types of packets which are parsed and other provided +/// as unparsed slice of bytes. +#[derive(Debug)] +#[allow(missing_docs)] // should be covered by spec +pub struct ResourceRecord<'a> { + pub name: Name<'a>, + /// Whether or not the set of resource records is fully contained in the + /// packet, or whether there will be more resource records in future + /// packets. Only used for multicast DNS. + pub multicast_unique: bool, + pub cls: Class, + pub ttl: u32, + pub data: RData<'a>, +} |