2020-10-02 08:47:00 +00:00
|
|
|
//! This houses various NAT hole punching strategies.
|
|
|
|
//!
|
|
|
|
//! Currently supported strategies:
|
|
|
|
//! - UPnP
|
|
|
|
|
|
|
|
use crate::{NetworkConfig, NetworkMessage};
|
2020-11-06 04:14:14 +00:00
|
|
|
use if_addrs::get_if_addrs;
|
2020-11-28 05:30:57 +00:00
|
|
|
use slog::{debug, info};
|
2020-10-02 08:47:00 +00:00
|
|
|
use std::net::{IpAddr, SocketAddr, SocketAddrV4};
|
|
|
|
use tokio::sync::mpsc;
|
|
|
|
use types::EthSpec;
|
|
|
|
|
|
|
|
/// Configuration required to construct the UPnP port mappings.
|
|
|
|
pub struct UPnPConfig {
|
|
|
|
/// The local tcp port.
|
|
|
|
tcp_port: u16,
|
|
|
|
/// The local udp port.
|
|
|
|
udp_port: u16,
|
|
|
|
/// Whether discovery is enabled or not.
|
|
|
|
disable_discovery: bool,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<&NetworkConfig> for UPnPConfig {
|
|
|
|
fn from(config: &NetworkConfig) -> Self {
|
|
|
|
UPnPConfig {
|
|
|
|
tcp_port: config.libp2p_port,
|
|
|
|
udp_port: config.discovery_port,
|
|
|
|
disable_discovery: config.disable_discovery,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Attempts to construct external port mappings with UPnP.
|
|
|
|
pub fn construct_upnp_mappings<T: EthSpec>(
|
|
|
|
config: UPnPConfig,
|
|
|
|
network_send: mpsc::UnboundedSender<NetworkMessage<T>>,
|
|
|
|
log: slog::Logger,
|
|
|
|
) {
|
2020-10-23 03:01:33 +00:00
|
|
|
info!(log, "UPnP Attempting to initialise routes");
|
2020-10-02 08:47:00 +00:00
|
|
|
match igd::search_gateway(Default::default()) {
|
2020-10-23 03:01:33 +00:00
|
|
|
Err(e) => info!(log, "UPnP not available"; "error" => %e),
|
2020-10-02 08:47:00 +00:00
|
|
|
Ok(gateway) => {
|
|
|
|
// Need to find the local listening address matched with the router subnet
|
|
|
|
let interfaces = match get_if_addrs() {
|
|
|
|
Ok(v) => v,
|
|
|
|
Err(e) => {
|
2020-10-23 03:01:33 +00:00
|
|
|
info!(log, "UPnP failed to get local interfaces"; "error" => %e);
|
2020-10-02 08:47:00 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
};
|
2020-10-23 03:01:33 +00:00
|
|
|
let local_ip = interfaces.iter().find_map(|interface| {
|
|
|
|
// Just use the first IP of the first interface that is not a loopback and not an
|
|
|
|
// ipv6 address.
|
2020-10-02 08:47:00 +00:00
|
|
|
if !interface.is_loopback() {
|
2021-02-15 00:09:12 +00:00
|
|
|
interface.ip().is_ipv4().then(|| interface.ip())
|
2020-10-23 03:01:33 +00:00
|
|
|
} else {
|
|
|
|
None
|
2020-10-02 08:47:00 +00:00
|
|
|
}
|
2020-10-23 03:01:33 +00:00
|
|
|
});
|
2020-10-02 08:47:00 +00:00
|
|
|
|
2020-10-23 03:01:33 +00:00
|
|
|
let local_ip = match local_ip {
|
|
|
|
None => {
|
|
|
|
info!(log, "UPnP failed to find local IP address");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
Some(v) => v,
|
|
|
|
};
|
2020-10-02 08:47:00 +00:00
|
|
|
|
2020-11-28 05:30:57 +00:00
|
|
|
debug!(log, "UPnP Local IP Discovered"; "ip" => ?local_ip);
|
|
|
|
|
2020-10-02 08:47:00 +00:00
|
|
|
match local_ip {
|
|
|
|
IpAddr::V4(address) => {
|
|
|
|
let libp2p_socket = SocketAddrV4::new(address, config.tcp_port);
|
|
|
|
let external_ip = gateway.get_external_ip();
|
|
|
|
// We add specific port mappings rather than getting the router to arbitrary assign
|
|
|
|
// one.
|
|
|
|
// I've found this to be more reliable. If multiple users are behind a single
|
|
|
|
// router, they should ideally try to set different port numbers.
|
2020-11-28 05:30:57 +00:00
|
|
|
let tcp_socket = add_port_mapping(
|
|
|
|
&gateway,
|
2020-10-02 08:47:00 +00:00
|
|
|
igd::PortMappingProtocol::TCP,
|
|
|
|
libp2p_socket,
|
2020-11-28 05:30:57 +00:00
|
|
|
"tcp",
|
|
|
|
&log,
|
|
|
|
).and_then(|_| {
|
2021-06-18 05:58:01 +00:00
|
|
|
let external_socket = external_ip.as_ref().map(|ip| SocketAddr::new((*ip).into(), config.tcp_port)).map_err(|_| ());
|
2020-11-28 05:30:57 +00:00
|
|
|
info!(log, "UPnP TCP route established"; "external_socket" => format!("{}:{}", external_socket.as_ref().map(|ip| ip.to_string()).unwrap_or_else(|_| "".into()), config.tcp_port));
|
|
|
|
external_socket
|
|
|
|
}).ok();
|
2020-10-02 08:47:00 +00:00
|
|
|
|
|
|
|
let udp_socket = if !config.disable_discovery {
|
|
|
|
let discovery_socket = SocketAddrV4::new(address, config.udp_port);
|
2020-11-28 05:30:57 +00:00
|
|
|
add_port_mapping(
|
|
|
|
&gateway,
|
2020-10-02 08:47:00 +00:00
|
|
|
igd::PortMappingProtocol::UDP,
|
|
|
|
discovery_socket,
|
2020-11-28 05:30:57 +00:00
|
|
|
"udp",
|
|
|
|
&log,
|
|
|
|
).and_then(|_| {
|
|
|
|
let external_socket = external_ip
|
|
|
|
.map(|ip| SocketAddr::new(ip.into(), config.udp_port)).map_err(|_| ());
|
|
|
|
info!(log, "UPnP UDP route established"; "external_socket" => format!("{}:{}", external_socket.as_ref().map(|ip| ip.to_string()).unwrap_or_else(|_| "".into()), config.udp_port));
|
|
|
|
external_socket
|
|
|
|
}).ok()
|
2020-10-02 08:47:00 +00:00
|
|
|
} else {
|
|
|
|
None
|
|
|
|
};
|
|
|
|
|
|
|
|
// report any updates to the network service.
|
|
|
|
network_send.send(NetworkMessage::UPnPMappingEstablished{ tcp_socket, udp_socket })
|
2020-11-28 05:30:57 +00:00
|
|
|
.unwrap_or_else(|e| debug!(log, "Could not send message to the network service"; "error" => %e));
|
2020-10-02 08:47:00 +00:00
|
|
|
}
|
|
|
|
_ => debug!(log, "UPnP no routes constructed. IPv6 not supported"),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-11-28 05:30:57 +00:00
|
|
|
/// Sets up a port mapping for a protocol returning the mapped port if successful.
|
|
|
|
fn add_port_mapping(
|
|
|
|
gateway: &igd::Gateway,
|
|
|
|
protocol: igd::PortMappingProtocol,
|
|
|
|
socket: SocketAddrV4,
|
|
|
|
protocol_string: &'static str,
|
|
|
|
log: &slog::Logger,
|
|
|
|
) -> Result<(), ()> {
|
|
|
|
// We add specific port mappings rather than getting the router to arbitrary assign
|
|
|
|
// one.
|
|
|
|
// I've found this to be more reliable. If multiple users are behind a single
|
|
|
|
// router, they should ideally try to set different port numbers.
|
|
|
|
let mapping_string = &format!("lighthouse-{}", protocol_string);
|
|
|
|
for _ in 0..2 {
|
|
|
|
match gateway.add_port(protocol, socket.port(), socket, 0, mapping_string) {
|
|
|
|
Err(e) => {
|
|
|
|
match e {
|
|
|
|
igd::AddPortError::PortInUse => {
|
|
|
|
// Try and remove and re-create
|
|
|
|
debug!(log, "UPnP port in use, attempting to remap"; "protocol" => protocol_string, "port" => socket.port());
|
|
|
|
match gateway.remove_port(protocol, socket.port()) {
|
|
|
|
Ok(()) => {
|
|
|
|
debug!(log, "UPnP Removed port mapping"; "protocol" => protocol_string, "port" => socket.port())
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
debug!(log, "UPnP Port remove failure"; "protocol" => protocol_string, "port" => socket.port(), "error" => %e);
|
|
|
|
return Err(());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
e => {
|
|
|
|
info!(log, "UPnP TCP route not set"; "error" => %e);
|
|
|
|
return Err(());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Ok(_) => {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(())
|
|
|
|
}
|
|
|
|
|
2020-10-02 08:47:00 +00:00
|
|
|
/// Removes the specified TCP and UDP port mappings.
|
|
|
|
pub fn remove_mappings(tcp_port: Option<u16>, udp_port: Option<u16>, log: &slog::Logger) {
|
|
|
|
if tcp_port.is_some() || udp_port.is_some() {
|
|
|
|
debug!(log, "Removing UPnP port mappings");
|
|
|
|
match igd::search_gateway(Default::default()) {
|
|
|
|
Ok(gateway) => {
|
|
|
|
if let Some(tcp_port) = tcp_port {
|
|
|
|
match gateway.remove_port(igd::PortMappingProtocol::TCP, tcp_port) {
|
|
|
|
Ok(()) => debug!(log, "UPnP Removed TCP port mapping"; "port" => tcp_port),
|
|
|
|
Err(e) => {
|
|
|
|
debug!(log, "UPnP Failed to remove TCP port mapping"; "port" => tcp_port, "error" => %e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if let Some(udp_port) = udp_port {
|
|
|
|
match gateway.remove_port(igd::PortMappingProtocol::UDP, udp_port) {
|
|
|
|
Ok(()) => debug!(log, "UPnP Removed UDP port mapping"; "port" => udp_port),
|
|
|
|
Err(e) => {
|
|
|
|
debug!(log, "UPnP Failed to remove UDP port mapping"; "port" => udp_port, "error" => %e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(e) => debug!(log, "UPnP failed to remove mappings"; "error" => %e),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|