use std::net::IpAddr;
use async_trait::async_trait;
use util::vnet::net::*;
use super::*;
use crate::error::*;
const DEFAULT_MAX_RETRIES: u16 = 10;
pub struct RelayAddressGeneratorRanges {
pub relay_address: IpAddr,
pub min_port: u16,
pub max_port: u16,
pub max_retries: u16,
pub address: String,
pub net: Arc<Net>,
}
#[async_trait]
impl RelayAddressGenerator for RelayAddressGeneratorRanges {
fn validate(&self) -> Result<()> {
if self.min_port == 0 {
Err(Error::ErrMinPortNotZero)
} else if self.max_port == 0 {
Err(Error::ErrMaxPortNotZero)
} else if self.max_port < self.min_port {
Err(Error::ErrMaxPortLessThanMinPort)
} else if self.address.is_empty() {
Err(Error::ErrListeningAddressInvalid)
} else {
Ok(())
}
}
async fn allocate_conn(
&self,
use_ipv4: bool,
requested_port: u16,
) -> Result<(Arc<dyn Conn + Send + Sync>, SocketAddr)> {
let max_retries = if self.max_retries == 0 {
DEFAULT_MAX_RETRIES
} else {
self.max_retries
};
if requested_port != 0 {
let addr = self
.net
.resolve_addr(use_ipv4, &format!("{}:{}", self.address, requested_port))
.await?;
let conn = self.net.bind(addr).await?;
let mut relay_addr = conn.local_addr()?;
relay_addr.set_ip(self.relay_address);
return Ok((conn, relay_addr));
}
for _ in 0..max_retries {
let port = self.min_port + rand::random::<u16>() % (self.max_port - self.min_port + 1);
let addr = self
.net
.resolve_addr(use_ipv4, &format!("{}:{}", self.address, port))
.await?;
let conn = match self.net.bind(addr).await {
Ok(conn) => conn,
Err(_) => continue,
};
let mut relay_addr = conn.local_addr()?;
relay_addr.set_ip(self.relay_address);
return Ok((conn, relay_addr));
}
Err(Error::ErrMaxRetriesExceeded)
}
}