#![cfg(any(target_os = "macos", target_os = "ios"))]
use security_framework::secure_transport::ClientHandshakeError;
use security_framework::secure_transport::HandshakeError;
use security_framework::secure_transport::MidHandshakeClientBuilder;
use security_framework::secure_transport::MidHandshakeSslStream;
use security_framework::secure_transport::SslConnectionType;
use security_framework::secure_transport::SslContext;
use security_framework::secure_transport::SslProtocolSide;
use security_framework::secure_transport::SslStream;
use std::future::Future;
use std::mem;
use std::pin::Pin;
use std::task::Context;
use std::task::Poll;
use tls_api::async_as_sync::AsyncIoAsSyncIo;
use tls_api::AsyncSocket;
use tls_api::BoxFuture;
use crate::TlsAcceptor;
use tls_api::spi::save_context;
enum ClientHandshakeFuture<F, S: Unpin> {
Initial(F, AsyncIoAsSyncIo<S>),
MidHandshake(MidHandshakeClientBuilder<AsyncIoAsSyncIo<S>>),
Done,
}
pub(crate) fn new_slient_handshake<'a, S>(
connector: &'a crate::TlsConnector,
domain: &'a str,
stream: S,
) -> impl Future<Output = anyhow::Result<crate::TlsStream<S>>> + 'a
where
S: AsyncSocket,
{
ClientHandshakeFuture::Initial(
move |stream| connector.0.handshake(domain, stream),
AsyncIoAsSyncIo::new(stream),
)
}
impl<F, S> Future for ClientHandshakeFuture<F, S>
where
S: AsyncSocket,
F: FnOnce(
AsyncIoAsSyncIo<S>,
)
-> Result<SslStream<AsyncIoAsSyncIo<S>>, ClientHandshakeError<AsyncIoAsSyncIo<S>>>,
Self: Unpin,
{
type Output = anyhow::Result<crate::TlsStream<S>>;
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
save_context(cx, || {
let self_mut = self.get_mut();
match mem::replace(self_mut, ClientHandshakeFuture::Done) {
ClientHandshakeFuture::Initial(f, stream) => match f(stream) {
Ok(stream) => {
return Poll::Ready(Ok(crate::TlsStream::new(stream)));
}
Err(ClientHandshakeError::Interrupted(mid)) => {
*self_mut = ClientHandshakeFuture::MidHandshake(mid);
return Poll::Pending;
}
Err(ClientHandshakeError::Failure(e)) => {
return Poll::Ready(Err(anyhow::Error::new(e)))
}
},
ClientHandshakeFuture::MidHandshake(stream) => match stream.handshake() {
Ok(stream) => {
return Poll::Ready(Ok(crate::TlsStream::new(stream)));
}
Err(ClientHandshakeError::Interrupted(mid)) => {
*self_mut = ClientHandshakeFuture::MidHandshake(mid);
return Poll::Pending;
}
Err(ClientHandshakeError::Failure(e)) => {
return Poll::Ready(Err(anyhow::Error::new(e)))
}
},
ClientHandshakeFuture::Done => panic!("Future must not be polled after ready"),
}
})
}
}
enum ServerHandshakeFuture<F, S: Unpin> {
Initial(F, AsyncIoAsSyncIo<S>),
MidHandshake(MidHandshakeSslStream<AsyncIoAsSyncIo<S>>),
Done,
}
pub(crate) fn new_server_handshake<'a, S>(
acceptor: &'a TlsAcceptor,
stream: S,
) -> impl Future<Output = anyhow::Result<crate::TlsStream<S>>> + 'a
where
S: AsyncSocket,
{
BoxFuture::new(async move {
let mut ctx = SslContext::new(SslProtocolSide::SERVER, SslConnectionType::STREAM)
.map_err(anyhow::Error::new)?;
ctx.set_certificate(&acceptor.0.identity, &acceptor.0.certs)
.map_err(anyhow::Error::new)?;
ServerHandshakeFuture::Initial(move |s| ctx.handshake(s), AsyncIoAsSyncIo::new(stream))
.await
})
}
impl<F, S> Future for ServerHandshakeFuture<F, S>
where
S: AsyncSocket,
F: FnOnce(
AsyncIoAsSyncIo<S>,
) -> Result<SslStream<AsyncIoAsSyncIo<S>>, HandshakeError<AsyncIoAsSyncIo<S>>>,
Self: Unpin,
{
type Output = anyhow::Result<crate::TlsStream<S>>;
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
save_context(cx, || {
let self_mut = self.get_mut();
match mem::replace(self_mut, ServerHandshakeFuture::Done) {
ServerHandshakeFuture::Initial(f, stream) => match f(stream) {
Ok(stream) => {
return Poll::Ready(Ok(crate::TlsStream::new(stream)));
}
Err(HandshakeError::Interrupted(mid)) => {
*self_mut = ServerHandshakeFuture::MidHandshake(mid);
return Poll::Pending;
}
Err(HandshakeError::Failure(e)) => {
return Poll::Ready(Err(anyhow::Error::new(e)))
}
},
ServerHandshakeFuture::MidHandshake(stream) => match stream.handshake() {
Ok(stream) => {
return Poll::Ready(Ok(crate::TlsStream::new(stream)));
}
Err(HandshakeError::Interrupted(mid)) => {
*self_mut = ServerHandshakeFuture::MidHandshake(mid);
return Poll::Pending;
}
Err(HandshakeError::Failure(e)) => {
return Poll::Ready(Err(anyhow::Error::new(e)))
}
},
ServerHandshakeFuture::Done => panic!("Future must not be polled after ready"),
}
})
}
}