Skip to content

Commit

Permalink
simplify QAD addresses in NodeMap
Browse files Browse the repository at this point in the history
  • Loading branch information
“ramfox” committed Jan 8, 2025
1 parent 6d49482 commit 33fd158
Show file tree
Hide file tree
Showing 9 changed files with 73 additions and 88 deletions.
1 change: 0 additions & 1 deletion iroh-net-report/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,6 @@ tokio = { version = "1", default-features = false, features = ["test-util"] }
default = ["metrics"]
metrics = ["iroh-metrics/metrics", "portmapper/metrics"]
stun-utils = []
iroh = []

[package.metadata.docs.rs]
all-features = true
Expand Down
1 change: 0 additions & 1 deletion iroh-net-report/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,6 @@ mod ping;
mod reportgen;

pub use metrics::Metrics;
pub use reportgen::MappedAddr;
use reportgen::ProbeProto;
pub use reportgen::QuicConfig;
#[cfg(feature = "stun-utils")]
Expand Down
26 changes: 0 additions & 26 deletions iroh-net-report/src/reportgen.rs
Original file line number Diff line number Diff line change
Expand Up @@ -700,25 +700,8 @@ pub struct QuicConfig {
pub ipv4: bool,
/// Enable ipv6 QUIC address discovery probes
pub ipv6: bool,
/// A map of RelayUrls to addresses
///
/// Only makes sense in the context of iroh, which uses QuicMappedAddrs
/// to allow dialing by NodeId
#[cfg(feature = "iroh")]
pub mapped_addrs: MappedRelayAddrs,
}

/// Holds a QuicMappedAddr
#[cfg(feature = "iroh")]
#[derive(Debug, Clone, Copy)]
pub struct MappedAddr(pub SocketAddr);

/// A relationship between a socket address and it's QUIC mapped address
///
/// Only relevant when using the net-report with iroh.
#[cfg(feature = "iroh")]
type MappedRelayAddrs = std::collections::HashMap<SocketAddr, MappedAddr>;

/// Executes a particular [`Probe`], including using a delayed start if needed.
///
/// If *stun_sock4* and *stun_sock6* are `None` the STUN probes are disabled.
Expand Down Expand Up @@ -940,14 +923,6 @@ async fn run_quic_probe(
));
}
};

// if we are using net-report with iroh, we must dial using the mapped address
#[cfg(feature = "iroh")]
let relay_addr = quic_config
.mapped_addrs
.get(&relay_addr)
.map_or(relay_addr, |mapped_addr| mapped_addr.0.clone());

let quic_client = iroh_relay::quic::QuicClient::new(quic_config.ep, quic_config.client_config)
.map_err(|e| ProbeError::Error(e, probe.clone()))?;
let (addr, latency) = quic_client
Expand Down Expand Up @@ -1617,7 +1592,6 @@ mod tests {
client_config,
ipv4: true,
ipv6: true,
mapped_addrs: std::collections::HashMap::new(),
};
let url = relay.url.clone();
let port = server.quic_addr().unwrap().port();
Expand Down
2 changes: 1 addition & 1 deletion iroh/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -95,7 +95,7 @@ webpki = { package = "rustls-webpki", version = "0.102" }
webpki-roots = "0.26"
x509-parser = "0.16"
z32 = "1.0.3"
net-report = { package = "iroh-net-report", path = "../iroh-net-report", version = "0.30", default-features = false, features = ["iroh"] }
net-report = { package = "iroh-net-report", path = "../iroh-net-report", version = "0.29", default-features = false }

# metrics
iroh-metrics = { version = "0.30", default-features = false }
Expand Down
2 changes: 1 addition & 1 deletion iroh/examples/listen.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ async fn main() -> anyhow::Result<()> {
// Use `RelayMode::Custom` to pass in a `RelayMap` with custom relay urls.
// Use `RelayMode::Disable` to disable holepunching and relaying over HTTPS
// If you want to experiment with relaying using your own relay server, you must pass in the same custom relay url to both the `listen` code AND the `connect` code
.relay_mode(RelayMode::Staging)
.relay_mode(RelayMode::Default)
// you can choose a port to bind to, but passing in `0` will bind the socket to a random available port
.bind()
.await?;
Expand Down
3 changes: 2 additions & 1 deletion iroh/src/endpoint.rs
Original file line number Diff line number Diff line change
Expand Up @@ -445,6 +445,7 @@ impl Builder {
self
}
}

