use std::{fmt::Display, future::pending, io, net::SocketAddr};
use once_cell::sync::Lazy;
use hickory_resolver::{TokioResolver, net::runtime::TokioRuntimeProvider, proto::rr::IntoName};
static GLOBAL_DNS_RESOLVER: Lazy<TokioResolver> = Lazy::new(|| {
use std::sync::{Arc, Condvar, Mutex};
use std::thread;
let pair = Arc::new((Mutex::new(None::<TokioResolver>), Condvar::new()));
let pair2 = pair.clone();
thread::spawn(move || {
let runtime = tokio::runtime::Runtime::new().expect("failed to launch Runtime");
let resolver = {
#[cfg(any(unix, windows))]
{
TokioResolver::builder(TokioRuntimeProvider::default())
.map(|builder| builder.build().unwrap())
}
#[cfg(not(any(unix, windows)))]
{
use hickory_resolver::config::{GOOGLE, ResolverConfig, ResolverOpts};
TokioResolver::new(
ResolverConfig::udp_and_tcp(&GOOGLE),
ResolverOpts::default(),
runtime.handle().clone(),
)
}
};
let (lock, cvar) = &*pair2;
let mut started = lock.lock().unwrap();
let resolver = resolver.expect("failed to create hickory-resolver");
*started = Some(resolver);
cvar.notify_one();
drop(started);
runtime.block_on(pending::<()>())
});
let (lock, cvar) = &*pair;
let mut resolver = lock.lock().unwrap();
while resolver.is_none() {
resolver = cvar.wait(resolver).unwrap();
}
let resolver = resolver.take();
resolver.expect("resolver should not be none")
});
pub async fn resolve<N: IntoName + Display + 'static>(
host: N,
port: u16,
) -> io::Result<Vec<SocketAddr>> {
let name = host.to_string();
let result = GLOBAL_DNS_RESOLVER.lookup_ip(host).await;
result
.map_err(move |err| {
io::Error::new(
io::ErrorKind::AddrNotAvailable,
format!("dns resolution error for {name}: {err}"),
)
})
.map(move |lookup_ip| {
lookup_ip
.iter()
.map(|ip| SocketAddr::new(ip, port))
.collect::<Vec<_>>()
})
}
fn main() {
tracing_subscriber::fmt::init();
run();
}
fn run() {
use std::thread;
let names = &["www.google.com", "www.reddit.com", "www.wikipedia.org"];
let threads = names
.iter()
.map(|name| {
let join = thread::spawn(move || {
let runtime = tokio::runtime::Runtime::new().expect("failed to launch Runtime");
runtime.block_on(resolve(*name, 443))
});
(name, join)
})
.collect::<Vec<_>>();
for (name, join) in threads {
let result = join
.join()
.expect("resolution thread failed")
.expect("resolution failed");
println!("{name} resolved to {result:?}");
}
}
#[test]
fn test_global_resolver() {
test_support::subscribe();
run()
}