|
|
|
@ -45,6 +45,20 @@ use std::{ |
|
|
|
#[cfg(feature = "conduit_bin")] |
|
|
|
#[cfg(feature = "conduit_bin")] |
|
|
|
use rocket::{get, post, put}; |
|
|
|
use rocket::{get, post, put}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// Wraps either an literal IP address plus port, or a hostname plus complement
|
|
|
|
|
|
|
|
/// (colon-plus-port if it was specified).
|
|
|
|
|
|
|
|
///
|
|
|
|
|
|
|
|
/// Note: A `FedDest::Named` might contain an IP address in string form if there
|
|
|
|
|
|
|
|
/// was no port specified to construct a SocketAddr with.
|
|
|
|
|
|
|
|
///
|
|
|
|
|
|
|
|
/// # Examples:
|
|
|
|
|
|
|
|
/// ```rust,ignore
|
|
|
|
|
|
|
|
/// FedDest::Literal("198.51.100.3:8448".parse()?);
|
|
|
|
|
|
|
|
/// FedDest::Literal("[2001:db8::4:5]:443".parse()?);
|
|
|
|
|
|
|
|
/// FedDest::Named("matrix.example.org".to_owned(), "".to_owned());
|
|
|
|
|
|
|
|
/// FedDest::Named("matrix.example.org".to_owned(), ":8448".to_owned());
|
|
|
|
|
|
|
|
/// FedDest::Named("198.51.100.5".to_owned(), "".to_owned());
|
|
|
|
|
|
|
|
/// ```
|
|
|
|
#[derive(Clone, Debug, PartialEq)] |
|
|
|
#[derive(Clone, Debug, PartialEq)] |
|
|
|
enum FedDest { |
|
|
|
enum FedDest { |
|
|
|
Literal(SocketAddr), |
|
|
|
Literal(SocketAddr), |
|
|
|
@ -52,21 +66,21 @@ enum FedDest { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
impl FedDest { |
|
|
|
impl FedDest { |
|
|
|
fn into_https_url(self) -> String { |
|
|
|
fn into_https_string(self) -> String { |
|
|
|
match self { |
|
|
|
match self { |
|
|
|
Self::Literal(addr) => format!("https://{}", addr), |
|
|
|
Self::Literal(addr) => format!("https://{}", addr), |
|
|
|
Self::Named(host, port) => format!("https://{}{}", host, port), |
|
|
|
Self::Named(host, port) => format!("https://{}{}", host, port), |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
fn into_uri(self) -> String { |
|
|
|
fn into_uri_string(self) -> String { |
|
|
|
match self { |
|
|
|
match self { |
|
|
|
Self::Literal(addr) => addr.to_string(), |
|
|
|
Self::Literal(addr) => addr.to_string(), |
|
|
|
Self::Named(host, ref port) => host + port, |
|
|
|
Self::Named(host, ref port) => host + port, |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
fn host(&self) -> String { |
|
|
|
fn hostname(&self) -> String { |
|
|
|
match &self { |
|
|
|
match &self { |
|
|
|
Self::Literal(addr) => addr.ip().to_string(), |
|
|
|
Self::Literal(addr) => addr.ip().to_string(), |
|
|
|
Self::Named(host, _) => host.clone(), |
|
|
|
Self::Named(host, _) => host.clone(), |
|
|
|
@ -99,21 +113,23 @@ where |
|
|
|
} else { |
|
|
|
} else { |
|
|
|
let result = find_actual_destination(globals, &destination).await; |
|
|
|
let result = find_actual_destination(globals, &destination).await; |
|
|
|
let (actual_destination, host) = result.clone(); |
|
|
|
let (actual_destination, host) = result.clone(); |
|
|
|
let result = (result.0.into_https_url(), result.1.into_uri()); |
|
|
|
let result_string = (result.0.into_https_string(), result.1.into_uri_string()); |
|
|
|
globals |
|
|
|
globals |
|
|
|
.actual_destination_cache |
|
|
|
.actual_destination_cache |
|
|
|
.write() |
|
|
|
.write() |
|
|
|
.unwrap() |
|
|
|
.unwrap() |
|
|
|
.insert(Box::<ServerName>::from(destination), result.clone()); |
|
|
|
.insert(Box::<ServerName>::from(destination), result_string.clone()); |
|
|
|
if actual_destination.host() != host.host() { |
|
|
|
let dest_hostname = actual_destination.hostname(); |
|
|
|
|
|
|
|
let host_hostname = host.hostname(); |
|
|
|
|
|
|
|
if dest_hostname != host_hostname { |
|
|
|
globals.tls_name_override.write().unwrap().insert( |
|
|
|
globals.tls_name_override.write().unwrap().insert( |
|
|
|
actual_destination.host(), |
|
|
|
dest_hostname, |
|
|
|
webpki::DNSNameRef::try_from_ascii_str(&host.host()) |
|
|
|
webpki::DNSNameRef::try_from_ascii_str(&host_hostname) |
|
|
|
.unwrap() |
|
|
|
.unwrap() |
|
|
|
.to_owned(), |
|
|
|
.to_owned(), |
|
|
|
); |
|
|
|
); |
|
|
|
} |
|
|
|
} |
|
|
|
result |
|
|
|
result_string |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
let mut http_request = request |
|
|
|
let mut http_request = request |
|
|
|
@ -317,6 +333,8 @@ async fn find_actual_destination( |
|
|
|
} |
|
|
|
} |
|
|
|
}; |
|
|
|
}; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Can't use get_ip_with_port here because we don't want to add a port
|
|
|
|
|
|
|
|
// to an IP address if it wasn't specified
|
|
|
|
let hostname = if let Ok(addr) = hostname.parse::<SocketAddr>() { |
|
|
|
let hostname = if let Ok(addr) = hostname.parse::<SocketAddr>() { |
|
|
|
FedDest::Literal(addr) |
|
|
|
FedDest::Literal(addr) |
|
|
|
} else if let Ok(addr) = hostname.parse::<IpAddr>() { |
|
|
|
} else if let Ok(addr) = hostname.parse::<IpAddr>() { |
|
|
|
@ -1743,45 +1761,45 @@ pub async fn fetch_required_signing_keys( |
|
|
|
|
|
|
|
|
|
|
|
#[cfg(test)] |
|
|
|
#[cfg(test)] |
|
|
|
mod tests { |
|
|
|
mod tests { |
|
|
|
use super::{add_port_to_hostname, get_ip_with_port}; |
|
|
|
use super::{FedDest, add_port_to_hostname, get_ip_with_port}; |
|
|
|
|
|
|
|
|
|
|
|
#[test] |
|
|
|
#[test] |
|
|
|
fn ips_get_default_ports() { |
|
|
|
fn ips_get_default_ports() { |
|
|
|
assert_eq!( |
|
|
|
assert_eq!( |
|
|
|
get_ip_with_port(String::from("1.1.1.1")), |
|
|
|
get_ip_with_port("1.1.1.1"), |
|
|
|
Some(String::from("1.1.1.1:8448")) |
|
|
|
Some(FedDest::Literal("1.1.1.1:8448".parse().unwrap())) |
|
|
|
); |
|
|
|
); |
|
|
|
assert_eq!( |
|
|
|
assert_eq!( |
|
|
|
get_ip_with_port(String::from("dead:beef::")), |
|
|
|
get_ip_with_port("dead:beef::"), |
|
|
|
Some(String::from("[dead:beef::]:8448")) |
|
|
|
Some(FedDest::Literal("[dead:beef::]:8448".parse().unwrap())) |
|
|
|
); |
|
|
|
); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#[test] |
|
|
|
#[test] |
|
|
|
fn ips_keep_custom_ports() { |
|
|
|
fn ips_keep_custom_ports() { |
|
|
|
assert_eq!( |
|
|
|
assert_eq!( |
|
|
|
get_ip_with_port(String::from("1.1.1.1:1234")), |
|
|
|
get_ip_with_port("1.1.1.1:1234"), |
|
|
|
Some(String::from("1.1.1.1:1234")) |
|
|
|
Some(FedDest::Literal("1.1.1.1:1234".parse().unwrap())) |
|
|
|
); |
|
|
|
); |
|
|
|
assert_eq!( |
|
|
|
assert_eq!( |
|
|
|
get_ip_with_port(String::from("[dead::beef]:8933")), |
|
|
|
get_ip_with_port("[dead::beef]:8933"), |
|
|
|
Some(String::from("[dead::beef]:8933")) |
|
|
|
Some(FedDest::Literal("[dead::beef]:8933".parse().unwrap())) |
|
|
|
); |
|
|
|
); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#[test] |
|
|
|
#[test] |
|
|
|
fn hostnames_get_default_ports() { |
|
|
|
fn hostnames_get_default_ports() { |
|
|
|
assert_eq!( |
|
|
|
assert_eq!( |
|
|
|
add_port_to_hostname(String::from("example.com")), |
|
|
|
add_port_to_hostname("example.com"), |
|
|
|
"example.com:8448" |
|
|
|
FedDest::Named(String::from("example.com"), String::from(":8448")) |
|
|
|
) |
|
|
|
) |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#[test] |
|
|
|
#[test] |
|
|
|
fn hostnames_keep_custom_ports() { |
|
|
|
fn hostnames_keep_custom_ports() { |
|
|
|
assert_eq!( |
|
|
|
assert_eq!( |
|
|
|
add_port_to_hostname(String::from("example.com:1337")), |
|
|
|
add_port_to_hostname("example.com:1337"), |
|
|
|
"example.com:1337" |
|
|
|
FedDest::Named(String::from("example.com"), String::from(":1337")) |
|
|
|
) |
|
|
|
) |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|