1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
use std::fmt;
use std::pin::Pin;
use std::sync::{Arc, Mutex, MutexGuard};
use std::task::{Context, Poll};
use anyhow::{anyhow, Error, Result};
use bytes::{Bytes, BytesMut};
use futures_util::stream::{Stream, TryStreamExt};
use crate::State;
pub struct Field<T> {
pub name: String,
pub index: Option<usize>,
pub filename: Option<String>,
pub content_type: Option<mime::Mime>,
pub headers: Option<http::HeaderMap>,
pub length: u64,
state: Option<Arc<Mutex<State<T>>>>,
}
impl<T> Field<T> {
pub fn empty() -> Self {
Self {
name: String::new(),
content_type: None,
filename: None,
headers: None,
state: None,
index: None,
length: 0,
}
}
pub fn headers_mut(&mut self) -> &mut Option<http::HeaderMap> {
&mut self.headers
}
pub fn state_mut(&mut self) -> &mut Option<Arc<Mutex<State<T>>>> {
&mut self.state
}
pub fn state(&self) -> Result<MutexGuard<'_, State<T>>> {
self.state
.as_ref()
.unwrap()
.try_lock()
.map_err(|e| anyhow!(e.to_string()))
}
pub fn consumed(&self) -> bool {
self.state.is_none()
}
pub async fn bytes<O, E>(&mut self) -> Result<Bytes>
where
T: Stream<Item = Result<O, E>> + Unpin + Send + 'static,
O: Into<Bytes>,
E: Into<Error>,
{
let mut bytes = BytesMut::new();
while let Some(buf) = self.try_next().await? {
bytes.extend_from_slice(&buf);
}
Ok(bytes.freeze())
}
}
impl<T> fmt::Debug for Field<T> {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("Field")
.field("name", &self.name)
.field("filename", &self.filename)
.field("content_type", &self.content_type)
.field("index", &self.index)
.field("length", &self.length)
.field("headers", &self.headers)
.field("consumed", &self.state.is_none())
.finish()
}
}
impl<T, O, E> Stream for Field<T>
where
T: Stream<Item = Result<O, E>> + Unpin + Send + 'static,
O: Into<Bytes>,
E: Into<Error>,
{
type Item = Result<Bytes>;
fn poll_next(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Option<Self::Item>> {
log::debug!("polling {} {}", self.index.unwrap_or_default(), self.state.is_some());
if self.state.is_none() {
return Poll::Ready(None);
}
let state = self.state.clone().unwrap();
let mut state = state.try_lock().map_err(|e| anyhow!(e.to_string()))?;
match Pin::new(&mut *state).poll_next(cx)? {
Poll::Ready(res) => match res {
Some(0) | None => {
if let Some(waker) = state.waker_mut().take() {
waker.wake();
}
log::debug!("polled {}", self.index.unwrap_or_default());
drop(self.state.take());
Poll::Ready(None)
}
Some(len) => {
self.length += len as u64;
log::debug!("polled bytes {}/{}", len, self.length);
Poll::Ready(Some(Ok(state.buffer_mut().split_to(len).freeze())))
}
},
Poll::Pending => Poll::Pending,
}
}
}