2021-03-29 15:59:14 +08:00
|
|
|
use crate::{bail, ResultType};
|
2022-01-02 22:55:33 +08:00
|
|
|
use anyhow::anyhow;
|
|
|
|
use bytes::{Bytes, BytesMut};
|
2021-06-25 19:42:51 +08:00
|
|
|
use futures::{SinkExt, StreamExt};
|
2021-03-29 15:59:14 +08:00
|
|
|
use protobuf::Message;
|
2021-06-25 19:42:51 +08:00
|
|
|
use socket2::{Domain, Socket, Type};
|
2022-01-04 00:44:50 +08:00
|
|
|
use std::net::SocketAddr;
|
2022-01-02 22:55:33 +08:00
|
|
|
use tokio::net::{ToSocketAddrs, UdpSocket};
|
2022-01-04 00:44:50 +08:00
|
|
|
use tokio_socks::{udp::Socks5UdpFramed, IntoTargetAddr, TargetAddr, ToProxyAddrs};
|
2021-03-29 15:59:14 +08:00
|
|
|
use tokio_util::{codec::BytesCodec, udp::UdpFramed};
|
|
|
|
|
2022-01-04 00:44:50 +08:00
|
|
|
pub enum FramedSocket {
|
|
|
|
Direct(UdpFramed<BytesCodec>),
|
|
|
|
ProxySocks(Socks5UdpFramed),
|
2022-01-02 22:55:33 +08:00
|
|
|
}
|
|
|
|
|
2021-06-25 19:42:51 +08:00
|
|
|
fn new_socket(addr: SocketAddr, reuse: bool) -> Result<Socket, std::io::Error> {
|
|
|
|
let socket = match addr {
|
|
|
|
SocketAddr::V4(..) => Socket::new(Domain::ipv4(), Type::dgram(), None),
|
|
|
|
SocketAddr::V6(..) => Socket::new(Domain::ipv6(), Type::dgram(), None),
|
|
|
|
}?;
|
|
|
|
if reuse {
|
|
|
|
// windows has no reuse_port, but it's reuse_address
|
|
|
|
// almost equals to unix's reuse_port + reuse_address,
|
2021-11-14 23:22:05 +08:00
|
|
|
// though may introduce nondeterministic behavior
|
2021-06-25 19:42:51 +08:00
|
|
|
#[cfg(unix)]
|
|
|
|
socket.set_reuse_port(true)?;
|
|
|
|
socket.set_reuse_address(true)?;
|
|
|
|
}
|
|
|
|
socket.bind(&addr.into())?;
|
|
|
|
Ok(socket)
|
|
|
|
}
|
|
|
|
|
2022-01-04 00:44:50 +08:00
|
|
|
impl FramedSocket {
|
2021-03-29 15:59:14 +08:00
|
|
|
pub async fn new<T: ToSocketAddrs>(addr: T) -> ResultType<Self> {
|
|
|
|
let socket = UdpSocket::bind(addr).await?;
|
2022-01-04 00:44:50 +08:00
|
|
|
Ok(Self::Direct(UdpFramed::new(socket, BytesCodec::new())))
|
2021-03-29 15:59:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(clippy::never_loop)]
|
2021-06-25 19:42:51 +08:00
|
|
|
pub async fn new_reuse<T: std::net::ToSocketAddrs>(addr: T) -> ResultType<Self> {
|
|
|
|
for addr in addr.to_socket_addrs()? {
|
2022-01-02 22:55:33 +08:00
|
|
|
let socket = new_socket(addr, true)?.into_udp_socket();
|
2022-01-04 00:44:50 +08:00
|
|
|
return Ok(Self::Direct(UdpFramed::new(
|
2022-01-02 22:55:33 +08:00
|
|
|
UdpSocket::from_std(socket)?,
|
2021-03-29 15:59:14 +08:00
|
|
|
BytesCodec::new(),
|
2022-01-04 00:44:50 +08:00
|
|
|
)));
|
2021-03-29 15:59:14 +08:00
|
|
|
}
|
|
|
|
bail!("could not resolve to any address");
|
|
|
|
}
|
2022-01-02 22:55:33 +08:00
|
|
|
|
2022-01-04 19:49:44 +08:00
|
|
|
pub async fn new_proxy<'a, 't, P: ToProxyAddrs, T1: IntoTargetAddr<'t>, T2: ToSocketAddrs>(
|
2022-01-02 22:55:33 +08:00
|
|
|
proxy: P,
|
|
|
|
target: T1,
|
|
|
|
local: T2,
|
|
|
|
username: &'a str,
|
|
|
|
password: &'a str,
|
|
|
|
ms_timeout: u64,
|
|
|
|
) -> ResultType<(Self, SocketAddr)> {
|
|
|
|
let framed = if username.trim().is_empty() {
|
|
|
|
super::timeout(
|
|
|
|
ms_timeout,
|
|
|
|
Socks5UdpFramed::connect(proxy, target, Some(local)),
|
|
|
|
)
|
|
|
|
.await??
|
|
|
|
} else {
|
|
|
|
super::timeout(
|
|
|
|
ms_timeout,
|
|
|
|
Socks5UdpFramed::connect_with_password(
|
|
|
|
proxy,
|
|
|
|
target,
|
|
|
|
Some(local),
|
|
|
|
username,
|
|
|
|
password,
|
|
|
|
),
|
|
|
|
)
|
|
|
|
.await??
|
|
|
|
};
|
|
|
|
let addr = if let TargetAddr::Ip(c) = framed.target_addr() {
|
|
|
|
c
|
|
|
|
} else {
|
|
|
|
unreachable!()
|
|
|
|
};
|
|
|
|
log::trace!(
|
|
|
|
"Socks5 udp connected, local addr: {}, target addr: {}",
|
|
|
|
framed.local_addr().unwrap(),
|
|
|
|
&addr
|
|
|
|
);
|
2022-01-04 00:44:50 +08:00
|
|
|
Ok((Self::ProxySocks(framed), addr))
|
2022-01-02 22:55:33 +08:00
|
|
|
}
|
2021-03-29 15:59:14 +08:00
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub async fn send(&mut self, msg: &impl Message, addr: SocketAddr) -> ResultType<()> {
|
2022-01-04 00:44:50 +08:00
|
|
|
let send_data = (Bytes::from(msg.write_to_bytes().unwrap()), addr);
|
|
|
|
let _ = match self {
|
|
|
|
Self::Direct(f) => f.send(send_data).await?,
|
|
|
|
Self::ProxySocks(f) => f.send(send_data).await?,
|
|
|
|
};
|
2021-03-29 15:59:14 +08:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub async fn send_raw(&mut self, msg: &'static [u8], addr: SocketAddr) -> ResultType<()> {
|
2022-01-04 00:44:50 +08:00
|
|
|
let _ = match self {
|
|
|
|
Self::Direct(f) => f.send((Bytes::from(msg), addr)).await?,
|
|
|
|
Self::ProxySocks(f) => f.send((Bytes::from(msg), addr)).await?,
|
|
|
|
};
|
2021-03-29 15:59:14 +08:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2022-01-04 00:44:50 +08:00
|
|
|
pub async fn next(&mut self) -> Option<ResultType<(BytesMut, SocketAddr)>> {
|
|
|
|
match self {
|
|
|
|
Self::Direct(f) => match f.next().await {
|
|
|
|
Some(Ok((data, addr))) => Some(Ok((data, addr))),
|
|
|
|
Some(Err(e)) => Some(Err(anyhow!(e))),
|
|
|
|
None => None,
|
|
|
|
},
|
|
|
|
Self::ProxySocks(f) => match f.next().await {
|
|
|
|
Some(Ok((data, addr))) => Some(Ok((data.data, addr))),
|
|
|
|
Some(Err(e)) => Some(Err(anyhow!(e))),
|
|
|
|
None => None,
|
|
|
|
},
|
|
|
|
}
|
2021-03-29 15:59:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2022-01-04 00:44:50 +08:00
|
|
|
pub async fn next_timeout(&mut self, ms: u64) -> Option<ResultType<(BytesMut, SocketAddr)>> {
|
2021-03-29 15:59:14 +08:00
|
|
|
if let Ok(res) =
|
2022-01-04 00:44:50 +08:00
|
|
|
tokio::time::timeout(std::time::Duration::from_millis(ms), self.next()).await
|
2021-03-29 15:59:14 +08:00
|
|
|
{
|
|
|
|
res
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|