use real_tokio_rustls::rustls::{ClientConfig, RootCertStore, ServerName}; use real_tokio_rustls::{client::TlsStream, TlsConnector}; use tungstenite::client::{uri_mode, IntoClientRequest}; use tungstenite::error::TlsError; use tungstenite::handshake::client::Request; use tungstenite::stream::Mode; use tungstenite::Error; use std::convert::TryFrom; use crate::stream::Stream as StreamSwitcher; use crate::{client_async_with_config, domain, Response, WebSocketConfig, WebSocketStream}; use super::TokioAdapter; /// A stream that might be protected with TLS. pub type MaybeTlsStream = StreamSwitcher, TokioAdapter>>; pub type AutoStream = MaybeTlsStream; pub type Connector = TlsConnector; async fn wrap_stream( socket: S, domain: String, connector: Option, mode: Mode, ) -> Result, Error> where S: 'static + tokio::io::AsyncRead + tokio::io::AsyncWrite + Unpin, { match mode { Mode::Plain => Ok(StreamSwitcher::Plain(TokioAdapter::new(socket))), Mode::Tls => { let stream = { let connector = if let Some(connector) = connector { connector } else { let mut root_store = RootCertStore::empty(); #[cfg(feature = "tokio-rustls-native-certs")] { use real_tokio_rustls::rustls::Certificate; for cert in rustls_native_certs::load_native_certs()? { root_store .add(&Certificate(cert.0)) .map_err(TlsError::Webpki)?; } } #[cfg(all( feature = "tokio-rustls-webpki-roots", not(feature = "tokio-rustls-native-certs") ))] { use real_tokio_rustls::rustls::OwnedTrustAnchor; root_store.add_server_trust_anchors( webpki_roots::TLS_SERVER_ROOTS.0.iter().map(|ta| { OwnedTrustAnchor::from_subject_spki_name_constraints( ta.subject, ta.spki, ta.name_constraints, ) }), ); } TlsConnector::from(std::sync::Arc::new( ClientConfig::builder() .with_safe_defaults() .with_root_certificates(root_store) .with_no_client_auth(), )) }; let domain = ServerName::try_from(domain.as_str()) .map_err(|_| Error::Tls(TlsError::InvalidDnsName))?; connector.connect(domain, socket).await? }; Ok(StreamSwitcher::Tls(TokioAdapter::new(stream))) } } } /// Creates a WebSocket handshake from a request and a stream, /// upgrading the stream to TLS if required and using the given /// connector and WebSocket configuration. pub async fn client_async_tls_with_connector_and_config( request: R, stream: S, connector: Option, config: Option, ) -> Result<(WebSocketStream>, Response), Error> where R: IntoClientRequest + Unpin, S: 'static + tokio::io::AsyncRead + tokio::io::AsyncWrite + Unpin, AutoStream: Unpin, { let request: Request = request.into_client_request()?; let domain = domain(&request)?; // Make sure we check domain and mode first. URL must be valid. let mode = uri_mode(request.uri())?; let stream = wrap_stream(stream, domain, connector, mode).await?; client_async_with_config(request, stream, config).await }