1use crate::{FusedMultipartWrite, MultipartWrite};
2
3use futures::ready;
4use std::fmt::{self, Debug, Formatter};
5use std::pin::Pin;
6use std::task::{Context, Poll};
7
8#[must_use = "futures do nothing unless polled"]
10#[pin_project::pin_project]
11pub struct With<Wr, Part, U, Fut, F> {
12 #[pin]
13 writer: Wr,
14 f: F,
15 #[pin]
16 future: Option<Fut>,
17 _f: std::marker::PhantomData<fn(U) -> Part>,
18}
19
20impl<Wr, Part, U, Fut, F> With<Wr, Part, U, Fut, F>
21where
22 Wr: MultipartWrite<Part>,
23 F: FnMut(U) -> Fut,
24 Fut: Future,
25{
26 pub(super) fn new<E>(writer: Wr, f: F) -> Self
27 where
28 Fut: Future<Output = Result<Part, E>>,
29 E: From<Wr::Error>,
30 {
31 Self {
32 writer,
33 f,
34 future: None,
35 _f: std::marker::PhantomData,
36 }
37 }
38
39 pub fn get_ref(&self) -> &Wr {
41 &self.writer
42 }
43
44 pub fn get_mut(&mut self) -> &mut Wr {
48 &mut self.writer
49 }
50
51 pub fn get_pin_mut(self: Pin<&mut Self>) -> Pin<&mut Wr> {
55 self.project().writer
56 }
57}
58
59impl<Wr, Part, U, Fut, F, E> With<Wr, Part, U, Fut, F>
60where
61 Wr: MultipartWrite<Part>,
62 F: FnMut(U) -> Fut,
63 Fut: Future<Output = Result<Part, E>>,
64 E: From<Wr::Error>,
65{
66 fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<(), E>> {
67 let mut this = self.project();
68
69 let part = match this.future.as_mut().as_pin_mut() {
70 None => return Poll::Ready(Ok(())),
71 Some(fut) => ready!(fut.poll(cx))?,
72 };
73 this.future.set(None);
74 this.writer.start_send(part)?;
75 Poll::Ready(Ok(()))
76 }
77}
78
79impl<Wr, Part, U, Fut, F, E> FusedMultipartWrite<U> for With<Wr, Part, U, Fut, F>
80where
81 Wr: FusedMultipartWrite<Part>,
82 F: FnMut(U) -> Fut,
83 Fut: Future<Output = Result<Part, E>>,
84 E: From<Wr::Error>,
85{
86 fn is_terminated(&self) -> bool {
87 self.writer.is_terminated()
88 }
89}
90
91impl<Wr, Part, U, Fut, F, E> MultipartWrite<U> for With<Wr, Part, U, Fut, F>
92where
93 Wr: MultipartWrite<Part>,
94 F: FnMut(U) -> Fut,
95 Fut: Future<Output = Result<Part, E>>,
96 E: From<Wr::Error>,
97{
98 type Ret = ();
99 type Output = Wr::Output;
100 type Error = E;
101
102 fn poll_ready(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
103 ready!(self.as_mut().poll(cx))?;
104 ready!(self.project().writer.poll_ready(cx)?);
105 Poll::Ready(Ok(()))
106 }
107
108 fn start_send(self: Pin<&mut Self>, part: U) -> Result<Self::Ret, Self::Error> {
109 let mut this = self.project();
110 this.future.set(Some((this.f)(part)));
111 Ok(())
112 }
113
114 fn poll_flush(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
115 ready!(self.as_mut().poll(cx))?;
116 ready!(self.project().writer.poll_flush(cx)?);
117 Poll::Ready(Ok(()))
118 }
119
120 fn poll_complete(
121 mut self: Pin<&mut Self>,
122 cx: &mut Context<'_>,
123 ) -> Poll<Result<Self::Output, Self::Error>> {
124 ready!(self.as_mut().poll(cx))?;
125 let ret = ready!(self.project().writer.poll_complete(cx)?);
126 Poll::Ready(Ok(ret))
127 }
128}
129
130impl<Wr, Part, U, Fut, F> Debug for With<Wr, Part, U, Fut, F>
131where
132 Wr: Debug,
133 Fut: Debug,
134{
135 fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
136 f.debug_struct("With")
137 .field("writer", &self.writer)
138 .field("future", &self.future)
139 .finish()
140 }
141}