use std::pin::Pin;
use async_trait::async_trait;
use pin_project::pin_project;
use rustls_pki_types::ServerName;
use tokio::io::{AsyncRead, AsyncWrite, ReadBuf};
use tokio::net::TcpStream;
use tokio_rustls::TlsConnector as RustlsTlsConnector;
use tokio_rustls::client::TlsStream;
use super::{AsyncTlsConnector, TokioTlsStream};
use crate::{FtpError, FtpResult};
pub struct AsyncRustlsConnector {
connector: RustlsTlsConnector,
}
impl std::fmt::Debug for AsyncRustlsConnector {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(f, "<?>")
}
}
impl From<RustlsTlsConnector> for AsyncRustlsConnector {
fn from(connector: RustlsTlsConnector) -> Self {
Self { connector }
}
}
#[async_trait]
impl AsyncTlsConnector for AsyncRustlsConnector {
type Stream = AsyncRustlsStream;
async fn connect(&self, domain: &str, stream: TcpStream) -> FtpResult<Self::Stream> {
let domain = domain.to_owned();
let server_name =
ServerName::try_from(domain).map_err(|e| FtpError::SecureError(e.to_string()))?;
self.connector
.connect(server_name, stream)
.await
.map(AsyncRustlsStream::from)
.map_err(|e| FtpError::SecureError(e.to_string()))
}
}
#[derive(Debug)]
#[pin_project(project = AsyncRustlsStreamProj)]
pub struct AsyncRustlsStream {
#[pin]
stream: TlsStream<TcpStream>,
}
impl From<TlsStream<TcpStream>> for AsyncRustlsStream {
fn from(stream: TlsStream<TcpStream>) -> Self {
Self { stream }
}
}
impl AsyncRead for AsyncRustlsStream {
fn poll_read(
self: std::pin::Pin<&mut Self>,
cx: &mut std::task::Context<'_>,
buf: &mut ReadBuf<'_>,
) -> std::task::Poll<std::io::Result<()>> {
self.project().stream.poll_read(cx, buf)
}
}
impl AsyncWrite for AsyncRustlsStream {
fn poll_write(
self: std::pin::Pin<&mut Self>,
cx: &mut std::task::Context<'_>,
buf: &[u8],
) -> std::task::Poll<std::io::Result<usize>> {
self.project().stream.poll_write(cx, buf)
}
fn poll_flush(
self: Pin<&mut Self>,
cx: &mut std::task::Context<'_>,
) -> std::task::Poll<std::io::Result<()>> {
self.project().stream.poll_flush(cx)
}
fn poll_shutdown(
self: Pin<&mut Self>,
cx: &mut std::task::Context<'_>,
) -> std::task::Poll<std::io::Result<()>> {
self.project().stream.poll_shutdown(cx)
}
}
impl TokioTlsStream for AsyncRustlsStream {
type InnerStream = TlsStream<TcpStream>;
fn get_ref(&self) -> &TcpStream {
self.stream.get_ref().0
}
fn mut_ref(&mut self) -> &mut Self::InnerStream {
&mut self.stream
}
fn tcp_stream(self) -> TcpStream {
self.stream.into_inner().0
}
}