Skip to main content

ntex_server/net/
socket.rs

1use std::{fmt, io, net};
2
3use ntex_io::Io;
4use ntex_service::cfg::SharedCfg;
5
6use super::Token;
7
8#[derive(Debug)]
9pub enum Stream {
10    Tcp(net::TcpStream),
11    #[cfg(unix)]
12    Uds(std::os::unix::net::UnixStream),
13}
14
15impl Stream {
16    pub(crate) fn convert(self, cfg: SharedCfg) -> Result<Io, io::Error> {
17        match self {
18            Stream::Tcp(stream) => ntex_net::from_tcp_stream(stream, cfg),
19            #[cfg(unix)]
20            Stream::Uds(stream) => ntex_net::from_unix_stream(stream, cfg),
21        }
22    }
23}
24
25#[derive(Debug)]
26pub struct Connection {
27    pub(crate) io: Stream,
28    pub(crate) token: Token,
29}
30
31pub enum Listener {
32    Tcp(net::TcpListener),
33    #[cfg(unix)]
34    Uds(std::os::unix::net::UnixListener),
35}
36
37impl fmt::Debug for Listener {
38    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
39        match *self {
40            Listener::Tcp(ref lst) => write!(f, "{lst:?}"),
41            #[cfg(unix)]
42            Listener::Uds(ref lst) => write!(f, "{lst:?}"),
43        }
44    }
45}
46
47impl fmt::Display for Listener {
48    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
49        match *self {
50            Listener::Tcp(ref lst) => write!(f, "{}", lst.local_addr().ok().unwrap()),
51            #[cfg(unix)]
52            Listener::Uds(ref lst) => {
53                write!(f, "{:?}", lst.local_addr().ok().unwrap())
54            }
55        }
56    }
57}
58
59pub(crate) enum SocketAddr {
60    Tcp(net::SocketAddr),
61    #[cfg(unix)]
62    Uds(std::os::unix::net::SocketAddr),
63}
64
65impl fmt::Display for SocketAddr {
66    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
67        match *self {
68            SocketAddr::Tcp(ref addr) => write!(f, "{addr}"),
69            #[cfg(unix)]
70            SocketAddr::Uds(ref addr) => write!(f, "{addr:?}"),
71        }
72    }
73}
74
75impl fmt::Debug for SocketAddr {
76    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
77        match *self {
78            SocketAddr::Tcp(ref addr) => write!(f, "{addr:?}"),
79            #[cfg(unix)]
80            SocketAddr::Uds(ref addr) => write!(f, "{addr:?}"),
81        }
82    }
83}
84
85impl Listener {
86    pub(super) fn from_tcp(lst: net::TcpListener) -> Self {
87        let _ = lst.set_nonblocking(true);
88        Listener::Tcp(lst)
89    }
90
91    #[cfg(unix)]
92    pub(super) fn from_uds(lst: std::os::unix::net::UnixListener) -> Self {
93        let _ = lst.set_nonblocking(true);
94        Listener::Uds(lst)
95    }
96
97    pub(crate) fn local_addr(&self) -> SocketAddr {
98        match self {
99            Listener::Tcp(lst) => SocketAddr::Tcp(lst.local_addr().unwrap()),
100            #[cfg(unix)]
101            Listener::Uds(lst) => SocketAddr::Uds(lst.local_addr().unwrap()),
102        }
103    }
104
105    pub(crate) fn accept(&self) -> io::Result<Option<Stream>> {
106        match *self {
107            Listener::Tcp(ref lst) => {
108                lst.accept().map(|(stream, _)| Some(Stream::Tcp(stream)))
109            }
110            #[cfg(unix)]
111            Listener::Uds(ref lst) => {
112                lst.accept().map(|(stream, _)| Some(Stream::Uds(stream)))
113            }
114        }
115    }
116
117    pub(crate) fn remove_source(&self) {
118        match *self {
119            Listener::Tcp(_) => (),
120            #[cfg(unix)]
121            Listener::Uds(ref lst) => {
122                // cleanup file path
123                if let Ok(addr) = lst.local_addr()
124                    && let Some(path) = addr.as_pathname()
125                {
126                    let _ = std::fs::remove_file(path);
127                }
128            }
129        }
130    }
131}
132
133#[cfg(unix)]
134mod listener_impl {
135    use std::os::fd::{AsFd, BorrowedFd};
136    use std::os::unix::io::{AsRawFd, RawFd};
137
138    impl AsFd for super::Listener {
139        fn as_fd(&self) -> BorrowedFd<'_> {
140            match *self {
141                super::Listener::Tcp(ref lst) => lst.as_fd(),
142                super::Listener::Uds(ref lst) => lst.as_fd(),
143            }
144        }
145    }
146
147    impl AsRawFd for super::Listener {
148        fn as_raw_fd(&self) -> RawFd {
149            match *self {
150                super::Listener::Tcp(ref lst) => lst.as_raw_fd(),
151                super::Listener::Uds(ref lst) => lst.as_raw_fd(),
152            }
153        }
154    }
155}
156
157#[cfg(windows)]
158mod listener_impl {
159    use std::os::windows::io::{AsRawSocket, AsSocket, BorrowedSocket, RawSocket};
160
161    impl AsSocket for super::Listener {
162        fn as_socket(&self) -> BorrowedSocket<'_> {
163            match *self {
164                super::Listener::Tcp(ref lst) => lst.as_socket(),
165            }
166        }
167    }
168
169    impl AsRawSocket for super::Listener {
170        fn as_raw_socket(&self) -> RawSocket {
171            match *self {
172                super::Listener::Tcp(ref lst) => lst.as_raw_socket(),
173            }
174        }
175    }
176}
177
178#[cfg(test)]
179mod tests {
180    use super::*;
181
182    #[test]
183    fn socket_addr() {
184        use socket2::{Domain, SockAddr, Socket, Type};
185
186        let addr = SocketAddr::Tcp("127.0.0.1:8080".parse().unwrap());
187        assert!(format!("{addr:?}").contains("127.0.0.1:8080"));
188        assert_eq!(format!("{addr}"), "127.0.0.1:8080");
189
190        let addr: net::SocketAddr = "127.0.0.1:0".parse().unwrap();
191        let socket = Socket::new(Domain::IPV4, Type::STREAM, None).unwrap();
192        socket.set_reuse_address(true).unwrap();
193        socket.bind(&SockAddr::from(addr)).unwrap();
194        let lst = Listener::Tcp(net::TcpListener::from(socket));
195        assert!(format!("{lst:?}").contains("TcpListener"));
196        assert!(format!("{lst}").contains("127.0.0.1"));
197    }
198
199    #[test]
200    #[cfg(unix)]
201    fn uds() {
202        use std::os::unix::net::UnixListener;
203
204        let _ = std::fs::remove_file("/tmp/sock.xxxxx");
205        if let Ok(lst) = UnixListener::bind("/tmp/sock.xxxxx") {
206            let addr = lst.local_addr().expect("Couldn't get local address");
207            let a = SocketAddr::Uds(addr);
208            assert!(format!("{a:?}").contains("/tmp/sock.xxxxx"));
209            assert!(format!("{a}").contains("/tmp/sock.xxxxx"));
210
211            let lst = Listener::Uds(lst);
212            assert!(format!("{lst:?}").contains("/tmp/sock.xxxxx"));
213            assert!(format!("{lst}").contains("/tmp/sock.xxxxx"));
214        }
215    }
216}