/// Configuration for a [`quinn::Endpoint`] that cannot be changed at runtime.
#[derive(Debug)]
struct StaticConfig {
Expand All @@ -455,7 +456,7 @@ struct StaticConfig {

impl StaticConfig {
/// Create a [`quinn::ServerConfig`] with the specified ALPN protocols.
fn create_server_config(&self, alpn_protocols: Vec<Vec<u8>>) -> Result<quinn::ServerConfig> {
fn create_server_config(&self, alpn_protocols: Vec<Vec<u8>>) -> Result<ServerConfig> {
let server_config = make_server_config(
&self.secret_key,
alpn_protocols,
Expand Down
49 changes: 21 additions & 28 deletions iroh/src/magicsock.rs
Original file line number Diff line number Diff line change
Expand Up @@ -162,8 +162,7 @@ impl Default for Options {
}
}

/// Generate a server config with no ALPNS and a default
/// transport configuration
/// Generate a server config with no ALPNS and a default transport configuration
fn make_default_server_config(secret_key: &SecretKey) -> ServerConfig {
let quic_server_config = crate::tls::make_server_config(secret_key, vec![], false)
.expect("should generate valid config");
Expand Down Expand Up @@ -574,11 +573,11 @@ impl MagicSock {
}
}
None => {
// Check if this is a QUIC address discovery packet
if let Some(addr) = self.node_map.get_qad_addr(&dest) {
// send udp
// Check if this is addr is used to perform QUIC Address Discovery
if let Some(addr) = self.node_map.qad_addr_for_send(&dest.0) {
// rewrite target address
transmit.destination = addr;
// send udp
match self.try_send_udp(addr, &transmit) {
Ok(()) => {
trace!(dst = %addr,
Expand Down Expand Up @@ -837,16 +836,16 @@ impl MagicSock {
// Update the NodeMap and remap RecvMeta to the QuicMappedAddr.
match self.node_map.receive_udp(meta.addr) {
None => {
// Check if this is QUIC address discovery response
if let Some(quic_mapped_addr) = self.node_map.receive_qad(meta.addr) {
// Check if this address is used for QUIC address discovery
if let Some(addr) = self.node_map.qad_addr_for_recv(&meta.addr) {
trace!(
src = ?meta.addr,
count = %quic_datagram_count,
len = meta.len,
"UDP recv QUIC address discovery packets",
);
quic_packets_total += quic_datagram_count;
meta.addr = quic_mapped_addr.0;
meta.addr = addr;
} else {
warn!(
src = ?meta.addr,
Expand Down Expand Up @@ -1960,7 +1959,7 @@ impl AsyncUdpSocket for MagicSock {
self.try_send(transmit)
}

/// NOTE: Receiving on a [`Self::close`]d socket will return [`Poll::Pending`] indefinitely.
/// NOTE: Receiving on a closed socket will return [`Poll::Pending`] indefinitely.
fn poll_recv(
&self,
cx: &mut Context,
Expand Down Expand Up @@ -2123,6 +2122,9 @@ impl Actor {
}
}

// kick off resolving the URLs for relay addresses
self.resolve_qad_addrs(Duration::from_millis(10)).await;

let mut receiver_closed = false;
let mut portmap_watcher_closed = false;
let mut link_change_closed = false;
Expand Down Expand Up @@ -2501,13 +2503,11 @@ impl Actor {
.stun_v4(Some(self.pconn4.clone()))
.stun_v6(self.pconn6.clone());

// Need to add Quic Mapped Addrs for the relay nodes to use for
// QUIC Address Discovery
let mapped_addrs = self
.resolve_qad_addrs(std::time::Duration::from_millis(500))
// Need to get the SocketAddrs of the relay urls and add them to the node map
// so the socket knows to treat them special
self.resolve_qad_addrs(std::time::Duration::from_millis(300))
.await;
// create a client config for the endpoint to
// use for QUIC address discovery
// create a client config for the endpoint to use for QUIC address discovery
let root_store =
rustls::RootCertStore::from_iter(webpki_roots::TLS_SERVER_ROOTS.iter().cloned());
let client_config = rustls::ClientConfig::builder()
Expand All @@ -2518,7 +2518,6 @@ impl Actor {
client_config,
ipv4: true,
ipv6: self.pconn6.is_some(),
mapped_addrs,
});
opts = opts.quic_config(quic_config);

Expand Down Expand Up @@ -2620,7 +2619,7 @@ impl Actor {

match relay_node.url.host() {
Some(url::Host::Domain(hostname)) => {
error!(%hostname, "Performing DNS A lookup for relay addr");
debug!(%hostname, "Performing DNS A lookup for relay addr");
let mut set = JoinSet::new();
let resolver = dns_resolver.clone();
let ipv4_resolver = resolver.clone();
Expand All @@ -2640,13 +2639,13 @@ impl Actor {
Err(_) => {}
Ok(resolved_addrs) => {
for addr in resolved_addrs {
addrs.insert(SocketAddr::new(addr.into(), port));
addrs.insert(SocketAddr::new(addr, port));
}
}
}
}
if addrs.is_empty() {
error!(%hostname, "Unable to resolve ip addresses for relay node");
debug!(%hostname, "Unable to resolve ip addresses for relay node");
}
}
Some(url::Host::Ipv4(addr)) => {
Expand All @@ -2659,15 +2658,11 @@ impl Actor {
error!(?relay_node.url, "No hostname for relay node, cannot resolve ip");
}
}
return addrs;
addrs
}

/// Resolve the relay addresses used for QUIC address discovery.
async fn resolve_qad_addrs(
&mut self,
duration: Duration,
) -> HashMap<SocketAddr, net_report::MappedAddr> {
let mut mapped_addrs = HashMap::new();
async fn resolve_qad_addrs(&mut self, duration: Duration) {
let mut set = JoinSet::new();
let resolver = self.msock.dns_resolver();
for relay_node in self.msock.relay_map.nodes() {
Expand All @@ -2680,11 +2675,9 @@ impl Actor {
let res = set.join_all().await;
for addrs in res {
addrs.iter().for_each(|addr| {
let mapped_addr = self.msock.node_map.add_qad_addr(*addr);
mapped_addrs.insert(*addr, net_report::MappedAddr(mapped_addr.0));
self.msock.node_map.add_qad_addr(*addr);
});
}
mapped_addrs
}

fn set_nearest_relay(&mut self, relay_url: Option<RelayUrl>) -> bool {
Expand Down
75 changes: 47 additions & 28 deletions iroh/src/magicsock/node_map.rs
Original file line number Diff line number Diff line change
Expand Up @@ -69,8 +69,8 @@ pub(super) struct NodeMapInner {
next_id: usize,
#[cfg(any(test, feature = "test-utils"))]
path_selection: PathSelection,
// special mapping for relay socket addresses so that we can do quic address discovery
qad_mapped_addrs: HashMap<SocketAddr, QuicMappedAddr>,
// special set of relay socket addresses that we use to do quic address discovery
qad_addrs: BTreeSet<SocketAddr>,
}

/// Identifier to look up a [`NodeState`] in the [`NodeMap`].
Expand Down Expand Up @@ -155,47 +155,66 @@ impl NodeMap {
.add_node_addr(node_addr, source)
}

/// Add a the SocketAddr used to preform QUIC Address Discovery to the nodemap
pub(super) fn add_qad_addr(&self, udp_addr: SocketAddr) -> QuicMappedAddr {
let quic_mapped_addr = QuicMappedAddr::generate();
/// Add a the SocketAddr used to perform QUIC Address Discovery to the nodemap
pub(super) fn add_qad_addr(&self, udp_addr: SocketAddr) {
self.inner
.lock()
.expect("poisoned")
.qad_mapped_addrs
.insert(udp_addr, quic_mapped_addr);
quic_mapped_addr
.qad_addrs
.insert(udp_addr);
}

/// Get the socket address used to preform QUIC Address Discovery
pub(super) fn get_qad_addr(&self, addr: &QuicMappedAddr) -> Option<SocketAddr> {
self.inner
/// Return a correctly canonicalized SocketAddr if this address is one
/// used to perform QUIC Address Discovery
pub(super) fn qad_addr_for_send(&self, addr: &SocketAddr) -> Option<SocketAddr> {
// all addresses given to the endpoint are Ipv6 addresses, so we need to
// canonicalize before we check for the actual addr we are trying to send to
let canonicalized_addr = SocketAddr::new(addr.ip().to_canonical(), addr.port());
if self
.inner
.lock()
.expect("poisoned")
.qad_addrs
.contains(&canonicalized_addr)
{
Some(canonicalized_addr)
} else {
None
}
}

/// Return a correctly formed SocketAddr if this address is one used to
/// perform QUIC Address Discovery
pub(super) fn qad_addr_for_recv(&self, addr: &SocketAddr) -> Option<SocketAddr> {
if self
.inner
.lock()
.expect("poisoned")
.qad_mapped_addrs
.iter()
.find_map(|(udp_addr, quic_mapped_addr)| {
if addr == quic_mapped_addr {
Some(*udp_addr)
} else {
None
.qad_addrs
.contains(addr)
{
match addr.ip() {
IpAddr::V4(ipv4_addr) => {
// if this is an ipv4 addr, we need to map it back to
// an ipv6 addr, since all addresses we use to dial on
// the underlying quinn endpoint are mapped ipv6 addrs
Some(SocketAddr::new(
ipv4_addr.to_ipv6_mapped().into(),
addr.port(),
))
}
})
IpAddr::V6(_) => Some(*addr),
}
} else {
None
}
}

/// Number of nodes currently listed.
pub(super) fn node_count(&self) -> usize {
self.inner.lock().expect("poisoned").node_count()
}

pub(super) fn receive_qad(&self, udp_addr: SocketAddr) -> Option<QuicMappedAddr> {
self.inner
.lock()
.expect("poisoned")
.qad_mapped_addrs
.get(&udp_addr)
.map(|addr| *addr)
}

pub(super) fn receive_udp(&self, udp_addr: SocketAddr) -> Option<(PublicKey, QuicMappedAddr)> {
self.inner.lock().expect("poisoned").receive_udp(udp_addr)
}
Expand Down
2 changes: 1 addition & 1 deletion iroh/src/magicsock/node_map/node_state.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1691,7 +1691,7 @@ mod tests {
]),
next_id: 5,
path_selection: PathSelection::default(),
qad_mapped_addrs: HashMap::new(),
qad_addrs: BTreeSet::new(),
});
let mut got = node_map.list_remote_infos(later);
got.sort_by_key(|p| p.node_id);
Expand Down

0 comments on commit 33fd158

Please sign in to comment.