mirror of
https://github.com/actix/actix-web.git
synced 2025-01-26 17:08:07 +00:00
244 lines
5.8 KiB
Rust
244 lines
5.8 KiB
Rust
//! Payload stream
|
|
use std::cell::RefCell;
|
|
use std::collections::VecDeque;
|
|
use std::pin::Pin;
|
|
use std::rc::{Rc, Weak};
|
|
use std::task::{Context, Poll};
|
|
|
|
use actix_utils::task::LocalWaker;
|
|
use bytes::Bytes;
|
|
use futures_core::Stream;
|
|
|
|
use crate::error::PayloadError;
|
|
|
|
/// max buffer size 32k
|
|
pub(crate) const MAX_BUFFER_SIZE: usize = 32_768;
|
|
|
|
#[derive(Debug, PartialEq)]
|
|
pub enum PayloadStatus {
|
|
Read,
|
|
Pause,
|
|
Dropped,
|
|
}
|
|
|
|
/// Buffered stream of bytes chunks
|
|
///
|
|
/// Payload stores chunks in a vector. First chunk can be received with
|
|
/// `.readany()` method. Payload stream is not thread safe. Payload does not
|
|
/// notify current task when new data is available.
|
|
///
|
|
/// Payload stream can be used as `Response` body stream.
|
|
#[derive(Debug)]
|
|
pub struct Payload {
|
|
inner: Rc<RefCell<Inner>>,
|
|
}
|
|
|
|
impl Payload {
|
|
/// Create payload stream.
|
|
///
|
|
/// This method construct two objects responsible for bytes stream
|
|
/// generation.
|
|
///
|
|
/// * `PayloadSender` - *Sender* side of the stream
|
|
///
|
|
/// * `Payload` - *Receiver* side of the stream
|
|
pub fn create(eof: bool) -> (PayloadSender, Payload) {
|
|
let shared = Rc::new(RefCell::new(Inner::new(eof)));
|
|
|
|
(
|
|
PayloadSender {
|
|
inner: Rc::downgrade(&shared),
|
|
},
|
|
Payload { inner: shared },
|
|
)
|
|
}
|
|
|
|
/// Create empty payload
|
|
#[doc(hidden)]
|
|
pub fn empty() -> Payload {
|
|
Payload {
|
|
inner: Rc::new(RefCell::new(Inner::new(true))),
|
|
}
|
|
}
|
|
|
|
/// Length of the data in this payload
|
|
#[cfg(test)]
|
|
pub fn len(&self) -> usize {
|
|
self.inner.borrow().len()
|
|
}
|
|
|
|
/// Is payload empty
|
|
#[cfg(test)]
|
|
pub fn is_empty(&self) -> bool {
|
|
self.inner.borrow().len() == 0
|
|
}
|
|
|
|
/// Put unused data back to payload
|
|
#[inline]
|
|
pub fn unread_data(&mut self, data: Bytes) {
|
|
self.inner.borrow_mut().unread_data(data);
|
|
}
|
|
|
|
#[inline]
|
|
pub fn readany(
|
|
&mut self,
|
|
cx: &mut Context<'_>,
|
|
) -> Poll<Option<Result<Bytes, PayloadError>>> {
|
|
self.inner.borrow_mut().readany(cx)
|
|
}
|
|
}
|
|
|
|
impl Stream for Payload {
|
|
type Item = Result<Bytes, PayloadError>;
|
|
|
|
fn poll_next(
|
|
self: Pin<&mut Self>,
|
|
cx: &mut Context<'_>,
|
|
) -> Poll<Option<Result<Bytes, PayloadError>>> {
|
|
self.inner.borrow_mut().readany(cx)
|
|
}
|
|
}
|
|
|
|
/// Sender part of the payload stream
|
|
pub struct PayloadSender {
|
|
inner: Weak<RefCell<Inner>>,
|
|
}
|
|
|
|
impl PayloadSender {
|
|
#[inline]
|
|
pub fn set_error(&mut self, err: PayloadError) {
|
|
if let Some(shared) = self.inner.upgrade() {
|
|
shared.borrow_mut().set_error(err)
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
pub fn feed_eof(&mut self) {
|
|
if let Some(shared) = self.inner.upgrade() {
|
|
shared.borrow_mut().feed_eof()
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
pub fn feed_data(&mut self, data: Bytes) {
|
|
if let Some(shared) = self.inner.upgrade() {
|
|
shared.borrow_mut().feed_data(data)
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
pub fn need_read(&self, cx: &mut Context<'_>) -> PayloadStatus {
|
|
// we check need_read only if Payload (other side) is alive,
|
|
// otherwise always return true (consume payload)
|
|
if let Some(shared) = self.inner.upgrade() {
|
|
if shared.borrow().need_read {
|
|
PayloadStatus::Read
|
|
} else {
|
|
shared.borrow_mut().io_task.register(cx.waker());
|
|
PayloadStatus::Pause
|
|
}
|
|
} else {
|
|
PayloadStatus::Dropped
|
|
}
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
struct Inner {
|
|
len: usize,
|
|
eof: bool,
|
|
err: Option<PayloadError>,
|
|
need_read: bool,
|
|
items: VecDeque<Bytes>,
|
|
task: LocalWaker,
|
|
io_task: LocalWaker,
|
|
}
|
|
|
|
impl Inner {
|
|
fn new(eof: bool) -> Self {
|
|
Inner {
|
|
eof,
|
|
len: 0,
|
|
err: None,
|
|
items: VecDeque::new(),
|
|
need_read: true,
|
|
task: LocalWaker::new(),
|
|
io_task: LocalWaker::new(),
|
|
}
|
|
}
|
|
|
|
#[inline]
|
|
fn set_error(&mut self, err: PayloadError) {
|
|
self.err = Some(err);
|
|
}
|
|
|
|
#[inline]
|
|
fn feed_eof(&mut self) {
|
|
self.eof = true;
|
|
}
|
|
|
|
#[inline]
|
|
fn feed_data(&mut self, data: Bytes) {
|
|
self.len += data.len();
|
|
self.items.push_back(data);
|
|
self.need_read = self.len < MAX_BUFFER_SIZE;
|
|
if let Some(task) = self.task.take() {
|
|
task.wake()
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
fn len(&self) -> usize {
|
|
self.len
|
|
}
|
|
|
|
fn readany(
|
|
&mut self,
|
|
cx: &mut Context<'_>,
|
|
) -> Poll<Option<Result<Bytes, PayloadError>>> {
|
|
if let Some(data) = self.items.pop_front() {
|
|
self.len -= data.len();
|
|
self.need_read = self.len < MAX_BUFFER_SIZE;
|
|
|
|
if self.need_read && !self.eof {
|
|
self.task.register(cx.waker());
|
|
}
|
|
self.io_task.wake();
|
|
Poll::Ready(Some(Ok(data)))
|
|
} else if let Some(err) = self.err.take() {
|
|
Poll::Ready(Some(Err(err)))
|
|
} else if self.eof {
|
|
Poll::Ready(None)
|
|
} else {
|
|
self.need_read = true;
|
|
self.task.register(cx.waker());
|
|
self.io_task.wake();
|
|
Poll::Pending
|
|
}
|
|
}
|
|
|
|
fn unread_data(&mut self, data: Bytes) {
|
|
self.len += data.len();
|
|
self.items.push_front(data);
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::*;
|
|
use futures_util::future::poll_fn;
|
|
|
|
#[actix_rt::test]
|
|
async fn test_unread_data() {
|
|
let (_, mut payload) = Payload::create(false);
|
|
|
|
payload.unread_data(Bytes::from("data"));
|
|
assert!(!payload.is_empty());
|
|
assert_eq!(payload.len(), 4);
|
|
|
|
assert_eq!(
|
|
Bytes::from("data"),
|
|
poll_fn(|cx| payload.readany(cx)).await.unwrap().unwrap()
|
|
);
|
|
}
|
|
}
|