1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
use super::*;
use crate::errors::*;
use std::net::IpAddr;
use tokio::net::UdpSocket;
use async_trait::async_trait;
pub struct RelayAddressGeneratorRanges {
pub relay_address: IpAddr,
pub min_port: u16,
pub max_port: u16,
pub max_retries: u16,
pub address: String,
}
#[async_trait]
impl RelayAddressGenerator for RelayAddressGeneratorRanges {
fn validate(&self) -> Result<(), Error> {
if self.min_port == 0 {
Err(ERR_MIN_PORT_NOT_ZERO.to_owned())
} else if self.max_port == 0 {
Err(ERR_MAX_PORT_NOT_ZERO.to_owned())
} else if self.max_port < self.min_port {
Err(ERR_MAX_PORT_LESS_THAN_MIN_PORT.to_owned())
} else if self.address.is_empty() {
Err(ERR_LISTENING_ADDRESS_INVALID.to_owned())
} else {
Ok(())
}
}
async fn allocate_conn(
&self,
_network: &str,
requested_port: u16,
) -> Result<(Arc<dyn Conn + Send + Sync>, SocketAddr), Error> {
let max_retries = if self.max_retries == 0 {
10
} else {
self.max_retries
};
if requested_port != 0 {
let conn = UdpSocket::bind(format!("{}:{}", self.address, requested_port)).await?;
let mut relay_addr = conn.local_addr()?;
relay_addr.set_ip(self.relay_address);
return Ok((Arc::new(conn), relay_addr));
}
for _ in 0..max_retries {
let port = self.min_port + rand::random::<u16>() % (self.max_port + 1 - self.min_port);
let conn = match UdpSocket::bind(format!("{}:{}", self.address, port)).await {
Ok(conn) => conn,
Err(_) => continue,
};
let mut relay_addr = conn.local_addr()?;
relay_addr.set_ip(self.relay_address);
return Ok((Arc::new(conn), relay_addr));
}
Err(ERR_MAX_RETRIES_EXCEEDED.to_owned())
}
}