use std::{
net::SocketAddr,
pin::Pin,
sync::atomic::{AtomicUsize, Ordering},
task::{Context, Poll},
time::Duration,
};
use bytes::{Bytes, BytesMut};
use tokio::{
io::{AsyncRead, AsyncReadExt, AsyncWrite, AsyncWriteExt, ReadBuf},
net::{TcpStream, ToSocketAddrs, UdpSocket},
};
use super::proto::{Address, AsyncStreamOperation, Reply, Response, StreamOperation, UdpHeader};
#[derive(Debug)]
pub struct UdpAssociate<S> {
stream: TcpStream,
_state: S,
}
impl<S: Default> UdpAssociate<S> {
#[inline]
pub(super) fn new(stream: TcpStream) -> Self {
Self {
stream,
_state: S::default(),
}
}
pub async fn reply(
mut self,
reply: Reply,
addr: Address,
) -> std::io::Result<UdpAssociate<Ready>> {
let resp = Response::new(reply, addr);
resp.write_to_async_stream(&mut self.stream).await?;
Ok(UdpAssociate::<Ready>::new(self.stream))
}
#[inline]
pub async fn shutdown(&mut self) -> std::io::Result<()> {
self.stream.shutdown().await
}
#[inline]
pub fn local_addr(&self) -> std::io::Result<SocketAddr> {
self.stream.local_addr()
}
#[inline]
pub fn peer_addr(&self) -> std::io::Result<SocketAddr> {
self.stream.peer_addr()
}
#[inline]
pub fn linger(&self) -> std::io::Result<Option<Duration>> {
self.stream.linger()
}
#[inline]
pub fn set_linger(&self, dur: Option<Duration>) -> std::io::Result<()> {
self.stream.set_linger(dur)
}
#[inline]
pub fn nodelay(&self) -> std::io::Result<bool> {
self.stream.nodelay()
}
#[inline]
pub fn set_nodelay(&self, nodelay: bool) -> std::io::Result<()> {
self.stream.set_nodelay(nodelay)
}
#[inline]
pub fn ttl(&self) -> std::io::Result<u32> {
self.stream.ttl()
}
#[inline]
pub fn set_ttl(&self, ttl: u32) -> std::io::Result<()> {
self.stream.set_ttl(ttl)
}
}
#[derive(Debug, Default)]
pub struct NeedReply;
#[derive(Debug, Default)]
pub struct Ready;
impl UdpAssociate<Ready> {
pub async fn wait_until_closed(&mut self) -> std::io::Result<()> {
loop {
match self.stream.read(&mut [0]).await {
Ok(0) => break Ok(()),
Ok(_) => {}
Err(err) => break Err(err),
}
}
}
}
impl std::ops::Deref for UdpAssociate<Ready> {
type Target = TcpStream;
#[inline]
fn deref(&self) -> &Self::Target {
&self.stream
}
}
impl std::ops::DerefMut for UdpAssociate<Ready> {
#[inline]
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.stream
}
}
impl AsyncRead for UdpAssociate<Ready> {
#[inline]
fn poll_read(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &mut ReadBuf<'_>,
) -> Poll<std::io::Result<()>> {
Pin::new(&mut self.stream).poll_read(cx, buf)
}
}
impl AsyncWrite for UdpAssociate<Ready> {
#[inline]
fn poll_write(
mut self: Pin<&mut Self>,
cx: &mut Context<'_>,
buf: &[u8],
) -> Poll<std::io::Result<usize>> {
Pin::new(&mut self.stream).poll_write(cx, buf)
}
#[inline]
fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<std::io::Result<()>> {
Pin::new(&mut self.stream).poll_flush(cx)
}
#[inline]
fn poll_shutdown(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<std::io::Result<()>> {
Pin::new(&mut self.stream).poll_shutdown(cx)
}
}
impl<S> From<UdpAssociate<S>> for TcpStream {
#[inline]
fn from(conn: UdpAssociate<S>) -> Self {
conn.stream
}
}
#[derive(Debug)]
pub struct AssociatedUdpSocket {
socket: UdpSocket,
buf_size: AtomicUsize,
}
impl AssociatedUdpSocket {
#[inline]
pub async fn connect<A: ToSocketAddrs>(&self, addr: A) -> std::io::Result<()> {
self.socket.connect(addr).await
}
#[inline]
pub fn get_max_packet_size(&self) -> usize {
self.buf_size.load(Ordering::Relaxed)
}
#[inline]
pub fn set_max_packet_size(&self, size: usize) {
self.buf_size.store(size, Ordering::Release);
}
pub async fn recv(&self) -> std::io::Result<(Bytes, u8, Address)> {
loop {
let max_packet_size = self.buf_size.load(Ordering::Acquire);
let mut buf = vec![0; max_packet_size];
let len = self.socket.recv(&mut buf).await?;
buf.truncate(len);
let pkt = Bytes::from(buf);
if let Ok(header) = UdpHeader::retrieve_from_async_stream(&mut pkt.as_ref()).await {
let pkt = pkt.slice(header.len()..);
return Ok((pkt, header.frag, header.address));
}
}
}
pub async fn recv_from(&self) -> std::io::Result<(Bytes, u8, Address, SocketAddr)> {
loop {
let max_packet_size = self.buf_size.load(Ordering::Acquire);
let mut buf = vec![0; max_packet_size];
let (len, src_addr) = self.socket.recv_from(&mut buf).await?;
buf.truncate(len);
let pkt = Bytes::from(buf);
if let Ok(header) = UdpHeader::retrieve_from_async_stream(&mut pkt.as_ref()).await {
let pkt = pkt.slice(header.len()..);
return Ok((pkt, header.frag, header.address, src_addr));
}
}
}
pub async fn send<P: AsRef<[u8]>>(
&self,
pkt: P,
frag: u8,
from_addr: Address,
) -> std::io::Result<usize> {
let header = UdpHeader::new(frag, from_addr);
let mut buf = BytesMut::with_capacity(header.len() + pkt.as_ref().len());
header.write_to_buf(&mut buf);
buf.extend_from_slice(pkt.as_ref());
self.socket.send(&buf).await.map(|len| len - header.len())
}
pub async fn send_to<P: AsRef<[u8]>>(
&self,
pkt: P,
frag: u8,
from_addr: Address,
to_addr: SocketAddr,
) -> std::io::Result<usize> {
let header = UdpHeader::new(frag, from_addr);
let mut buf = BytesMut::with_capacity(header.len() + pkt.as_ref().len());
header.write_to_buf(&mut buf);
buf.extend_from_slice(pkt.as_ref());
self.socket
.send_to(&buf, to_addr)
.await
.map(|len| len - header.len())
}
}
impl From<(UdpSocket, usize)> for AssociatedUdpSocket {
#[inline]
fn from(from: (UdpSocket, usize)) -> Self {
AssociatedUdpSocket {
socket: from.0,
buf_size: AtomicUsize::new(from.1),
}
}
}
impl From<AssociatedUdpSocket> for UdpSocket {
#[inline]
fn from(from: AssociatedUdpSocket) -> Self {
from.socket
}
}
impl AsRef<UdpSocket> for AssociatedUdpSocket {
#[inline]
fn as_ref(&self) -> &UdpSocket {
&self.socket
}
}
impl AsMut<UdpSocket> for AssociatedUdpSocket {
#[inline]
fn as_mut(&mut self) -> &mut UdpSocket {
&mut self.socket
}
}