#[cfg(feature = "__rustls-tls")]
use std::ops::Deref;
use std::{
fmt::Debug,
io::{Read, Result as IoResult, Write},
net::TcpStream,
};
#[cfg(feature = "native-tls")]
use native_tls_crate::TlsStream;
#[cfg(feature = "__rustls-tls")]
use rustls::StreamOwned;
#[derive(Debug, Clone, Copy)]
pub enum Mode {
Plain,
Tls,
}
pub trait NoDelay {
fn set_nodelay(&mut self, no_delay: bool) -> IoResult<()>;
}
impl NoDelay for TcpStream {
fn set_nodelay(&mut self, no_delay: bool) -> IoResult<()> {
TcpStream::set_nodelay(self, no_delay)
}
}
#[cfg(feature = "native-tls")]
impl<S: Read + Write + NoDelay> NoDelay for TlsStream<S> {
fn set_nodelay(&mut self, no_delay: bool) -> IoResult<()> {
self.get_mut().set_nodelay(no_delay)
}
}
#[cfg(feature = "__rustls-tls")]
impl<S, SD, T> NoDelay for StreamOwned<S, T>
where
S: Deref<Target = rustls::ConnectionCommon<SD>>,
SD: rustls::SideData,
T: Read + Write + NoDelay,
{
fn set_nodelay(&mut self, no_delay: bool) -> IoResult<()> {
self.sock.set_nodelay(no_delay)
}
}
#[non_exhaustive]
#[allow(clippy::large_enum_variant)]
pub enum SimplifiedStream<S: Read + Write> {
Plain(S),
#[cfg(feature = "native-tls")]
NativeTls(native_tls_crate::TlsStream<S>),
#[cfg(feature = "__rustls-tls")]
Rustls(rustls::StreamOwned<rustls::ClientConnection, S>),
}
impl<S: Read + Write + Debug> Debug for SimplifiedStream<S> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
match self {
Self::Plain(s) => f.debug_tuple("SimplifiedStream::Plain").field(s).finish(),
#[cfg(feature = "native-tls")]
Self::NativeTls(s) => f.debug_tuple("SimplifiedStream::NativeTls").field(s).finish(),
#[cfg(feature = "__rustls-tls")]
Self::Rustls(s) => {
struct RustlsStreamDebug<'a, S: Read + Write>(
&'a rustls::StreamOwned<rustls::ClientConnection, S>,
);
impl<S: Read + Write + Debug> Debug for RustlsStreamDebug<'_, S> {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("StreamOwned")
.field("conn", &self.0.conn)
.field("sock", &self.0.sock)
.finish()
}
}
f.debug_tuple("SimplifiedStream::Rustls").field(&RustlsStreamDebug(s)).finish()
}
}
}
}
impl<S: Read + Write> Read for SimplifiedStream<S> {
fn read(&mut self, buf: &mut [u8]) -> IoResult<usize> {
match self {
Self::Plain(ref mut s) => s.read(buf),
#[cfg(feature = "native-tls")]
Self::NativeTls(ref mut s) => s.read(buf),
#[cfg(feature = "__rustls-tls")]
Self::Rustls(ref mut s) => s.read(buf),
}
}
}
impl<S: Read + Write> Write for SimplifiedStream<S> {
fn write(&mut self, buf: &[u8]) -> IoResult<usize> {
match self {
Self::Plain(ref mut s) => s.write(buf),
#[cfg(feature = "native-tls")]
Self::NativeTls(ref mut s) => s.write(buf),
#[cfg(feature = "__rustls-tls")]
Self::Rustls(ref mut s) => s.write(buf),
}
}
fn flush(&mut self) -> IoResult<()> {
match self {
Self::Plain(ref mut s) => s.flush(),
#[cfg(feature = "native-tls")]
Self::NativeTls(ref mut s) => s.flush(),
#[cfg(feature = "__rustls-tls")]
Self::Rustls(ref mut s) => s.flush(),
}
}
}
impl<S: Read + Write + NoDelay> NoDelay for SimplifiedStream<S> {
fn set_nodelay(&mut self, no_delay: bool) -> IoResult<()> {
match self {
Self::Plain(ref mut s) => s.set_nodelay(no_delay),
#[cfg(feature = "native-tls")]
Self::NativeTls(ref mut s) => s.set_nodelay(no_delay),
#[cfg(feature = "__rustls-tls")]
Self::Rustls(ref mut s) => s.set_nodelay(no_delay),
}
}
}