mirror of
https://github.com/actix/actix-web.git
synced 2025-01-03 05:48:45 +00:00
Merge branch 'master' into issue-271-cors-origin-validation
This commit is contained in:
commit
3efcd14291
7 changed files with 64 additions and 25 deletions
|
@ -25,6 +25,8 @@
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
|
||||||
|
* Support chunked encoding for UrlEncoded body #262
|
||||||
|
|
||||||
* `HttpRequest::url_for()` for a named route with no variables segments #265
|
* `HttpRequest::url_for()` for a named route with no variables segments #265
|
||||||
|
|
||||||
|
|
||||||
|
|
24
src/error.rs
24
src/error.rs
|
@ -293,8 +293,18 @@ impl From<IoError> for PayloadError {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `InternalServerError` for `PayloadError`
|
/// `PayloadError` returns two possible results:
|
||||||
impl ResponseError for PayloadError {}
|
///
|
||||||
|
/// - `Overflow` returns `PayloadTooLarge`
|
||||||
|
/// - Other errors returns `BadRequest`
|
||||||
|
impl ResponseError for PayloadError {
|
||||||
|
fn error_response(&self) -> HttpResponse {
|
||||||
|
match *self {
|
||||||
|
PayloadError::Overflow => HttpResponse::new(StatusCode::PAYLOAD_TOO_LARGE),
|
||||||
|
_ => HttpResponse::new(StatusCode::BAD_REQUEST),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Return `BadRequest` for `cookie::ParseError`
|
/// Return `BadRequest` for `cookie::ParseError`
|
||||||
impl ResponseError for cookie::ParseError {
|
impl ResponseError for cookie::ParseError {
|
||||||
|
@ -417,8 +427,10 @@ pub enum UrlencodedError {
|
||||||
/// Can not decode chunked transfer encoding
|
/// Can not decode chunked transfer encoding
|
||||||
#[fail(display = "Can not decode chunked transfer encoding")]
|
#[fail(display = "Can not decode chunked transfer encoding")]
|
||||||
Chunked,
|
Chunked,
|
||||||
/// Payload size is bigger than 256k
|
/// Payload size is bigger than allowed. (default: 256kB)
|
||||||
#[fail(display = "Payload size is bigger than 256k")]
|
#[fail(
|
||||||
|
display = "Urlencoded payload size is bigger than allowed. (default: 256kB)"
|
||||||
|
)]
|
||||||
Overflow,
|
Overflow,
|
||||||
/// Payload size is now known
|
/// Payload size is now known
|
||||||
#[fail(display = "Payload size is now known")]
|
#[fail(display = "Payload size is now known")]
|
||||||
|
@ -458,8 +470,8 @@ impl From<PayloadError> for UrlencodedError {
|
||||||
/// A set of errors that can occur during parsing json payloads
|
/// A set of errors that can occur during parsing json payloads
|
||||||
#[derive(Fail, Debug)]
|
#[derive(Fail, Debug)]
|
||||||
pub enum JsonPayloadError {
|
pub enum JsonPayloadError {
|
||||||
/// Payload size is bigger than 256k
|
/// Payload size is bigger than allowed. (default: 256kB)
|
||||||
#[fail(display = "Payload size is bigger than 256k")]
|
#[fail(display = "Json payload size is bigger than allowed. (default: 256kB)")]
|
||||||
Overflow,
|
Overflow,
|
||||||
/// Content type error
|
/// Content type error
|
||||||
#[fail(display = "Content type error")]
|
#[fail(display = "Content type error")]
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
use std::ops::{Deref, DerefMut};
|
use std::ops::{Deref, DerefMut};
|
||||||
use std::str;
|
use std::{fmt, str};
|
||||||
|
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
use encoding::all::UTF_8;
|
use encoding::all::UTF_8;
|
||||||
|
@ -115,6 +115,18 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T: fmt::Debug> fmt::Debug for Path<T> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
self.inner.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: fmt::Display> fmt::Display for Path<T> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
self.inner.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Extract typed information from from the request's query.
|
/// Extract typed information from from the request's query.
|
||||||
///
|
///
|
||||||
/// ## Example
|
/// ## Example
|
||||||
|
@ -175,13 +187,24 @@ where
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn from_request(req: &HttpRequest<S>, _: &Self::Config) -> Self::Result {
|
fn from_request(req: &HttpRequest<S>, _: &Self::Config) -> Self::Result {
|
||||||
let req = req.clone();
|
|
||||||
serde_urlencoded::from_str::<T>(req.query_string())
|
serde_urlencoded::from_str::<T>(req.query_string())
|
||||||
.map_err(|e| e.into())
|
.map_err(|e| e.into())
|
||||||
.map(Query)
|
.map(Query)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T: fmt::Debug> fmt::Debug for Query<T> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
self.0.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: fmt::Display> fmt::Display for Query<T> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
self.0.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Extract typed information from the request's body.
|
/// Extract typed information from the request's body.
|
||||||
///
|
///
|
||||||
/// To extract typed information from request's body, the type `T` must
|
/// To extract typed information from request's body, the type `T` must
|
||||||
|
@ -252,6 +275,18 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T: fmt::Debug> fmt::Debug for Form<T> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
self.0.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: fmt::Display> fmt::Display for Form<T> {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
self.0.fmt(f)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Form extractor configuration
|
/// Form extractor configuration
|
||||||
///
|
///
|
||||||
/// ```rust
|
/// ```rust
|
||||||
|
|
|
@ -149,7 +149,6 @@ pub trait HttpMessage {
|
||||||
/// Returns error:
|
/// Returns error:
|
||||||
///
|
///
|
||||||
/// * content type is not `application/x-www-form-urlencoded`
|
/// * content type is not `application/x-www-form-urlencoded`
|
||||||
/// * transfer encoding is `chunked`.
|
|
||||||
/// * content-length is greater than 256k
|
/// * content-length is greater than 256k
|
||||||
///
|
///
|
||||||
/// ## Server example
|
/// ## Server example
|
||||||
|
@ -367,9 +366,7 @@ where
|
||||||
|
|
||||||
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
fn poll(&mut self) -> Poll<Self::Item, Self::Error> {
|
||||||
if let Some(req) = self.req.take() {
|
if let Some(req) = self.req.take() {
|
||||||
if req.chunked().unwrap_or(false) {
|
if let Some(len) = req.headers().get(header::CONTENT_LENGTH) {
|
||||||
return Err(UrlencodedError::Chunked);
|
|
||||||
} else if let Some(len) = req.headers().get(header::CONTENT_LENGTH) {
|
|
||||||
if let Ok(s) = len.to_str() {
|
if let Ok(s) = len.to_str() {
|
||||||
if let Ok(len) = s.parse::<u64>() {
|
if let Ok(len) = s.parse::<u64>() {
|
||||||
if len > 262_144 {
|
if len > 262_144 {
|
||||||
|
@ -577,13 +574,6 @@ mod tests {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_urlencoded_error() {
|
fn test_urlencoded_error() {
|
||||||
let req =
|
|
||||||
TestRequest::with_header(header::TRANSFER_ENCODING, "chunked").finish();
|
|
||||||
assert_eq!(
|
|
||||||
req.urlencoded::<Info>().poll().err().unwrap(),
|
|
||||||
UrlencodedError::Chunked
|
|
||||||
);
|
|
||||||
|
|
||||||
let req = TestRequest::with_header(
|
let req = TestRequest::with_header(
|
||||||
header::CONTENT_TYPE,
|
header::CONTENT_TYPE,
|
||||||
"application/x-www-form-urlencoded",
|
"application/x-www-form-urlencoded",
|
||||||
|
|
|
@ -283,9 +283,9 @@ impl<S> HttpRequest<S> {
|
||||||
/// Generate url for named resource
|
/// Generate url for named resource
|
||||||
///
|
///
|
||||||
/// ```rust
|
/// ```rust
|
||||||
/// //#### # extern crate actix_web;
|
/// # extern crate actix_web;
|
||||||
/// //#### # use actix_web::{App, HttpRequest, HttpResponse, http};
|
/// # use actix_web::{App, HttpRequest, HttpResponse, http};
|
||||||
/// //#### #
|
/// #
|
||||||
/// fn index(req: HttpRequest) -> HttpResponse {
|
/// fn index(req: HttpRequest) -> HttpResponse {
|
||||||
/// let url = req.url_for("foo", &["1", "2", "3"]); // <- generate url for "foo" resource
|
/// let url = req.url_for("foo", &["1", "2", "3"]); // <- generate url for "foo" resource
|
||||||
/// HttpResponse::Ok().into()
|
/// HttpResponse::Ok().into()
|
||||||
|
|
|
@ -2,10 +2,10 @@
|
||||||
//! for Rust.
|
//! for Rust.
|
||||||
//!
|
//!
|
||||||
//! ```rust
|
//! ```rust
|
||||||
//! use actix_web::{server, App, Path};
|
//! use actix_web::{server, App, Path, Responder};
|
||||||
//! # use std::thread;
|
//! # use std::thread;
|
||||||
//!
|
//!
|
||||||
//! fn index(info: Path<(String, u32)>) -> String {
|
//! fn index(info: Path<(String, u32)>) -> impl Responder {
|
||||||
//! format!("Hello {}! id:{}", info.0, info.1)
|
//! format!("Hello {}! id:{}", info.0, info.1)
|
||||||
//! }
|
//! }
|
||||||
//!
|
//!
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
//! [**IdentityService**](struct.IdentityService.html) middleware can be
|
//! [**IdentityService**](struct.IdentityService.html) middleware can be
|
||||||
//! used with different policies types to store identity information.
|
//! used with different policies types to store identity information.
|
||||||
//!
|
//!
|
||||||
//! Bu default, only cookie identity policy is implemented. Other backend
|
//! By default, only cookie identity policy is implemented. Other backend
|
||||||
//! implementations can be added separately.
|
//! implementations can be added separately.
|
||||||
//!
|
//!
|
||||||
//! [**CookieIdentityPolicy**](struct.CookieIdentityPolicy.html)
|
//! [**CookieIdentityPolicy**](struct.CookieIdentityPolicy.html)
|
||||||
|
|
Loading…
Reference in a new issue