mirror of
https://github.com/actix/actix-extras.git
synced 2024-11-24 07:53:00 +01:00
add content-encoding decompression
This commit is contained in:
parent
9451ba71f4
commit
1904b01fc0
16
Cargo.toml
16
Cargo.toml
@ -36,7 +36,7 @@ members = [
|
|||||||
]
|
]
|
||||||
|
|
||||||
[package.metadata.docs.rs]
|
[package.metadata.docs.rs]
|
||||||
features = ["ssl", "tls", "rust-tls", "brotli", "flate2-c", "cookies"]
|
features = ["ssl", "tls", "rust-tls", "brotli", "flate2-c", "cookies", "client"]
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["brotli", "flate2-c", "cookies", "client"]
|
default = ["brotli", "flate2-c", "cookies", "client"]
|
||||||
@ -45,13 +45,13 @@ default = ["brotli", "flate2-c", "cookies", "client"]
|
|||||||
client = ["awc"]
|
client = ["awc"]
|
||||||
|
|
||||||
# brotli encoding, requires c compiler
|
# brotli encoding, requires c compiler
|
||||||
brotli = ["brotli2"]
|
brotli = ["actix-http/brotli2"]
|
||||||
|
|
||||||
# miniz-sys backend for flate2 crate
|
# miniz-sys backend for flate2 crate
|
||||||
flate2-c = ["flate2/miniz-sys"]
|
flate2-c = ["actix-http/flate2-c"]
|
||||||
|
|
||||||
# rust backend for flate2 crate
|
# rust backend for flate2 crate
|
||||||
flate2-rust = ["flate2/rust_backend"]
|
flate2-rust = ["actix-http/flate2-rust"]
|
||||||
|
|
||||||
# sessions feature, session require "ring" crate and c compiler
|
# sessions feature, session require "ring" crate and c compiler
|
||||||
cookies = ["cookie", "actix-http/cookies"]
|
cookies = ["cookie", "actix-http/cookies"]
|
||||||
@ -96,22 +96,20 @@ url = { version="1.7", features=["query_encoding"] }
|
|||||||
# cookies support
|
# cookies support
|
||||||
cookie = { version="0.11", features=["secure", "percent-encode"], optional = true }
|
cookie = { version="0.11", features=["secure", "percent-encode"], optional = true }
|
||||||
|
|
||||||
# compression
|
|
||||||
brotli2 = { version="^0.3.2", optional = true }
|
|
||||||
flate2 = { version="^1.0.2", optional = true, default-features = false }
|
|
||||||
|
|
||||||
# ssl support
|
# ssl support
|
||||||
native-tls = { version="0.2", optional = true }
|
native-tls = { version="0.2", optional = true }
|
||||||
openssl = { version="0.10", optional = true }
|
openssl = { version="0.10", optional = true }
|
||||||
# rustls = { version = "^0.15", optional = true }
|
# rustls = { version = "^0.15", optional = true }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
actix-http = { path = "actix-http", features=["ssl"] }
|
actix-http = { path = "actix-http", features=["ssl", "brotli", "flate2-c"] }
|
||||||
actix-http-test = { path = "test-server", features=["ssl"] }
|
actix-http-test = { path = "test-server", features=["ssl"] }
|
||||||
rand = "0.6"
|
rand = "0.6"
|
||||||
env_logger = "0.6"
|
env_logger = "0.6"
|
||||||
serde_derive = "1.0"
|
serde_derive = "1.0"
|
||||||
tokio-timer = "0.2.8"
|
tokio-timer = "0.2.8"
|
||||||
|
brotli2 = { version="^0.3.2" }
|
||||||
|
flate2 = { version="^1.0.2" }
|
||||||
|
|
||||||
[profile.release]
|
[profile.release]
|
||||||
lto = true
|
lto = true
|
||||||
|
@ -36,6 +36,15 @@ ssl = ["openssl", "actix-connect/ssl"]
|
|||||||
# cookies integration
|
# cookies integration
|
||||||
cookies = ["cookie"]
|
cookies = ["cookie"]
|
||||||
|
|
||||||
|
# brotli encoding, requires c compiler
|
||||||
|
brotli = ["brotli2"]
|
||||||
|
|
||||||
|
# miniz-sys backend for flate2 crate
|
||||||
|
flate2-c = ["flate2/miniz-sys"]
|
||||||
|
|
||||||
|
# rust backend for flate2 crate
|
||||||
|
flate2-rust = ["flate2/rust_backend"]
|
||||||
|
|
||||||
# failure integration. actix does not use failure anymore
|
# failure integration. actix does not use failure anymore
|
||||||
fail = ["failure"]
|
fail = ["failure"]
|
||||||
|
|
||||||
@ -77,6 +86,10 @@ tokio-timer = "0.2"
|
|||||||
tokio-current-thread = "0.1"
|
tokio-current-thread = "0.1"
|
||||||
trust-dns-resolver = { version="0.11.0-alpha.2", default-features = false }
|
trust-dns-resolver = { version="0.11.0-alpha.2", default-features = false }
|
||||||
|
|
||||||
|
# compression
|
||||||
|
brotli2 = { version="^0.3.2", optional = true }
|
||||||
|
flate2 = { version="^1.0.2", optional = true, default-features = false }
|
||||||
|
|
||||||
# optional deps
|
# optional deps
|
||||||
cookie = { version="0.11", features=["percent-encode"], optional = true }
|
cookie = { version="0.11", features=["percent-encode"], optional = true }
|
||||||
failure = { version = "0.1.5", optional = true }
|
failure = { version = "0.1.5", optional = true }
|
||||||
|
191
actix-http/src/encoding/decoder.rs
Normal file
191
actix-http/src/encoding/decoder.rs
Normal file
@ -0,0 +1,191 @@
|
|||||||
|
use std::io::{self, Write};
|
||||||
|
|
||||||
|
use bytes::Bytes;
|
||||||
|
use futures::{Async, Poll, Stream};
|
||||||
|
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
use brotli2::write::BrotliDecoder;
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
use flate2::write::{GzDecoder, ZlibDecoder};
|
||||||
|
|
||||||
|
use super::Writer;
|
||||||
|
use crate::error::PayloadError;
|
||||||
|
use crate::http::header::{ContentEncoding, HeaderMap, CONTENT_ENCODING};
|
||||||
|
|
||||||
|
pub struct Decoder<T> {
|
||||||
|
stream: T,
|
||||||
|
decoder: Option<ContentDecoder>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Decoder<T>
|
||||||
|
where
|
||||||
|
T: Stream<Item = Bytes, Error = PayloadError>,
|
||||||
|
{
|
||||||
|
pub fn new(stream: T, encoding: ContentEncoding) -> Self {
|
||||||
|
let decoder = match encoding {
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
ContentEncoding::Br => Some(ContentDecoder::Br(Box::new(
|
||||||
|
BrotliDecoder::new(Writer::new()),
|
||||||
|
))),
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoding::Deflate => Some(ContentDecoder::Deflate(Box::new(
|
||||||
|
ZlibDecoder::new(Writer::new()),
|
||||||
|
))),
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoding::Gzip => Some(ContentDecoder::Gzip(Box::new(
|
||||||
|
GzDecoder::new(Writer::new()),
|
||||||
|
))),
|
||||||
|
_ => None,
|
||||||
|
};
|
||||||
|
Decoder { stream, decoder }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn from_headers(headers: &HeaderMap, stream: T) -> Self {
|
||||||
|
// check content-encoding
|
||||||
|
let encoding = if let Some(enc) = headers.get(CONTENT_ENCODING) {
|
||||||
|
if let Ok(enc) = enc.to_str() {
|
||||||
|
ContentEncoding::from(enc)
|
||||||
|
} else {
|
||||||
|
ContentEncoding::Identity
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ContentEncoding::Identity
|
||||||
|
};
|
||||||
|
|
||||||
|
Self::new(stream, encoding)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Stream for Decoder<T>
|
||||||
|
where
|
||||||
|
T: Stream<Item = Bytes, Error = PayloadError>,
|
||||||
|
{
|
||||||
|
type Item = Bytes;
|
||||||
|
type Error = PayloadError;
|
||||||
|
|
||||||
|
fn poll(&mut self) -> Poll<Option<Self::Item>, Self::Error> {
|
||||||
|
loop {
|
||||||
|
match self.stream.poll()? {
|
||||||
|
Async::Ready(Some(chunk)) => {
|
||||||
|
if let Some(ref mut decoder) = self.decoder {
|
||||||
|
match decoder.feed_data(chunk) {
|
||||||
|
Ok(Some(chunk)) => return Ok(Async::Ready(Some(chunk))),
|
||||||
|
Ok(None) => continue,
|
||||||
|
Err(e) => return Err(e.into()),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Async::Ready(None) => {
|
||||||
|
return if let Some(mut decoder) = self.decoder.take() {
|
||||||
|
match decoder.feed_eof() {
|
||||||
|
Ok(chunk) => Ok(Async::Ready(chunk)),
|
||||||
|
Err(e) => Err(e.into()),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Ok(Async::Ready(None))
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Async::NotReady => break,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(Async::NotReady)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum ContentDecoder {
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
Deflate(Box<ZlibDecoder<Writer>>),
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
Gzip(Box<GzDecoder<Writer>>),
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
Br(Box<BrotliDecoder<Writer>>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ContentDecoder {
|
||||||
|
fn feed_eof(&mut self) -> io::Result<Option<Bytes>> {
|
||||||
|
match self {
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
ContentDecoder::Br(ref mut decoder) => match decoder.finish() {
|
||||||
|
Ok(mut writer) => {
|
||||||
|
let b = writer.take();
|
||||||
|
if !b.is_empty() {
|
||||||
|
Ok(Some(b))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => Err(e),
|
||||||
|
},
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentDecoder::Gzip(ref mut decoder) => match decoder.try_finish() {
|
||||||
|
Ok(_) => {
|
||||||
|
let b = decoder.get_mut().take();
|
||||||
|
if !b.is_empty() {
|
||||||
|
Ok(Some(b))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => Err(e),
|
||||||
|
},
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentDecoder::Deflate(ref mut decoder) => match decoder.try_finish() {
|
||||||
|
Ok(_) => {
|
||||||
|
let b = decoder.get_mut().take();
|
||||||
|
if !b.is_empty() {
|
||||||
|
Ok(Some(b))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => Err(e),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn feed_data(&mut self, data: Bytes) -> io::Result<Option<Bytes>> {
|
||||||
|
match self {
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentDecoder::Br(ref mut decoder) => match decoder.write_all(&data) {
|
||||||
|
Ok(_) => {
|
||||||
|
decoder.flush()?;
|
||||||
|
let b = decoder.get_mut().take();
|
||||||
|
if !b.is_empty() {
|
||||||
|
Ok(Some(b))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => Err(e),
|
||||||
|
},
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentDecoder::Gzip(ref mut decoder) => match decoder.write_all(&data) {
|
||||||
|
Ok(_) => {
|
||||||
|
decoder.flush()?;
|
||||||
|
let b = decoder.get_mut().take();
|
||||||
|
if !b.is_empty() {
|
||||||
|
Ok(Some(b))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => Err(e),
|
||||||
|
},
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentDecoder::Deflate(ref mut decoder) => match decoder.write_all(&data) {
|
||||||
|
Ok(_) => {
|
||||||
|
decoder.flush()?;
|
||||||
|
let b = decoder.get_mut().take();
|
||||||
|
if !b.is_empty() {
|
||||||
|
Ok(Some(b))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Err(e) => Err(e),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
234
actix-http/src/encoding/encoder.rs
Normal file
234
actix-http/src/encoding/encoder.rs
Normal file
@ -0,0 +1,234 @@
|
|||||||
|
//! Stream encoder
|
||||||
|
use std::io::{self, Write};
|
||||||
|
|
||||||
|
use bytes::Bytes;
|
||||||
|
use futures::{Async, Poll};
|
||||||
|
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
use brotli2::write::BrotliEncoder;
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
use flate2::write::{GzEncoder, ZlibEncoder};
|
||||||
|
|
||||||
|
use crate::body::{Body, BodyLength, MessageBody, ResponseBody};
|
||||||
|
use crate::http::header::{ContentEncoding, CONTENT_ENCODING};
|
||||||
|
use crate::http::{HeaderValue, HttpTryFrom, StatusCode};
|
||||||
|
use crate::{Error, Head, ResponseHead};
|
||||||
|
|
||||||
|
use super::Writer;
|
||||||
|
|
||||||
|
pub struct Encoder<B> {
|
||||||
|
body: EncoderBody<B>,
|
||||||
|
encoder: Option<ContentEncoder>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<B: MessageBody> Encoder<B> {
|
||||||
|
pub fn response(
|
||||||
|
encoding: ContentEncoding,
|
||||||
|
head: &mut ResponseHead,
|
||||||
|
body: ResponseBody<B>,
|
||||||
|
) -> ResponseBody<Encoder<B>> {
|
||||||
|
let has_ce = head.headers().contains_key(CONTENT_ENCODING);
|
||||||
|
match body {
|
||||||
|
ResponseBody::Other(b) => match b {
|
||||||
|
Body::None => ResponseBody::Other(Body::None),
|
||||||
|
Body::Empty => ResponseBody::Other(Body::Empty),
|
||||||
|
Body::Bytes(buf) => {
|
||||||
|
if !(has_ce
|
||||||
|
|| encoding == ContentEncoding::Identity
|
||||||
|
|| encoding == ContentEncoding::Auto)
|
||||||
|
{
|
||||||
|
let mut enc = ContentEncoder::encoder(encoding).unwrap();
|
||||||
|
|
||||||
|
// TODO return error!
|
||||||
|
let _ = enc.write(buf.as_ref());
|
||||||
|
let body = enc.finish().unwrap();
|
||||||
|
update_head(encoding, head);
|
||||||
|
ResponseBody::Other(Body::Bytes(body))
|
||||||
|
} else {
|
||||||
|
ResponseBody::Other(Body::Bytes(buf))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Body::Message(stream) => {
|
||||||
|
if has_ce || head.status == StatusCode::SWITCHING_PROTOCOLS {
|
||||||
|
ResponseBody::Body(Encoder {
|
||||||
|
body: EncoderBody::Other(stream),
|
||||||
|
encoder: None,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
update_head(encoding, head);
|
||||||
|
head.no_chunking = false;
|
||||||
|
ResponseBody::Body(Encoder {
|
||||||
|
body: EncoderBody::Other(stream),
|
||||||
|
encoder: ContentEncoder::encoder(encoding),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
ResponseBody::Body(stream) => {
|
||||||
|
if has_ce || head.status == StatusCode::SWITCHING_PROTOCOLS {
|
||||||
|
ResponseBody::Body(Encoder {
|
||||||
|
body: EncoderBody::Body(stream),
|
||||||
|
encoder: None,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
update_head(encoding, head);
|
||||||
|
head.no_chunking = false;
|
||||||
|
ResponseBody::Body(Encoder {
|
||||||
|
body: EncoderBody::Body(stream),
|
||||||
|
encoder: ContentEncoder::encoder(encoding),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum EncoderBody<B> {
|
||||||
|
Body(B),
|
||||||
|
Other(Box<dyn MessageBody>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<B: MessageBody> MessageBody for Encoder<B> {
|
||||||
|
fn length(&self) -> BodyLength {
|
||||||
|
if self.encoder.is_none() {
|
||||||
|
match self.body {
|
||||||
|
EncoderBody::Body(ref b) => b.length(),
|
||||||
|
EncoderBody::Other(ref b) => b.length(),
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
BodyLength::Stream
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn poll_next(&mut self) -> Poll<Option<Bytes>, Error> {
|
||||||
|
loop {
|
||||||
|
let result = match self.body {
|
||||||
|
EncoderBody::Body(ref mut b) => b.poll_next()?,
|
||||||
|
EncoderBody::Other(ref mut b) => b.poll_next()?,
|
||||||
|
};
|
||||||
|
match result {
|
||||||
|
Async::NotReady => return Ok(Async::NotReady),
|
||||||
|
Async::Ready(Some(chunk)) => {
|
||||||
|
if let Some(ref mut encoder) = self.encoder {
|
||||||
|
if encoder.write(&chunk)? {
|
||||||
|
return Ok(Async::Ready(Some(encoder.take())));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return Ok(Async::Ready(Some(chunk)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Async::Ready(None) => {
|
||||||
|
if let Some(encoder) = self.encoder.take() {
|
||||||
|
let chunk = encoder.finish()?;
|
||||||
|
if chunk.is_empty() {
|
||||||
|
return Ok(Async::Ready(None));
|
||||||
|
} else {
|
||||||
|
return Ok(Async::Ready(Some(chunk)));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return Ok(Async::Ready(None));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn update_head(encoding: ContentEncoding, head: &mut ResponseHead) {
|
||||||
|
head.headers_mut().insert(
|
||||||
|
CONTENT_ENCODING,
|
||||||
|
HeaderValue::try_from(Bytes::from_static(encoding.as_str().as_bytes())).unwrap(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
enum ContentEncoder {
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
Deflate(ZlibEncoder<Writer>),
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
Gzip(GzEncoder<Writer>),
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
Br(BrotliEncoder<Writer>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ContentEncoder {
|
||||||
|
fn encoder(encoding: ContentEncoding) -> Option<Self> {
|
||||||
|
match encoding {
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoding::Deflate => Some(ContentEncoder::Deflate(ZlibEncoder::new(
|
||||||
|
Writer::new(),
|
||||||
|
flate2::Compression::fast(),
|
||||||
|
))),
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoding::Gzip => Some(ContentEncoder::Gzip(GzEncoder::new(
|
||||||
|
Writer::new(),
|
||||||
|
flate2::Compression::fast(),
|
||||||
|
))),
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
ContentEncoding::Br => {
|
||||||
|
Some(ContentEncoder::Br(BrotliEncoder::new(Writer::new(), 3)))
|
||||||
|
}
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub(crate) fn take(&mut self) -> Bytes {
|
||||||
|
match *self {
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
ContentEncoder::Br(ref mut encoder) => encoder.get_mut().take(),
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoder::Deflate(ref mut encoder) => encoder.get_mut().take(),
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoder::Gzip(ref mut encoder) => encoder.get_mut().take(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn finish(self) -> Result<Bytes, io::Error> {
|
||||||
|
match self {
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
ContentEncoder::Br(encoder) => match encoder.finish() {
|
||||||
|
Ok(writer) => Ok(writer.buf.freeze()),
|
||||||
|
Err(err) => Err(err),
|
||||||
|
},
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoder::Gzip(encoder) => match encoder.finish() {
|
||||||
|
Ok(writer) => Ok(writer.buf.freeze()),
|
||||||
|
Err(err) => Err(err),
|
||||||
|
},
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoder::Deflate(encoder) => match encoder.finish() {
|
||||||
|
Ok(writer) => Ok(writer.buf.freeze()),
|
||||||
|
Err(err) => Err(err),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn write(&mut self, data: &[u8]) -> Result<bool, io::Error> {
|
||||||
|
match *self {
|
||||||
|
#[cfg(feature = "brotli")]
|
||||||
|
ContentEncoder::Br(ref mut encoder) => match encoder.write_all(data) {
|
||||||
|
Ok(_) => Ok(!encoder.get_ref().buf.is_empty()),
|
||||||
|
Err(err) => {
|
||||||
|
trace!("Error decoding br encoding: {}", err);
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoder::Gzip(ref mut encoder) => match encoder.write_all(data) {
|
||||||
|
Ok(_) => Ok(!encoder.get_ref().buf.is_empty()),
|
||||||
|
Err(err) => {
|
||||||
|
trace!("Error decoding gzip encoding: {}", err);
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
||||||
|
ContentEncoder::Deflate(ref mut encoder) => match encoder.write_all(data) {
|
||||||
|
Ok(_) => Ok(!encoder.get_ref().buf.is_empty()),
|
||||||
|
Err(err) => {
|
||||||
|
trace!("Error decoding deflate encoding: {}", err);
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
35
actix-http/src/encoding/mod.rs
Normal file
35
actix-http/src/encoding/mod.rs
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
//! Content-Encoding support
|
||||||
|
use std::io;
|
||||||
|
|
||||||
|
use bytes::{Bytes, BytesMut};
|
||||||
|
|
||||||
|
mod decoder;
|
||||||
|
mod encoder;
|
||||||
|
|
||||||
|
pub use self::decoder::Decoder;
|
||||||
|
pub use self::encoder::Encoder;
|
||||||
|
|
||||||
|
pub(self) struct Writer {
|
||||||
|
buf: BytesMut,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Writer {
|
||||||
|
fn new() -> Writer {
|
||||||
|
Writer {
|
||||||
|
buf: BytesMut::with_capacity(8192),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fn take(&mut self) -> Bytes {
|
||||||
|
self.buf.take().freeze()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl io::Write for Writer {
|
||||||
|
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
||||||
|
self.buf.extend_from_slice(buf);
|
||||||
|
Ok(buf.len())
|
||||||
|
}
|
||||||
|
fn flush(&mut self) -> io::Result<()> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
@ -12,6 +12,7 @@ pub mod body;
|
|||||||
mod builder;
|
mod builder;
|
||||||
pub mod client;
|
pub mod client;
|
||||||
mod config;
|
mod config;
|
||||||
|
pub mod encoding;
|
||||||
mod extensions;
|
mod extensions;
|
||||||
mod header;
|
mod header;
|
||||||
mod helpers;
|
mod helpers;
|
||||||
|
@ -193,10 +193,10 @@ where
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Register a request modifier. It can modify any request parameters
|
/// Register a request modifier. It can modify any request parameters
|
||||||
/// including payload stream type.
|
/// including request payload type.
|
||||||
pub fn chain<C, F, P1>(
|
pub fn chain<C, F, P1>(
|
||||||
self,
|
self,
|
||||||
chain: C,
|
chain: F,
|
||||||
) -> App<
|
) -> App<
|
||||||
P1,
|
P1,
|
||||||
impl NewService<
|
impl NewService<
|
||||||
|
@ -380,7 +380,7 @@ impl<P> Service for AppRouting<P> {
|
|||||||
} else if let Some(ref mut default) = self.default {
|
} else if let Some(ref mut default) = self.default {
|
||||||
Either::A(default.call(req))
|
Either::A(default.call(req))
|
||||||
} else {
|
} else {
|
||||||
let req = req.into_request();
|
let req = req.into_parts().0;
|
||||||
Either::B(ok(ServiceResponse::new(req, Response::NotFound().finish())))
|
Either::B(ok(ServiceResponse::new(req, Response::NotFound().finish())))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,25 +1,14 @@
|
|||||||
/// `Middleware` for compressing response body.
|
//! `Middleware` for compressing response body.
|
||||||
use std::io::Write;
|
use std::cmp;
|
||||||
use std::marker::PhantomData;
|
use std::marker::PhantomData;
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
use std::{cmp, fmt, io};
|
|
||||||
|
|
||||||
use actix_http::body::{Body, BodyLength, MessageBody, ResponseBody};
|
use actix_http::body::MessageBody;
|
||||||
use actix_http::http::header::{
|
use actix_http::encoding::Encoder;
|
||||||
ContentEncoding, HeaderValue, ACCEPT_ENCODING, CONTENT_ENCODING,
|
use actix_http::http::header::{ContentEncoding, ACCEPT_ENCODING};
|
||||||
};
|
|
||||||
use actix_http::http::{HttpTryFrom, StatusCode};
|
|
||||||
use actix_http::{Error, Head, ResponseHead};
|
|
||||||
use actix_service::{Service, Transform};
|
use actix_service::{Service, Transform};
|
||||||
use bytes::{Bytes, BytesMut};
|
|
||||||
use futures::future::{ok, FutureResult};
|
use futures::future::{ok, FutureResult};
|
||||||
use futures::{Async, Future, Poll};
|
use futures::{Async, Future, Poll};
|
||||||
use log::trace;
|
|
||||||
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
use brotli2::write::BrotliEncoder;
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
use flate2::write::{GzEncoder, ZlibEncoder};
|
|
||||||
|
|
||||||
use crate::service::{ServiceRequest, ServiceResponse};
|
use crate::service::{ServiceRequest, ServiceResponse};
|
||||||
|
|
||||||
@ -130,266 +119,11 @@ where
|
|||||||
let resp = futures::try_ready!(self.fut.poll());
|
let resp = futures::try_ready!(self.fut.poll());
|
||||||
|
|
||||||
Ok(Async::Ready(resp.map_body(move |head, body| {
|
Ok(Async::Ready(resp.map_body(move |head, body| {
|
||||||
Encoder::body(self.encoding, head, body)
|
Encoder::response(self.encoding, head, body)
|
||||||
})))
|
})))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
enum EncoderBody<B> {
|
|
||||||
Body(B),
|
|
||||||
Other(Box<dyn MessageBody>),
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Encoder<B> {
|
|
||||||
body: EncoderBody<B>,
|
|
||||||
encoder: Option<ContentEncoder>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<B: MessageBody> MessageBody for Encoder<B> {
|
|
||||||
fn length(&self) -> BodyLength {
|
|
||||||
if self.encoder.is_none() {
|
|
||||||
match self.body {
|
|
||||||
EncoderBody::Body(ref b) => b.length(),
|
|
||||||
EncoderBody::Other(ref b) => b.length(),
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
BodyLength::Stream
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn poll_next(&mut self) -> Poll<Option<Bytes>, Error> {
|
|
||||||
loop {
|
|
||||||
let result = match self.body {
|
|
||||||
EncoderBody::Body(ref mut b) => b.poll_next()?,
|
|
||||||
EncoderBody::Other(ref mut b) => b.poll_next()?,
|
|
||||||
};
|
|
||||||
match result {
|
|
||||||
Async::NotReady => return Ok(Async::NotReady),
|
|
||||||
Async::Ready(Some(chunk)) => {
|
|
||||||
if let Some(ref mut encoder) = self.encoder {
|
|
||||||
if encoder.write(&chunk)? {
|
|
||||||
return Ok(Async::Ready(Some(encoder.take())));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return Ok(Async::Ready(Some(chunk)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Async::Ready(None) => {
|
|
||||||
if let Some(encoder) = self.encoder.take() {
|
|
||||||
let chunk = encoder.finish()?;
|
|
||||||
if chunk.is_empty() {
|
|
||||||
return Ok(Async::Ready(None));
|
|
||||||
} else {
|
|
||||||
return Ok(Async::Ready(Some(chunk)));
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return Ok(Async::Ready(None));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn update_head(encoding: ContentEncoding, head: &mut ResponseHead) {
|
|
||||||
head.headers_mut().insert(
|
|
||||||
CONTENT_ENCODING,
|
|
||||||
HeaderValue::try_from(Bytes::from_static(encoding.as_str().as_bytes())).unwrap(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<B: MessageBody> Encoder<B> {
|
|
||||||
fn body(
|
|
||||||
encoding: ContentEncoding,
|
|
||||||
head: &mut ResponseHead,
|
|
||||||
body: ResponseBody<B>,
|
|
||||||
) -> ResponseBody<Encoder<B>> {
|
|
||||||
let has_ce = head.headers().contains_key(CONTENT_ENCODING);
|
|
||||||
match body {
|
|
||||||
ResponseBody::Other(b) => match b {
|
|
||||||
Body::None => ResponseBody::Other(Body::None),
|
|
||||||
Body::Empty => ResponseBody::Other(Body::Empty),
|
|
||||||
Body::Bytes(buf) => {
|
|
||||||
if !(has_ce
|
|
||||||
|| encoding == ContentEncoding::Identity
|
|
||||||
|| encoding == ContentEncoding::Auto)
|
|
||||||
{
|
|
||||||
let mut enc = ContentEncoder::encoder(encoding).unwrap();
|
|
||||||
|
|
||||||
// TODO return error!
|
|
||||||
let _ = enc.write(buf.as_ref());
|
|
||||||
let body = enc.finish().unwrap();
|
|
||||||
update_head(encoding, head);
|
|
||||||
ResponseBody::Other(Body::Bytes(body))
|
|
||||||
} else {
|
|
||||||
ResponseBody::Other(Body::Bytes(buf))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Body::Message(stream) => {
|
|
||||||
if has_ce || head.status == StatusCode::SWITCHING_PROTOCOLS {
|
|
||||||
ResponseBody::Body(Encoder {
|
|
||||||
body: EncoderBody::Other(stream),
|
|
||||||
encoder: None,
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
update_head(encoding, head);
|
|
||||||
head.no_chunking = false;
|
|
||||||
ResponseBody::Body(Encoder {
|
|
||||||
body: EncoderBody::Other(stream),
|
|
||||||
encoder: ContentEncoder::encoder(encoding),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
ResponseBody::Body(stream) => {
|
|
||||||
if has_ce || head.status == StatusCode::SWITCHING_PROTOCOLS {
|
|
||||||
ResponseBody::Body(Encoder {
|
|
||||||
body: EncoderBody::Body(stream),
|
|
||||||
encoder: None,
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
update_head(encoding, head);
|
|
||||||
head.no_chunking = false;
|
|
||||||
ResponseBody::Body(Encoder {
|
|
||||||
body: EncoderBody::Body(stream),
|
|
||||||
encoder: ContentEncoder::encoder(encoding),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) struct Writer {
|
|
||||||
buf: BytesMut,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Writer {
|
|
||||||
fn new() -> Writer {
|
|
||||||
Writer {
|
|
||||||
buf: BytesMut::with_capacity(8192),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
fn take(&mut self) -> Bytes {
|
|
||||||
self.buf.take().freeze()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl io::Write for Writer {
|
|
||||||
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
|
||||||
self.buf.extend_from_slice(buf);
|
|
||||||
Ok(buf.len())
|
|
||||||
}
|
|
||||||
fn flush(&mut self) -> io::Result<()> {
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) enum ContentEncoder {
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
Deflate(ZlibEncoder<Writer>),
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
Gzip(GzEncoder<Writer>),
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
Br(BrotliEncoder<Writer>),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Debug for ContentEncoder {
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(_) => writeln!(f, "ContentEncoder(Brotli)"),
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoder::Deflate(_) => writeln!(f, "ContentEncoder(Deflate)"),
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoder::Gzip(_) => writeln!(f, "ContentEncoder(Gzip)"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ContentEncoder {
|
|
||||||
fn encoder(encoding: ContentEncoding) -> Option<Self> {
|
|
||||||
match encoding {
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoding::Deflate => Some(ContentEncoder::Deflate(ZlibEncoder::new(
|
|
||||||
Writer::new(),
|
|
||||||
flate2::Compression::fast(),
|
|
||||||
))),
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoding::Gzip => Some(ContentEncoder::Gzip(GzEncoder::new(
|
|
||||||
Writer::new(),
|
|
||||||
flate2::Compression::fast(),
|
|
||||||
))),
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoding::Br => {
|
|
||||||
Some(ContentEncoder::Br(BrotliEncoder::new(Writer::new(), 3)))
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub(crate) fn take(&mut self) -> Bytes {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(ref mut encoder) => encoder.get_mut().take(),
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoder::Deflate(ref mut encoder) => encoder.get_mut().take(),
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoder::Gzip(ref mut encoder) => encoder.get_mut().take(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn finish(self) -> Result<Bytes, io::Error> {
|
|
||||||
match self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(encoder) => match encoder.finish() {
|
|
||||||
Ok(writer) => Ok(writer.buf.freeze()),
|
|
||||||
Err(err) => Err(err),
|
|
||||||
},
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoder::Gzip(encoder) => match encoder.finish() {
|
|
||||||
Ok(writer) => Ok(writer.buf.freeze()),
|
|
||||||
Err(err) => Err(err),
|
|
||||||
},
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoder::Deflate(encoder) => match encoder.finish() {
|
|
||||||
Ok(writer) => Ok(writer.buf.freeze()),
|
|
||||||
Err(err) => Err(err),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn write(&mut self, data: &[u8]) -> Result<bool, io::Error> {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(ref mut encoder) => match encoder.write_all(data) {
|
|
||||||
Ok(_) => Ok(!encoder.get_ref().buf.is_empty()),
|
|
||||||
Err(err) => {
|
|
||||||
trace!("Error decoding br encoding: {}", err);
|
|
||||||
Err(err)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoder::Gzip(ref mut encoder) => match encoder.write_all(data) {
|
|
||||||
Ok(_) => Ok(!encoder.get_ref().buf.is_empty()),
|
|
||||||
Err(err) => {
|
|
||||||
trace!("Error decoding gzip encoding: {}", err);
|
|
||||||
Err(err)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
#[cfg(any(feature = "flate2-c", feature = "flate2-rust"))]
|
|
||||||
ContentEncoder::Deflate(ref mut encoder) => match encoder.write_all(data) {
|
|
||||||
Ok(_) => Ok(!encoder.get_ref().buf.is_empty()),
|
|
||||||
Err(err) => {
|
|
||||||
trace!("Error decoding deflate encoding: {}", err);
|
|
||||||
Err(err)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct AcceptEncoding {
|
struct AcceptEncoding {
|
||||||
encoding: ContentEncoding,
|
encoding: ContentEncoding,
|
||||||
quality: f64,
|
quality: f64,
|
||||||
|
60
src/middleware/decompress.rs
Normal file
60
src/middleware/decompress.rs
Normal file
@ -0,0 +1,60 @@
|
|||||||
|
//! Chain service for decompressing request payload.
|
||||||
|
use std::marker::PhantomData;
|
||||||
|
|
||||||
|
use actix_http::encoding::Decoder;
|
||||||
|
use actix_service::{NewService, Service};
|
||||||
|
use bytes::Bytes;
|
||||||
|
use futures::future::{ok, FutureResult};
|
||||||
|
use futures::{Async, Poll, Stream};
|
||||||
|
|
||||||
|
use crate::dev::Payload;
|
||||||
|
use crate::error::{Error, PayloadError};
|
||||||
|
use crate::service::ServiceRequest;
|
||||||
|
use crate::HttpMessage;
|
||||||
|
|
||||||
|
pub struct Decompress<P>(PhantomData<P>);
|
||||||
|
|
||||||
|
impl<P> Decompress<P>
|
||||||
|
where
|
||||||
|
P: Stream<Item = Bytes, Error = PayloadError>,
|
||||||
|
{
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Decompress(PhantomData)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<P> NewService for Decompress<P>
|
||||||
|
where
|
||||||
|
P: Stream<Item = Bytes, Error = PayloadError>,
|
||||||
|
{
|
||||||
|
type Request = ServiceRequest<P>;
|
||||||
|
type Response = ServiceRequest<Decoder<Payload<P>>>;
|
||||||
|
type Error = Error;
|
||||||
|
type InitError = ();
|
||||||
|
type Service = Decompress<P>;
|
||||||
|
type Future = FutureResult<Self::Service, Self::InitError>;
|
||||||
|
|
||||||
|
fn new_service(&self, _: &()) -> Self::Future {
|
||||||
|
ok(Decompress(PhantomData))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<P> Service for Decompress<P>
|
||||||
|
where
|
||||||
|
P: Stream<Item = Bytes, Error = PayloadError>,
|
||||||
|
{
|
||||||
|
type Request = ServiceRequest<P>;
|
||||||
|
type Response = ServiceRequest<Decoder<Payload<P>>>;
|
||||||
|
type Error = Error;
|
||||||
|
type Future = FutureResult<Self::Response, Self::Error>;
|
||||||
|
|
||||||
|
fn poll_ready(&mut self) -> Poll<(), Self::Error> {
|
||||||
|
Ok(Async::Ready(()))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn call(&mut self, req: ServiceRequest<P>) -> Self::Future {
|
||||||
|
let (req, payload) = req.into_parts();
|
||||||
|
let payload = Decoder::from_headers(req.headers(), payload);
|
||||||
|
ok(ServiceRequest::from_parts(req, Payload::Stream(payload)))
|
||||||
|
}
|
||||||
|
}
|
@ -4,6 +4,11 @@ mod compress;
|
|||||||
#[cfg(any(feature = "brotli", feature = "flate2"))]
|
#[cfg(any(feature = "brotli", feature = "flate2"))]
|
||||||
pub use self::compress::Compress;
|
pub use self::compress::Compress;
|
||||||
|
|
||||||
|
#[cfg(any(feature = "brotli", feature = "flate2"))]
|
||||||
|
mod decompress;
|
||||||
|
#[cfg(any(feature = "brotli", feature = "flate2"))]
|
||||||
|
pub use self::decompress::Decompress;
|
||||||
|
|
||||||
pub mod cors;
|
pub mod cors;
|
||||||
mod defaultheaders;
|
mod defaultheaders;
|
||||||
pub mod errhandlers;
|
pub mod errhandlers;
|
||||||
|
@ -507,7 +507,7 @@ impl<P> Service for ResourceService<P> {
|
|||||||
if let Some(ref mut default) = self.default {
|
if let Some(ref mut default) = self.default {
|
||||||
Either::B(Either::A(default.call(req)))
|
Either::B(Either::A(default.call(req)))
|
||||||
} else {
|
} else {
|
||||||
let req = req.into_request();
|
let req = req.into_parts().0;
|
||||||
Either::B(Either::B(ok(ServiceResponse::new(
|
Either::B(Either::B(ok(ServiceResponse::new(
|
||||||
req,
|
req,
|
||||||
Response::MethodNotAllowed().finish(),
|
Response::MethodNotAllowed().finish(),
|
||||||
|
@ -489,7 +489,7 @@ impl<P> Service for ScopeService<P> {
|
|||||||
} else if let Some(ref mut default) = self.default {
|
} else if let Some(ref mut default) = self.default {
|
||||||
Either::A(default.call(req))
|
Either::A(default.call(req))
|
||||||
} else {
|
} else {
|
||||||
let req = req.into_request();
|
let req = req.into_parts().0;
|
||||||
Either::B(ok(ServiceResponse::new(req, Response::NotFound().finish())))
|
Either::B(ok(ServiceResponse::new(req, Response::NotFound().finish())))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -69,9 +69,14 @@ impl<P> ServiceRequest<P> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
/// Construct service request from parts
|
||||||
pub fn into_request(self) -> HttpRequest {
|
pub fn from_parts(req: HttpRequest, payload: Payload<P>) -> Self {
|
||||||
self.req
|
ServiceRequest { req, payload }
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Deconstruct request into parts
|
||||||
|
pub fn into_parts(self) -> (HttpRequest, Payload<P>) {
|
||||||
|
(self.req, self.payload)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Create service response
|
/// Create service response
|
||||||
@ -162,11 +167,6 @@ impl<P> ServiceRequest<P> {
|
|||||||
pub fn app_config(&self) -> &AppConfig {
|
pub fn app_config(&self) -> &AppConfig {
|
||||||
self.req.config()
|
self.req.config()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Deconstruct request into parts
|
|
||||||
pub fn into_parts(self) -> (HttpRequest, Payload<P>) {
|
|
||||||
(self.req, self.payload)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<P> Resource<Url> for ServiceRequest<P> {
|
impl<P> Resource<Url> for ServiceRequest<P> {
|
||||||
|
@ -350,7 +350,8 @@ impl TestRequest {
|
|||||||
Rc::new(self.rmap),
|
Rc::new(self.rmap),
|
||||||
AppConfig::new(self.config),
|
AppConfig::new(self.config),
|
||||||
)
|
)
|
||||||
.into_request()
|
.into_parts()
|
||||||
|
.0
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Complete request creation and generate `ServiceFromRequest` instance
|
/// Complete request creation and generate `ServiceFromRequest` instance
|
||||||
|
@ -1,14 +1,16 @@
|
|||||||
use std::io::{Read, Write};
|
use std::io::{Read, Write};
|
||||||
|
|
||||||
use actix_http::http::header::{
|
use actix_http::http::header::{
|
||||||
ContentEncoding, ACCEPT_ENCODING, CONTENT_LENGTH, TRANSFER_ENCODING,
|
ContentEncoding, ACCEPT_ENCODING, CONTENT_ENCODING, CONTENT_LENGTH,
|
||||||
|
TRANSFER_ENCODING,
|
||||||
};
|
};
|
||||||
use actix_http::{h1, Error, Response};
|
use actix_http::{h1, Error, HttpService, Response};
|
||||||
use actix_http_test::TestServer;
|
use actix_http_test::TestServer;
|
||||||
use brotli2::write::BrotliDecoder;
|
use brotli2::write::{BrotliDecoder, BrotliEncoder};
|
||||||
use bytes::Bytes;
|
use bytes::Bytes;
|
||||||
use flate2::read::GzDecoder;
|
use flate2::read::GzDecoder;
|
||||||
use flate2::write::ZlibDecoder;
|
use flate2::write::{GzEncoder, ZlibDecoder, ZlibEncoder};
|
||||||
|
use flate2::Compression;
|
||||||
use futures::stream::once; //Future, Stream
|
use futures::stream::once; //Future, Stream
|
||||||
use rand::{distributions::Alphanumeric, Rng};
|
use rand::{distributions::Alphanumeric, Rng};
|
||||||
|
|
||||||
@ -297,278 +299,246 @@ fn test_body_brotli() {
|
|||||||
assert_eq!(Bytes::from(dec), Bytes::from_static(STR.as_ref()));
|
assert_eq!(Bytes::from(dec), Bytes::from_static(STR.as_ref()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_gzip_encoding() {
|
fn test_gzip_encoding() {
|
||||||
// let mut srv = test::TestServer::new(|app| {
|
let mut srv = TestServer::new(move || {
|
||||||
// app.handler(|req: &HttpRequest| {
|
HttpService::new(
|
||||||
// req.body()
|
App::new().chain(middleware::Decompress::new()).service(
|
||||||
// .and_then(|bytes: Bytes| {
|
web::resource("/")
|
||||||
// Ok(HttpResponse::Ok()
|
.route(web::to(move |body: Bytes| Response::Ok().body(body))),
|
||||||
// .content_encoding(http::ContentEncoding::Identity)
|
),
|
||||||
// .body(bytes))
|
)
|
||||||
// })
|
});
|
||||||
// .responder()
|
|
||||||
// })
|
|
||||||
// });
|
|
||||||
|
|
||||||
// // client request
|
// client request
|
||||||
// let mut e = GzEncoder::new(Vec::new(), Compression::default());
|
let mut e = GzEncoder::new(Vec::new(), Compression::default());
|
||||||
// e.write_all(STR.as_ref()).unwrap();
|
e.write_all(STR.as_ref()).unwrap();
|
||||||
// let enc = e.finish().unwrap();
|
let enc = e.finish().unwrap();
|
||||||
|
|
||||||
// let request = srv
|
let request = srv
|
||||||
// .post()
|
.post()
|
||||||
// .header(http::header::CONTENT_ENCODING, "gzip")
|
.header(CONTENT_ENCODING, "gzip")
|
||||||
// .body(enc.clone())
|
.send_body(enc.clone());
|
||||||
// .unwrap();
|
let mut response = srv.block_on(request).unwrap();
|
||||||
// let response = srv.block_on(request.send()).unwrap();
|
assert!(response.status().is_success());
|
||||||
// assert!(response.status().is_success());
|
|
||||||
|
|
||||||
// // read response
|
// read response
|
||||||
// let bytes = srv.block_on(response.body()).unwrap();
|
let bytes = srv.block_on(HttpMessageBody::new(&mut response)).unwrap();
|
||||||
// assert_eq!(bytes, Bytes::from_static(STR.as_ref()));
|
assert_eq!(bytes, Bytes::from_static(STR.as_ref()));
|
||||||
// }
|
}
|
||||||
|
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_gzip_encoding_large() {
|
fn test_gzip_encoding_large() {
|
||||||
// let data = STR.repeat(10);
|
let data = STR.repeat(10);
|
||||||
// let mut srv = test::TestServer::new(|app| {
|
let mut srv = TestServer::new(move || {
|
||||||
// app.handler(|req: &HttpRequest| {
|
h1::H1Service::new(
|
||||||
// req.body()
|
App::new().chain(middleware::Decompress::new()).service(
|
||||||
// .and_then(|bytes: Bytes| {
|
web::resource("/")
|
||||||
// Ok(HttpResponse::Ok()
|
.route(web::to(move |body: Bytes| Response::Ok().body(body))),
|
||||||
// .content_encoding(http::ContentEncoding::Identity)
|
),
|
||||||
// .body(bytes))
|
)
|
||||||
// })
|
});
|
||||||
// .responder()
|
|
||||||
// })
|
|
||||||
// });
|
|
||||||
|
|
||||||
// // client request
|
// client request
|
||||||
// let mut e = GzEncoder::new(Vec::new(), Compression::default());
|
let mut e = GzEncoder::new(Vec::new(), Compression::default());
|
||||||
// e.write_all(data.as_ref()).unwrap();
|
e.write_all(data.as_ref()).unwrap();
|
||||||
// let enc = e.finish().unwrap();
|
let enc = e.finish().unwrap();
|
||||||
|
|
||||||
// let request = srv
|
let request = srv
|
||||||
// .post()
|
.post()
|
||||||
// .header(http::header::CONTENT_ENCODING, "gzip")
|
.header(CONTENT_ENCODING, "gzip")
|
||||||
// .body(enc.clone())
|
.send_body(enc.clone());
|
||||||
// .unwrap();
|
let mut response = srv.block_on(request).unwrap();
|
||||||
// let response = srv.block_on(request.send()).unwrap();
|
assert!(response.status().is_success());
|
||||||
// assert!(response.status().is_success());
|
|
||||||
|
|
||||||
// // read response
|
// read response
|
||||||
// let bytes = srv.block_on(response.body()).unwrap();
|
let bytes = srv.block_on(HttpMessageBody::new(&mut response)).unwrap();
|
||||||
// assert_eq!(bytes, Bytes::from(data));
|
assert_eq!(bytes, Bytes::from(data));
|
||||||
// }
|
}
|
||||||
|
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_reading_gzip_encoding_large_random() {
|
fn test_reading_gzip_encoding_large_random() {
|
||||||
// let data = rand::thread_rng()
|
let data = rand::thread_rng()
|
||||||
// .sample_iter(&Alphanumeric)
|
.sample_iter(&Alphanumeric)
|
||||||
// .take(60_000)
|
.take(60_000)
|
||||||
// .collect::<String>();
|
.collect::<String>();
|
||||||
|
|
||||||
// let mut srv = test::TestServer::new(|app| {
|
let mut srv = TestServer::new(move || {
|
||||||
// app.handler(|req: &HttpRequest| {
|
HttpService::new(
|
||||||
// req.body()
|
App::new().chain(middleware::Decompress::new()).service(
|
||||||
// .and_then(|bytes: Bytes| {
|
web::resource("/")
|
||||||
// Ok(HttpResponse::Ok()
|
.route(web::to(move |body: Bytes| Response::Ok().body(body))),
|
||||||
// .content_encoding(http::ContentEncoding::Identity)
|
),
|
||||||
// .body(bytes))
|
)
|
||||||
// })
|
});
|
||||||
// .responder()
|
|
||||||
// })
|
|
||||||
// });
|
|
||||||
|
|
||||||
// // client request
|
// client request
|
||||||
// let mut e = GzEncoder::new(Vec::new(), Compression::default());
|
let mut e = GzEncoder::new(Vec::new(), Compression::default());
|
||||||
// e.write_all(data.as_ref()).unwrap();
|
e.write_all(data.as_ref()).unwrap();
|
||||||
// let enc = e.finish().unwrap();
|
let enc = e.finish().unwrap();
|
||||||
|
|
||||||
// let request = srv
|
let request = srv
|
||||||
// .post()
|
.post()
|
||||||
// .header(http::header::CONTENT_ENCODING, "gzip")
|
.header(CONTENT_ENCODING, "gzip")
|
||||||
// .body(enc.clone())
|
.send_body(enc.clone());
|
||||||
// .unwrap();
|
let mut response = srv.block_on(request).unwrap();
|
||||||
// let response = srv.block_on(request.send()).unwrap();
|
assert!(response.status().is_success());
|
||||||
// assert!(response.status().is_success());
|
|
||||||
|
|
||||||
// // read response
|
// read response
|
||||||
// let bytes = srv.block_on(response.body()).unwrap();
|
let bytes = srv.block_on(HttpMessageBody::new(&mut response)).unwrap();
|
||||||
// assert_eq!(bytes.len(), data.len());
|
assert_eq!(bytes.len(), data.len());
|
||||||
// assert_eq!(bytes, Bytes::from(data));
|
assert_eq!(bytes, Bytes::from(data));
|
||||||
// }
|
}
|
||||||
|
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_reading_deflate_encoding() {
|
fn test_reading_deflate_encoding() {
|
||||||
// let mut srv = test::TestServer::new(|app| {
|
let mut srv = TestServer::new(move || {
|
||||||
// app.handler(|req: &HttpRequest| {
|
h1::H1Service::new(
|
||||||
// req.body()
|
App::new().chain(middleware::Decompress::new()).service(
|
||||||
// .and_then(|bytes: Bytes| {
|
web::resource("/")
|
||||||
// Ok(HttpResponse::Ok()
|
.route(web::to(move |body: Bytes| Response::Ok().body(body))),
|
||||||
// .content_encoding(http::ContentEncoding::Identity)
|
),
|
||||||
// .body(bytes))
|
)
|
||||||
// })
|
});
|
||||||
// .responder()
|
|
||||||
// })
|
|
||||||
// });
|
|
||||||
|
|
||||||
// let mut e = ZlibEncoder::new(Vec::new(), Compression::default());
|
let mut e = ZlibEncoder::new(Vec::new(), Compression::default());
|
||||||
// e.write_all(STR.as_ref()).unwrap();
|
e.write_all(STR.as_ref()).unwrap();
|
||||||
// let enc = e.finish().unwrap();
|
let enc = e.finish().unwrap();
|
||||||
|
|
||||||
// // client request
|
// client request
|
||||||
// let request = srv
|
let request = srv
|
||||||
// .post()
|
.post()
|
||||||
// .header(http::header::CONTENT_ENCODING, "deflate")
|
.header(CONTENT_ENCODING, "deflate")
|
||||||
// .body(enc)
|
.send_body(enc.clone());
|
||||||
// .unwrap();
|
let mut response = srv.block_on(request).unwrap();
|
||||||
// let response = srv.block_on(request.send()).unwrap();
|
assert!(response.status().is_success());
|
||||||
// assert!(response.status().is_success());
|
|
||||||
|
|
||||||
// // read response
|
// read response
|
||||||
// let bytes = srv.block_on(response.body()).unwrap();
|
let bytes = srv.block_on(HttpMessageBody::new(&mut response)).unwrap();
|
||||||
// assert_eq!(bytes, Bytes::from_static(STR.as_ref()));
|
assert_eq!(bytes, Bytes::from_static(STR.as_ref()));
|
||||||
// }
|
}
|
||||||
|
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_reading_deflate_encoding_large() {
|
fn test_reading_deflate_encoding_large() {
|
||||||
// let data = STR.repeat(10);
|
let data = STR.repeat(10);
|
||||||
// let mut srv = test::TestServer::new(|app| {
|
let mut srv = TestServer::new(move || {
|
||||||
// app.handler(|req: &HttpRequest| {
|
h1::H1Service::new(
|
||||||
// req.body()
|
App::new().chain(middleware::Decompress::new()).service(
|
||||||
// .and_then(|bytes: Bytes| {
|
web::resource("/")
|
||||||
// Ok(HttpResponse::Ok()
|
.route(web::to(move |body: Bytes| Response::Ok().body(body))),
|
||||||
// .content_encoding(http::ContentEncoding::Identity)
|
),
|
||||||
// .body(bytes))
|
)
|
||||||
// })
|
});
|
||||||
// .responder()
|
|
||||||
// })
|
|
||||||
// });
|
|
||||||
|
|
||||||
// let mut e = ZlibEncoder::new(Vec::new(), Compression::default());
|
let mut e = ZlibEncoder::new(Vec::new(), Compression::default());
|
||||||
// e.write_all(data.as_ref()).unwrap();
|
e.write_all(data.as_ref()).unwrap();
|
||||||
// let enc = e.finish().unwrap();
|
let enc = e.finish().unwrap();
|
||||||
|
|
||||||
// // client request
|
// client request
|
||||||
// let request = srv
|
let request = srv
|
||||||
// .post()
|
.post()
|
||||||
// .header(http::header::CONTENT_ENCODING, "deflate")
|
.header(CONTENT_ENCODING, "deflate")
|
||||||
// .body(enc)
|
.send_body(enc.clone());
|
||||||
// .unwrap();
|
let mut response = srv.block_on(request).unwrap();
|
||||||
// let response = srv.block_on(request.send()).unwrap();
|
assert!(response.status().is_success());
|
||||||
// assert!(response.status().is_success());
|
|
||||||
|
|
||||||
// // read response
|
// read response
|
||||||
// let bytes = srv.block_on(response.body()).unwrap();
|
let bytes = srv.block_on(HttpMessageBody::new(&mut response)).unwrap();
|
||||||
// assert_eq!(bytes, Bytes::from(data));
|
assert_eq!(bytes, Bytes::from(data));
|
||||||
// }
|
}
|
||||||
|
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_reading_deflate_encoding_large_random() {
|
fn test_reading_deflate_encoding_large_random() {
|
||||||
// let data = rand::thread_rng()
|
let data = rand::thread_rng()
|
||||||
// .sample_iter(&Alphanumeric)
|
.sample_iter(&Alphanumeric)
|
||||||
// .take(160_000)
|
.take(160_000)
|
||||||
// .collect::<String>();
|
.collect::<String>();
|
||||||
|
|
||||||
// let mut srv = test::TestServer::new(|app| {
|
let mut srv = TestServer::new(move || {
|
||||||
// app.handler(|req: &HttpRequest| {
|
h1::H1Service::new(
|
||||||
// req.body()
|
App::new().chain(middleware::Decompress::new()).service(
|
||||||
// .and_then(|bytes: Bytes| {
|
web::resource("/")
|
||||||
// Ok(HttpResponse::Ok()
|
.route(web::to(move |body: Bytes| Response::Ok().body(body))),
|
||||||
// .content_encoding(http::ContentEncoding::Identity)
|
),
|
||||||
// .body(bytes))
|
)
|
||||||
// })
|
});
|
||||||
// .responder()
|
|
||||||
// })
|
|
||||||
// });
|
|
||||||
|
|
||||||
// let mut e = ZlibEncoder::new(Vec::new(), Compression::default());
|
let mut e = ZlibEncoder::new(Vec::new(), Compression::default());
|
||||||
// e.write_all(data.as_ref()).unwrap();
|
e.write_all(data.as_ref()).unwrap();
|
||||||
// let enc = e.finish().unwrap();
|
let enc = e.finish().unwrap();
|
||||||
|
|
||||||
// // client request
|
// client request
|
||||||
// let request = srv
|
let request = srv
|
||||||
// .post()
|
.post()
|
||||||
// .header(http::header::CONTENT_ENCODING, "deflate")
|
.header(CONTENT_ENCODING, "deflate")
|
||||||
// .body(enc)
|
.send_body(enc.clone());
|
||||||
// .unwrap();
|
let mut response = srv.block_on(request).unwrap();
|
||||||
// let response = srv.block_on(request.send()).unwrap();
|
assert!(response.status().is_success());
|
||||||
// assert!(response.status().is_success());
|
|
||||||
|
|
||||||
// // read response
|
// read response
|
||||||
// let bytes = srv.execute(response.body()).unwrap();
|
let bytes = srv.block_on(HttpMessageBody::new(&mut response)).unwrap();
|
||||||
// assert_eq!(bytes.len(), data.len());
|
assert_eq!(bytes.len(), data.len());
|
||||||
// assert_eq!(bytes, Bytes::from(data));
|
assert_eq!(bytes, Bytes::from(data));
|
||||||
// }
|
}
|
||||||
|
|
||||||
// #[cfg(feature = "brotli")]
|
#[cfg(feature = "brotli")]
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_brotli_encoding() {
|
fn test_brotli_encoding() {
|
||||||
// let mut srv = test::TestServer::new(|app| {
|
let mut srv = TestServer::new(move || {
|
||||||
// app.handler(|req: &HttpRequest| {
|
h1::H1Service::new(
|
||||||
// req.body()
|
App::new().chain(middleware::Decompress::new()).service(
|
||||||
// .and_then(|bytes: Bytes| {
|
web::resource("/")
|
||||||
// Ok(HttpResponse::Ok()
|
.route(web::to(move |body: Bytes| Response::Ok().body(body))),
|
||||||
// .content_encoding(http::ContentEncoding::Identity)
|
),
|
||||||
// .body(bytes))
|
)
|
||||||
// })
|
});
|
||||||
// .responder()
|
|
||||||
// })
|
|
||||||
// });
|
|
||||||
|
|
||||||
// let mut e = BrotliEncoder::new(Vec::new(), 5);
|
let mut e = BrotliEncoder::new(Vec::new(), 5);
|
||||||
// e.write_all(STR.as_ref()).unwrap();
|
e.write_all(STR.as_ref()).unwrap();
|
||||||
// let enc = e.finish().unwrap();
|
let enc = e.finish().unwrap();
|
||||||
|
|
||||||
// // client request
|
// client request
|
||||||
// let request = srv
|
let request = srv
|
||||||
// .post()
|
.post()
|
||||||
// .header(http::header::CONTENT_ENCODING, "br")
|
.header(CONTENT_ENCODING, "br")
|
||||||
// .body(enc)
|
.send_body(enc.clone());
|
||||||
// .unwrap();
|
let mut response = srv.block_on(request).unwrap();
|
||||||
// let response = srv.execute(request.send()).unwrap();
|
assert!(response.status().is_success());
|
||||||
// assert!(response.status().is_success());
|
|
||||||
|
|
||||||
// // read response
|
// read response
|
||||||
// let bytes = srv.execute(response.body()).unwrap();
|
let bytes = srv.block_on(HttpMessageBody::new(&mut response)).unwrap();
|
||||||
// assert_eq!(bytes, Bytes::from_static(STR.as_ref()));
|
assert_eq!(bytes, Bytes::from_static(STR.as_ref()));
|
||||||
// }
|
}
|
||||||
|
|
||||||
// #[cfg(feature = "brotli")]
|
#[cfg(feature = "brotli")]
|
||||||
// #[test]
|
#[test]
|
||||||
// fn test_brotli_encoding_large() {
|
fn test_brotli_encoding_large() {
|
||||||
// let data = STR.repeat(10);
|
let data = STR.repeat(10);
|
||||||
// let mut srv = test::TestServer::new(|app| {
|
let mut srv = TestServer::new(move || {
|
||||||
// app.handler(|req: &HttpRequest| {
|
h1::H1Service::new(
|
||||||
// req.body()
|
App::new().chain(middleware::Decompress::new()).service(
|
||||||
// .and_then(|bytes: Bytes| {
|
web::resource("/")
|
||||||
// Ok(HttpResponse::Ok()
|
.route(web::to(move |body: Bytes| Response::Ok().body(body))),
|
||||||
// .content_encoding(http::ContentEncoding::Identity)
|
),
|
||||||
// .body(bytes))
|
)
|
||||||
// })
|
});
|
||||||
// .responder()
|
|
||||||
// })
|
|
||||||
// });
|
|
||||||
|
|
||||||
// let mut e = BrotliEncoder::new(Vec::new(), 5);
|
let mut e = BrotliEncoder::new(Vec::new(), 5);
|
||||||
// e.write_all(data.as_ref()).unwrap();
|
e.write_all(data.as_ref()).unwrap();
|
||||||
// let enc = e.finish().unwrap();
|
let enc = e.finish().unwrap();
|
||||||
|
|
||||||
// // client request
|
// client request
|
||||||
// let request = srv
|
let request = srv
|
||||||
// .post()
|
.post()
|
||||||
// .header(http::header::CONTENT_ENCODING, "br")
|
.header(CONTENT_ENCODING, "br")
|
||||||
// .body(enc)
|
.send_body(enc.clone());
|
||||||
// .unwrap();
|
let mut response = srv.block_on(request).unwrap();
|
||||||
// let response = srv.execute(request.send()).unwrap();
|
assert!(response.status().is_success());
|
||||||
// assert!(response.status().is_success());
|
|
||||||
|
|
||||||
// // read response
|
// read response
|
||||||
// let bytes = srv.execute(response.body()).unwrap();
|
let bytes = srv.block_on(HttpMessageBody::new(&mut response)).unwrap();
|
||||||
// assert_eq!(bytes, Bytes::from(data));
|
assert_eq!(bytes, Bytes::from(data));
|
||||||
// }
|
}
|
||||||
|
|
||||||
// #[cfg(all(feature = "brotli", feature = "ssl"))]
|
// #[cfg(all(feature = "brotli", feature = "ssl"))]
|
||||||
// #[test]
|
// #[test]
|
||||||
|
Loading…
Reference in New Issue
Block a user