diff --git a/.travis.yml b/.travis.yml
index 5dac530c4..0419ed11e 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -69,7 +69,7 @@ script:
# Upload docs
after_success:
- |
- if [[ "$TRAVIS_OS_NAME" == "linux" && "$TRAVIS_PULL_REQUEST" = "false" && "$TRAVIS_BRANCH" == "master" && "$TRAVIS_RUST_VERSION" == "nightly-2018-01-03" ]]; then
+ if [[ "$TRAVIS_OS_NAME" == "linux" && "$TRAVIS_PULL_REQUEST" = "false" && "$TRAVIS_BRANCH" == "master" && "$TRAVIS_RUST_VERSION" == "1.20.0" ]]; then
cargo doc --features alpn --no-deps &&
echo "" > target/doc/index.html &&
cargo install mdbook &&
diff --git a/CHANGES.md b/CHANGES.md
index 22b422667..df1308f87 100644
--- a/CHANGES.md
+++ b/CHANGES.md
@@ -1,7 +1,7 @@
# Changes
-## 0.3.0 (2017-xx-xx)
+## 0.3.0 (2018-01-12)
* HTTP/2 Support
diff --git a/Cargo.toml b/Cargo.toml
index 80a53a9c4..cc87f7247 100644
--- a/Cargo.toml
+++ b/Cargo.toml
@@ -36,10 +36,11 @@ alpn = ["openssl", "openssl/v102", "openssl/v110", "tokio-openssl"]
log = "0.4"
failure = "0.1"
failure_derive = "0.1"
-time = "0.1"
+h2 = "0.1"
http = "^0.1.2"
httparse = "1.2"
http-range = "0.1"
+time = "0.1"
mime = "0.3"
mime_guess = "1.8"
regex = "0.2"
@@ -54,8 +55,7 @@ smallvec = "0.6"
bitflags = "1.0"
num_cpus = "1.0"
-#flate2 = "1.0"
-flate2 = { git="https://github.com/fafhrd91/flate2-rs.git" }
+flate2 = "1.0"
# temp solution
# cookie = { version="0.10", features=["percent-encode", "secure"] }
@@ -69,8 +69,6 @@ futures = "0.1"
tokio-io = "0.1"
tokio-core = "0.1"
-h2 = { git = 'https://github.com/carllerche/h2' }
-
# native-tls
native-tls = { version="0.1", optional = true }
tokio-tls = { version="0.1", optional = true }
diff --git a/src/server/encoding.rs b/src/server/encoding.rs
index f9dbd64c3..deb4a5435 100644
--- a/src/server/encoding.rs
+++ b/src/server/encoding.rs
@@ -1,5 +1,5 @@
use std::{io, cmp, mem};
-use std::io::Write;
+use std::io::{Read, Write};
use std::fmt::Write as FmtWrite;
use std::str::FromStr;
@@ -8,7 +8,8 @@ use http::header::{HeaderMap, HeaderValue,
ACCEPT_ENCODING, CONNECTION,
CONTENT_ENCODING, CONTENT_LENGTH, TRANSFER_ENCODING};
use flate2::Compression;
-use flate2::write::{GzDecoder, GzEncoder, DeflateDecoder, DeflateEncoder};
+use flate2::read::GzDecoder;
+use flate2::write::{GzEncoder, DeflateDecoder, DeflateEncoder};
use brotli2::write::{BrotliDecoder, BrotliEncoder};
use bytes::{Bytes, BytesMut, BufMut, Writer};
@@ -122,15 +123,50 @@ impl PayloadWriter for PayloadType {
enum Decoder {
Deflate(Box>>),
- Gzip(Box>>),
+ Gzip(Option>>),
Br(Box>>),
Identity,
}
+// should go after write::GzDecoder get implemented
+#[derive(Debug)]
+struct Wrapper {
+ buf: BytesMut,
+ eof: bool,
+}
+
+impl io::Read for Wrapper {
+ fn read(&mut self, buf: &mut [u8]) -> io::Result {
+ 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 {
+ self.buf.extend_from_slice(buf);
+ Ok(buf.len())
+ }
+ fn flush(&mut self) -> io::Result<()> {
+ Ok(())
+ }
+}
+
/// Payload wrapper with content decompression support
pub(crate) struct EncodedPayload {
inner: PayloadSender,
decoder: Decoder,
+ dst: BytesMut,
error: bool,
}
@@ -141,11 +177,10 @@ impl EncodedPayload {
Box::new(BrotliDecoder::new(BytesMut::with_capacity(8192).writer()))),
ContentEncoding::Deflate => Decoder::Deflate(
Box::new(DeflateDecoder::new(BytesMut::with_capacity(8192).writer()))),
- ContentEncoding::Gzip => Decoder::Gzip(
- Box::new(GzDecoder::new(BytesMut::with_capacity(8192).writer()))),
+ ContentEncoding::Gzip => Decoder::Gzip(None),
_ => Decoder::Identity,
};
- EncodedPayload{ inner: inner, decoder: dec, error: false }
+ EncodedPayload{ inner: inner, decoder: dec, error: false, dst: BytesMut::new() }
}
}
@@ -174,16 +209,28 @@ impl PayloadWriter for EncodedPayload {
}
},
Decoder::Gzip(ref mut decoder) => {
- match decoder.try_finish() {
- Ok(_) => {
- let b = decoder.get_mut().get_mut().take().freeze();
- if !b.is_empty() {
- self.inner.feed_data(b);
+ if let Some(ref mut decoder) = *decoder {
+ decoder.as_mut().get_mut().eof = true;
+
+ loop {
+ self.dst.reserve(8192);
+ match decoder.read(unsafe{self.dst.bytes_mut()}) {
+ Ok(n) => {
+ if n == 0 {
+ self.inner.feed_eof();
+ return
+ } else {
+ unsafe{self.dst.set_len(n)};
+ self.inner.feed_data(self.dst.split_to(n).freeze());
+ }
+ }
+ Err(err) => {
+ break Some(err);
+ }
}
- self.inner.feed_eof();
- return
- },
- Err(err) => Some(err),
+ }
+ } else {
+ return
}
},
Decoder::Deflate(ref mut decoder) => {
@@ -231,14 +278,33 @@ impl PayloadWriter for EncodedPayload {
}
Decoder::Gzip(ref mut decoder) => {
- if decoder.write(&data).is_ok() && decoder.flush().is_ok() {
- let b = decoder.get_mut().get_mut().take().freeze();
- if !b.is_empty() {
- self.inner.feed_data(b);
- }
- return
+ if decoder.is_none() {
+ *decoder = Some(
+ Box::new(GzDecoder::new(
+ Wrapper{buf: BytesMut::from(data), eof: false})));
+ } else {
+ let _ = decoder.as_mut().unwrap().write(&data);
+ }
+
+ loop {
+ self.dst.reserve(8192);
+ match decoder.as_mut().as_mut().unwrap().read(unsafe{self.dst.bytes_mut()}) {
+ Ok(n) => {
+ if n == 0 {
+ return
+ } else {
+ unsafe{self.dst.set_len(n)};
+ self.inner.feed_data(self.dst.split_to(n).freeze());
+ }
+ }
+ Err(e) => {
+ if e.kind() == io::ErrorKind::WouldBlock {
+ return
+ }
+ break
+ }
+ }
}
- trace!("Error decoding gzip encoding");
}
Decoder::Deflate(ref mut decoder) => {