mirror of
https://github.com/actix/actix-web.git
synced 2025-01-10 01:05:30 +00:00
use new gzdecoder, fixes gz streaming #228
This commit is contained in:
parent
4dba531bf9
commit
3bd43090fb
4 changed files with 32 additions and 99 deletions
|
@ -10,6 +10,8 @@
|
||||||
|
|
||||||
* Do not override HOST header for client request #428
|
* Do not override HOST header for client request #428
|
||||||
|
|
||||||
|
* Gz streaming, use `flate2::write::GzDecoder` #228
|
||||||
|
|
||||||
|
|
||||||
## [0.7.2] - 2018-07-26
|
## [0.7.2] - 2018-07-26
|
||||||
|
|
||||||
|
|
|
@ -81,7 +81,7 @@ parking_lot = "0.6"
|
||||||
url = { version="1.7", features=["query_encoding"] }
|
url = { version="1.7", features=["query_encoding"] }
|
||||||
cookie = { version="0.11", features=["percent-encode"] }
|
cookie = { version="0.11", features=["percent-encode"] }
|
||||||
brotli2 = { version="^0.3.2", optional = true }
|
brotli2 = { version="^0.3.2", optional = true }
|
||||||
flate2 = { version="1.0", optional = true, default-features = false }
|
flate2 = { version="^1.0.2", optional = true, default-features = false }
|
||||||
|
|
||||||
failure = "=0.1.1"
|
failure = "=0.1.1"
|
||||||
|
|
||||||
|
|
|
@ -1,14 +1,11 @@
|
||||||
use std::io::{Read, Write};
|
use std::io::{self, Write};
|
||||||
use std::{cmp, io};
|
|
||||||
|
|
||||||
#[cfg(feature = "brotli")]
|
#[cfg(feature = "brotli")]
|
||||||
use brotli2::write::BrotliDecoder;
|
use brotli2::write::BrotliDecoder;
|
||||||
use bytes::{BufMut, Bytes, BytesMut};
|
use bytes::{Bytes, BytesMut};
|
||||||
use error::PayloadError;
|
use error::PayloadError;
|
||||||
#[cfg(feature = "flate2")]
|
#[cfg(feature = "flate2")]
|
||||||
use flate2::read::GzDecoder;
|
use flate2::write::{DeflateDecoder, GzDecoder};
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
use flate2::write::DeflateDecoder;
|
|
||||||
use header::ContentEncoding;
|
use header::ContentEncoding;
|
||||||
use http::header::{HeaderMap, CONTENT_ENCODING};
|
use http::header::{HeaderMap, CONTENT_ENCODING};
|
||||||
use payload::{PayloadSender, PayloadStatus, PayloadWriter};
|
use payload::{PayloadSender, PayloadStatus, PayloadWriter};
|
||||||
|
@ -144,46 +141,12 @@ pub(crate) enum Decoder {
|
||||||
#[cfg(feature = "flate2")]
|
#[cfg(feature = "flate2")]
|
||||||
Deflate(Box<DeflateDecoder<Writer>>),
|
Deflate(Box<DeflateDecoder<Writer>>),
|
||||||
#[cfg(feature = "flate2")]
|
#[cfg(feature = "flate2")]
|
||||||
Gzip(Option<Box<GzDecoder<Wrapper>>>),
|
Gzip(Box<GzDecoder<Writer>>),
|
||||||
#[cfg(feature = "brotli")]
|
#[cfg(feature = "brotli")]
|
||||||
Br(Box<BrotliDecoder<Writer>>),
|
Br(Box<BrotliDecoder<Writer>>),
|
||||||
Identity,
|
Identity,
|
||||||
}
|
}
|
||||||
|
|
||||||
// should go after write::GzDecoder get implemented
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub(crate) struct Wrapper {
|
|
||||||
pub buf: BytesMut,
|
|
||||||
pub eof: bool,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl io::Read for Wrapper {
|
|
||||||
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
|
||||||
let len = cmp::min(buf.len(), self.buf.len());
|
|
||||||
buf[..len].copy_from_slice(&self.buf[..len]);
|
|
||||||
self.buf.split_to(len);
|
|
||||||
if len == 0 {
|
|
||||||
if self.eof {
|
|
||||||
Ok(0)
|
|
||||||
} else {
|
|
||||||
Err(io::Error::new(io::ErrorKind::WouldBlock, ""))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Ok(len)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl io::Write for Wrapper {
|
|
||||||
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) struct Writer {
|
pub(crate) struct Writer {
|
||||||
buf: BytesMut,
|
buf: BytesMut,
|
||||||
}
|
}
|
||||||
|
@ -212,12 +175,11 @@ impl io::Write for Writer {
|
||||||
/// Payload stream with decompression support
|
/// Payload stream with decompression support
|
||||||
pub(crate) struct PayloadStream {
|
pub(crate) struct PayloadStream {
|
||||||
decoder: Decoder,
|
decoder: Decoder,
|
||||||
dst: BytesMut,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl PayloadStream {
|
impl PayloadStream {
|
||||||
pub fn new(enc: ContentEncoding) -> PayloadStream {
|
pub fn new(enc: ContentEncoding) -> PayloadStream {
|
||||||
let dec = match enc {
|
let decoder = match enc {
|
||||||
#[cfg(feature = "brotli")]
|
#[cfg(feature = "brotli")]
|
||||||
ContentEncoding::Br => {
|
ContentEncoding::Br => {
|
||||||
Decoder::Br(Box::new(BrotliDecoder::new(Writer::new())))
|
Decoder::Br(Box::new(BrotliDecoder::new(Writer::new())))
|
||||||
|
@ -227,13 +189,12 @@ impl PayloadStream {
|
||||||
Decoder::Deflate(Box::new(DeflateDecoder::new(Writer::new())))
|
Decoder::Deflate(Box::new(DeflateDecoder::new(Writer::new())))
|
||||||
}
|
}
|
||||||
#[cfg(feature = "flate2")]
|
#[cfg(feature = "flate2")]
|
||||||
ContentEncoding::Gzip => Decoder::Gzip(None),
|
ContentEncoding::Gzip => {
|
||||||
|
Decoder::Gzip(Box::new(GzDecoder::new(Writer::new())))
|
||||||
|
}
|
||||||
_ => Decoder::Identity,
|
_ => Decoder::Identity,
|
||||||
};
|
};
|
||||||
PayloadStream {
|
PayloadStream { decoder }
|
||||||
decoder: dec,
|
|
||||||
dst: BytesMut::new(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -253,22 +214,17 @@ impl PayloadStream {
|
||||||
Err(e) => Err(e),
|
Err(e) => Err(e),
|
||||||
},
|
},
|
||||||
#[cfg(feature = "flate2")]
|
#[cfg(feature = "flate2")]
|
||||||
Decoder::Gzip(ref mut decoder) => {
|
Decoder::Gzip(ref mut decoder) => match decoder.try_finish() {
|
||||||
if let Some(ref mut decoder) = *decoder {
|
Ok(_) => {
|
||||||
decoder.as_mut().get_mut().eof = true;
|
let b = decoder.get_mut().take();
|
||||||
|
if !b.is_empty() {
|
||||||
self.dst.reserve(8192);
|
Ok(Some(b))
|
||||||
match decoder.read(unsafe { self.dst.bytes_mut() }) {
|
|
||||||
Ok(n) => {
|
|
||||||
unsafe { self.dst.advance_mut(n) };
|
|
||||||
return Ok(Some(self.dst.take().freeze()));
|
|
||||||
}
|
|
||||||
Err(e) => return Err(e),
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Err(e) => Err(e),
|
||||||
|
},
|
||||||
#[cfg(feature = "flate2")]
|
#[cfg(feature = "flate2")]
|
||||||
Decoder::Deflate(ref mut decoder) => match decoder.try_finish() {
|
Decoder::Deflate(ref mut decoder) => match decoder.try_finish() {
|
||||||
Ok(_) => {
|
Ok(_) => {
|
||||||
|
@ -301,43 +257,18 @@ impl PayloadStream {
|
||||||
Err(e) => Err(e),
|
Err(e) => Err(e),
|
||||||
},
|
},
|
||||||
#[cfg(feature = "flate2")]
|
#[cfg(feature = "flate2")]
|
||||||
Decoder::Gzip(ref mut decoder) => {
|
Decoder::Gzip(ref mut decoder) => match decoder.write_all(&data) {
|
||||||
if decoder.is_none() {
|
Ok(_) => {
|
||||||
*decoder = Some(Box::new(GzDecoder::new(Wrapper {
|
decoder.flush()?;
|
||||||
buf: BytesMut::from(data),
|
let b = decoder.get_mut().take();
|
||||||
eof: false,
|
if !b.is_empty() {
|
||||||
})));
|
Ok(Some(b))
|
||||||
} else {
|
} else {
|
||||||
let _ = decoder.as_mut().unwrap().write(&data);
|
Ok(None)
|
||||||
}
|
|
||||||
|
|
||||||
loop {
|
|
||||||
self.dst.reserve(8192);
|
|
||||||
match decoder
|
|
||||||
.as_mut()
|
|
||||||
.as_mut()
|
|
||||||
.unwrap()
|
|
||||||
.read(unsafe { self.dst.bytes_mut() })
|
|
||||||
{
|
|
||||||
Ok(n) => {
|
|
||||||
if n != 0 {
|
|
||||||
unsafe { self.dst.advance_mut(n) };
|
|
||||||
}
|
|
||||||
if n == 0 {
|
|
||||||
return Ok(Some(self.dst.take().freeze()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
if e.kind() == io::ErrorKind::WouldBlock
|
|
||||||
&& !self.dst.is_empty()
|
|
||||||
{
|
|
||||||
return Ok(Some(self.dst.take().freeze()));
|
|
||||||
}
|
|
||||||
return Err(e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Err(e) => Err(e),
|
||||||
|
},
|
||||||
#[cfg(feature = "flate2")]
|
#[cfg(feature = "flate2")]
|
||||||
Decoder::Deflate(ref mut decoder) => match decoder.write_all(&data) {
|
Decoder::Deflate(ref mut decoder) => match decoder.write_all(&data) {
|
||||||
Ok(_) => {
|
Ok(_) => {
|
||||||
|
|
|
@ -438,7 +438,7 @@ fn test_default_headers() {
|
||||||
let repr = format!("{:?}", request);
|
let repr = format!("{:?}", request);
|
||||||
assert!(repr.contains("\"accept-encoding\": \"gzip, deflate\""));
|
assert!(repr.contains("\"accept-encoding\": \"gzip, deflate\""));
|
||||||
assert!(repr.contains(concat!(
|
assert!(repr.contains(concat!(
|
||||||
"\"user-agent\": \"Actix-web/",
|
"\"user-agent\": \"actix-web/",
|
||||||
env!("CARGO_PKG_VERSION"),
|
env!("CARGO_PKG_VERSION"),
|
||||||
"\""
|
"\""
|
||||||
)));
|
)));
|
||||||
|
|
Loading…
Reference in a new issue