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
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
|
use std::net::SocketAddr;
use async_trait::async_trait;
use smol::{
io::{split, AsyncReadExt, AsyncWriteExt, ReadHalf, WriteHalf},
lock::Mutex,
net::{TcpListener, TcpStream},
};
use crate::{
connection::{Connection, ToConn},
endpoint::Endpoint,
listener::{ConnListener, ToListener},
Error, Result,
};
/// TCP network connection implementation of the [`Connection`] trait.
pub struct TcpConn {
inner: TcpStream,
read: Mutex<ReadHalf<TcpStream>>,
write: Mutex<WriteHalf<TcpStream>>,
}
impl TcpConn {
/// Creates a new TcpConn
pub fn new(conn: TcpStream) -> Self {
let (read, write) = split(conn.clone());
Self {
inner: conn,
read: Mutex::new(read),
write: Mutex::new(write),
}
}
}
#[async_trait]
impl Connection for TcpConn {
fn peer_endpoint(&self) -> Result<Endpoint> {
Ok(Endpoint::new_tcp_addr(&self.inner.peer_addr()?))
}
fn local_endpoint(&self) -> Result<Endpoint> {
Ok(Endpoint::new_tcp_addr(&self.inner.local_addr()?))
}
async fn read(&self, buf: &mut [u8]) -> Result<usize> {
self.read.lock().await.read(buf).await.map_err(Error::from)
}
async fn write(&self, buf: &[u8]) -> Result<usize> {
self.write
.lock()
.await
.write(buf)
.await
.map_err(Error::from)
}
}
#[async_trait]
impl ConnListener for TcpListener {
fn local_endpoint(&self) -> Result<Endpoint> {
Ok(Endpoint::new_tcp_addr(&self.local_addr()?))
}
async fn accept(&self) -> Result<Box<dyn Connection>> {
let (conn, _) = self.accept().await?;
conn.set_nodelay(true)?;
Ok(Box::new(TcpConn::new(conn)))
}
}
/// Connects to the given TCP address and port.
pub async fn dial(endpoint: &Endpoint) -> Result<TcpConn> {
let addr = SocketAddr::try_from(endpoint.clone())?;
let conn = TcpStream::connect(addr).await?;
conn.set_nodelay(true)?;
Ok(TcpConn::new(conn))
}
/// Listens on the given TCP address and port.
pub async fn listen(endpoint: &Endpoint) -> Result<TcpListener> {
let addr = SocketAddr::try_from(endpoint.clone())?;
let listener = TcpListener::bind(addr).await?;
Ok(listener)
}
impl From<TcpStream> for Box<dyn Connection> {
fn from(conn: TcpStream) -> Self {
Box::new(TcpConn::new(conn))
}
}
impl From<TcpListener> for Box<dyn ConnListener> {
fn from(listener: TcpListener) -> Self {
Box::new(listener)
}
}
impl ToConn for TcpStream {
fn to_conn(self) -> Box<dyn Connection> {
self.into()
}
}
impl ToConn for TcpConn {
fn to_conn(self) -> Box<dyn Connection> {
Box::new(self)
}
}
impl ToListener for TcpListener {
fn to_listener(self) -> Box<dyn ConnListener> {
self.into()
}
}
|