1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2025-01-24 16:08:06 +00:00

use new gzdecoder, fixes gz streaming #228

This commit is contained in:
Nikolay Kim 2018-07-31 09:06:05 -07:00
parent 4dba531bf9
commit 3bd43090fb
4 changed files with 32 additions and 99 deletions

View file

@ -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

View file

@ -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"

View file

@ -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() }) { } else {
Ok(n) => { Ok(None)
unsafe { self.dst.advance_mut(n) };
return Ok(Some(self.dst.take().freeze()));
}
Err(e) => return Err(e),
} }
} else {
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(_) => {

View file

@ -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"),
"\"" "\""
))); )));