requests.rs 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. use crate::connect::TransportType;
  2. use crate::resolve::Resolver;
  3. use crate::txid::TxidGenerator;
  4. /// All the information necessary to generate requests for one or more
  5. /// queries, nameservers, or transport types.
  6. #[derive(PartialEq, Debug)]
  7. pub struct RequestGenerator {
  8. /// The input parameter matrix.
  9. pub inputs: Inputs,
  10. /// How to generate transaction IDs.
  11. pub txid_generator: TxidGenerator,
  12. /// Whether to OPT in to DNS extensions.
  13. pub edns: UseEDNS,
  14. /// Other weird protocol options.
  15. pub protocol_tweaks: ProtocolTweaks,
  16. }
  17. /// Which things the user has specified they want queried.
  18. #[derive(PartialEq, Debug, Default)]
  19. pub struct Inputs {
  20. /// The list of domain names to query.
  21. pub domains: Vec<String>,
  22. /// The list of DNS record types to query for.
  23. pub types: Vec<u16>,
  24. /// The list of DNS classes to query for.
  25. pub classes: Vec<dns::QClass>,
  26. /// The list of resolvers to send queries to.
  27. pub resolvers: Vec<Resolver>,
  28. /// The list of transport types to send queries over.
  29. pub transport_types: Vec<TransportType>,
  30. }
  31. /// Weird protocol options that are allowed by the spec but are not common.
  32. #[derive(PartialEq, Debug, Default)]
  33. pub struct ProtocolTweaks {
  34. /// Set the `AD` flag (Authentic Data) in the header of each request.
  35. pub set_authentic_flag: bool,
  36. }
  37. /// Whether to send or display OPT packets.
  38. #[derive(PartialEq, Debug, Copy, Clone)]
  39. pub enum UseEDNS {
  40. /// Do not send an OPT query in requests, and do not display them.
  41. Disable,
  42. /// Send an OPT query in requests, but hide the result. This is the
  43. /// default, because the information is usually not useful to the user.
  44. SendAndHide,
  45. /// Send an OPT query in requests, _and_ display any OPT records in the
  46. /// response we receive.
  47. SendAndShow,
  48. }
  49. impl RequestGenerator {
  50. /// Iterate through the inputs matrix, returning pairs of DNS requests and
  51. /// the details of the transport to send them down.
  52. pub fn generate(self) -> Vec<(dns::Request, Box<dyn dns_transport::Transport>)> {
  53. let nameservers = self.inputs.resolvers.into_iter()
  54. .map(|e| e.lookup().expect("Failed to get nameserver").expect("No nameserver found"))
  55. .collect::<Vec<_>>();
  56. let mut requests = Vec::new();
  57. for domain in &self.inputs.domains {
  58. for qtype in self.inputs.types.iter().copied() {
  59. for qclass in self.inputs.classes.iter().copied() {
  60. for nameserver in &nameservers {
  61. for transport_type in &self.inputs.transport_types {
  62. let transaction_id = self.txid_generator.generate();
  63. let mut flags = dns::Flags::query();
  64. if self.protocol_tweaks.set_authentic_flag {
  65. flags.authentic_data = true;
  66. }
  67. let mut additional = None;
  68. if self.edns.should_send() {
  69. additional = Some(dns::Request::additional_record());
  70. }
  71. let query = dns::Query { qname: domain.clone(), qtype, qclass };
  72. let request = dns::Request { transaction_id, flags, query, additional };
  73. let transport = transport_type.make_transport(nameserver.clone());
  74. requests.push((request, transport));
  75. }
  76. }
  77. }
  78. }
  79. }
  80. requests
  81. }
  82. }
  83. impl UseEDNS {
  84. /// Whether the user wants to send OPT records.
  85. pub fn should_send(self) -> bool {
  86. self != Self::Disable
  87. }
  88. /// Whether the user wants to display sent OPT records.
  89. pub fn should_show(self) -> bool {
  90. self == Self::SendAndShow
  91. }
  92. }