1
0
mirror of https://github.com/fafhrd91/actix-web synced 2024-11-24 08:22:59 +01:00
actix-web/src/types/payload.rs

456 lines
13 KiB
Rust
Raw Normal View History

//! Payload/Bytes/String extractors
use std::str;
use actix_http::error::{Error, ErrorBadRequest, PayloadError};
use actix_http::HttpMessage;
2019-03-17 08:48:40 +01:00
use bytes::{Bytes, BytesMut};
use encoding::all::UTF_8;
use encoding::types::{DecoderTrap, Encoding};
use futures::future::{err, Either, FutureResult};
use futures::{Future, Poll, Stream};
use mime::Mime;
2019-04-07 23:43:07 +02:00
use crate::dev;
use crate::extract::FromRequest;
2019-03-17 08:48:40 +01:00
use crate::http::header;
2019-04-07 23:43:07 +02:00
use crate::request::HttpRequest;
/// Payload extractor returns request 's payload stream.
///
/// ## Example
///
/// ```rust
/// use futures::{Future, Stream};
/// use actix_web::{web, error, App, Error, HttpResponse};
///
/// /// extract binary data from request
2019-03-25 01:00:59 +01:00
/// fn index(body: web::Payload) -> impl Future<Item = HttpResponse, Error = Error>
/// {
/// body.map_err(Error::from)
/// .fold(web::BytesMut::new(), move |mut body, chunk| {
/// body.extend_from_slice(&chunk);
/// Ok::<_, Error>(body)
/// })
/// .and_then(|body| {
/// format!("Body {:?}!", body);
/// Ok(HttpResponse::Ok().finish())
/// })
/// }
///
/// fn main() {
/// let app = App::new().service(
/// web::resource("/index.html").route(
/// web::get().to_async(index))
/// );
/// }
/// ```
pub struct Payload(crate::dev::Payload);
2019-03-25 01:00:59 +01:00
impl Stream for Payload {
type Item = Bytes;
type Error = PayloadError;
#[inline]
fn poll(&mut self) -> Poll<Option<Self::Item>, PayloadError> {
self.0.poll()
}
}
/// Get request's payload stream
///
/// ## Example
///
/// ```rust
/// use futures::{Future, Stream};
/// use actix_web::{web, error, App, Error, HttpResponse};
///
/// /// extract binary data from request
2019-03-25 01:00:59 +01:00
/// fn index(body: web::Payload) -> impl Future<Item = HttpResponse, Error = Error>
/// {
/// body.map_err(Error::from)
/// .fold(web::BytesMut::new(), move |mut body, chunk| {
/// body.extend_from_slice(&chunk);
/// Ok::<_, Error>(body)
/// })
/// .and_then(|body| {
/// format!("Body {:?}!", body);
/// Ok(HttpResponse::Ok().finish())
/// })
/// }
///
/// fn main() {
/// let app = App::new().service(
/// web::resource("/index.html").route(
/// web::get().to_async(index))
/// );
/// }
/// ```
impl FromRequest for Payload {
type Error = Error;
2019-03-25 01:00:59 +01:00
type Future = Result<Payload, Error>;
#[inline]
fn from_request(_: &HttpRequest, payload: &mut dev::Payload) -> Self::Future {
Ok(Payload(payload.take()))
}
}
/// Request binary data from a request's payload.
///
/// Loads request's payload and construct Bytes instance.
///
/// [**PayloadConfig**](struct.PayloadConfig.html) allows to configure
/// extraction process.
///
/// ## Example
///
/// ```rust
/// use bytes::Bytes;
/// use actix_web::{web, App};
///
/// /// extract binary data from request
/// fn index(body: Bytes) -> String {
/// format!("Body {:?}!", body)
/// }
///
/// fn main() {
/// let app = App::new().service(
/// web::resource("/index.html").route(
/// web::get().to(index))
/// );
/// }
/// ```
impl FromRequest for Bytes {
type Error = Error;
type Future =
Either<Box<Future<Item = Bytes, Error = Error>>, FutureResult<Bytes, Error>>;
#[inline]
fn from_request(req: &HttpRequest, payload: &mut dev::Payload) -> Self::Future {
let mut tmp;
2019-03-17 05:09:11 +01:00
let cfg = if let Some(cfg) = req.route_data::<PayloadConfig>() {
cfg
} else {
tmp = PayloadConfig::default();
&tmp
};
if let Err(e) = cfg.check_mimetype(req) {
return Either::B(err(e));
}
let limit = cfg.limit;
2019-04-07 23:43:07 +02:00
Either::A(Box::new(
HttpMessageBody::new(req, payload).limit(limit).from_err(),
))
}
}
/// Extract text information from a request's body.
///
/// Text extractor automatically decode body according to the request's charset.
///
/// [**PayloadConfig**](struct.PayloadConfig.html) allows to configure
/// extraction process.
///
/// ## Example
///
/// ```rust
/// use actix_web::{web, App};
///
/// /// extract text data from request
/// fn index(text: String) -> String {
/// format!("Body {}!", text)
/// }
///
/// fn main() {
/// let app = App::new().service(
/// web::resource("/index.html").route(
/// web::get()
/// .data(web::PayloadConfig::new(4096)) // <- limit size of the payload
/// .to(index)) // <- register handler with extractor params
/// );
/// }
/// ```
impl FromRequest for String {
type Error = Error;
type Future =
Either<Box<Future<Item = String, Error = Error>>, FutureResult<String, Error>>;
#[inline]
fn from_request(req: &HttpRequest, payload: &mut dev::Payload) -> Self::Future {
let mut tmp;
2019-03-17 05:09:11 +01:00
let cfg = if let Some(cfg) = req.route_data::<PayloadConfig>() {
cfg
} else {
tmp = PayloadConfig::default();
&tmp
};
// check content-type
if let Err(e) = cfg.check_mimetype(req) {
return Either::B(err(e));
}
// check charset
let encoding = match req.encoding() {
Ok(enc) => enc,
Err(e) => return Either::B(err(e.into())),
};
let limit = cfg.limit;
Either::A(Box::new(
2019-04-07 23:43:07 +02:00
HttpMessageBody::new(req, payload)
.limit(limit)
.from_err()
.and_then(move |body| {
let enc: *const Encoding = encoding as *const Encoding;
if enc == UTF_8 {
Ok(str::from_utf8(body.as_ref())
.map_err(|_| ErrorBadRequest("Can not decode body"))?
.to_owned())
} else {
Ok(encoding
.decode(&body, DecoderTrap::Strict)
.map_err(|_| ErrorBadRequest("Can not decode body"))?)
}
}),
))
}
}
/// Payload configuration for request's payload.
#[derive(Clone)]
pub struct PayloadConfig {
limit: usize,
mimetype: Option<Mime>,
}
impl PayloadConfig {
/// Create `PayloadConfig` instance and set max size of payload.
pub fn new(limit: usize) -> Self {
Self::default().limit(limit)
}
/// Change max size of payload. By default max size is 256Kb
pub fn limit(mut self, limit: usize) -> Self {
self.limit = limit;
self
}
/// Set required mime-type of the request. By default mime type is not
/// enforced.
pub fn mimetype(mut self, mt: Mime) -> Self {
self.mimetype = Some(mt);
self
}
2019-04-07 23:43:07 +02:00
fn check_mimetype(&self, req: &HttpRequest) -> Result<(), Error> {
// check content-type
if let Some(ref mt) = self.mimetype {
match req.mime_type() {
Ok(Some(ref req_mt)) => {
if mt != req_mt {
return Err(ErrorBadRequest("Unexpected Content-Type"));
}
}
Ok(None) => {
return Err(ErrorBadRequest("Content-Type is expected"));
}
Err(err) => {
return Err(err.into());
}
}
}
Ok(())
}
}
impl Default for PayloadConfig {
fn default() -> Self {
PayloadConfig {
limit: 262_144,
mimetype: None,
}
}
}
2019-03-17 08:48:40 +01:00
/// Future that resolves to a complete http message body.
///
/// Load http message body.
///
/// By default only 256Kb payload reads to a memory, then
/// `PayloadError::Overflow` get returned. Use `MessageBody::limit()`
/// method to change upper limit.
pub struct HttpMessageBody {
2019-03-17 08:48:40 +01:00
limit: usize,
length: Option<usize>,
stream: Option<dev::Decompress<dev::Payload>>,
2019-03-17 08:48:40 +01:00
err: Option<PayloadError>,
fut: Option<Box<Future<Item = Bytes, Error = PayloadError>>>,
}
impl HttpMessageBody {
2019-03-17 08:48:40 +01:00
/// Create `MessageBody` for request.
pub fn new(req: &HttpRequest, payload: &mut dev::Payload) -> HttpMessageBody {
2019-03-17 08:48:40 +01:00
let mut len = None;
if let Some(l) = req.headers().get(&header::CONTENT_LENGTH) {
2019-03-17 08:48:40 +01:00
if let Ok(s) = l.to_str() {
if let Ok(l) = s.parse::<usize>() {
len = Some(l)
} else {
return Self::err(PayloadError::UnknownLength);
}
} else {
return Self::err(PayloadError::UnknownLength);
}
}
HttpMessageBody {
stream: Some(dev::Decompress::from_headers(payload.take(), req.headers())),
2019-03-17 08:48:40 +01:00
limit: 262_144,
length: len,
fut: None,
err: None,
}
}
/// Change max size of payload. By default max size is 256Kb
pub fn limit(mut self, limit: usize) -> Self {
self.limit = limit;
self
}
fn err(e: PayloadError) -> Self {
HttpMessageBody {
stream: None,
2019-03-17 08:48:40 +01:00
limit: 262_144,
fut: None,
err: Some(e),
length: None,
}
}
}
impl Future for HttpMessageBody {
2019-03-17 08:48:40 +01:00
type Item = Bytes;
type Error = PayloadError;
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
if let Some(ref mut fut) = self.fut {
return fut.poll();
}
if let Some(err) = self.err.take() {
return Err(err);
}
if let Some(len) = self.length.take() {
if len > self.limit {
return Err(PayloadError::Overflow);
}
}
// future
let limit = self.limit;
self.fut = Some(Box::new(
self.stream
.take()
.unwrap()
2019-03-17 08:48:40 +01:00
.from_err()
.fold(BytesMut::with_capacity(8192), move |mut body, chunk| {
if (body.len() + chunk.len()) > limit {
Err(PayloadError::Overflow)
} else {
body.extend_from_slice(&chunk);
Ok(body)
}
})
.map(|body| body.freeze()),
));
self.poll()
}
}
#[cfg(test)]
mod tests {
2019-03-10 19:04:50 +01:00
use bytes::Bytes;
use super::*;
use crate::http::header;
2019-03-10 19:04:50 +01:00
use crate::test::{block_on, TestRequest};
#[test]
fn test_payload_config() {
2019-04-07 23:43:07 +02:00
let req = TestRequest::default().to_http_request();
let cfg = PayloadConfig::default().mimetype(mime::APPLICATION_JSON);
assert!(cfg.check_mimetype(&req).is_err());
let req = TestRequest::with_header(
header::CONTENT_TYPE,
"application/x-www-form-urlencoded",
)
2019-04-07 23:43:07 +02:00
.to_http_request();
assert!(cfg.check_mimetype(&req).is_err());
2019-04-07 23:43:07 +02:00
let req = TestRequest::with_header(header::CONTENT_TYPE, "application/json")
.to_http_request();
assert!(cfg.check_mimetype(&req).is_ok());
}
2019-03-10 19:04:50 +01:00
#[test]
fn test_bytes() {
2019-04-07 23:43:07 +02:00
let (req, mut pl) = TestRequest::with_header(header::CONTENT_LENGTH, "11")
2019-03-10 19:04:50 +01:00
.set_payload(Bytes::from_static(b"hello=world"))
2019-04-07 23:43:07 +02:00
.to_http_parts();
2019-03-10 19:04:50 +01:00
2019-04-07 23:43:07 +02:00
let s = block_on(Bytes::from_request(&req, &mut pl)).unwrap();
2019-03-10 19:04:50 +01:00
assert_eq!(s, Bytes::from_static(b"hello=world"));
}
#[test]
fn test_string() {
2019-04-07 23:43:07 +02:00
let (req, mut pl) = TestRequest::with_header(header::CONTENT_LENGTH, "11")
2019-03-10 19:04:50 +01:00
.set_payload(Bytes::from_static(b"hello=world"))
2019-04-07 23:43:07 +02:00
.to_http_parts();
2019-03-10 19:04:50 +01:00
2019-04-07 23:43:07 +02:00
let s = block_on(String::from_request(&req, &mut pl)).unwrap();
2019-03-10 19:04:50 +01:00
assert_eq!(s, "hello=world");
}
2019-03-17 08:48:40 +01:00
#[test]
fn test_message_body() {
2019-04-07 23:43:07 +02:00
let (req, mut pl) = TestRequest::with_header(header::CONTENT_LENGTH, "xxxx")
.to_srv_request()
.into_parts();
let res = block_on(HttpMessageBody::new(&req, &mut pl));
2019-03-17 08:48:40 +01:00
match res.err().unwrap() {
PayloadError::UnknownLength => (),
_ => unreachable!("error"),
}
2019-04-07 23:43:07 +02:00
let (req, mut pl) = TestRequest::with_header(header::CONTENT_LENGTH, "1000000")
.to_srv_request()
.into_parts();
let res = block_on(HttpMessageBody::new(&req, &mut pl));
2019-03-17 08:48:40 +01:00
match res.err().unwrap() {
PayloadError::Overflow => (),
_ => unreachable!("error"),
}
2019-04-07 23:43:07 +02:00
let (req, mut pl) = TestRequest::default()
2019-03-17 08:48:40 +01:00
.set_payload(Bytes::from_static(b"test"))
2019-04-07 23:43:07 +02:00
.to_http_parts();
let res = block_on(HttpMessageBody::new(&req, &mut pl));
2019-03-17 08:48:40 +01:00
assert_eq!(res.ok().unwrap(), Bytes::from_static(b"test"));
2019-04-07 23:43:07 +02:00
let (req, mut pl) = TestRequest::default()
2019-03-17 08:48:40 +01:00
.set_payload(Bytes::from_static(b"11111111111111"))
2019-04-07 23:43:07 +02:00
.to_http_parts();
let res = block_on(HttpMessageBody::new(&req, &mut pl).limit(5));
2019-03-17 08:48:40 +01:00
match res.err().unwrap() {
PayloadError::Overflow => (),
_ => unreachable!("error"),
}
}
}