wire.rs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364
  1. //! Parsing the DNS wire protocol.
  2. pub(crate) use std::io::Cursor;
  3. pub(crate) use byteorder::{BigEndian, ReadBytesExt, WriteBytesExt};
  4. use std::io;
  5. use log::{error, info, debug};
  6. use crate::record::{Record, OPT};
  7. use crate::strings::{ReadLabels, WriteLabels};
  8. use crate::types::*;
  9. impl Request {
  10. /// Converts this request to a vector of bytes.
  11. pub fn to_bytes(&self) -> io::Result<Vec<u8>> {
  12. let mut bytes = Vec::with_capacity(32);
  13. bytes.write_u16::<BigEndian>(self.transaction_id)?;
  14. bytes.write_u16::<BigEndian>(self.flags.to_u16())?;
  15. bytes.write_u16::<BigEndian>(self.queries.len() as u16)?;
  16. bytes.write_u16::<BigEndian>(0)?; // usually answers
  17. bytes.write_u16::<BigEndian>(0)?; // usually authority RRs
  18. bytes.write_u16::<BigEndian>(if self.additional.is_some() { 1 } else { 0 })?; // additional RRs
  19. for query in &self.queries {
  20. bytes.write_labels(&query.qname)?;
  21. bytes.write_u16::<BigEndian>(query.qtype)?;
  22. bytes.write_u16::<BigEndian>(query.qclass.to_u16())?;
  23. }
  24. if let Some(opt) = &self.additional {
  25. bytes.write_u8(0)?; // usually a name
  26. bytes.write_u16::<BigEndian>(OPT::RR_TYPE)?;
  27. bytes.extend(opt.to_bytes()?);
  28. }
  29. Ok(bytes)
  30. }
  31. /// Returns the OPT record to be sent as part of requests.
  32. pub fn additional_record() -> OPT {
  33. OPT {
  34. udp_payload_size: 512,
  35. higher_bits: 0,
  36. edns0_version: 0,
  37. flags: 0,
  38. data: Vec::new(),
  39. }
  40. }
  41. }
  42. impl Response {
  43. /// Reads bytes off of the given slice, parsing them into a response.
  44. pub fn from_bytes(bytes: &[u8]) -> Result<Self, WireError> {
  45. debug!("Parsing bytes -> {:?}", bytes);
  46. let mut c = Cursor::new(bytes);
  47. let transaction_id = c.read_u16::<BigEndian>()?;
  48. let flags = Flags::from_u16(c.read_u16::<BigEndian>()?);
  49. debug!("Read flags: {:#?}", flags);
  50. let query_count = c.read_u16::<BigEndian>()?;
  51. let answer_count = c.read_u16::<BigEndian>()?;
  52. let authority_count = c.read_u16::<BigEndian>()?;
  53. let additional_count = c.read_u16::<BigEndian>()?;
  54. let mut queries = Vec::new();
  55. debug!("Reading {}x query from response", query_count);
  56. for _ in 0 .. query_count {
  57. let qname = c.read_labels()?;
  58. queries.push(Query::from_bytes(qname, &mut c)?);
  59. }
  60. let mut answers = Vec::new();
  61. debug!("Reading {}x answer from response", answer_count);
  62. for _ in 0 .. answer_count {
  63. let qname = c.read_labels()?;
  64. answers.push(Answer::from_bytes(qname, &mut c)?);
  65. }
  66. let mut authorities = Vec::new();
  67. debug!("Reading {}x authority from response", authority_count);
  68. for _ in 0 .. authority_count {
  69. let qname = c.read_labels()?;
  70. authorities.push(Answer::from_bytes(qname, &mut c)?);
  71. }
  72. let mut additionals = Vec::new();
  73. debug!("Reading {}x additional answer from response", additional_count);
  74. for _ in 0 .. additional_count {
  75. let qname = c.read_labels()?;
  76. additionals.push(Answer::from_bytes(qname, &mut c)?);
  77. }
  78. Ok(Response { transaction_id, flags, queries, answers, authorities, additionals })
  79. }
  80. }
  81. impl Query {
  82. /// Reads bytes from the given cursor, and parses them into a query with
  83. /// the given domain name.
  84. fn from_bytes(qname: String, c: &mut Cursor<&[u8]>) -> Result<Self, WireError> {
  85. let qtype = c.read_u16::<BigEndian>()?;
  86. let qclass = QClass::from_u16(c.read_u16::<BigEndian>()?);
  87. Ok(Query { qtype, qclass, qname })
  88. }
  89. }
  90. impl Answer {
  91. /// Reads bytes from the given cursor, and parses them into an answer with
  92. /// the given domain name.
  93. fn from_bytes(qname: String, c: &mut Cursor<&[u8]>) -> Result<Self, WireError> {
  94. let qtype = c.read_u16::<BigEndian>()?;
  95. if qtype == OPT::RR_TYPE {
  96. let opt = OPT::read(c)?;
  97. Ok(Answer::Pseudo { qname, opt })
  98. }
  99. else {
  100. let qclass = QClass::from_u16(c.read_u16::<BigEndian>()?);
  101. let ttl = c.read_u32::<BigEndian>()?;
  102. let len = c.read_u16::<BigEndian>()?;
  103. let record = Record::from_bytes(qtype, len, c)?;
  104. Ok(Answer::Standard { qclass, qname, record, ttl })
  105. }
  106. }
  107. }
  108. impl Record {
  109. /// Reads at most `len` bytes from the given curser, and parses them into
  110. /// a record structure depending on the type number, which has already been read.
  111. fn from_bytes(qtype: TypeInt, len: u16, c: &mut Cursor<&[u8]>) -> Result<Record, WireError> {
  112. use crate::record::*;
  113. macro_rules! try_record {
  114. ($record:tt) => {
  115. if $record::RR_TYPE == qtype {
  116. info!("Deciphering {} record (type {}, len {})", $record::NAME, qtype, len);
  117. return Wire::read(len, c).map(Record::$record)
  118. }
  119. }
  120. }
  121. // Try all the records, one type at a time, returning early if the
  122. // type number matches.
  123. try_record!(A);
  124. try_record!(AAAA);
  125. try_record!(CAA);
  126. try_record!(CNAME);
  127. try_record!(MX);
  128. try_record!(NS);
  129. // OPT is handled separately
  130. try_record!(PTR);
  131. try_record!(SOA);
  132. try_record!(SRV);
  133. try_record!(TXT);
  134. // Otherwise, collect the bytes into a vector and return an unknown
  135. // record type.
  136. let mut bytes = Vec::new();
  137. for _ in 0 .. len {
  138. bytes.push(c.read_u8()?);
  139. }
  140. let type_number = UnknownQtype::from(qtype);
  141. Ok(Record::Other { type_number, bytes })
  142. }
  143. }
  144. impl QClass {
  145. fn from_u16(uu: u16) -> Self {
  146. match uu {
  147. 0x0001 => QClass::IN,
  148. 0x0003 => QClass::CH,
  149. 0x0004 => QClass::HS,
  150. _ => QClass::Other(uu),
  151. }
  152. }
  153. fn to_u16(self) -> u16 {
  154. match self {
  155. QClass::IN => 0x0001,
  156. QClass::CH => 0x0003,
  157. QClass::HS => 0x0004,
  158. QClass::Other(uu) => uu,
  159. }
  160. }
  161. }
  162. /// Determines the record type number to signify a record with the given name.
  163. pub fn find_qtype_number(record_type: &str) -> Option<TypeInt> {
  164. use crate::record::*;
  165. macro_rules! try_record {
  166. ($record:tt) => {
  167. if $record::NAME == record_type {
  168. return Some($record::RR_TYPE);
  169. }
  170. }
  171. }
  172. try_record!(A);
  173. try_record!(AAAA);
  174. try_record!(CAA);
  175. try_record!(CNAME);
  176. try_record!(MX);
  177. try_record!(NS);
  178. // OPT is elsewhere
  179. try_record!(PTR);
  180. try_record!(SOA);
  181. try_record!(SRV);
  182. try_record!(TXT);
  183. None
  184. }
  185. impl Flags {
  186. /// The set of flags that represents a query packet.
  187. pub fn query() -> Self {
  188. Self::from_u16(0b_0000_0001_0000_0000)
  189. }
  190. /// Converts the flags into a two-byte number.
  191. pub fn to_u16(self) -> u16 { // 0123 4567 89AB CDEF
  192. let mut bits = 0b_0000_0000_0000_0000;
  193. if self.response { bits += 0b_1000_0000_0000_0000; }
  194. match self.opcode {
  195. _ => { bits += 0b_0000_0000_0000_0000; }
  196. }
  197. if self.authoritative { bits += 0b_0000_0100_0000_0000; }
  198. if self.truncated { bits += 0b_0000_0010_0000_0000; }
  199. if self.recursion_desired { bits += 0b_0000_0001_0000_0000; }
  200. if self.recursion_available { bits += 0b_0000_0000_1000_0000; }
  201. // (the Z bit is reserved) 0b_0000_0000_0100_0000
  202. if self.authentic_data { bits += 0b_0000_0000_0010_0000; }
  203. if self.checking_disabled { bits += 0b_0000_0000_0001_0000; }
  204. bits
  205. }
  206. /// Extracts the flags from the given two-byte number.
  207. pub fn from_u16(bits: u16) -> Self {
  208. let has_bit = |bit| { bits & bit == bit };
  209. Flags {
  210. response: has_bit(0b_1000_0000_0000_0000),
  211. opcode: 0,
  212. authoritative: has_bit(0b_0000_0100_0000_0000),
  213. truncated: has_bit(0b_0000_0010_0000_0000),
  214. recursion_desired: has_bit(0b_0000_0001_0000_0000),
  215. recursion_available: has_bit(0b_0000_0000_1000_0000),
  216. authentic_data: has_bit(0b_0000_0000_0010_0000),
  217. checking_disabled: has_bit(0b_0000_0000_0001_0000),
  218. error_code: ErrorCode::from_bits(bits & 0b_1111),
  219. }
  220. }
  221. }
  222. impl ErrorCode {
  223. /// Extracts the rcode from the last four bits of the flags field.
  224. fn from_bits(bits: u16) -> Option<Self> {
  225. match bits {
  226. 0 => None,
  227. 1 => Some(Self::FormatError),
  228. 2 => Some(Self::ServerFailure),
  229. 3 => Some(Self::NXDomain),
  230. 4 => Some(Self::NotImplemented),
  231. 5 => Some(Self::QueryRefused),
  232. 16 => Some(Self::BadVersion),
  233. n => Some(Self::Other(n)),
  234. }
  235. }
  236. }
  237. /// Trait for decoding DNS record structures from bytes read over the wire.
  238. pub trait Wire: Sized {
  239. /// This record’s type as a string, such as `"A"` or `"CNAME"`.
  240. const NAME: &'static str;
  241. /// The number signifying that a record is of this type.
  242. /// See <https://www.iana.org/assignments/dns-parameters/dns-parameters.xhtml#dns-parameters-4>
  243. const RR_TYPE: u16;
  244. /// Read at most `len` bytes from the given `Cursor`. This cursor travels
  245. /// throughout the complete data — by this point, we have read the entire
  246. /// response into a buffer.
  247. fn read(len: u16, c: &mut Cursor<&[u8]>) -> Result<Self, WireError>;
  248. }
  249. /// Helper macro to get the qtype number of a record type at compile-time.
  250. ///
  251. /// # Examples
  252. ///
  253. /// ```
  254. /// use dns::{qtype, record::MX};
  255. ///
  256. /// assert_eq!(15, qtype!(MX));
  257. /// ```
  258. #[macro_export]
  259. macro_rules! qtype {
  260. ($type:ty) => {
  261. <$type as $crate::Wire>::RR_TYPE
  262. }
  263. }
  264. /// Something that can go wrong deciphering a record.
  265. #[derive(PartialEq, Debug)]
  266. pub enum WireError {
  267. /// There was an IO error reading from the cursor.
  268. /// Almost all the time, this means that the buffer was too short.
  269. IO,
  270. // (io::Error is not PartialEq so we don’t propagate it)
  271. /// When this record expected the data to be a certain size, but it was
  272. /// a different one.
  273. WrongLength {
  274. /// The expected size.
  275. expected: u16,
  276. /// The size that was actually received.
  277. got: u16,
  278. },
  279. /// When the data contained a string containing a cycle of pointers.
  280. /// Contains the vector of indexes that was being checked.
  281. TooMuchRecursion(Vec<u16>),
  282. /// When the data contained a string with a pointer to an index outside of
  283. /// the packet. Contains the invalid index.
  284. OutOfBounds(u16),
  285. }
  286. impl From<io::Error> for WireError {
  287. fn from(ioe: io::Error) -> Self {
  288. error!("IO error -> {:?}", ioe);
  289. WireError::IO
  290. }
  291. }