use std::fmt::{self, Debug, Formatter};
use std::marker::PhantomData;
use std::pin::Pin;
use std::task::{Context, Poll};
use futures_core::ready;
use crate::{FusedMultipartWrite, MultipartWrite};
pin_project_lite::pin_project! {
#[must_use = "futures do nothing unless polled"]
pub struct FoldSent<Wr, T, F, Part> {
#[pin]
writer: Wr,
acc: Option<T>,
f: F,
_p: PhantomData<Part>,
}
}
impl<Wr, T, F, Part> FoldSent<Wr, T, F, Part> {
pub(super) fn new(writer: Wr, id: T, f: F) -> Self {
Self { writer, acc: Some(id), f, _p: PhantomData }
}
pub fn into_inner(self) -> Wr {
self.writer
}
pub fn get_ref(&self) -> &Wr {
&self.writer
}
pub fn get_mut(&mut self) -> &mut Wr {
&mut self.writer
}
pub fn get_pin_mut(self: Pin<&mut Self>) -> Pin<&mut Wr> {
self.project().writer
}
}
impl<Wr, T, F, Part> FusedMultipartWrite<Part> for FoldSent<Wr, T, F, Part>
where
Wr: FusedMultipartWrite<Part>,
F: FnMut(T, &Wr::Recv) -> T,
{
fn is_terminated(&self) -> bool {
self.writer.is_terminated()
}
}
impl<Wr, T, F, Part> MultipartWrite<Part> for FoldSent<Wr, T, F, Part>
where
Wr: MultipartWrite<Part>,
F: FnMut(T, &Wr::Recv) -> T,
{
type Error = Wr::Error;
type Output = (T, Wr::Output);
type Recv = Wr::Recv;
fn poll_ready(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<Result<(), Self::Error>> {
self.project().writer.poll_ready(cx)
}
fn start_send(
self: Pin<&mut Self>,
part: Part,
) -> Result<Self::Recv, Self::Error> {
let mut this = self.project();
let ret = this.writer.as_mut().start_send(part)?;
let new_acc = this.acc.take().map(|acc| (this.f)(acc, &ret));
*this.acc = new_acc;
Ok(ret)
}
fn poll_flush(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<Result<(), Self::Error>> {
self.project().writer.poll_flush(cx)
}
fn poll_complete(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<Result<Self::Output, Self::Error>> {
let mut this = self.project();
let out = ready!(this.writer.as_mut().poll_complete(cx))?;
let acc = this.acc.take().expect("polled FoldSent after completion");
Poll::Ready(Ok((acc, out)))
}
}
impl<Wr, T, F, Part> Debug for FoldSent<Wr, T, F, Part>
where
Wr: Debug,
T: Debug,
{
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
f.debug_struct("FoldSent")
.field("writer", &self.writer)
.field("acc", &self.acc)
.finish()
}
}