-
Notifications
You must be signed in to change notification settings - Fork 19
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
6 changed files
with
345 additions
and
2 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,168 @@ | ||
//! IMAP COMPRESS extension specified in [RFC4978](https://www.rfc-editor.org/rfc/rfc4978.html). | ||
use std::fmt; | ||
use std::pin::Pin; | ||
use std::task::{Context, Poll}; | ||
|
||
use pin_project::pin_project; | ||
|
||
use crate::client::Session; | ||
use crate::error::Result; | ||
use crate::imap_stream::ImapStream; | ||
use crate::shared_stream::SharedStream; | ||
use crate::types::IdGenerator; | ||
use crate::Connection; | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
use async_std::io::{BufReader, Read, Write}; | ||
#[cfg(feature = "runtime-tokio")] | ||
use tokio::io::{AsyncRead as Read, AsyncWrite as Write, BufReader, ReadBuf}; | ||
|
||
#[cfg(feature = "runtime-tokio")] | ||
use async_compression::tokio::bufread::DeflateDecoder; | ||
#[cfg(feature = "runtime-tokio")] | ||
use async_compression::tokio::write::DeflateEncoder; | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
use async_compression::futures::bufread::DeflateDecoder; | ||
#[cfg(feature = "runtime-async-std")] | ||
use async_compression::futures::write::DeflateEncoder; | ||
|
||
/// IMAP stream | ||
#[derive(Debug)] | ||
#[pin_project] | ||
pub struct DeflateStream<T: Read + Write + Unpin + fmt::Debug> { | ||
/// Shared stream reference to allow direct access | ||
/// to the underlying stream. | ||
stream: SharedStream<T>, | ||
|
||
#[pin] | ||
decoder: DeflateDecoder<BufReader<SharedStream<T>>>, | ||
|
||
#[pin] | ||
encoder: DeflateEncoder<SharedStream<T>>, | ||
} | ||
|
||
impl<T: Read + Write + Unpin + fmt::Debug> DeflateStream<T> { | ||
pub(crate) fn new(stream: T) -> Self { | ||
let stream = SharedStream::new(stream); | ||
let decoder = DeflateDecoder::new(BufReader::new(stream.clone())); | ||
let encoder = DeflateEncoder::new(stream.clone()); | ||
Self { | ||
stream, | ||
decoder, | ||
encoder, | ||
} | ||
} | ||
|
||
/// Runs provided function while holding the lock on the underlying stream. | ||
/// | ||
/// This allows to access the underlying stream while ensuring | ||
/// that no data is read from the stream or written into the stream at the same time. | ||
pub fn with_lock<R>(&self, f: impl FnOnce(Pin<&mut T>) -> R) -> R { | ||
self.stream.with_lock(f) | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-tokio")] | ||
impl<T: Read + Write + Unpin + fmt::Debug> Read for DeflateStream<T> { | ||
fn poll_read( | ||
self: Pin<&mut Self>, | ||
cx: &mut Context<'_>, | ||
buf: &mut ReadBuf<'_>, | ||
) -> Poll<std::io::Result<()>> { | ||
self.project().decoder.poll_read(cx, buf) | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
impl<T: Read + Write + Unpin + fmt::Debug> Read for DeflateStream<T> { | ||
fn poll_read( | ||
self: Pin<&mut Self>, | ||
cx: &mut Context<'_>, | ||
buf: &mut [u8], | ||
) -> Poll<async_std::io::Result<usize>> { | ||
self.project().decoder.poll_read(cx, buf) | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-tokio")] | ||
impl<T: Read + Write + Unpin + fmt::Debug> Write for DeflateStream<T> { | ||
fn poll_write( | ||
self: Pin<&mut Self>, | ||
cx: &mut std::task::Context<'_>, | ||
buf: &[u8], | ||
) -> Poll<std::io::Result<usize>> { | ||
self.project().encoder.poll_write(cx, buf) | ||
} | ||
|
||
fn poll_flush( | ||
self: Pin<&mut Self>, | ||
cx: &mut std::task::Context<'_>, | ||
) -> Poll<std::io::Result<()>> { | ||
self.project().encoder.poll_flush(cx) | ||
} | ||
|
||
fn poll_shutdown( | ||
self: Pin<&mut Self>, | ||
cx: &mut std::task::Context<'_>, | ||
) -> Poll<std::io::Result<()>> { | ||
self.project().encoder.poll_shutdown(cx) | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
impl<T: Read + Write + Unpin + fmt::Debug> Write for DeflateStream<T> { | ||
fn poll_write( | ||
self: Pin<&mut Self>, | ||
cx: &mut std::task::Context<'_>, | ||
buf: &[u8], | ||
) -> Poll<async_std::io::Result<usize>> { | ||
self.project().encoder.poll_write(cx, buf) | ||
} | ||
|
||
fn poll_flush( | ||
self: Pin<&mut Self>, | ||
cx: &mut std::task::Context<'_>, | ||
) -> Poll<async_std::io::Result<()>> { | ||
self.project().encoder.poll_flush(cx) | ||
} | ||
|
||
fn poll_close( | ||
self: Pin<&mut Self>, | ||
cx: &mut std::task::Context<'_>, | ||
) -> Poll<async_std::io::Result<()>> { | ||
self.project().encoder.poll_close(cx) | ||
} | ||
} | ||
|
||
impl<T: Read + Write + Unpin + fmt::Debug + Send> Session<T> { | ||
/// Runs `COMPRESS DEFLATE` command. | ||
pub async fn compress<F, S>(self, f: F) -> Result<Session<S>> | ||
where | ||
S: Read + Write + Unpin + fmt::Debug, | ||
F: FnOnce(DeflateStream<T>) -> S, | ||
{ | ||
let Self { | ||
mut conn, | ||
unsolicited_responses_tx, | ||
unsolicited_responses, | ||
} = self; | ||
conn.run_command_and_check_ok("COMPRESS DEFLATE", Some(unsolicited_responses_tx.clone())) | ||
.await?; | ||
|
||
let stream = conn.into_inner(); | ||
let deflate_stream = DeflateStream::new(stream); | ||
let stream = ImapStream::new(f(deflate_stream)); | ||
let conn = Connection { | ||
stream, | ||
request_ids: IdGenerator::new(), | ||
}; | ||
let session = Session { | ||
conn, | ||
unsolicited_responses_tx, | ||
unsolicited_responses, | ||
}; | ||
Ok(session) | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,157 @@ | ||
use std::pin::Pin; | ||
use std::sync::{Arc, Mutex}; | ||
use std::task::{Context, Poll}; | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
use async_std::io::{Read, Write}; | ||
#[cfg(feature = "runtime-tokio")] | ||
use tokio::io::{AsyncRead as Read, AsyncWrite as Write, ReadBuf}; | ||
|
||
#[cfg(feature = "runtime-tokio")] | ||
#[derive(Debug)] | ||
pub(crate) struct SharedStream<T: std::fmt::Debug> { | ||
inner: Arc<Mutex<T>>, | ||
is_write_vectored: bool, | ||
} | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
#[derive(Debug)] | ||
pub(crate) struct SharedStream<T: std::fmt::Debug> { | ||
inner: Arc<Mutex<T>>, | ||
} | ||
|
||
#[cfg(feature = "runtime-tokio")] | ||
impl<T: std::fmt::Debug> Clone for SharedStream<T> { | ||
fn clone(&self) -> Self { | ||
Self { | ||
inner: Arc::clone(&self.inner), | ||
is_write_vectored: self.is_write_vectored, | ||
} | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
impl<T: std::fmt::Debug> Clone for SharedStream<T> { | ||
fn clone(&self) -> Self { | ||
Self { | ||
inner: Arc::clone(&self.inner), | ||
} | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-tokio")] | ||
impl<T: std::fmt::Debug> SharedStream<T> | ||
where | ||
T: Read + Write, | ||
{ | ||
pub(crate) fn new(stream: T) -> SharedStream<T> { | ||
let is_write_vectored = stream.is_write_vectored(); | ||
|
||
let inner = Arc::new(Mutex::new(stream)); | ||
|
||
Self { | ||
inner, | ||
is_write_vectored, | ||
} | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
impl<T: std::fmt::Debug> SharedStream<T> | ||
where | ||
T: Read + Write, | ||
{ | ||
pub(crate) fn new(stream: T) -> SharedStream<T> { | ||
let inner = Arc::new(Mutex::new(stream)); | ||
|
||
Self { inner } | ||
} | ||
} | ||
|
||
impl<T: Unpin + std::fmt::Debug> SharedStream<T> { | ||
pub(crate) fn with_lock<R>(&self, f: impl FnOnce(Pin<&mut T>) -> R) -> R { | ||
let mut guard = self.inner.lock().unwrap(); | ||
let stream = Pin::new(&mut *guard); | ||
f(stream) | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-tokio")] | ||
impl<T: Read + Unpin + std::fmt::Debug> Read for SharedStream<T> { | ||
fn poll_read( | ||
self: Pin<&mut Self>, | ||
cx: &mut Context<'_>, | ||
buf: &mut ReadBuf<'_>, | ||
) -> Poll<std::io::Result<()>> { | ||
self.with_lock(|stream| stream.poll_read(cx, buf)) | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
impl<T: Read + Unpin + std::fmt::Debug> Read for SharedStream<T> { | ||
fn poll_read( | ||
self: Pin<&mut Self>, | ||
cx: &mut Context<'_>, | ||
buf: &mut [u8], | ||
) -> Poll<async_std::io::Result<usize>> { | ||
self.with_lock(|stream| stream.poll_read(cx, buf)) | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-tokio")] | ||
impl<T: Write + Unpin + std::fmt::Debug> Write for SharedStream<T> { | ||
fn poll_write( | ||
self: Pin<&mut Self>, | ||
cx: &mut Context<'_>, | ||
buf: &[u8], | ||
) -> Poll<std::io::Result<usize>> { | ||
self.with_lock(|stream| stream.poll_write(cx, buf)) | ||
} | ||
|
||
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<std::io::Result<()>> { | ||
self.with_lock(|stream| stream.poll_flush(cx)) | ||
} | ||
|
||
fn poll_shutdown(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<std::io::Result<()>> { | ||
self.with_lock(|stream| stream.poll_shutdown(cx)) | ||
} | ||
|
||
fn poll_write_vectored( | ||
self: Pin<&mut Self>, | ||
cx: &mut Context<'_>, | ||
bufs: &[std::io::IoSlice<'_>], | ||
) -> Poll<std::io::Result<usize>> { | ||
self.with_lock(|stream| stream.poll_write_vectored(cx, bufs)) | ||
} | ||
|
||
fn is_write_vectored(&self) -> bool { | ||
self.is_write_vectored | ||
} | ||
} | ||
|
||
#[cfg(feature = "runtime-async-std")] | ||
impl<T: Write + Unpin + std::fmt::Debug> Write for SharedStream<T> { | ||
fn poll_write( | ||
self: Pin<&mut Self>, | ||
cx: &mut Context<'_>, | ||
buf: &[u8], | ||
) -> Poll<std::io::Result<usize>> { | ||
self.with_lock(|stream| stream.poll_write(cx, buf)) | ||
} | ||
|
||
fn poll_flush(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<async_std::io::Result<()>> { | ||
self.with_lock(|stream| stream.poll_flush(cx)) | ||
} | ||
|
||
fn poll_close(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<async_std::io::Result<()>> { | ||
self.with_lock(|stream| stream.poll_close(cx)) | ||
} | ||
|
||
fn poll_write_vectored( | ||
self: Pin<&mut Self>, | ||
cx: &mut Context<'_>, | ||
bufs: &[async_std::io::IoSlice<'_>], | ||
) -> Poll<async_std::io::Result<usize>> { | ||
self.with_lock(|stream| stream.poll_write_vectored(cx, bufs)) | ||
} | ||
} |