1
0
mirror of https://github.com/actix/actix-extras.git synced 2024-11-28 09:42:40 +01:00
actix-extras/src/payload.rs

623 lines
17 KiB
Rust
Raw Normal View History

2017-12-19 09:18:57 +01:00
//! Payload stream
2017-11-07 01:23:58 +01:00
use std::{fmt, cmp};
2017-10-09 05:16:48 +02:00
use std::rc::{Rc, Weak};
use std::cell::RefCell;
use std::collections::VecDeque;
2017-11-29 04:51:39 +01:00
use std::ops::{Deref, DerefMut};
2017-11-07 01:23:58 +01:00
use bytes::{Bytes, BytesMut};
2017-12-19 09:18:57 +01:00
use futures::{Future, Async, Poll, Stream};
2017-10-09 05:16:48 +02:00
use futures::task::{Task, current as current_task};
2017-12-14 07:36:28 +01:00
use body::BodyStream;
2017-10-27 08:14:33 +02:00
use actix::ResponseType;
2017-11-16 07:06:28 +01:00
use error::PayloadError;
2017-10-27 08:14:33 +02:00
2017-11-06 10:24:49 +01:00
pub(crate) const DEFAULT_BUFFER_SIZE: usize = 65_536; // max buffer size 64k
2017-10-09 05:16:48 +02:00
2017-10-14 01:33:23 +02:00
/// Just Bytes object
2017-11-20 05:26:05 +01:00
#[derive(PartialEq)]
2017-10-27 08:14:33 +02:00
pub struct PayloadItem(pub Bytes);
impl ResponseType for PayloadItem {
type Item = ();
type Error = ();
}
2017-10-14 01:33:23 +02:00
2017-11-29 04:51:39 +01:00
impl Deref for PayloadItem {
type Target = Bytes;
fn deref(&self) -> &Bytes {
&self.0
}
}
impl DerefMut for PayloadItem {
fn deref_mut(&mut self) -> &mut Bytes {
&mut self.0
}
}
2017-11-06 10:24:49 +01:00
impl fmt::Debug for PayloadItem {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
fmt::Debug::fmt(&self.0, f)
}
}
2017-12-19 09:29:25 +01:00
/// Buffered stream of bytes chunks
2017-10-09 05:16:48 +02:00
///
2017-12-19 09:29:25 +01:00
/// Payload stores chunks in a vector. First chunk can be received with `.readany()` method.
2017-12-19 06:58:38 +01:00
/// Payload stream is not thread safe.
2017-12-19 09:29:25 +01:00
///
/// Payload stream can be used as `HttpResponse` body stream.
2017-10-19 08:43:50 +02:00
#[derive(Debug)]
2017-10-09 05:16:48 +02:00
pub struct Payload {
inner: Rc<RefCell<Inner>>,
}
impl Payload {
2017-12-19 06:58:38 +01:00
/// 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 new(eof: bool) -> (PayloadSender, Payload) {
2017-10-09 05:16:48 +02:00
let shared = Rc::new(RefCell::new(Inner::new(eof)));
2017-10-23 04:58:50 +02:00
(PayloadSender{inner: Rc::downgrade(&shared)}, Payload{inner: shared})
2017-10-09 05:16:48 +02:00
}
2017-11-27 04:00:57 +01:00
/// Create empty payload
#[doc(hidden)]
pub fn empty() -> Payload {
Payload{inner: Rc::new(RefCell::new(Inner::new(true)))}
}
2017-10-09 05:16:48 +02:00
/// Indicates EOF of payload
pub fn eof(&self) -> bool {
self.inner.borrow().eof()
}
/// Length of the data in this payload
pub fn len(&self) -> usize {
self.inner.borrow().len()
}
/// Is payload empty
pub fn is_empty(&self) -> bool {
self.inner.borrow().len() == 0
}
2017-10-09 05:55:44 +02:00
/// Get first available chunk of data.
2017-12-19 09:18:57 +01:00
pub fn readany(&mut self) -> ReadAny {
ReadAny(Rc::clone(&self.inner))
2017-10-09 05:16:48 +02:00
}
2017-12-19 09:18:57 +01:00
/// Get exact number of bytes
pub fn readexactly(&mut self, size: usize) -> ReadExactly {
ReadExactly(Rc::clone(&self.inner), size)
2017-10-19 08:43:50 +02:00
}
/// Read until `\n`
2017-12-19 09:18:57 +01:00
pub fn readline(&mut self) -> ReadLine {
ReadLine(Rc::clone(&self.inner))
2017-10-19 08:43:50 +02:00
}
/// Read until match line
2017-12-19 09:18:57 +01:00
pub fn readuntil(&mut self, line: &[u8]) -> ReadUntil {
ReadUntil(Rc::clone(&self.inner), line.to_vec())
2017-10-19 08:43:50 +02:00
}
2017-10-14 09:11:12 +02:00
#[doc(hidden)]
pub fn readall(&mut self) -> Option<Bytes> {
self.inner.borrow_mut().readall()
}
2017-10-09 05:16:48 +02:00
/// Put unused data back to payload
2017-10-14 01:33:23 +02:00
pub fn unread_data(&mut self, data: Bytes) {
2017-10-09 05:16:48 +02:00
self.inner.borrow_mut().unread_data(data);
}
2017-11-04 17:07:44 +01:00
/// Get size of payload buffer
pub fn buffer_size(&self) -> usize {
self.inner.borrow().buffer_size()
}
/// Set size of payload buffer
pub fn set_buffer_size(&self, size: usize) {
self.inner.borrow_mut().set_buffer_size(size)
}
2017-12-14 07:36:28 +01:00
2017-12-19 09:29:25 +01:00
/// Convert payload into compatible `HttpResponse` body stream
2017-12-14 07:36:28 +01:00
pub fn stream(self) -> BodyStream {
2017-12-19 09:18:57 +01:00
Box::new(self.map_err(|e| e.into()))
2017-12-14 07:36:28 +01:00
}
2017-10-09 05:16:48 +02:00
}
impl Stream for Payload {
2017-12-19 09:18:57 +01:00
type Item = Bytes;
2017-10-27 08:14:33 +02:00
type Error = PayloadError;
2017-10-09 05:16:48 +02:00
2017-12-19 09:18:57 +01:00
fn poll(&mut self) -> Poll<Option<Bytes>, PayloadError> {
match self.inner.borrow_mut().readany()? {
Async::Ready(Some(item)) => Ok(Async::Ready(Some(item.0))),
Async::Ready(None) => Ok(Async::Ready(None)),
Async::NotReady => Ok(Async::NotReady),
}
}
}
impl Clone for Payload {
fn clone(&self) -> Payload {
Payload{inner: Rc::clone(&self.inner)}
}
}
/// Get first available chunk of data
pub struct ReadAny(Rc<RefCell<Inner>>);
impl Stream for ReadAny {
type Item = Bytes;
type Error = PayloadError;
fn poll(&mut self) -> Poll<Option<Bytes>, Self::Error> {
match self.0.borrow_mut().readany()? {
Async::Ready(Some(item)) => Ok(Async::Ready(Some(item.0))),
Async::Ready(None) => Ok(Async::Ready(None)),
Async::NotReady => Ok(Async::NotReady),
}
}
}
/// Get exact number of bytes
pub struct ReadExactly(Rc<RefCell<Inner>>, usize);
impl Future for ReadExactly {
type Item = Bytes;
type Error = PayloadError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
match self.0.borrow_mut().readexactly(self.1)? {
Async::Ready(chunk) => Ok(Async::Ready(chunk)),
Async::NotReady => Ok(Async::NotReady),
}
}
}
/// Read until `\n`
pub struct ReadLine(Rc<RefCell<Inner>>);
impl Future for ReadLine {
type Item = Bytes;
type Error = PayloadError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
match self.0.borrow_mut().readline()? {
Async::Ready(chunk) => Ok(Async::Ready(chunk)),
Async::NotReady => Ok(Async::NotReady),
}
}
}
/// Read until match line
pub struct ReadUntil(Rc<RefCell<Inner>>, Vec<u8>);
impl Future for ReadUntil {
type Item = Bytes;
type Error = PayloadError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
match self.0.borrow_mut().readuntil(&self.1)? {
Async::Ready(chunk) => Ok(Async::Ready(chunk)),
Async::NotReady => Ok(Async::NotReady),
}
2017-10-09 05:16:48 +02:00
}
}
2017-12-19 09:18:57 +01:00
/// Payload writer interface.
2017-12-19 06:58:38 +01:00
pub trait PayloadWriter {
/// Set stream error.
2017-11-06 10:24:49 +01:00
fn set_error(&mut self, err: PayloadError);
2017-12-19 06:58:38 +01:00
/// Write eof into a stream which closes reading side of a stream.
2017-11-06 10:24:49 +01:00
fn feed_eof(&mut self);
2017-12-19 06:58:38 +01:00
/// Feed bytes into a payload stream
2017-11-06 10:24:49 +01:00
fn feed_data(&mut self, data: Bytes);
2017-12-19 06:58:38 +01:00
/// Get estimated available capacity
2017-11-06 10:24:49 +01:00
fn capacity(&self) -> usize;
}
2017-12-19 06:58:38 +01:00
/// Sender part of the payload stream
pub struct PayloadSender {
2017-10-09 05:16:48 +02:00
inner: Weak<RefCell<Inner>>,
}
2017-11-06 10:24:49 +01:00
impl PayloadWriter for PayloadSender {
fn set_error(&mut self, err: PayloadError) {
2017-10-14 01:33:23 +02:00
if let Some(shared) = self.inner.upgrade() {
shared.borrow_mut().set_error(err)
}
}
2017-11-06 10:24:49 +01:00
fn feed_eof(&mut self) {
2017-10-09 05:16:48 +02:00
if let Some(shared) = self.inner.upgrade() {
shared.borrow_mut().feed_eof()
}
}
2017-11-06 10:24:49 +01:00
fn feed_data(&mut self, data: Bytes) {
2017-10-09 05:16:48 +02:00
if let Some(shared) = self.inner.upgrade() {
shared.borrow_mut().feed_data(data)
}
}
2017-11-06 10:24:49 +01:00
fn capacity(&self) -> usize {
if let Some(shared) = self.inner.upgrade() {
shared.borrow().capacity()
} else {
0
}
}
}
2017-10-09 05:16:48 +02:00
2017-10-19 08:43:50 +02:00
#[derive(Debug)]
2017-10-09 05:16:48 +02:00
struct Inner {
len: usize,
eof: bool,
2017-10-14 01:33:23 +02:00
err: Option<PayloadError>,
2017-10-09 05:16:48 +02:00
task: Option<Task>,
items: VecDeque<Bytes>,
2017-11-04 17:07:44 +01:00
buf_size: usize,
2017-10-09 05:16:48 +02:00
}
impl Inner {
fn new(eof: bool) -> Self {
Inner {
len: 0,
eof: eof,
2017-10-14 01:33:23 +02:00
err: None,
2017-10-09 05:16:48 +02:00
task: None,
items: VecDeque::new(),
2017-11-04 17:07:44 +01:00
buf_size: DEFAULT_BUFFER_SIZE,
2017-10-09 05:16:48 +02:00
}
}
2017-10-14 01:33:23 +02:00
fn set_error(&mut self, err: PayloadError) {
self.err = Some(err);
if let Some(task) = self.task.take() {
task.notify()
}
}
2017-10-09 05:16:48 +02:00
fn feed_eof(&mut self) {
self.eof = true;
if let Some(task) = self.task.take() {
task.notify()
}
}
fn feed_data(&mut self, data: Bytes) {
self.len += data.len();
self.items.push_back(data);
if let Some(task) = self.task.take() {
task.notify()
}
}
fn eof(&self) -> bool {
2017-10-14 09:11:12 +02:00
self.items.is_empty() && self.eof
2017-10-09 05:16:48 +02:00
}
fn len(&self) -> usize {
self.len
}
2017-10-27 08:14:33 +02:00
fn readany(&mut self) -> Poll<Option<PayloadItem>, PayloadError> {
2017-10-09 05:16:48 +02:00
if let Some(data) = self.items.pop_front() {
self.len -= data.len();
2017-10-27 08:14:33 +02:00
Ok(Async::Ready(Some(PayloadItem(data))))
2017-10-09 05:16:48 +02:00
} else if self.eof {
2017-10-27 08:14:33 +02:00
Ok(Async::Ready(None))
2017-10-14 01:33:23 +02:00
} else if let Some(err) = self.err.take() {
2017-10-27 08:14:33 +02:00
Err(err)
2017-10-09 05:16:48 +02:00
} else {
self.task = Some(current_task());
2017-10-27 08:14:33 +02:00
Ok(Async::NotReady)
2017-10-09 05:16:48 +02:00
}
}
2017-10-19 08:43:50 +02:00
fn readexactly(&mut self, size: usize) -> Result<Async<Bytes>, PayloadError> {
if size <= self.len {
let mut buf = BytesMut::with_capacity(size);
while buf.len() < size {
let mut chunk = self.items.pop_front().unwrap();
2017-10-23 04:58:50 +02:00
let rem = cmp::min(size - buf.len(), chunk.len());
self.len -= rem;
2017-12-14 01:44:35 +01:00
buf.extend_from_slice(&chunk.split_to(rem));
2017-10-19 08:43:50 +02:00
if !chunk.is_empty() {
self.items.push_front(chunk);
return Ok(Async::Ready(buf.freeze()))
}
}
}
if let Some(err) = self.err.take() {
Err(err)
} else {
self.task = Some(current_task());
Ok(Async::NotReady)
}
}
fn readuntil(&mut self, line: &[u8]) -> Result<Async<Bytes>, PayloadError> {
let mut idx = 0;
let mut num = 0;
let mut offset = 0;
let mut found = false;
let mut length = 0;
for no in 0..self.items.len() {
{
let chunk = &self.items[no];
for (pos, ch) in chunk.iter().enumerate() {
if *ch == line[idx] {
idx += 1;
if idx == line.len() {
num = no;
offset = pos+1;
2017-10-23 05:07:18 +02:00
length += pos+1;
2017-10-19 08:43:50 +02:00
found = true;
break;
}
} else {
idx = 0
}
}
if !found {
length += chunk.len()
}
}
if found {
let mut buf = BytesMut::with_capacity(length);
if num > 0 {
for _ in 0..num {
2017-12-14 01:44:35 +01:00
buf.extend_from_slice(&self.items.pop_front().unwrap());
2017-10-19 08:43:50 +02:00
}
}
if offset > 0 {
let mut chunk = self.items.pop_front().unwrap();
2017-12-14 01:44:35 +01:00
buf.extend_from_slice(&chunk.split_to(offset));
2017-10-19 08:43:50 +02:00
if !chunk.is_empty() {
self.items.push_front(chunk)
}
}
self.len -= length;
return Ok(Async::Ready(buf.freeze()))
}
}
if let Some(err) = self.err.take() {
Err(err)
} else {
self.task = Some(current_task());
Ok(Async::NotReady)
}
}
fn readline(&mut self) -> Result<Async<Bytes>, PayloadError> {
self.readuntil(b"\n")
}
2017-10-14 09:11:12 +02:00
pub fn readall(&mut self) -> Option<Bytes> {
let len = self.items.iter().fold(0, |cur, item| cur + item.len());
if len > 0 {
let mut buf = BytesMut::with_capacity(len);
for item in &self.items {
2017-12-14 01:44:35 +01:00
buf.extend_from_slice(item);
2017-10-14 09:11:12 +02:00
}
self.items = VecDeque::new();
2017-10-19 08:43:50 +02:00
self.len = 0;
2017-10-14 09:11:12 +02:00
Some(buf.take().freeze())
} else {
None
}
}
2017-11-04 17:07:44 +01:00
fn unread_data(&mut self, data: Bytes) {
2017-10-09 05:16:48 +02:00
self.len += data.len();
2017-12-21 05:30:54 +01:00
self.items.push_front(data);
2017-10-09 05:16:48 +02:00
}
2017-11-04 17:07:44 +01:00
fn capacity(&self) -> usize {
if self.len > self.buf_size {
0
} else {
self.buf_size - self.len
}
}
fn buffer_size(&self) -> usize {
self.buf_size
}
fn set_buffer_size(&mut self, size: usize) {
self.buf_size = size
}
2017-10-09 05:16:48 +02:00
}
2017-10-23 04:58:50 +02:00
#[cfg(test)]
2017-10-23 06:40:41 +02:00
mod tests {
2017-10-23 04:58:50 +02:00
use super::*;
2017-10-23 23:08:11 +02:00
use std::io;
2017-11-16 07:06:28 +01:00
use failure::Fail;
2017-10-23 04:58:50 +02:00
use futures::future::{lazy, result};
use tokio_core::reactor::Core;
2017-10-23 23:08:11 +02:00
#[test]
fn test_error() {
2017-11-16 07:06:28 +01:00
let err: PayloadError = io::Error::new(io::ErrorKind::Other, "ParseError").into();
2017-10-23 23:08:11 +02:00
assert_eq!(format!("{}", err), "ParseError");
2017-11-16 07:06:28 +01:00
assert_eq!(format!("{}", err.cause().unwrap()), "ParseError");
2017-10-23 23:08:11 +02:00
let err = PayloadError::Incomplete;
assert_eq!(format!("{}", err), "A payload reached EOF, but is not complete.");
}
2017-10-23 04:58:50 +02:00
#[test]
fn test_basic() {
Core::new().unwrap().run(lazy(|| {
let (_, mut payload) = Payload::new(false);
assert!(!payload.eof());
assert!(payload.is_empty());
assert_eq!(payload.len(), 0);
2017-12-19 09:18:57 +01:00
assert_eq!(Async::NotReady, payload.readany().poll().ok().unwrap());
2017-10-23 04:58:50 +02:00
let res: Result<(), ()> = Ok(());
result(res)
})).unwrap();
}
#[test]
fn test_eof() {
Core::new().unwrap().run(lazy(|| {
let (mut sender, mut payload) = Payload::new(false);
2017-12-19 09:18:57 +01:00
assert_eq!(Async::NotReady, payload.readany().poll().ok().unwrap());
2017-10-23 04:58:50 +02:00
assert!(!payload.eof());
sender.feed_data(Bytes::from("data"));
sender.feed_eof();
assert!(!payload.eof());
2017-12-19 09:18:57 +01:00
assert_eq!(Async::Ready(Some(Bytes::from("data"))),
payload.readany().poll().ok().unwrap());
2017-10-23 04:58:50 +02:00
assert!(payload.is_empty());
assert!(payload.eof());
2017-10-23 05:07:18 +02:00
assert_eq!(payload.len(), 0);
2017-10-23 04:58:50 +02:00
2017-12-19 09:18:57 +01:00
assert_eq!(Async::Ready(None), payload.readany().poll().ok().unwrap());
2017-10-23 04:58:50 +02:00
let res: Result<(), ()> = Ok(());
result(res)
})).unwrap();
}
#[test]
fn test_err() {
Core::new().unwrap().run(lazy(|| {
let (mut sender, mut payload) = Payload::new(false);
2017-12-19 09:18:57 +01:00
assert_eq!(Async::NotReady, payload.readany().poll().ok().unwrap());
2017-10-23 04:58:50 +02:00
sender.set_error(PayloadError::Incomplete);
2017-12-19 09:18:57 +01:00
payload.readany().poll().err().unwrap();
2017-10-23 04:58:50 +02:00
let res: Result<(), ()> = Ok(());
result(res)
})).unwrap();
}
#[test]
fn test_readany() {
Core::new().unwrap().run(lazy(|| {
let (mut sender, mut payload) = Payload::new(false);
sender.feed_data(Bytes::from("line1"));
assert!(!payload.is_empty());
assert_eq!(payload.len(), 5);
sender.feed_data(Bytes::from("line2"));
assert!(!payload.is_empty());
assert_eq!(payload.len(), 10);
2017-12-19 09:18:57 +01:00
assert_eq!(Async::Ready(Some(Bytes::from("line1"))),
payload.readany().poll().ok().unwrap());
2017-10-23 04:58:50 +02:00
assert!(!payload.is_empty());
assert_eq!(payload.len(), 5);
let res: Result<(), ()> = Ok(());
result(res)
})).unwrap();
}
#[test]
fn test_readexactly() {
Core::new().unwrap().run(lazy(|| {
let (mut sender, mut payload) = Payload::new(false);
2017-12-19 09:18:57 +01:00
assert_eq!(Async::NotReady, payload.readexactly(2).poll().ok().unwrap());
2017-10-23 04:58:50 +02:00
sender.feed_data(Bytes::from("line1"));
sender.feed_data(Bytes::from("line2"));
2017-10-23 05:07:18 +02:00
assert_eq!(payload.len(), 10);
2017-10-23 04:58:50 +02:00
2017-12-19 09:18:57 +01:00
assert_eq!(Async::Ready(Bytes::from("li")),
payload.readexactly(2).poll().ok().unwrap());
2017-10-23 05:07:18 +02:00
assert_eq!(payload.len(), 8);
2017-10-23 04:58:50 +02:00
2017-12-19 09:18:57 +01:00
assert_eq!(Async::Ready(Bytes::from("ne1l")),
payload.readexactly(4).poll().ok().unwrap());
2017-10-23 05:07:18 +02:00
assert_eq!(payload.len(), 4);
2017-10-23 04:58:50 +02:00
sender.set_error(PayloadError::Incomplete);
2017-12-19 09:18:57 +01:00
payload.readexactly(10).poll().err().unwrap();
2017-10-23 04:58:50 +02:00
let res: Result<(), ()> = Ok(());
result(res)
})).unwrap();
}
#[test]
fn test_readuntil() {
Core::new().unwrap().run(lazy(|| {
let (mut sender, mut payload) = Payload::new(false);
2017-12-19 09:18:57 +01:00
assert_eq!(Async::NotReady, payload.readuntil(b"ne").poll().ok().unwrap());
2017-10-23 04:58:50 +02:00
sender.feed_data(Bytes::from("line1"));
sender.feed_data(Bytes::from("line2"));
2017-10-23 05:07:18 +02:00
assert_eq!(payload.len(), 10);
2017-10-23 04:58:50 +02:00
2017-11-20 05:26:05 +01:00
assert_eq!(Async::Ready(Bytes::from("line")),
2017-12-19 09:18:57 +01:00
payload.readuntil(b"ne").poll().ok().unwrap());
2017-10-23 05:07:18 +02:00
assert_eq!(payload.len(), 6);
2017-10-23 04:58:50 +02:00
2017-11-20 05:26:05 +01:00
assert_eq!(Async::Ready(Bytes::from("1line2")),
2017-12-19 09:18:57 +01:00
payload.readuntil(b"2").poll().ok().unwrap());
2017-10-23 05:07:18 +02:00
assert_eq!(payload.len(), 0);
2017-10-23 04:58:50 +02:00
sender.set_error(PayloadError::Incomplete);
2017-12-19 09:18:57 +01:00
payload.readuntil(b"b").poll().err().unwrap();
2017-10-23 04:58:50 +02:00
let res: Result<(), ()> = Ok(());
result(res)
})).unwrap();
}
2017-10-23 05:19:20 +02:00
#[test]
fn test_unread_data() {
Core::new().unwrap().run(lazy(|| {
let (_, mut payload) = Payload::new(false);
payload.unread_data(Bytes::from("data"));
assert!(!payload.is_empty());
assert_eq!(payload.len(), 4);
2017-12-19 09:18:57 +01:00
assert_eq!(Async::Ready(Some(Bytes::from("data"))),
payload.readany().poll().ok().unwrap());
2017-10-23 05:19:20 +02:00
let res: Result<(), ()> = Ok(());
result(res)
})).unwrap();
}
2017-10-23 04:58:50 +02:00
}