improvement: always use port from SRV lookups
Also query SRV records when well-known is not found, fixes #29next
parent
f12fbca3c5
commit
8dcc1dfe56
|
@ -24,30 +24,12 @@ use std::{
|
||||||
collections::BTreeMap,
|
collections::BTreeMap,
|
||||||
convert::TryFrom,
|
convert::TryFrom,
|
||||||
fmt::Debug,
|
fmt::Debug,
|
||||||
|
net::{IpAddr, SocketAddr},
|
||||||
time::{Duration, SystemTime},
|
time::{Duration, SystemTime},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub async fn request_well_known(
|
|
||||||
globals: &crate::database::globals::Globals,
|
|
||||||
destination: &str,
|
|
||||||
) -> Option<String> {
|
|
||||||
let body: serde_json::Value = serde_json::from_str(
|
|
||||||
&globals
|
|
||||||
.reqwest_client()
|
|
||||||
.get(&format!(
|
|
||||||
"https://{}/.well-known/matrix/server",
|
|
||||||
destination
|
|
||||||
))
|
|
||||||
.send()
|
|
||||||
.await
|
|
||||||
.ok()?
|
|
||||||
.text()
|
|
||||||
.await
|
|
||||||
.ok()?,
|
|
||||||
)
|
|
||||||
.ok()?;
|
|
||||||
Some(body.get("m.server")?.as_str()?.to_owned())
|
|
||||||
}
|
|
||||||
|
|
||||||
pub async fn send_request<T: OutgoingRequest>(
|
pub async fn send_request<T: OutgoingRequest>(
|
||||||
globals: &crate::database::globals::Globals,
|
globals: &crate::database::globals::Globals,
|
||||||
|
@ -215,42 +197,130 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn get_ip_with_port(destination_str: String) -> Option<String> {
|
||||||
|
if destination_str.parse::<SocketAddr>().is_ok() {
|
||||||
|
Some(destination_str)
|
||||||
|
} else if let Ok(ip_addr) = destination_str.parse::<IpAddr>() {
|
||||||
|
Some(SocketAddr::new(ip_addr, 8448).to_string())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_port_to_hostname(destination_str: String) -> String {
|
||||||
|
match destination_str.find(':') {
|
||||||
|
None => destination_str.to_owned() + ":8448",
|
||||||
|
Some(_) => destination_str.to_string(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Returns: actual_destination, host header
|
/// Returns: actual_destination, host header
|
||||||
|
/// Implemented according to the specification at https://matrix.org/docs/spec/server_server/r0.1.4#resolving-server-names
|
||||||
|
/// Numbers in comments below refer to bullet points in linked section of specification
|
||||||
async fn find_actual_destination(
|
async fn find_actual_destination(
|
||||||
globals: &crate::database::globals::Globals,
|
globals: &crate::database::globals::Globals,
|
||||||
destination: &Box<ServerName>,
|
destination: &Box<ServerName>,
|
||||||
) -> (String, Option<String>) {
|
) -> (String, Option<String>) {
|
||||||
let mut host = None;
|
let mut host = None;
|
||||||
|
|
||||||
|
let destination_str = destination.as_str().to_owned();
|
||||||
let actual_destination = "https://".to_owned()
|
let actual_destination = "https://".to_owned()
|
||||||
+ &if let Some(mut delegated_hostname) =
|
+ &match get_ip_with_port(destination_str.clone()) {
|
||||||
request_well_known(globals, destination.as_str()).await
|
Some(host_port) => {
|
||||||
{
|
// 1: IP literal with provided or default port
|
||||||
if let Ok(Some(srv)) = globals
|
host_port
|
||||||
.dns_resolver()
|
}
|
||||||
.srv_lookup(format!("_matrix._tcp.{}", delegated_hostname))
|
None => {
|
||||||
.await
|
if destination_str.find(':').is_some() {
|
||||||
.map(|srv| srv.iter().next().map(|result| result.target().to_string()))
|
// 2: Hostname with included port
|
||||||
{
|
destination_str
|
||||||
host = Some(delegated_hostname);
|
|
||||||
srv.trim_end_matches('.').to_owned()
|
|
||||||
} else {
|
} else {
|
||||||
if delegated_hostname.find(':').is_none() {
|
match request_well_known(globals, &destination.as_str()).await {
|
||||||
delegated_hostname += ":8448";
|
// 3: A .well-known file is available
|
||||||
}
|
Some(delegated_hostname) => {
|
||||||
delegated_hostname
|
match get_ip_with_port(delegated_hostname.clone()) {
|
||||||
}
|
Some(host_and_port) => host_and_port, // 3.1: IP literal in .well-known file
|
||||||
|
None => {
|
||||||
|
if destination_str.find(':').is_some() {
|
||||||
|
// 3.2: Hostname with port in .well-known file
|
||||||
|
destination_str
|
||||||
} else {
|
} else {
|
||||||
let mut destination = destination.as_str().to_owned();
|
match query_srv_record(globals, &delegated_hostname).await {
|
||||||
if destination.find(':').is_none() {
|
// 3.3: SRV lookup successful
|
||||||
destination += ":8448";
|
Some(hostname) => hostname,
|
||||||
|
// 3.4: No SRV records, just use the hostname from .well-known
|
||||||
|
None => add_port_to_hostname(delegated_hostname),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// 4: No .well-known or an error occured
|
||||||
|
None => {
|
||||||
|
match query_srv_record(globals, &destination_str).await {
|
||||||
|
// 4: SRV record found
|
||||||
|
Some(hostname) => {
|
||||||
|
host = Some(destination_str.to_owned());
|
||||||
|
hostname
|
||||||
|
}
|
||||||
|
// 5: No SRV record found
|
||||||
|
None => add_port_to_hostname(destination_str.to_string()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
destination
|
|
||||||
};
|
};
|
||||||
|
|
||||||
(actual_destination, host)
|
(actual_destination, host)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async fn query_srv_record<'a>(
|
||||||
|
globals: &crate::database::globals::Globals,
|
||||||
|
hostname: &'a str,
|
||||||
|
) -> Option<String> {
|
||||||
|
if let Ok(Some(host_port)) = globals
|
||||||
|
.dns_resolver()
|
||||||
|
.srv_lookup(format!("_matrix._tcp.{}", hostname))
|
||||||
|
.await
|
||||||
|
.map(|srv| {
|
||||||
|
srv.iter().next().map(|result| {
|
||||||
|
format!(
|
||||||
|
"{}:{}",
|
||||||
|
result.target().to_string().trim_end_matches('.'),
|
||||||
|
result.port().to_string()
|
||||||
|
)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
{
|
||||||
|
Some(host_port)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn request_well_known(
|
||||||
|
globals: &crate::database::globals::Globals,
|
||||||
|
destination: &str,
|
||||||
|
) -> Option<String> {
|
||||||
|
let body: serde_json::Value = serde_json::from_str(
|
||||||
|
&globals
|
||||||
|
.reqwest_client()
|
||||||
|
.get(&format!(
|
||||||
|
"https://{}/.well-known/matrix/server",
|
||||||
|
destination
|
||||||
|
))
|
||||||
|
.send()
|
||||||
|
.await
|
||||||
|
.ok()?
|
||||||
|
.text()
|
||||||
|
.await
|
||||||
|
.ok()?,
|
||||||
|
)
|
||||||
|
.ok()?;
|
||||||
|
Some(body.get("m.server")?.as_str()?.to_owned())
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg_attr(feature = "conduit_bin", get("/_matrix/federation/v1/version"))]
|
#[cfg_attr(feature = "conduit_bin", get("/_matrix/federation/v1/version"))]
|
||||||
pub fn get_server_version_route(
|
pub fn get_server_version_route(
|
||||||
db: State<'_, Database>,
|
db: State<'_, Database>,
|
||||||
|
@ -622,3 +692,48 @@ pub fn get_user_devices_route<'a>(
|
||||||
.into())
|
.into())
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::{add_port_to_hostname, get_ip_with_port};
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn ips_get_default_ports() {
|
||||||
|
assert_eq!(
|
||||||
|
get_ip_with_port(String::from("1.1.1.1")),
|
||||||
|
Some(String::from("1.1.1.1:8448"))
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
get_ip_with_port(String::from("dead:beef::")),
|
||||||
|
Some(String::from("[dead:beef::]:8448"))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn ips_keep_custom_ports() {
|
||||||
|
assert_eq!(
|
||||||
|
get_ip_with_port(String::from("1.1.1.1:1234")),
|
||||||
|
Some(String::from("1.1.1.1:1234"))
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
get_ip_with_port(String::from("[dead::beef]:8933")),
|
||||||
|
Some(String::from("[dead::beef]:8933"))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn hostnames_get_default_ports() {
|
||||||
|
assert_eq!(
|
||||||
|
add_port_to_hostname(String::from("example.com")),
|
||||||
|
"example.com:8448"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn hostnames_keep_custom_ports() {
|
||||||
|
assert_eq!(
|
||||||
|
add_port_to_hostname(String::from("example.com:1337")),
|
||||||
|
"example.com:1337"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in New Issue