use crate::{
traits::AsyncToSocketAddrs,
util::{self, SocketAddrsFromIpAddrs},
};
use hickory_resolver::{IntoName, Resolver, lookup_ip::LookupIpIntoIter};
use std::{
io,
net::{SocketAddr, ToSocketAddrs},
str::FromStr,
};
#[derive(Debug, Clone)]
pub struct HickoryToSocketAddrs<T: IntoName + Send + 'static> {
host: T,
port: u16,
}
impl<H: IntoName + Send + 'static> HickoryToSocketAddrs<H> {
pub fn new(host: H, port: u16) -> Self {
Self { host, port }
}
async fn lookup(self) -> io::Result<SocketAddrsFromIpAddrs<LookupIpIntoIter>> {
if !util::inside_tokio() {
return Err(io::Error::other(
"hickory-dns is only supported in a tokio context",
));
}
Ok(SocketAddrsFromIpAddrs(
Resolver::builder_tokio()?
.build()
.lookup_ip(self.host)
.await?
.into_iter(),
self.port,
))
}
}
impl FromStr for HickoryToSocketAddrs<String> {
type Err = io::Error;
fn from_str(s: &str) -> io::Result<Self> {
let (host, port_str) = s
.rsplit_once(':')
.ok_or_else(|| io::Error::new(io::ErrorKind::InvalidInput, "invalid socket address"))?;
let port = port_str
.parse()
.map_err(|_| io::Error::new(io::ErrorKind::InvalidInput, "invalid port value"))?;
Ok(Self::new(host.to_owned(), port))
}
}
impl<T: IntoName + Clone + Send + 'static> ToSocketAddrs for HickoryToSocketAddrs<T> {
type Iter = SocketAddrsFromIpAddrs<LookupIpIntoIter>;
fn to_socket_addrs(&self) -> io::Result<Self::Iter> {
util::block_on_tokio(self.clone().lookup())
}
}
impl<T: IntoName + Send + 'static> AsyncToSocketAddrs for HickoryToSocketAddrs<T> {
fn to_socket_addrs(
self,
) -> impl Future<Output = io::Result<impl Iterator<Item = SocketAddr> + Send + 'static>>
+ Send
+ 'static {
self.lookup()
}
}