requests.rs 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  1. //! Request generation based on the user’s input arguments.
  2. use crate::connect::TransportType;
  3. use crate::resolve::{ResolverType, ResolverLookupError};
  4. use crate::txid::TxidGenerator;
  5. /// All the information necessary to generate requests for one or more
  6. /// queries, nameservers, or transport types.
  7. #[derive(PartialEq, Debug)]
  8. pub struct RequestGenerator {
  9. /// The input parameter matrix.
  10. pub inputs: Inputs,
  11. /// How to generate transaction IDs.
  12. pub txid_generator: TxidGenerator,
  13. /// Whether to OPT in to DNS extensions.
  14. pub edns: UseEDNS,
  15. /// Other weird protocol options.
  16. pub protocol_tweaks: ProtocolTweaks,
  17. }
  18. /// Which things the user has specified they want queried.
  19. #[derive(PartialEq, Debug, Default)]
  20. pub struct Inputs {
  21. /// The list of domain names to query.
  22. pub domains: Vec<dns::Labels>,
  23. /// The list of DNS record types to query for.
  24. pub record_types: Vec<dns::record::RecordType>,
  25. /// The list of DNS classes to query for.
  26. pub classes: Vec<dns::QClass>,
  27. /// The list of resolvers to send queries to.
  28. pub resolver_types: Vec<ResolverType>,
  29. /// The list of transport types to send queries over.
  30. pub transport_types: Vec<TransportType>,
  31. }
  32. /// Weird protocol options that are allowed by the spec but are not common.
  33. #[derive(PartialEq, Debug, Default, Copy, Clone)]
  34. pub struct ProtocolTweaks {
  35. /// Set the `AA` (Authoritative Answer) flag in the header of each request.
  36. pub set_authoritative_flag: bool,
  37. /// Set the `AD` (Authentic Data) flag in the header of each request.
  38. pub set_authentic_flag: bool,
  39. /// Set the `CD` (Checking Disabled) flag in the header of each request.
  40. pub set_checking_disabled_flag: bool,
  41. /// Set the buffer size field in the OPT record of each request.
  42. pub udp_payload_size: Option<u16>,
  43. }
  44. /// Whether to send or display OPT packets.
  45. #[derive(PartialEq, Debug, Copy, Clone)]
  46. pub enum UseEDNS {
  47. /// Do not send an OPT query in requests, and do not display them.
  48. Disable,
  49. /// Send an OPT query in requests, but hide the result. This is the
  50. /// default, because the information is usually not useful to the user.
  51. SendAndHide,
  52. /// Send an OPT query in requests, _and_ display any OPT records in the
  53. /// response we receive.
  54. SendAndShow,
  55. }
  56. /// The entry type for `RequestGenerator`: a transport to send a request, and
  57. /// a list of one or more DNS queries to send over it, as determined by the
  58. /// search path in the resolver.
  59. pub type RequestSet = (Box<dyn dns_transport::Transport>, Vec<dns::Request>);
  60. impl RequestGenerator {
  61. /// Iterate through the inputs matrix, returning pairs of DNS request list
  62. /// and the details of the transport to send them down.
  63. pub fn generate(self) -> Result<Vec<RequestSet>, ResolverLookupError> {
  64. let mut requests = Vec::new();
  65. let resolvers = self.inputs.resolver_types.into_iter()
  66. .map(ResolverType::obtain)
  67. .collect::<Result<Vec<_>, _>>()?;
  68. for domain in &self.inputs.domains {
  69. for qtype in self.inputs.record_types.iter().copied() {
  70. for qclass in self.inputs.classes.iter().copied() {
  71. for resolver in &resolvers {
  72. for transport_type in &self.inputs.transport_types {
  73. let mut flags = dns::Flags::query();
  74. self.protocol_tweaks.set_request_flags(&mut flags);
  75. let mut additional = None;
  76. if self.edns.should_send() {
  77. let mut opt = dns::Request::additional_record();
  78. self.protocol_tweaks.set_request_opt_fields(&mut opt);
  79. additional = Some(opt);
  80. }
  81. let nameserver = resolver.nameserver();
  82. let transport = transport_type.make_transport(nameserver);
  83. let mut request_list = Vec::new();
  84. for qname in resolver.name_list(domain) {
  85. let transaction_id = self.txid_generator.generate();
  86. let query = dns::Query { qname, qtype, qclass };
  87. let request = dns::Request { transaction_id, flags, query, additional: additional.clone() };
  88. request_list.push(request);
  89. }
  90. requests.push((transport, request_list));
  91. }
  92. }
  93. }
  94. }
  95. }
  96. Ok(requests)
  97. }
  98. }
  99. impl UseEDNS {
  100. /// Whether the user wants to send OPT records.
  101. pub fn should_send(self) -> bool {
  102. self != Self::Disable
  103. }
  104. /// Whether the user wants to display sent OPT records.
  105. pub fn should_show(self) -> bool {
  106. self == Self::SendAndShow
  107. }
  108. }
  109. impl ProtocolTweaks {
  110. /// Sets fields in the DNS flags based on the user’s requested tweaks.
  111. pub fn set_request_flags(self, flags: &mut dns::Flags) {
  112. if self.set_authoritative_flag {
  113. flags.authoritative = true;
  114. }
  115. if self.set_authentic_flag {
  116. flags.authentic_data = true;
  117. }
  118. if self.set_checking_disabled_flag {
  119. flags.checking_disabled = true;
  120. }
  121. }
  122. /// Set the payload size field in the outgoing OPT record, if the user has
  123. /// requested to do so.
  124. pub fn set_request_opt_fields(self, opt: &mut dns::record::OPT) {
  125. if let Some(bufsize) = self.udp_payload_size {
  126. opt.udp_payload_size = bufsize;
  127. }
  128. }
  129. }