xmpp-rs/tokio-xmpp/src/happy_eyeballs.rs

62 lines
1.8 KiB
Rust
Raw Normal View History

2018-12-18 18:04:31 +00:00
use crate::{ConnecterError, Error};
use std::net::SocketAddr;
use tokio::net::TcpStream;
2020-03-05 00:25:24 +00:00
use trust_dns_resolver::{IntoName, TokioAsyncResolver};
async fn connect_to_host(
resolver: &TokioAsyncResolver,
host: &str,
port: u16,
) -> Result<TcpStream, Error> {
let ips = resolver
.lookup_ip(host)
.await
.map_err(ConnecterError::Resolve)?;
for ip in ips.iter() {
match TcpStream::connect(&SocketAddr::new(ip, port)).await {
Ok(stream) => return Ok(stream),
Err(_) => {}
}
}
Err(Error::Disconnected)
2018-09-06 22:12:00 +00:00
}
2020-03-05 00:25:24 +00:00
pub async fn connect(
domain: &str,
srv: Option<&str>,
2018-09-01 19:59:02 +00:00
fallback_port: u16,
2020-03-05 00:25:24 +00:00
) -> Result<TcpStream, Error> {
if let Ok(ip) = domain.parse() {
return Ok(TcpStream::connect(&SocketAddr::new(ip, fallback_port)).await?);
2017-07-13 00:56:02 +00:00
}
2020-12-30 02:15:59 +00:00
let resolver = TokioAsyncResolver::tokio_from_system_conf().map_err(ConnecterError::Resolve)?;
2017-07-13 00:56:02 +00:00
2020-03-05 00:25:24 +00:00
let srv_records = match srv {
Some(srv) => {
let srv_domain = format!("{}.{}.", srv, domain)
.into_name()
.map_err(ConnecterError::Dns)?;
resolver.srv_lookup(srv_domain).await.ok()
}
None => None,
};
match srv_records {
Some(lookup) => {
// TODO: sort lookup records by priority/weight
for srv in lookup.iter() {
match connect_to_host(&resolver, &srv.target().to_ascii(), srv.port()).await {
Ok(stream) => return Ok(stream),
Err(_) => {}
2018-09-01 19:59:02 +00:00
}
2017-07-13 00:56:02 +00:00
}
2020-03-05 00:25:24 +00:00
Err(Error::Disconnected)
}
None => {
// SRV lookup error, retry with hostname
connect_to_host(&resolver, domain, fallback_port).await
2017-07-13 20:17:29 +00:00
}
2017-07-13 00:56:02 +00:00
}
}