1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2024-10-12 05:02:27 +00:00
actix-web/src/body.rs

386 lines
10 KiB
Rust
Raw Normal View History

2017-10-24 06:39:01 +00:00
use bytes::{Bytes, BytesMut};
2017-11-30 22:42:20 +00:00
use futures::Stream;
2018-04-13 23:02:01 +00:00
use std::rc::Rc;
use std::sync::Arc;
use std::{fmt, mem};
2017-10-24 06:25:32 +00:00
2018-01-01 01:26:32 +00:00
use context::ActorHttpContext;
2018-04-13 23:02:01 +00:00
use error::Error;
use handler::Responder;
use httprequest::HttpRequest;
use httpresponse::HttpResponse;
2017-12-14 06:36:28 +00:00
/// Type represent streaming body
2018-04-13 23:02:01 +00:00
pub type BodyStream = Box<Stream<Item = Bytes, Error = Error>>;
2017-10-24 06:25:32 +00:00
/// Represents various types of http message body.
pub enum Body {
/// Empty response. `Content-Length` header is set to `0`
Empty,
/// Specific response body.
2017-11-10 21:42:32 +00:00
Binary(Binary),
2017-10-24 06:25:32 +00:00
/// Unspecified streaming response. Developer is responsible for setting
/// right `Content-Length` or `Transfer-Encoding` headers.
2017-11-30 22:42:20 +00:00
Streaming(BodyStream),
2018-01-01 01:26:32 +00:00
/// Special body type for actor response.
Actor(Box<ActorHttpContext>),
2017-10-24 06:25:32 +00:00
}
/// Represents various types of binary body.
/// `Content-Length` header is set to length of the body.
2017-11-27 18:39:47 +00:00
#[derive(Debug, PartialEq)]
2017-11-10 21:42:32 +00:00
pub enum Binary {
2017-10-24 06:25:32 +00:00
/// Bytes body
Bytes(Bytes),
/// Static slice
Slice(&'static [u8]),
2018-01-15 21:47:25 +00:00
/// Shared string body
2017-10-24 06:49:27 +00:00
SharedString(Rc<String>),
/// Shared string body
#[doc(hidden)]
ArcSharedString(Arc<String>),
/// Shared vec body
SharedVec(Arc<Vec<u8>>),
2017-10-24 06:25:32 +00:00
}
impl Body {
2017-11-20 04:55:37 +00:00
/// Does this body streaming.
2017-12-14 00:44:35 +00:00
#[inline]
2017-11-20 04:55:37 +00:00
pub fn is_streaming(&self) -> bool {
2017-10-24 06:25:32 +00:00
match *self {
2018-01-01 01:26:32 +00:00
Body::Streaming(_) | Body::Actor(_) => true,
2018-04-13 23:02:01 +00:00
_ => false,
2017-10-24 06:25:32 +00:00
}
}
2017-11-09 00:44:23 +00:00
/// Is this binary body.
2017-12-14 00:44:35 +00:00
#[inline]
2017-11-09 00:44:23 +00:00
pub fn is_binary(&self) -> bool {
match *self {
Body::Binary(_) => true,
2018-04-13 23:02:01 +00:00
_ => false,
2017-11-09 00:44:23 +00:00
}
}
2017-10-24 06:25:32 +00:00
2018-05-15 23:41:46 +00:00
/// Is this binary empy.
#[inline]
pub fn is_empty(&self) -> bool {
match *self {
Body::Empty => true,
_ => false,
}
}
2017-10-24 06:39:01 +00:00
/// Create body from slice (copy)
2017-10-25 23:25:26 +00:00
pub fn from_slice(s: &[u8]) -> Body {
2017-11-10 21:42:32 +00:00
Body::Binary(Binary::Bytes(Bytes::from(s)))
2017-10-24 06:25:32 +00:00
}
2018-05-15 23:41:46 +00:00
/// Is this binary body.
#[inline]
pub(crate) fn binary(self) -> Binary {
match self {
Body::Binary(b) => b,
_ => panic!(),
}
}
2017-10-24 06:25:32 +00:00
}
2017-11-30 22:42:20 +00:00
impl PartialEq for Body {
fn eq(&self, other: &Body) -> bool {
match *self {
Body::Empty => match *other {
Body::Empty => true,
_ => false,
},
Body::Binary(ref b) => match *other {
Body::Binary(ref b2) => b == b2,
_ => false,
},
2018-01-01 01:26:32 +00:00
Body::Streaming(_) | Body::Actor(_) => false,
2017-11-30 22:42:20 +00:00
}
}
}
impl fmt::Debug for Body {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
match *self {
Body::Empty => write!(f, "Body::Empty"),
Body::Binary(ref b) => write!(f, "Body::Binary({:?})", b),
Body::Streaming(_) => write!(f, "Body::Streaming(_)"),
2018-01-01 01:26:32 +00:00
Body::Actor(_) => write!(f, "Body::Actor(_)"),
2017-11-30 22:42:20 +00:00
}
}
}
2018-04-13 23:02:01 +00:00
impl<T> From<T> for Body
where
T: Into<Binary>,
{
2017-10-24 06:39:01 +00:00
fn from(b: T) -> Body {
Body::Binary(b.into())
2017-10-24 06:25:32 +00:00
}
}
2018-01-01 01:26:32 +00:00
impl From<Box<ActorHttpContext>> for Body {
fn from(ctx: Box<ActorHttpContext>) -> Body {
Body::Actor(ctx)
}
}
2017-11-10 21:42:32 +00:00
impl Binary {
2017-12-14 00:44:35 +00:00
#[inline]
2017-10-25 23:25:26 +00:00
pub fn is_empty(&self) -> bool {
self.len() == 0
}
2017-12-14 00:44:35 +00:00
#[inline]
2017-10-24 06:39:01 +00:00
pub fn len(&self) -> usize {
2017-10-25 23:25:26 +00:00
match *self {
2017-11-10 21:42:32 +00:00
Binary::Bytes(ref bytes) => bytes.len(),
Binary::Slice(slice) => slice.len(),
Binary::SharedString(ref s) => s.len(),
Binary::ArcSharedString(ref s) => s.len(),
Binary::SharedVec(ref s) => s.len(),
2017-10-24 06:39:01 +00:00
}
}
/// Create binary body from slice
2017-11-10 21:42:32 +00:00
pub fn from_slice(s: &[u8]) -> Binary {
Binary::Bytes(Bytes::from(s))
2017-10-24 06:25:32 +00:00
}
2018-01-14 22:40:39 +00:00
/// Convert Binary to a Bytes instance
pub fn take(&mut self) -> Bytes {
mem::replace(self, Binary::Slice(b"")).into()
}
}
impl Clone for Binary {
fn clone(&self) -> Binary {
match *self {
Binary::Bytes(ref bytes) => Binary::Bytes(bytes.clone()),
Binary::Slice(slice) => Binary::Bytes(Bytes::from(slice)),
Binary::SharedString(ref s) => Binary::SharedString(s.clone()),
Binary::ArcSharedString(ref s) => Binary::ArcSharedString(s.clone()),
Binary::SharedVec(ref s) => Binary::SharedVec(s.clone()),
2018-01-14 22:40:39 +00:00
}
}
2017-10-24 06:25:32 +00:00
}
2018-01-10 23:28:33 +00:00
impl Into<Bytes> for Binary {
fn into(self) -> Bytes {
match self {
Binary::Bytes(bytes) => bytes,
Binary::Slice(slice) => Bytes::from(slice),
Binary::SharedString(s) => Bytes::from(s.as_str()),
Binary::ArcSharedString(s) => Bytes::from(s.as_str()),
Binary::SharedVec(s) => Bytes::from(AsRef::<[u8]>::as_ref(s.as_ref())),
2018-01-10 23:28:33 +00:00
}
}
}
2017-11-10 21:42:32 +00:00
impl From<&'static str> for Binary {
fn from(s: &'static str) -> Binary {
Binary::Slice(s.as_ref())
2017-10-24 06:25:32 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl From<&'static [u8]> for Binary {
fn from(s: &'static [u8]) -> Binary {
Binary::Slice(s)
2017-10-24 06:25:32 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl From<Vec<u8>> for Binary {
fn from(vec: Vec<u8>) -> Binary {
Binary::Bytes(Bytes::from(vec))
2017-10-24 06:25:32 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl From<String> for Binary {
fn from(s: String) -> Binary {
Binary::Bytes(Bytes::from(s))
2017-10-24 06:25:32 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl<'a> From<&'a String> for Binary {
fn from(s: &'a String) -> Binary {
Binary::Bytes(Bytes::from(AsRef::<[u8]>::as_ref(&s)))
2017-10-30 05:50:21 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl From<Bytes> for Binary {
fn from(s: Bytes) -> Binary {
Binary::Bytes(s)
2017-10-24 06:39:01 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl From<BytesMut> for Binary {
fn from(s: BytesMut) -> Binary {
Binary::Bytes(s.freeze())
2017-10-24 06:39:01 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl From<Rc<String>> for Binary {
fn from(body: Rc<String>) -> Binary {
Binary::SharedString(body)
2017-10-24 06:49:27 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl<'a> From<&'a Rc<String>> for Binary {
fn from(body: &'a Rc<String>) -> Binary {
Binary::SharedString(Rc::clone(body))
2017-10-24 06:49:27 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl From<Arc<String>> for Binary {
fn from(body: Arc<String>) -> Binary {
Binary::ArcSharedString(body)
2017-10-24 06:49:27 +00:00
}
}
2017-11-10 21:42:32 +00:00
impl<'a> From<&'a Arc<String>> for Binary {
fn from(body: &'a Arc<String>) -> Binary {
Binary::ArcSharedString(Arc::clone(body))
2017-10-24 06:49:27 +00:00
}
}
impl From<Arc<Vec<u8>>> for Binary {
fn from(body: Arc<Vec<u8>>) -> Binary {
Binary::SharedVec(body)
}
}
impl<'a> From<&'a Arc<Vec<u8>>> for Binary {
fn from(body: &'a Arc<Vec<u8>>) -> Binary {
Binary::SharedVec(Arc::clone(body))
}
}
2017-11-10 21:42:32 +00:00
impl AsRef<[u8]> for Binary {
2018-03-18 18:05:44 +00:00
#[inline]
2017-10-24 06:25:32 +00:00
fn as_ref(&self) -> &[u8] {
2017-10-25 23:25:26 +00:00
match *self {
2018-03-19 16:30:58 +00:00
Binary::Bytes(ref bytes) => bytes.as_ref(),
2017-11-10 21:42:32 +00:00
Binary::Slice(slice) => slice,
Binary::SharedString(ref s) => s.as_bytes(),
Binary::ArcSharedString(ref s) => s.as_bytes(),
Binary::SharedVec(ref s) => s.as_ref().as_ref(),
2017-10-24 06:25:32 +00:00
}
}
}
2017-10-24 07:09:52 +00:00
impl Responder for Binary {
type Item = HttpResponse;
type Error = Error;
2018-05-04 18:44:22 +00:00
fn respond_to<S>(self, req: &HttpRequest<S>) -> Result<HttpResponse, Error> {
Ok(HttpResponse::build_from(req)
2018-04-29 16:09:08 +00:00
.content_type("application/octet-stream")
.body(self))
}
}
2017-10-24 07:09:52 +00:00
#[cfg(test)]
mod tests {
use super::*;
2017-11-20 04:55:37 +00:00
#[test]
fn test_body_is_streaming() {
assert_eq!(Body::Empty.is_streaming(), false);
assert_eq!(Body::Binary(Binary::from("")).is_streaming(), false);
}
2017-11-20 04:32:37 +00:00
#[test]
fn test_is_empty() {
assert_eq!(Binary::from("").is_empty(), true);
assert_eq!(Binary::from("test").is_empty(), false);
}
2017-10-24 07:09:52 +00:00
#[test]
fn test_static_str() {
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from("test").len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from("test").as_ref(), b"test");
2017-10-24 07:09:52 +00:00
}
#[test]
fn test_static_bytes() {
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from(b"test".as_ref()).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(b"test".as_ref()).as_ref(), b"test");
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from_slice(b"test".as_ref()).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from_slice(b"test".as_ref()).as_ref(), b"test");
2017-10-24 07:09:52 +00:00
}
#[test]
fn test_vec() {
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from(Vec::from("test")).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(Vec::from("test")).as_ref(), b"test");
2017-10-24 07:09:52 +00:00
}
#[test]
fn test_bytes() {
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from(Bytes::from("test")).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(Bytes::from("test")).as_ref(), b"test");
2017-10-24 07:09:52 +00:00
}
2017-10-30 05:50:21 +00:00
#[test]
fn test_ref_string() {
let b = Rc::new("test".to_owned());
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from(&b).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(&b).as_ref(), b"test");
2017-10-30 05:50:21 +00:00
}
2017-10-24 07:09:52 +00:00
#[test]
fn test_rc_string() {
let b = Rc::new("test".to_owned());
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from(b.clone()).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(b.clone()).as_ref(), b"test");
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from(&b).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(&b).as_ref(), b"test");
2017-10-24 07:09:52 +00:00
}
#[test]
fn test_arc_string() {
let b = Arc::new("test".to_owned());
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from(b.clone()).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(b.clone()).as_ref(), b"test");
2017-11-10 21:42:32 +00:00
assert_eq!(Binary::from(&b).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(&b).as_ref(), b"test");
2017-10-24 07:09:52 +00:00
}
2017-11-20 04:32:37 +00:00
#[test]
fn test_string() {
let b = "test".to_owned();
assert_eq!(Binary::from(b.clone()).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(b.clone()).as_ref(), b"test");
2017-11-20 04:32:37 +00:00
assert_eq!(Binary::from(&b).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(&b).as_ref(), b"test");
2017-11-20 04:32:37 +00:00
}
#[test]
fn test_shared_vec() {
let b = Arc::new(Vec::from(&b"test"[..]));
assert_eq!(Binary::from(b.clone()).len(), 4);
assert_eq!(Binary::from(b.clone()).as_ref(), &b"test"[..]);
assert_eq!(Binary::from(&b).len(), 4);
assert_eq!(Binary::from(&b).as_ref(), &b"test"[..]);
}
2017-11-20 04:32:37 +00:00
#[test]
fn test_bytes_mut() {
2018-04-13 23:02:01 +00:00
let b = BytesMut::from("test");
2017-11-20 04:32:37 +00:00
assert_eq!(Binary::from(b.clone()).len(), 4);
2018-04-05 03:24:09 +00:00
assert_eq!(Binary::from(b).as_ref(), b"test");
2017-11-20 04:32:37 +00:00
}
2018-01-10 23:28:33 +00:00
#[test]
fn test_binary_into() {
let bytes = Bytes::from_static(b"test");
let b: Bytes = Binary::from("test").into();
assert_eq!(b, bytes);
let b: Bytes = Binary::from(bytes.clone()).into();
assert_eq!(b, bytes);
}
2017-10-24 07:09:52 +00:00
}