2020-03-05 00:25:24 +00:00
|
|
|
use futures::{sink::SinkExt, stream::StreamExt};
|
2021-02-15 19:45:58 +00:00
|
|
|
|
2021-02-15 18:59:46 +00:00
|
|
|
#[cfg(feature = "tls-rust")]
|
2021-02-15 19:45:58 +00:00
|
|
|
use {
|
|
|
|
idna,
|
|
|
|
std::sync::Arc,
|
|
|
|
tokio_rustls::{client::TlsStream, rustls::ClientConfig, TlsConnector},
|
|
|
|
webpki::DNSNameRef,
|
|
|
|
webpki_roots,
|
|
|
|
};
|
|
|
|
|
2021-02-15 18:59:46 +00:00
|
|
|
#[cfg(feature = "tls-native")]
|
2021-02-15 19:45:58 +00:00
|
|
|
use {
|
|
|
|
native_tls::TlsConnector as NativeTlsConnector,
|
|
|
|
tokio_native_tls::{TlsConnector, TlsStream},
|
|
|
|
};
|
|
|
|
|
2020-03-05 00:25:24 +00:00
|
|
|
use tokio::io::{AsyncRead, AsyncWrite};
|
2020-05-29 23:19:06 +00:00
|
|
|
use xmpp_parsers::{ns, Element};
|
2017-06-04 22:42:35 +00:00
|
|
|
|
2018-12-18 17:29:31 +00:00
|
|
|
use crate::xmpp_codec::Packet;
|
|
|
|
use crate::xmpp_stream::XMPPStream;
|
2020-03-05 00:25:24 +00:00
|
|
|
use crate::{Error, ProtocolError};
|
2017-06-04 22:42:35 +00:00
|
|
|
|
2021-02-15 18:59:46 +00:00
|
|
|
#[cfg(feature = "tls-native")]
|
|
|
|
async fn get_tls_stream<S: AsyncRead + AsyncWrite + Unpin>(
|
|
|
|
xmpp_stream: XMPPStream<S>,
|
|
|
|
) -> Result<TlsStream<S>, Error> {
|
|
|
|
let domain = &xmpp_stream.jid.clone().domain();
|
|
|
|
let stream = xmpp_stream.into_inner();
|
|
|
|
let tls_stream = TlsConnector::from(NativeTlsConnector::builder().build().unwrap())
|
|
|
|
.connect(&domain, stream)
|
|
|
|
.await?;
|
|
|
|
Ok(tls_stream)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(feature = "tls-rust")]
|
|
|
|
async fn get_tls_stream<S: AsyncRead + AsyncWrite + Unpin>(
|
|
|
|
xmpp_stream: XMPPStream<S>,
|
|
|
|
) -> Result<TlsStream<S>, Error> {
|
|
|
|
let domain = &xmpp_stream.jid.clone().domain();
|
|
|
|
let ascii_domain = idna::domain_to_ascii(domain).map_err(|_| Error::Idna)?;
|
|
|
|
let domain = DNSNameRef::try_from_ascii_str(&ascii_domain).unwrap();
|
|
|
|
let stream = xmpp_stream.into_inner();
|
2021-02-15 19:45:58 +00:00
|
|
|
let mut config = ClientConfig::new();
|
|
|
|
config
|
|
|
|
.root_store
|
|
|
|
.add_server_trust_anchors(&webpki_roots::TLS_SERVER_ROOTS);
|
|
|
|
let tls_stream = TlsConnector::from(Arc::new(config))
|
2021-02-15 18:59:46 +00:00
|
|
|
.connect(domain, stream)
|
|
|
|
.await?;
|
|
|
|
Ok(tls_stream)
|
|
|
|
}
|
|
|
|
|
2020-03-15 23:34:46 +00:00
|
|
|
/// Performs `<starttls/>` on an XMPPStream and returns a binary
|
|
|
|
/// TlsStream.
|
2020-03-05 00:25:24 +00:00
|
|
|
pub async fn starttls<S: AsyncRead + AsyncWrite + Unpin>(
|
|
|
|
mut xmpp_stream: XMPPStream<S>,
|
|
|
|
) -> Result<TlsStream<S>, Error> {
|
2020-05-29 23:19:06 +00:00
|
|
|
let nonza = Element::builder("starttls", ns::TLS).build();
|
2020-03-05 00:25:24 +00:00
|
|
|
let packet = Packet::Stanza(nonza);
|
|
|
|
xmpp_stream.send(packet).await?;
|
|
|
|
|
|
|
|
loop {
|
|
|
|
match xmpp_stream.next().await {
|
|
|
|
Some(Ok(Packet::Stanza(ref stanza))) if stanza.name() == "proceed" => break,
|
|
|
|
Some(Ok(Packet::Text(_))) => {}
|
|
|
|
Some(Err(e)) => return Err(e.into()),
|
|
|
|
_ => {
|
|
|
|
return Err(ProtocolError::NoTls.into());
|
|
|
|
}
|
2017-06-04 22:42:35 +00:00
|
|
|
}
|
|
|
|
}
|
2018-09-01 19:59:02 +00:00
|
|
|
|
2021-02-15 18:59:46 +00:00
|
|
|
get_tls_stream(xmpp_stream).await
|
2017-06-04 22:42:35 +00:00
|
|
|
}
|