From c0b99d394540b9e689be196f9854ecd255cc064b Mon Sep 17 00:00:00 2001 From: William Desportes Date: Sun, 22 Sep 2024 13:14:18 +0200 Subject: [PATCH] Move to hickory_resolver crate --- Cargo.toml | 2 +- src/lib.rs | 125 ++++++++++++++++++++++++++++------------------------ src/main.rs | 30 ++++++------- 3 files changed, 84 insertions(+), 73 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 3f5fe51..b45d805 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -36,6 +36,6 @@ path = "src/main.rs" [dependencies] rayon = "1.10.0" -hickory-client = { version = "0.24.1", default-features = false } +hickory-resolver = { version = "0.24.1", default-features = false, features = ["tokio-runtime", "dns-over-h3", "dns-over-https", "dns-over-quic"]} rustdns = "0.4.0" weighted-rs = "0.1.3" diff --git a/src/lib.rs b/src/lib.rs index d2a2b2b..52de4d4 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -1,7 +1,6 @@ -use hickory_client::client::{Client, SyncClient}; -use hickory_client::op::DnsResponse; -use hickory_client::rr::{DNSClass, Name, RData, Record, RecordType}; -use hickory_client::tcp::TcpClientConnection; +use hickory_resolver::proto::rr::{RData, Record, RecordType}; +use hickory_resolver::{Name, Resolver}; + use rustdns::util::reverse; use std::net::IpAddr; use std::str::FromStr; @@ -35,19 +34,31 @@ impl fmt::Display for ResolvingError { /** * Resolve a DNS IP adddress (IPv4/IPv6) into a DNS pointer * ``` - * use hickory_client::client::SyncClient; - * use hickory_client::rr::Name; - * use hickory_client::tcp::TcpClientConnection; + * use hickory_resolver::{Name, Resolver}; + * use hickory_resolver::config::{NameServerConfigGroup, ResolverOpts, ResolverConfig}; + * use std::time::Duration; + * use std::net::IpAddr; + * use std::str::FromStr;// IpAddr::from_str * * use dns_ptr_resolver::{get_ptr, ResolvedResult}; * - * let server = "1.1.1.1:53".parse().expect("To parse"); - * let conn = TcpClientConnection::with_timeout(server, std::time::Duration::new(5, 0)).unwrap(); - * let client = SyncClient::new(conn); + * let server_ip = "1.1.1.1"; + * + * let server = NameServerConfigGroup::from_ips_clear( + * &[IpAddr::from_str(server_ip).unwrap()], + * 53,// Port 53 + * true, + * ); + * let config = ResolverConfig::from_parts(None, vec![], server); + * let mut options = ResolverOpts::default(); + * options.timeout = Duration::from_secs(5); + * options.attempts = 1; // One try + * + * let resolver = Resolver::new(config, options).unwrap(); * let query_address = "8.8.8.8".parse().expect("To parse"); * * assert_eq!( - * get_ptr(query_address, client).unwrap(), + * get_ptr(query_address, resolver).unwrap(), * ResolvedResult { * query: Name::from_str_relaxed("8.8.8.8.in-addr.arpa.").unwrap(), * result: Some(Name::from_str_relaxed("dns.google.").unwrap()), @@ -56,13 +67,10 @@ impl fmt::Display for ResolvingError { * ); * ``` */ -pub fn get_ptr( - ip_address: IpAddr, - client: SyncClient, -) -> Result { +pub fn get_ptr(ip_address: IpAddr, resolver: Resolver) -> Result { // Specify the name, note the final '.' which specifies it's an FQDN match Name::from_str(&reverse(ip_address)) { - Ok(name) => ptr_resolve(name, client), + Ok(name) => ptr_resolve(name, resolver), Err(err) => Err(ResolvingError { message: format!( "Something went wrong while building the name ({}): {}", @@ -76,20 +84,32 @@ pub fn get_ptr( /** * This will resolve a name into its DNS pointer value * ``` - * use hickory_client::client::SyncClient; - * use hickory_client::rr::Name; - * use hickory_client::tcp::TcpClientConnection; + * use hickory_resolver::{Name, Resolver}; + * use hickory_resolver::config::{NameServerConfigGroup, ResolverOpts, ResolverConfig}; + * use std::time::Duration; + * use std::net::IpAddr; + * use std::str::FromStr;// IpAddr::from_str * * use dns_ptr_resolver::{ptr_resolve, ResolvedResult}; * - * let server = "8.8.8.8:53".parse().expect("To parse"); - * let conn = TcpClientConnection::with_timeout(server, std::time::Duration::new(5, 0)).unwrap(); - * let client = SyncClient::new(conn); + * let server_ip = "8.8.8.8"; + * + * let server = NameServerConfigGroup::from_ips_clear( + * &[IpAddr::from_str(server_ip).unwrap()], + * 53,// Port 53 + * true, + * ); + * let config = ResolverConfig::from_parts(None, vec![], server); + * let mut options = ResolverOpts::default(); + * options.timeout = Duration::from_secs(5); + * options.attempts = 1; // One try + * + * let resolver = Resolver::new(config, options).unwrap(); * * let name_to_resolve = Name::from_str_relaxed("1.1.1.1.in-addr.arpa.").unwrap(); * * assert_eq!( - * ptr_resolve(name_to_resolve.clone(), client).unwrap(), + * ptr_resolve(name_to_resolve.clone(), resolver).unwrap(), * ResolvedResult { * query: name_to_resolve, * result: Some(Name::from_str_relaxed("one.one.one.one.").unwrap()), @@ -98,11 +118,8 @@ pub fn get_ptr( * ); * ``` */ -pub fn ptr_resolve( - name: Name, - client: SyncClient, -) -> Result { - let response: DnsResponse = match client.query(&name, DNSClass::IN, RecordType::PTR) { +pub fn ptr_resolve(name: Name, resolver: Resolver) -> Result { + let response = match resolver.lookup(name.clone(), RecordType::PTR) { Ok(res) => res, Err(err) => { return Err(ResolvingError { @@ -111,7 +128,7 @@ pub fn ptr_resolve( } }; - let answers: &[Record] = response.answers(); + let answers: &[Record] = response.records(); if answers.len() == 0 { return Ok(ResolvedResult { @@ -134,7 +151,7 @@ pub fn ptr_resolve( // 75.7.246.87.in-addr.arpa. 3600 IN CNAME 75.0-255.7.246.87.in-addr.arpa. // 75.0-255.7.246.87.in-addr.arpa. 86400 IN PTR bulbank.linkbg.com. Some(RData::CNAME(res)) => { - return ptr_resolve(res.to_lowercase(), client); + return ptr_resolve(res.to_lowercase(), resolver); } Some(res) => { return Err(ResolvingError { @@ -152,28 +169,23 @@ pub fn ptr_resolve( #[cfg(test)] mod test { use super::*; - use std::process; + use hickory_resolver::config::{NameServerConfigGroup, ResolverConfig, ResolverOpts}; use std::time::Duration; #[test] fn test_get_ptr() { - let server = "8.8.8.8:53".parse().expect("To parse"); - let conn = match TcpClientConnection::with_timeout(server, Duration::new(5, 0)) { - Ok(conn) => conn, - Err(err) => { - eprintln!( - "Something went wrong with the UDP client connection: {}", - err - ); - process::exit(1); - } - }; - let client = SyncClient::new(conn); + let server = NameServerConfigGroup::google(); + let config = ResolverConfig::from_parts(None, vec![], server); + let mut options = ResolverOpts::default(); + options.timeout = Duration::from_secs(5); + options.attempts = 1; // One try + + let resolver = Resolver::new(config, options).unwrap(); let query_address = "8.8.8.8".parse().expect("To parse"); assert_eq!( - get_ptr(query_address, client).unwrap(), + get_ptr(query_address, resolver).unwrap(), ResolvedResult { query: Name::from_str_relaxed("8.8.8.8.in-addr.arpa.").unwrap(), result: Some(Name::from_str_relaxed("dns.google.").unwrap()), @@ -184,23 +196,22 @@ mod test { #[test] fn test_ptr_resolve() { - let server = "1.1.1.1:53".parse().expect("To parse"); - let conn = match TcpClientConnection::with_timeout(server, Duration::new(5, 0)) { - Ok(conn) => conn, - Err(err) => { - eprintln!( - "Something went wrong with the UDP client connection: {}", - err - ); - process::exit(1); - } - }; - let client = SyncClient::new(conn); + let server = NameServerConfigGroup::from_ips_clear( + &[IpAddr::from_str("1.1.1.1").unwrap()], + 53, + true, + ); + let config = ResolverConfig::from_parts(None, vec![], server); + let mut options = ResolverOpts::default(); + options.timeout = Duration::from_secs(5); + options.attempts = 1; // One try + + let resolver = Resolver::new(config, options).unwrap(); let name_to_resolve = Name::from_str_relaxed("1.1.1.1.in-addr.arpa.").unwrap(); assert_eq!( - ptr_resolve(name_to_resolve.clone(), client).unwrap(), + ptr_resolve(name_to_resolve.clone(), resolver).unwrap(), ResolvedResult { query: name_to_resolve, result: Some(Name::from_str_relaxed("one.one.one.one.").unwrap()), diff --git a/src/main.rs b/src/main.rs index 91fc43f..14131c3 100644 --- a/src/main.rs +++ b/src/main.rs @@ -1,8 +1,8 @@ use std::{env, thread}; use dns_ptr_resolver::get_ptr; -use hickory_client::client::SyncClient; -use hickory_client::tcp::TcpClientConnection; +use hickory_resolver::config::{NameServerConfigGroup, ResolverConfig, ResolverOpts}; +use hickory_resolver::Resolver; use rayon::prelude::*; use std::fs::read_to_string; use std::net::{IpAddr, SocketAddr}; @@ -75,19 +75,19 @@ fn resolve_file(filename: &str, dns_servers: Vec<&str>) { ips.into_par_iter() .enumerate() .for_each(|(_i, to_resolve)| { - let conn = - match TcpClientConnection::with_timeout(to_resolve.server, Duration::new(5, 0)) { - Ok(conn) => conn, - Err(err) => { - eprintln!( - "Something went wrong with the UDP client connection: {}", - err - ); - process::exit(1); - } - }; - let client = SyncClient::new(conn); - let ptr_result = get_ptr(to_resolve.address, client); + let server = NameServerConfigGroup::from_ips_clear( + &[to_resolve.server.ip()], + to_resolve.server.port(), + true, + ); + + let config = ResolverConfig::from_parts(None, vec![], server); + let mut options = ResolverOpts::default(); + options.timeout = Duration::from_secs(5); + options.attempts = 1; // One try + + let resolver = Resolver::new(config, options).unwrap(); + let ptr_result = get_ptr(to_resolve.address, resolver); match ptr_result { Ok(ptr) => match ptr.result { Some(res) => println!("{} # {}", to_resolve.address, res),