From 2379bcbf39a0b65ed3a5b94e8efca10dbd0cbdb8 Mon Sep 17 00:00:00 2001 From: Nikolay Kim Date: Mon, 6 Nov 2017 09:35:52 -0800 Subject: [PATCH] added content-encoding support to h2 --- src/h1.rs | 1 + src/h2.rs | 48 +++++++++++++++++++++++++++++++++++++++++------- src/payload.rs | 2 ++ 3 files changed, 44 insertions(+), 7 deletions(-) diff --git a/src/h1.rs b/src/h1.rs index 33a5313dc..63d878ffd 100644 --- a/src/h1.rs +++ b/src/h1.rs @@ -395,6 +395,7 @@ impl Reader { let payload = if let Some(decoder) = decoder { let (tx, rx) = Payload::new(false); + // Content-Encoding let enc = if let Some(enc) = msg.headers().get(CONTENT_ENCODING) { if let Ok(enc) = enc.to_str() { ContentEncoding::from(enc) diff --git a/src/h2.rs b/src/h2.rs index 51f03ef84..2de7e102d 100644 --- a/src/h2.rs +++ b/src/h2.rs @@ -7,6 +7,7 @@ use std::collections::VecDeque; use actix::Arbiter; use http::request::Parts; +use http::header::CONTENT_ENCODING; use http2::{Reason, RecvStream}; use http2::server::{Server, Handshake, Respond}; use bytes::{Buf, Bytes}; @@ -19,7 +20,8 @@ use h2writer::H2Writer; use channel::HttpHandler; use httpcodes::HTTPNotFound; use httprequest::HttpRequest; -use payload::{Payload, PayloadError, PayloadSender, PayloadWriter}; +use httpresponse::ContentEncoding; +use payload::{Payload, PayloadError, PayloadSender, PayloadWriter, EncodedPayload}; const KEEPALIVE_PERIOD: u64 = 15; // seconds @@ -193,10 +195,26 @@ impl Http2 } } +struct PayloadInfo(PayloadInfoItem); +enum PayloadInfoItem { + Sender(PayloadSender), + Encoding(EncodedPayload), +} + +impl PayloadInfo { + + fn as_mut(&mut self) -> &mut PayloadWriter { + match self.0 { + PayloadInfoItem::Sender(ref mut sender) => sender, + PayloadInfoItem::Encoding(ref mut enc) => enc, + } + } +} + struct Entry { task: Task, req: UnsafeCell, - payload: PayloadSender, + payload: PayloadInfo, recv: RecvStream, stream: H2Writer, eof: bool, @@ -218,7 +236,23 @@ impl Entry { let mut req = HttpRequest::new( parts.method, path, parts.version, parts.headers, query); + + // Payload and Content-Encoding let (psender, payload) = Payload::new(false); + let enc = if let Some(enc) = req.headers().get(CONTENT_ENCODING) { + if let Ok(enc) = enc.to_str() { + ContentEncoding::from(enc) + } else { + ContentEncoding::Auto + } + } else { + ContentEncoding::Auto + }; + let psender = match enc { + ContentEncoding::Auto | ContentEncoding::Identity => + PayloadInfoItem::Sender(psender), + _ => PayloadInfoItem::Encoding(EncodedPayload::new(psender, enc)), + }; // start request processing let mut task = None; @@ -231,7 +265,7 @@ impl Entry { Entry {task: task.unwrap_or_else(|| Task::reply(HTTPNotFound)), req: UnsafeCell::new(req), - payload: psender, + payload: PayloadInfo(psender), recv: recv, stream: H2Writer::new(resp), eof: false, @@ -246,22 +280,22 @@ impl Entry { if !self.reof { match self.recv.poll() { Ok(Async::Ready(Some(chunk))) => { - self.payload.feed_data(chunk); + self.payload.as_mut().feed_data(chunk); }, Ok(Async::Ready(None)) => { self.reof = true; }, Ok(Async::NotReady) => (), Err(err) => { - self.payload.set_error(PayloadError::Http2(err)) + self.payload.as_mut().set_error(PayloadError::Http2(err)) } } - let capacity = self.payload.capacity(); + let capacity = self.payload.as_mut().capacity(); if self.capacity != capacity { self.capacity = capacity; if let Err(err) = self.recv.release_capacity().release_capacity(capacity) { - self.payload.set_error(PayloadError::Http2(err)) + self.payload.as_mut().set_error(PayloadError::Http2(err)) } } } diff --git a/src/payload.rs b/src/payload.rs index 98adc1692..9299dc692 100644 --- a/src/payload.rs +++ b/src/payload.rs @@ -212,6 +212,7 @@ enum Decoder { Identity, } +// should go after write::GzDecoder get implemented #[derive(Debug)] struct Wrapper { buf: BytesMut @@ -247,6 +248,7 @@ impl io::Write for BytesWriter { } +// should go after brotli2::write::BrotliDecoder::get_mut get implemented #[derive(Debug)] struct WrapperRc { buf: Rc>,