use crate::net::{SocketAddr, TcpListener, TcpStream, UdpSocket};
use ipnet::IpNet;
use std::{io, net, time::Duration};
enum AddrSet {
Net(IpNet),
NameWildcard(String),
}
struct IpGrant {
set: AddrSet,
port: u16, }
pub(crate) struct Catalog {
grants: Vec<IpGrant>,
}
impl Catalog {
pub(crate) fn bind_tcp_listener(
&self,
addrs: impl Iterator<Item = SocketAddr>,
) -> io::Result<TcpListener> {
let mut last_err = None;
for addr in addrs {
self.check_addr(&addr)?;
match net::TcpListener::bind(addr) {
Ok(tcp_listener) => return Ok(TcpListener::from_std(tcp_listener)),
Err(e) => last_err = Some(e),
}
}
match last_err {
Some(e) => Err(e),
None => Err(net::TcpListener::bind(NO_SOCKET_ADDRS).unwrap_err()),
}
}
pub(crate) fn connect_tcp_stream(
&self,
addrs: impl Iterator<Item = SocketAddr>,
) -> io::Result<TcpStream> {
let mut last_err = None;
for addr in addrs {
self.check_addr(&addr)?;
match net::TcpStream::connect(addr) {
Ok(tcp_stream) => return Ok(TcpStream::from_std(tcp_stream)),
Err(e) => last_err = Some(e),
}
}
match last_err {
Some(e) => Err(e),
None => Err(net::TcpStream::connect(NO_SOCKET_ADDRS).unwrap_err()),
}
}
pub(crate) fn connect_timeout_tcp_stream(
&self,
addr: &SocketAddr,
timeout: Duration,
) -> io::Result<TcpStream> {
self.check_addr(&addr)?;
net::TcpStream::connect_timeout(addr, timeout).map(TcpStream::from_std)
}
pub(crate) fn bind_udp_socket(
&self,
addrs: impl Iterator<Item = SocketAddr>,
) -> io::Result<UdpSocket> {
let mut last_err = None;
for addr in addrs {
self.check_addr(&addr)?;
match net::UdpSocket::bind(addr) {
Ok(udp_socket) => return Ok(UdpSocket::from_std(udp_socket)),
Err(e) => last_err = Some(e),
}
}
match last_err {
Some(e) => Err(e),
None => Err(net::UdpSocket::bind(NO_SOCKET_ADDRS).unwrap_err()),
}
}
pub(crate) fn send_to_udp_socket_addr(
&self,
udp_socket: &UdpSocket,
buf: &[u8],
mut addrs: impl Iterator<Item = SocketAddr>,
) -> io::Result<usize> {
let addr = addrs
.next()
.ok_or_else(|| net::UdpSocket::bind(NO_SOCKET_ADDRS).unwrap_err())?;
self.check_addr(&addr)?;
udp_socket.std.send_to(buf, addr)
}
pub(crate) fn connect_udp_socket(
&self,
udp_socket: &UdpSocket,
addrs: impl Iterator<Item = SocketAddr>,
) -> io::Result<()> {
let mut last_err = None;
for addr in addrs {
self.check_addr(&addr)?;
match udp_socket.std.connect(addr) {
Ok(()) => return Ok(()),
Err(e) => last_err = Some(e),
}
}
match last_err {
Some(e) => Err(e),
None => Err(net::UdpSocket::bind(NO_SOCKET_ADDRS).unwrap_err()),
}
}
fn check_addr(&self, addr: &SocketAddr) -> io::Result<()> {
unimplemented!("Catalog::check_addr({:?})", addr)
}
}
const NO_SOCKET_ADDRS: &[net::SocketAddr] = &[];