use std::io::{Read, Write};
#[cfg(any(feature = "native-tls", feature = "__rustls-tls"))]
use crate::error::{Error, UrlError};
use crate::{
client::{client_with_config, uri_mode, IntoClientRequest},
error::Result,
handshake::{
client::{ClientHandshake, Response},
core::HandshakeError,
},
protocol::{config::WebSocketConfig, websocket::WebSocket},
stream::SimplifiedStream,
};
#[non_exhaustive]
#[allow(missing_debug_implementations)]
pub enum Connector {
Plain,
#[cfg(feature = "native-tls")]
NativeTls(native_tls_crate::TlsConnector),
#[cfg(feature = "__rustls-tls")]
Rustls(std::sync::Arc<rustls::ClientConfig>),
}
mod encryption {
#[cfg(feature = "native-tls")]
pub mod native_tls {
use crate::{
error::{Error, Result, TlsError},
stream::{Mode, SimplifiedStream},
};
use native_tls_crate::{HandshakeError as TlsHandshakeError, TlsConnector};
use std::io::{Read, Write};
pub fn wrap_stream<S>(
socket: S,
domain: &str,
mode: Mode,
tls_connection: Option<TlsConnector>,
) -> Result<SimplifiedStream<S>>
where
S: Read + Write,
{
match mode {
Mode::Plain => Ok(SimplifiedStream::Plain(socket)),
Mode::Tls => {
let try_connector = tls_connection.map_or_else(TlsConnector::new, Ok);
let connector = try_connector.map_err(TlsError::Native)?;
let connected = connector.connect(domain, socket);
match connected {
Err(e) => match e {
TlsHandshakeError::Failure(f) => Err(Error::Tls(f.into())),
TlsHandshakeError::WouldBlock(_) => {
panic!("Bug: TLS handshake not blocked")
}
},
Ok(s) => Ok(SimplifiedStream::NativeTls(s)),
}
}
}
}
}
#[cfg(feature = "__rustls-tls")]
pub mod rustls {
use crate::{
error::{Result, TlsError},
stream::{Mode, SimplifiedStream},
};
use rustls::{ClientConfig, ClientConnection, RootCertStore, StreamOwned};
use rustls_pki_types::ServerName;
use std::{
io::{Read, Write},
sync::Arc,
};
pub fn wrap_stream<S>(
socket: S,
domain: &str,
mode: Mode,
tls_connector: Option<Arc<ClientConfig>>,
) -> Result<SimplifiedStream<S>>
where
S: Read + Write,
{
match mode {
Mode::Plain => Ok(SimplifiedStream::Plain(socket)),
Mode::Tls => {
let config = match tls_connector {
Some(config) => config,
None => {
#[allow(unused_mut)]
let mut root_store = RootCertStore::empty();
#[cfg(feature = "rustls-tls-native-roots")]
{
let rustls_native_certs::CertificateResult {
certs, errors, ..
} = rustls_native_certs::load_native_certs();
if certs.is_empty() {
return Err(std::io::Error::new(
std::io::ErrorKind::NotFound,
format!("No native root CA certificates found (errors: {errors:?})")
).into());
}
}
#[cfg(feature = "rustls-tls-webpki-roots")]
{
root_store.extend(webpki_roots::TLS_SERVER_ROOTS.iter().cloned());
}
Arc::new(
ClientConfig::builder()
.with_root_certificates(root_store)
.with_no_client_auth(),
)
}
};
let domain = ServerName::try_from(domain)
.map_err(|_| TlsError::InvalidDnsName)?
.to_owned();
let client = ClientConnection::new(config, domain).map_err(TlsError::Rustls)?;
let stream = StreamOwned::new(client, socket);
Ok(SimplifiedStream::Rustls(stream))
}
}
}
}
pub mod plain {
use crate::{
error::{Error, Result, UrlError},
stream::{Mode, SimplifiedStream},
};
use std::io::{Read, Write};
pub fn wrap_stream<S>(socket: S, mode: Mode) -> Result<SimplifiedStream<S>>
where
S: Read + Write,
{
match mode {
Mode::Plain => Ok(SimplifiedStream::Plain(socket)),
Mode::Tls => Err(Error::Url(UrlError::TlsFeatureNotEnabled)),
}
}
}
}
type TlsErrorHandshake<S> = HandshakeError<ClientHandshake<SimplifiedStream<S>>>;
pub fn client_tls<R, S>(
request: R,
stream: S,
) -> Result<(WebSocket<SimplifiedStream<S>>, Response), TlsErrorHandshake<S>>
where
R: IntoClientRequest,
S: Read + Write,
{
client_tls_with_config(request, stream, None, None)
}
pub fn client_tls_with_config<R, S>(
request: R,
stream: S,
config: Option<WebSocketConfig>,
connector: Option<Connector>,
) -> Result<(WebSocket<SimplifiedStream<S>>, Response), TlsErrorHandshake<S>>
where
R: IntoClientRequest,
S: Read + Write,
{
let request = request.into_client_request()?;
#[cfg(any(feature = "native-tls", feature = "__rustls-tls"))]
let domain = match request.uri().host() {
Some(d) => Ok(d.to_string()),
None => Err(Error::Url(UrlError::MissingHost)),
}?;
let mode = uri_mode(request.uri())?;
let stream = match connector {
Some(conn) => match conn {
#[cfg(feature = "native-tls")]
Connector::NativeTls(conn) => {
self::encryption::native_tls::wrap_stream(stream, &domain, mode, Some(conn))
}
#[cfg(feature = "__rustls-tls")]
Connector::Rustls(conn) => {
self::encryption::rustls::wrap_stream(stream, &domain, mode, Some(conn))
}
Connector::Plain => self::encryption::plain::wrap_stream(stream, mode),
},
None => {
#[cfg(feature = "native-tls")]
{
self::encryption::native_tls::wrap_stream(stream, &domain, mode, None)
}
#[cfg(all(feature = "__rustls-tls", not(feature = "native-tls")))]
{
self::encryption::rustls::wrap_stream(stream, &domain, mode, None)
}
#[cfg(not(any(feature = "native-tls", feature = "__rustls-tls")))]
{
self::encryption::plain::wrap_stream(stream, mode)
}
}
}?;
client_with_config(request, stream, config)
}