// VpnCloud - Peer-to-Peer VPN // Copyright (C) 2015-2021 Dennis Schwerdel // This software is licensed under GPL-3 or newer (see LICENSE.md) use crate::config::DEFAULT_PORT; use crate::port_forwarding::PortForwarding; use crate::util::{MockTimeSource, MsgBuffer, Time, TimeSource}; use parking_lot::Mutex; use std::time::Duration; use std::{ collections::{HashMap, VecDeque}, io::{self, ErrorKind}, net::{UdpSocket, IpAddr, Ipv6Addr, SocketAddr}, sync::{ atomic::{AtomicBool, Ordering}, Arc, }, }; pub fn mapped_addr(addr: SocketAddr) -> SocketAddr { // HOT PATH match addr { SocketAddr::V4(addr4) => SocketAddr::new(IpAddr::V6(addr4.ip().to_ipv6_mapped()), addr4.port()), _ => addr, } } pub fn get_ip() -> IpAddr { let s = std::net::UdpSocket::bind("[::]:0").unwrap(); s.connect("8.8.8.8:0").unwrap(); s.local_addr().unwrap().ip() } pub trait Socket: Sized + Clone + Send + Sync + 'static { fn receive(&mut self, buffer: &mut MsgBuffer) -> Result; fn send(&mut self, data: &[u8], addr: SocketAddr) -> Result; fn address(&self) -> Result; fn create_port_forwarding(&self) -> Option; } pub fn parse_listen(addr: &str, default_port: u16) -> SocketAddr { if let Some(addr) = addr.strip_prefix("*:") { let port = try_fail!(addr.parse::(), "Invalid port: {}"); SocketAddr::new(IpAddr::V6(Ipv6Addr::UNSPECIFIED), port) } else if addr.contains(':') { try_fail!(addr.parse::(), "Invalid address: {}: {}", addr) } else if let Ok(port) = addr.parse::() { SocketAddr::new(IpAddr::V6(Ipv6Addr::UNSPECIFIED), port) } else { let ip = try_fail!(addr.parse::(), "Invalid addr: {}"); SocketAddr::new(ip, default_port) } } pub struct NetSocket(Arc); impl NetSocket { pub fn listen(addr: &str) -> Result { let addr = parse_listen(addr, DEFAULT_PORT); let sock = UdpSocket::bind(addr)?; sock.set_read_timeout(Some(Duration::from_secs(1)))?; Ok(Self(Arc::new(sock))) } } impl Clone for NetSocket { fn clone(&self) -> Self { Self(self.0.clone()) } } impl Socket for NetSocket { fn create_port_forwarding(&self) -> Option { PortForwarding::new(self.0.local_addr().unwrap().port()) } fn receive(&mut self, buffer: &mut MsgBuffer) -> Result { buffer.clear(); let (size, addr) = self.0.recv_from(buffer.buffer())?; buffer.set_length(size); Ok(addr) } fn send(&mut self, data: &[u8], addr: SocketAddr) -> Result { self.0.send_to(data, addr) } fn address(&self) -> Result { let mut addr = self.0.local_addr()?; addr.set_ip(get_ip()); Ok(addr) } } thread_local! { static MOCK_SOCKET_NAT: AtomicBool = AtomicBool::new(false); } type MsgQueue = Arc)>>>; #[derive(Clone)] pub struct MockSocket { nat: bool, nat_peers: Arc>>, address: SocketAddr, outbound: MsgQueue, inbound: MsgQueue, } impl MockSocket { pub fn new(address: SocketAddr) -> Self { Self { nat: Self::get_nat(), nat_peers: Default::default(), address, outbound: Arc::new(Mutex::new(VecDeque::with_capacity(10))), inbound: Arc::new(Mutex::new(VecDeque::with_capacity(10))), } } pub fn set_nat(nat: bool) { MOCK_SOCKET_NAT.with(|t| t.store(nat, Ordering::SeqCst)) } pub fn get_nat() -> bool { MOCK_SOCKET_NAT.with(|t| t.load(Ordering::SeqCst)) } pub fn put_inbound(&mut self, from: SocketAddr, data: Vec) -> bool { if !self.nat { self.inbound.lock().push_back((from, data)); return true; } if let Some(timeout) = self.nat_peers.lock().get(&from) { if *timeout >= MockTimeSource::now() { self.inbound.lock().push_back((from, data)); return true; } } warn!("Sender {:?} is filtered out by NAT", from); false } pub fn pop_outbound(&mut self) -> Option<(SocketAddr, Vec)> { self.outbound.lock().pop_front() } } impl Socket for MockSocket { fn receive(&mut self, buffer: &mut MsgBuffer) -> Result { if let Some((addr, data)) = self.inbound.lock().pop_front() { buffer.clear(); buffer.set_length(data.len()); buffer.message_mut().copy_from_slice(&data); Ok(addr) } else { Err(io::Error::new(ErrorKind::Other, "nothing in queue")) } } fn send(&mut self, data: &[u8], addr: SocketAddr) -> Result { self.outbound.lock().push_back((addr, data.into())); if self.nat { self.nat_peers.lock().insert(addr, MockTimeSource::now() + 300); } Ok(data.len()) } fn address(&self) -> Result { Ok(self.address) } fn create_port_forwarding(&self) -> Option { None } } #[cfg(feature = "bench")] mod bench { use std::net::{Ipv4Addr, SocketAddrV4, UdpSocket}; use test::Bencher; #[bench] fn udp_send(b: &mut Bencher) { let sock = UdpSocket::bind("127.0.0.1:0").unwrap(); let data = [0; 1400]; let addr = SocketAddrV4::new(Ipv4Addr::new(127, 0, 0, 1), 1); b.iter(|| sock.send_to(&data, &addr).unwrap()); b.bytes = 1400; } }