123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 |
- //! Request generation based on the user’s input arguments.
- use crate::connect::TransportType;
- use crate::resolve::{ResolverType, ResolverLookupError};
- use crate::txid::TxidGenerator;
- /// All the information necessary to generate requests for one or more
- /// queries, nameservers, or transport types.
- #[derive(PartialEq, Debug)]
- pub struct RequestGenerator {
- /// The input parameter matrix.
- pub inputs: Inputs,
- /// How to generate transaction IDs.
- pub txid_generator: TxidGenerator,
- /// Whether to OPT in to DNS extensions.
- pub edns: UseEDNS,
- /// Other weird protocol options.
- pub protocol_tweaks: ProtocolTweaks,
- }
- /// Which things the user has specified they want queried.
- #[derive(PartialEq, Debug, Default)]
- pub struct Inputs {
- /// The list of domain names to query.
- pub domains: Vec<dns::Labels>,
- /// The list of DNS record types to query for.
- pub record_types: Vec<dns::record::RecordType>,
- /// The list of DNS classes to query for.
- pub classes: Vec<dns::QClass>,
- /// The list of resolvers to send queries to.
- pub resolver_types: Vec<ResolverType>,
- /// The list of transport types to send queries over.
- pub transport_types: Vec<TransportType>,
- }
- /// Weird protocol options that are allowed by the spec but are not common.
- #[derive(PartialEq, Debug, Default, Copy, Clone)]
- pub struct ProtocolTweaks {
- /// Set the `AA` (Authoritative Answer) flag in the header of each request.
- pub set_authoritative_flag: bool,
- /// Set the `AD` (Authentic Data) flag in the header of each request.
- pub set_authentic_flag: bool,
- /// Set the `CD` (Checking Disabled) flag in the header of each request.
- pub set_checking_disabled_flag: bool,
- /// Set the buffer size field in the OPT record of each request.
- pub udp_payload_size: Option<u16>,
- }
- /// Whether to send or display OPT packets.
- #[derive(PartialEq, Debug, Copy, Clone)]
- pub enum UseEDNS {
- /// Do not send an OPT query in requests, and do not display them.
- Disable,
- /// Send an OPT query in requests, but hide the result. This is the
- /// default, because the information is usually not useful to the user.
- SendAndHide,
- /// Send an OPT query in requests, _and_ display any OPT records in the
- /// response we receive.
- SendAndShow,
- }
- /// The entry type for `RequestGenerator`: a transport to send a request, and
- /// a list of one or more DNS queries to send over it, as determined by the
- /// search path in the resolver.
- pub type RequestSet = (Box<dyn dns_transport::Transport>, Vec<dns::Request>);
- impl RequestGenerator {
- /// Iterate through the inputs matrix, returning pairs of DNS request list
- /// and the details of the transport to send them down.
- pub fn generate(self) -> Result<Vec<RequestSet>, ResolverLookupError> {
- let mut requests = Vec::new();
- let resolvers = self.inputs.resolver_types.into_iter()
- .map(ResolverType::obtain)
- .collect::<Result<Vec<_>, _>>()?;
- for domain in &self.inputs.domains {
- for qtype in self.inputs.record_types.iter().copied() {
- for qclass in self.inputs.classes.iter().copied() {
- for resolver in &resolvers {
- for transport_type in &self.inputs.transport_types {
- let mut flags = dns::Flags::query();
- self.protocol_tweaks.set_request_flags(&mut flags);
- let mut additional = None;
- if self.edns.should_send() {
- let mut opt = dns::Request::additional_record();
- self.protocol_tweaks.set_request_opt_fields(&mut opt);
- additional = Some(opt);
- }
- let nameserver = resolver.nameserver();
- let transport = transport_type.make_transport(nameserver);
- let mut request_list = Vec::new();
- for qname in resolver.name_list(domain) {
- let transaction_id = self.txid_generator.generate();
- let query = dns::Query { qname, qtype, qclass };
- let request = dns::Request { transaction_id, flags, query, additional: additional.clone() };
- request_list.push(request);
- }
- requests.push((transport, request_list));
- }
- }
- }
- }
- }
- Ok(requests)
- }
- }
- impl UseEDNS {
- /// Whether the user wants to send OPT records.
- pub fn should_send(self) -> bool {
- self != Self::Disable
- }
- /// Whether the user wants to display sent OPT records.
- pub fn should_show(self) -> bool {
- self == Self::SendAndShow
- }
- }
- impl ProtocolTweaks {
- /// Sets fields in the DNS flags based on the user’s requested tweaks.
- pub fn set_request_flags(self, flags: &mut dns::Flags) {
- if self.set_authoritative_flag {
- flags.authoritative = true;
- }
- if self.set_authentic_flag {
- flags.authentic_data = true;
- }
- if self.set_checking_disabled_flag {
- flags.checking_disabled = true;
- }
- }
- /// Set the payload size field in the outgoing OPT record, if the user has
- /// requested to do so.
- pub fn set_request_opt_fields(self, opt: &mut dns::record::OPT) {
- if let Some(bufsize) = self.udp_payload_size {
- opt.udp_payload_size = bufsize;
- }
- }
- }
|