mirror of
https://github.com/actix/actix-web.git
synced 2025-01-04 22:38:44 +00:00
calculate response parameters
This commit is contained in:
parent
e78014c65a
commit
7fdc18f9b9
6 changed files with 194 additions and 503 deletions
|
@ -21,7 +21,7 @@ pub struct ServiceConfig(Rc<Inner>);
|
||||||
struct Inner {
|
struct Inner {
|
||||||
keep_alive: Option<Duration>,
|
keep_alive: Option<Duration>,
|
||||||
client_timeout: u64,
|
client_timeout: u64,
|
||||||
client_shutdown: u64,
|
client_disconnect: u64,
|
||||||
ka_enabled: bool,
|
ka_enabled: bool,
|
||||||
date: UnsafeCell<(bool, Date)>,
|
date: UnsafeCell<(bool, Date)>,
|
||||||
}
|
}
|
||||||
|
@ -35,7 +35,7 @@ impl Clone for ServiceConfig {
|
||||||
impl ServiceConfig {
|
impl ServiceConfig {
|
||||||
/// Create instance of `ServiceConfig`
|
/// Create instance of `ServiceConfig`
|
||||||
pub(crate) fn new(
|
pub(crate) fn new(
|
||||||
keep_alive: KeepAlive, client_timeout: u64, client_shutdown: u64,
|
keep_alive: KeepAlive, client_timeout: u64, client_disconnect: u64,
|
||||||
) -> ServiceConfig {
|
) -> ServiceConfig {
|
||||||
let (keep_alive, ka_enabled) = match keep_alive {
|
let (keep_alive, ka_enabled) = match keep_alive {
|
||||||
KeepAlive::Timeout(val) => (val as u64, true),
|
KeepAlive::Timeout(val) => (val as u64, true),
|
||||||
|
@ -52,7 +52,7 @@ impl ServiceConfig {
|
||||||
keep_alive,
|
keep_alive,
|
||||||
ka_enabled,
|
ka_enabled,
|
||||||
client_timeout,
|
client_timeout,
|
||||||
client_shutdown,
|
client_disconnect,
|
||||||
date: UnsafeCell::new((false, Date::new())),
|
date: UnsafeCell::new((false, Date::new())),
|
||||||
}))
|
}))
|
||||||
}
|
}
|
||||||
|
@ -100,9 +100,9 @@ impl ServiceConfig {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Client shutdown timer
|
/// Client disconnect timer
|
||||||
pub fn client_shutdown_timer(&self) -> Option<Instant> {
|
pub fn client_disconnect_timer(&self) -> Option<Instant> {
|
||||||
let delay = self.0.client_shutdown;
|
let delay = self.0.client_disconnect;
|
||||||
if delay != 0 {
|
if delay != 0 {
|
||||||
Some(self.now() + Duration::from_millis(delay))
|
Some(self.now() + Duration::from_millis(delay))
|
||||||
} else {
|
} else {
|
||||||
|
@ -184,7 +184,7 @@ impl ServiceConfig {
|
||||||
pub struct ServiceConfigBuilder {
|
pub struct ServiceConfigBuilder {
|
||||||
keep_alive: KeepAlive,
|
keep_alive: KeepAlive,
|
||||||
client_timeout: u64,
|
client_timeout: u64,
|
||||||
client_shutdown: u64,
|
client_disconnect: u64,
|
||||||
host: String,
|
host: String,
|
||||||
addr: net::SocketAddr,
|
addr: net::SocketAddr,
|
||||||
secure: bool,
|
secure: bool,
|
||||||
|
@ -196,7 +196,7 @@ impl ServiceConfigBuilder {
|
||||||
ServiceConfigBuilder {
|
ServiceConfigBuilder {
|
||||||
keep_alive: KeepAlive::Timeout(5),
|
keep_alive: KeepAlive::Timeout(5),
|
||||||
client_timeout: 5000,
|
client_timeout: 5000,
|
||||||
client_shutdown: 5000,
|
client_disconnect: 0,
|
||||||
secure: false,
|
secure: false,
|
||||||
host: "localhost".to_owned(),
|
host: "localhost".to_owned(),
|
||||||
addr: "127.0.0.1:8080".parse().unwrap(),
|
addr: "127.0.0.1:8080".parse().unwrap(),
|
||||||
|
@ -204,10 +204,14 @@ impl ServiceConfigBuilder {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Enable secure flag for current server.
|
/// Enable secure flag for current server.
|
||||||
|
/// This flags also enables `client disconnect timeout`.
|
||||||
///
|
///
|
||||||
/// By default this flag is set to false.
|
/// By default this flag is set to false.
|
||||||
pub fn secure(mut self) -> Self {
|
pub fn secure(mut self) -> Self {
|
||||||
self.secure = true;
|
self.secure = true;
|
||||||
|
if self.client_disconnect == 0 {
|
||||||
|
self.client_disconnect = 3000;
|
||||||
|
}
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -233,16 +237,16 @@ impl ServiceConfigBuilder {
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Set server connection shutdown timeout in milliseconds.
|
/// Set server connection disconnect timeout in milliseconds.
|
||||||
///
|
///
|
||||||
/// Defines a timeout for shutdown connection. If a shutdown procedure does not complete
|
/// Defines a timeout for disconnect connection. If a disconnect procedure does not complete
|
||||||
/// within this time, the request is dropped. This timeout affects only secure connections.
|
/// within this time, the request get dropped. This timeout affects secure connections.
|
||||||
///
|
///
|
||||||
/// To disable timeout set value to 0.
|
/// To disable timeout set value to 0.
|
||||||
///
|
///
|
||||||
/// By default client timeout is set to 5000 milliseconds.
|
/// By default disconnect timeout is set to 3000 milliseconds.
|
||||||
pub fn client_shutdown(mut self, val: u64) -> Self {
|
pub fn client_disconnect(mut self, val: u64) -> Self {
|
||||||
self.client_shutdown = val;
|
self.client_disconnect = val;
|
||||||
self
|
self
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -277,9 +281,7 @@ impl ServiceConfigBuilder {
|
||||||
|
|
||||||
/// Finish service configuration and create `ServiceConfig` object.
|
/// Finish service configuration and create `ServiceConfig` object.
|
||||||
pub fn finish(self) -> ServiceConfig {
|
pub fn finish(self) -> ServiceConfig {
|
||||||
let client_shutdown = if self.secure { self.client_shutdown } else { 0 };
|
ServiceConfig::new(self.keep_alive, self.client_timeout, self.client_disconnect)
|
||||||
|
|
||||||
ServiceConfig::new(self.keep_alive, self.client_timeout, client_shutdown)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -397,9 +397,9 @@ pub enum DispatchError<E: fmt::Debug + fmt::Display> {
|
||||||
// #[fail(display = "The first request did not complete within the specified timeout")]
|
// #[fail(display = "The first request did not complete within the specified timeout")]
|
||||||
SlowRequestTimeout,
|
SlowRequestTimeout,
|
||||||
|
|
||||||
/// Shutdown timeout
|
/// Disconnect timeout. Makes sense for ssl streams.
|
||||||
// #[fail(display = "Connection shutdown timeout")]
|
// #[fail(display = "Connection shutdown timeout")]
|
||||||
ShutdownTimeout,
|
DisconnectTimeout,
|
||||||
|
|
||||||
/// Payload is not consumed
|
/// Payload is not consumed
|
||||||
// #[fail(display = "Task is completed but request's payload is not consumed")]
|
// #[fail(display = "Task is completed but request's payload is not consumed")]
|
||||||
|
|
|
@ -10,7 +10,7 @@ use body::Body;
|
||||||
use error::ParseError;
|
use error::ParseError;
|
||||||
use helpers;
|
use helpers;
|
||||||
use http::header::{HeaderValue, CONNECTION, CONTENT_LENGTH, DATE, TRANSFER_ENCODING};
|
use http::header::{HeaderValue, CONNECTION, CONTENT_LENGTH, DATE, TRANSFER_ENCODING};
|
||||||
use http::Version;
|
use http::{Method, Version};
|
||||||
use httpresponse::HttpResponse;
|
use httpresponse::HttpResponse;
|
||||||
use request::RequestPool;
|
use request::RequestPool;
|
||||||
use server::output::{ResponseInfo, ResponseLength};
|
use server::output::{ResponseInfo, ResponseLength};
|
||||||
|
@ -27,6 +27,8 @@ pub enum OutMessage {
|
||||||
pub struct Codec {
|
pub struct Codec {
|
||||||
decoder: H1Decoder,
|
decoder: H1Decoder,
|
||||||
encoder: H1Writer,
|
encoder: H1Writer,
|
||||||
|
head: bool,
|
||||||
|
version: Version,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Codec {
|
impl Codec {
|
||||||
|
@ -40,6 +42,8 @@ impl Codec {
|
||||||
Codec {
|
Codec {
|
||||||
decoder: H1Decoder::with_pool(pool),
|
decoder: H1Decoder::with_pool(pool),
|
||||||
encoder: H1Writer::new(),
|
encoder: H1Writer::new(),
|
||||||
|
head: false,
|
||||||
|
version: Version::HTTP_11,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -49,7 +53,17 @@ impl Decoder for Codec {
|
||||||
type Error = ParseError;
|
type Error = ParseError;
|
||||||
|
|
||||||
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
|
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
|
||||||
self.decoder.decode(src)
|
let res = self.decoder.decode(src);
|
||||||
|
|
||||||
|
match res {
|
||||||
|
Ok(Some(InMessage::Message(ref req)))
|
||||||
|
| Ok(Some(InMessage::MessageWithPayload(ref req))) => {
|
||||||
|
self.head = req.inner.method == Method::HEAD;
|
||||||
|
self.version = req.inner.version;
|
||||||
|
}
|
||||||
|
_ => (),
|
||||||
|
}
|
||||||
|
res
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -62,7 +76,7 @@ impl Encoder for Codec {
|
||||||
) -> Result<(), Self::Error> {
|
) -> Result<(), Self::Error> {
|
||||||
match item {
|
match item {
|
||||||
OutMessage::Response(res) => {
|
OutMessage::Response(res) => {
|
||||||
self.encoder.encode(res, dst)?;
|
self.encoder.encode(res, dst, self.head, self.version)?;
|
||||||
}
|
}
|
||||||
OutMessage::Payload(bytes) => {
|
OutMessage::Payload(bytes) => {
|
||||||
dst.extend_from_slice(&bytes);
|
dst.extend_from_slice(&bytes);
|
||||||
|
@ -87,6 +101,7 @@ struct H1Writer {
|
||||||
flags: Flags,
|
flags: Flags,
|
||||||
written: u64,
|
written: u64,
|
||||||
headers_size: u32,
|
headers_size: u32,
|
||||||
|
info: ResponseInfo,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl H1Writer {
|
impl H1Writer {
|
||||||
|
@ -95,6 +110,7 @@ impl H1Writer {
|
||||||
flags: Flags::empty(),
|
flags: Flags::empty(),
|
||||||
written: 0,
|
written: 0,
|
||||||
headers_size: 0,
|
headers_size: 0,
|
||||||
|
info: ResponseInfo::default(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -116,10 +132,11 @@ impl H1Writer {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn encode(
|
fn encode(
|
||||||
&mut self, mut msg: HttpResponse, buffer: &mut BytesMut,
|
&mut self, mut msg: HttpResponse, buffer: &mut BytesMut, head: bool,
|
||||||
|
version: Version,
|
||||||
) -> io::Result<()> {
|
) -> io::Result<()> {
|
||||||
// prepare task
|
// prepare task
|
||||||
let info = ResponseInfo::new(false); // req.inner.method == Method::HEAD);
|
self.info.update(&mut msg, head, version);
|
||||||
|
|
||||||
//if msg.keep_alive().unwrap_or_else(|| req.keep_alive()) {
|
//if msg.keep_alive().unwrap_or_else(|| req.keep_alive()) {
|
||||||
//self.flags = Flags::STARTED | Flags::KEEPALIVE;
|
//self.flags = Flags::STARTED | Flags::KEEPALIVE;
|
||||||
|
@ -166,7 +183,7 @@ impl H1Writer {
|
||||||
buffer.extend_from_slice(reason);
|
buffer.extend_from_slice(reason);
|
||||||
|
|
||||||
// content length
|
// content length
|
||||||
match info.length {
|
match self.info.length {
|
||||||
ResponseLength::Chunked => {
|
ResponseLength::Chunked => {
|
||||||
buffer.extend_from_slice(b"\r\ntransfer-encoding: chunked\r\n")
|
buffer.extend_from_slice(b"\r\ntransfer-encoding: chunked\r\n")
|
||||||
}
|
}
|
||||||
|
@ -183,11 +200,6 @@ impl H1Writer {
|
||||||
}
|
}
|
||||||
ResponseLength::None => buffer.extend_from_slice(b"\r\n"),
|
ResponseLength::None => buffer.extend_from_slice(b"\r\n"),
|
||||||
}
|
}
|
||||||
if let Some(ce) = info.content_encoding {
|
|
||||||
buffer.extend_from_slice(b"content-encoding: ");
|
|
||||||
buffer.extend_from_slice(ce.as_ref());
|
|
||||||
buffer.extend_from_slice(b"\r\n");
|
|
||||||
}
|
|
||||||
|
|
||||||
// write headers
|
// write headers
|
||||||
let mut pos = 0;
|
let mut pos = 0;
|
||||||
|
@ -197,7 +209,7 @@ impl H1Writer {
|
||||||
for (key, value) in msg.headers() {
|
for (key, value) in msg.headers() {
|
||||||
match *key {
|
match *key {
|
||||||
TRANSFER_ENCODING => continue,
|
TRANSFER_ENCODING => continue,
|
||||||
CONTENT_LENGTH => match info.length {
|
CONTENT_LENGTH => match self.info.length {
|
||||||
ResponseLength::None => (),
|
ResponseLength::None => (),
|
||||||
_ => continue,
|
_ => continue,
|
||||||
},
|
},
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
// #![allow(unused_imports, unused_variables, dead_code)]
|
// #![allow(unused_imports, unused_variables, dead_code)]
|
||||||
use std::collections::VecDeque;
|
use std::collections::VecDeque;
|
||||||
use std::fmt::{Debug, Display};
|
use std::fmt::{Debug, Display};
|
||||||
// use std::time::{Duration, Instant};
|
use std::time::Instant;
|
||||||
|
|
||||||
use actix_net::service::Service;
|
use actix_net::service::Service;
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ use futures::{Async, AsyncSink, Future, Poll, Sink, Stream};
|
||||||
use tokio_codec::Framed;
|
use tokio_codec::Framed;
|
||||||
// use tokio_current_thread::spawn;
|
// use tokio_current_thread::spawn;
|
||||||
use tokio_io::{AsyncRead, AsyncWrite};
|
use tokio_io::{AsyncRead, AsyncWrite};
|
||||||
// use tokio_timer::Delay;
|
use tokio_timer::Delay;
|
||||||
|
|
||||||
use error::{ParseError, PayloadError};
|
use error::{ParseError, PayloadError};
|
||||||
use payload::{Payload, PayloadStatus, PayloadWriter};
|
use payload::{Payload, PayloadStatus, PayloadWriter};
|
||||||
|
@ -47,12 +47,14 @@ where
|
||||||
flags: Flags,
|
flags: Flags,
|
||||||
framed: Framed<T, Codec>,
|
framed: Framed<T, Codec>,
|
||||||
error: Option<DispatchError<S::Error>>,
|
error: Option<DispatchError<S::Error>>,
|
||||||
|
config: ServiceConfig,
|
||||||
|
|
||||||
state: State<S>,
|
state: State<S>,
|
||||||
payload: Option<PayloadType>,
|
payload: Option<PayloadType>,
|
||||||
messages: VecDeque<Request>,
|
messages: VecDeque<Request>,
|
||||||
|
|
||||||
config: ServiceConfig,
|
ka_expire: Instant,
|
||||||
|
ka_timer: Option<Delay>,
|
||||||
}
|
}
|
||||||
|
|
||||||
enum State<S: Service> {
|
enum State<S: Service> {
|
||||||
|
@ -81,9 +83,28 @@ where
|
||||||
{
|
{
|
||||||
/// Create http/1 dispatcher.
|
/// Create http/1 dispatcher.
|
||||||
pub fn new(stream: T, config: ServiceConfig, service: S) -> Self {
|
pub fn new(stream: T, config: ServiceConfig, service: S) -> Self {
|
||||||
let flags = Flags::FLUSHED;
|
Dispatcher::with_timeout(stream, config, None, service)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create http/1 dispatcher with slow request timeout.
|
||||||
|
pub fn with_timeout(
|
||||||
|
stream: T, config: ServiceConfig, timeout: Option<Delay>, service: S,
|
||||||
|
) -> Self {
|
||||||
|
let flags = if config.keep_alive_enabled() {
|
||||||
|
Flags::KEEPALIVE | Flags::KEEPALIVE_ENABLED | Flags::FLUSHED
|
||||||
|
} else {
|
||||||
|
Flags::FLUSHED
|
||||||
|
};
|
||||||
let framed = Framed::new(stream, Codec::new());
|
let framed = Framed::new(stream, Codec::new());
|
||||||
|
|
||||||
|
let (ka_expire, ka_timer) = if let Some(delay) = timeout {
|
||||||
|
(delay.deadline(), Some(delay))
|
||||||
|
} else if let Some(delay) = config.keep_alive_timer() {
|
||||||
|
(delay.deadline(), Some(delay))
|
||||||
|
} else {
|
||||||
|
(config.now(), None)
|
||||||
|
};
|
||||||
|
|
||||||
Dispatcher {
|
Dispatcher {
|
||||||
payload: None,
|
payload: None,
|
||||||
state: State::None,
|
state: State::None,
|
||||||
|
@ -93,6 +114,8 @@ where
|
||||||
flags,
|
flags,
|
||||||
framed,
|
framed,
|
||||||
config,
|
config,
|
||||||
|
ka_expire,
|
||||||
|
ka_timer,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -358,8 +381,64 @@ where
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if self.ka_timer.is_some() && updated {
|
||||||
|
if let Some(expire) = self.config.keep_alive_expire() {
|
||||||
|
self.ka_expire = expire;
|
||||||
|
}
|
||||||
|
}
|
||||||
Ok(updated)
|
Ok(updated)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// keep-alive timer
|
||||||
|
fn poll_keepalive(&mut self) -> Result<(), DispatchError<S::Error>> {
|
||||||
|
if let Some(ref mut timer) = self.ka_timer {
|
||||||
|
match timer.poll() {
|
||||||
|
Ok(Async::Ready(_)) => {
|
||||||
|
if timer.deadline() >= self.ka_expire {
|
||||||
|
// check for any outstanding request handling
|
||||||
|
if self.state.is_empty() && self.messages.is_empty() {
|
||||||
|
// if we get timer during shutdown, just drop connection
|
||||||
|
if self.flags.contains(Flags::SHUTDOWN) {
|
||||||
|
return Err(DispatchError::DisconnectTimeout);
|
||||||
|
} else if !self.flags.contains(Flags::STARTED) {
|
||||||
|
// timeout on first request (slow request) return 408
|
||||||
|
trace!("Slow request timeout");
|
||||||
|
self.flags
|
||||||
|
.insert(Flags::STARTED | Flags::READ_DISCONNECTED);
|
||||||
|
self.state =
|
||||||
|
State::SendResponse(Some(OutMessage::Response(
|
||||||
|
HttpResponse::RequestTimeout().finish(),
|
||||||
|
)));
|
||||||
|
} else {
|
||||||
|
trace!("Keep-alive timeout, close connection");
|
||||||
|
self.flags.insert(Flags::SHUTDOWN);
|
||||||
|
|
||||||
|
// start shutdown timer
|
||||||
|
if let Some(deadline) =
|
||||||
|
self.config.client_disconnect_timer()
|
||||||
|
{
|
||||||
|
timer.reset(deadline)
|
||||||
|
} else {
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else if let Some(deadline) = self.config.keep_alive_expire() {
|
||||||
|
timer.reset(deadline)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
timer.reset(self.ka_expire)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(Async::NotReady) => (),
|
||||||
|
Err(e) => {
|
||||||
|
error!("Timer error {:?}", e);
|
||||||
|
return Err(DispatchError::Unknown);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T, S> Future for Dispatcher<T, S>
|
impl<T, S> Future for Dispatcher<T, S>
|
||||||
|
@ -373,6 +452,8 @@ where
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn poll(&mut self) -> Poll<(), Self::Error> {
|
fn poll(&mut self) -> Poll<(), Self::Error> {
|
||||||
|
self.poll_keepalive()?;
|
||||||
|
|
||||||
// shutdown
|
// shutdown
|
||||||
if self.flags.contains(Flags::SHUTDOWN) {
|
if self.flags.contains(Flags::SHUTDOWN) {
|
||||||
if self.flags.contains(Flags::WRITE_DISCONNECTED) {
|
if self.flags.contains(Flags::WRITE_DISCONNECTED) {
|
||||||
|
|
|
@ -79,7 +79,7 @@
|
||||||
//!
|
//!
|
||||||
#![cfg_attr(actix_nightly, feature(tool_lints))]
|
#![cfg_attr(actix_nightly, feature(tool_lints))]
|
||||||
// #![warn(missing_docs)]
|
// #![warn(missing_docs)]
|
||||||
// #![allow(unused_imports, unused_variables, dead_code)]
|
#![allow(dead_code)]
|
||||||
|
|
||||||
extern crate actix;
|
extern crate actix;
|
||||||
extern crate actix_net;
|
extern crate actix_net;
|
||||||
|
|
|
@ -4,26 +4,15 @@ use std::io::Write;
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
use std::{cmp, fmt, io, mem};
|
use std::{cmp, fmt, io, mem};
|
||||||
|
|
||||||
#[cfg(feature = "brotli")]
|
use bytes::{Bytes, BytesMut};
|
||||||
use brotli2::write::BrotliEncoder;
|
|
||||||
use bytes::BytesMut;
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
use flate2::write::{GzEncoder, ZlibEncoder};
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
use flate2::Compression;
|
|
||||||
use http::header::{HeaderValue, ACCEPT_ENCODING, CONTENT_LENGTH};
|
use http::header::{HeaderValue, ACCEPT_ENCODING, CONTENT_LENGTH};
|
||||||
use http::{StatusCode, Version};
|
use http::{StatusCode, Version};
|
||||||
|
|
||||||
use body::{Binary, Body};
|
use body::{Binary, Body};
|
||||||
use header::ContentEncoding;
|
use header::ContentEncoding;
|
||||||
|
use http::Method;
|
||||||
use httpresponse::HttpResponse;
|
use httpresponse::HttpResponse;
|
||||||
use request::InnerRequest;
|
use request::Request;
|
||||||
|
|
||||||
// #[derive(Debug)]
|
|
||||||
// pub(crate) struct RequestInfo {
|
|
||||||
// pub version: Version,
|
|
||||||
// pub accept_encoding: Option<HeaderValue>,
|
|
||||||
// }
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub(crate) enum ResponseLength {
|
pub(crate) enum ResponseLength {
|
||||||
|
@ -38,285 +27,91 @@ pub(crate) enum ResponseLength {
|
||||||
pub(crate) struct ResponseInfo {
|
pub(crate) struct ResponseInfo {
|
||||||
head: bool,
|
head: bool,
|
||||||
pub length: ResponseLength,
|
pub length: ResponseLength,
|
||||||
pub content_encoding: Option<&'static str>,
|
pub te: TransferEncoding,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for ResponseInfo {
|
||||||
|
fn default() -> Self {
|
||||||
|
ResponseInfo {
|
||||||
|
head: false,
|
||||||
|
length: ResponseLength::None,
|
||||||
|
te: TransferEncoding::empty(),
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ResponseInfo {
|
impl ResponseInfo {
|
||||||
pub fn new(head: bool) -> Self {
|
pub fn update(&mut self, resp: &mut HttpResponse, head: bool, version: Version) {
|
||||||
ResponseInfo {
|
self.head = head;
|
||||||
head,
|
|
||||||
length: ResponseLength::None,
|
|
||||||
content_encoding: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
let version = resp.version().unwrap_or_else(|| version);
|
||||||
pub(crate) enum Output {
|
|
||||||
Empty(BytesMut),
|
|
||||||
Buffer(BytesMut),
|
|
||||||
Encoder(ContentEncoder),
|
|
||||||
TE(TransferEncoding),
|
|
||||||
Done,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Output {
|
|
||||||
pub fn take(&mut self) -> BytesMut {
|
|
||||||
match mem::replace(self, Output::Done) {
|
|
||||||
Output::Empty(bytes) => bytes,
|
|
||||||
Output::Buffer(bytes) => bytes,
|
|
||||||
Output::Encoder(mut enc) => enc.take_buf(),
|
|
||||||
Output::TE(mut te) => te.take(),
|
|
||||||
Output::Done => panic!(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn take_option(&mut self) -> Option<BytesMut> {
|
|
||||||
match mem::replace(self, Output::Done) {
|
|
||||||
Output::Empty(bytes) => Some(bytes),
|
|
||||||
Output::Buffer(bytes) => Some(bytes),
|
|
||||||
Output::Encoder(mut enc) => Some(enc.take_buf()),
|
|
||||||
Output::TE(mut te) => Some(te.take()),
|
|
||||||
Output::Done => None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn as_ref(&mut self) -> &BytesMut {
|
|
||||||
match self {
|
|
||||||
Output::Empty(ref mut bytes) => bytes,
|
|
||||||
Output::Buffer(ref mut bytes) => bytes,
|
|
||||||
Output::Encoder(ref mut enc) => enc.buf_ref(),
|
|
||||||
Output::TE(ref mut te) => te.buf_ref(),
|
|
||||||
Output::Done => panic!(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn as_mut(&mut self) -> &mut BytesMut {
|
|
||||||
match self {
|
|
||||||
Output::Empty(ref mut bytes) => bytes,
|
|
||||||
Output::Buffer(ref mut bytes) => bytes,
|
|
||||||
Output::Encoder(ref mut enc) => enc.buf_mut(),
|
|
||||||
Output::TE(ref mut te) => te.buf_mut(),
|
|
||||||
Output::Done => panic!(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn split_to(&mut self, cap: usize) -> BytesMut {
|
|
||||||
match self {
|
|
||||||
Output::Empty(ref mut bytes) => bytes.split_to(cap),
|
|
||||||
Output::Buffer(ref mut bytes) => bytes.split_to(cap),
|
|
||||||
Output::Encoder(ref mut enc) => enc.buf_mut().split_to(cap),
|
|
||||||
Output::TE(ref mut te) => te.buf_mut().split_to(cap),
|
|
||||||
Output::Done => BytesMut::new(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn len(&self) -> usize {
|
|
||||||
match self {
|
|
||||||
Output::Empty(ref bytes) => bytes.len(),
|
|
||||||
Output::Buffer(ref bytes) => bytes.len(),
|
|
||||||
Output::Encoder(ref enc) => enc.len(),
|
|
||||||
Output::TE(ref te) => te.len(),
|
|
||||||
Output::Done => 0,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn is_empty(&self) -> bool {
|
|
||||||
match self {
|
|
||||||
Output::Empty(ref bytes) => bytes.is_empty(),
|
|
||||||
Output::Buffer(ref bytes) => bytes.is_empty(),
|
|
||||||
Output::Encoder(ref enc) => enc.is_empty(),
|
|
||||||
Output::TE(ref te) => te.is_empty(),
|
|
||||||
Output::Done => true,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn write(&mut self, data: &[u8]) -> Result<(), io::Error> {
|
|
||||||
match self {
|
|
||||||
Output::Buffer(ref mut bytes) => {
|
|
||||||
bytes.extend_from_slice(data);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
Output::Encoder(ref mut enc) => enc.write(data),
|
|
||||||
Output::TE(ref mut te) => te.encode(data).map(|_| ()),
|
|
||||||
Output::Empty(_) | Output::Done => Ok(()),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn write_eof(&mut self) -> Result<bool, io::Error> {
|
|
||||||
match self {
|
|
||||||
Output::Buffer(_) => Ok(true),
|
|
||||||
Output::Encoder(ref mut enc) => enc.write_eof(),
|
|
||||||
Output::TE(ref mut te) => Ok(te.encode_eof()),
|
|
||||||
Output::Empty(_) | Output::Done => Ok(true),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) fn for_server(
|
|
||||||
&mut self, info: &mut ResponseInfo, req: &InnerRequest, resp: &mut HttpResponse,
|
|
||||||
response_encoding: ContentEncoding,
|
|
||||||
) {
|
|
||||||
let buf = self.take();
|
|
||||||
let version = resp.version().unwrap_or_else(|| req.version);
|
|
||||||
let mut len = 0;
|
let mut len = 0;
|
||||||
|
|
||||||
let has_body = match resp.body() {
|
let has_body = match resp.body() {
|
||||||
Body::Empty => false,
|
Body::Empty => false,
|
||||||
Body::Binary(ref bin) => {
|
Body::Binary(ref bin) => {
|
||||||
len = bin.len();
|
len = bin.len();
|
||||||
!(response_encoding == ContentEncoding::Auto && len < 96)
|
true
|
||||||
}
|
}
|
||||||
_ => true,
|
_ => true,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Enable content encoding only if response does not contain Content-Encoding
|
let has_body = match resp.body() {
|
||||||
// header
|
Body::Empty => false,
|
||||||
#[cfg(any(feature = "brotli", feature = "flate2"))]
|
_ => true,
|
||||||
let mut encoding = if has_body {
|
|
||||||
let encoding = match response_encoding {
|
|
||||||
ContentEncoding::Auto => {
|
|
||||||
// negotiate content-encoding
|
|
||||||
if let Some(val) = req.headers.get(ACCEPT_ENCODING) {
|
|
||||||
if let Ok(enc) = val.to_str() {
|
|
||||||
AcceptEncoding::parse(enc)
|
|
||||||
} else {
|
|
||||||
ContentEncoding::Identity
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
ContentEncoding::Identity
|
|
||||||
}
|
|
||||||
}
|
|
||||||
encoding => encoding,
|
|
||||||
};
|
|
||||||
if encoding.is_compression() {
|
|
||||||
info.content_encoding = Some(encoding.as_str());
|
|
||||||
}
|
|
||||||
encoding
|
|
||||||
} else {
|
|
||||||
ContentEncoding::Identity
|
|
||||||
};
|
};
|
||||||
#[cfg(not(any(feature = "brotli", feature = "flate2")))]
|
|
||||||
let mut encoding = ContentEncoding::Identity;
|
|
||||||
|
|
||||||
let transfer = match resp.body() {
|
let transfer = match resp.body() {
|
||||||
Body::Empty => {
|
Body::Empty => {
|
||||||
if !info.head {
|
if !self.head {
|
||||||
info.length = match resp.status() {
|
self.length = match resp.status() {
|
||||||
StatusCode::NO_CONTENT
|
StatusCode::NO_CONTENT
|
||||||
| StatusCode::CONTINUE
|
| StatusCode::CONTINUE
|
||||||
| StatusCode::SWITCHING_PROTOCOLS
|
| StatusCode::SWITCHING_PROTOCOLS
|
||||||
| StatusCode::PROCESSING => ResponseLength::None,
|
| StatusCode::PROCESSING => ResponseLength::None,
|
||||||
_ => ResponseLength::Zero,
|
_ => ResponseLength::Zero,
|
||||||
};
|
};
|
||||||
|
} else {
|
||||||
|
self.length = ResponseLength::Zero;
|
||||||
}
|
}
|
||||||
*self = Output::Empty(buf);
|
TransferEncoding::empty()
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
Body::Binary(_) => {
|
Body::Binary(_) => {
|
||||||
#[cfg(any(feature = "brotli", feature = "flate2"))]
|
self.length = ResponseLength::Length(len);
|
||||||
{
|
TransferEncoding::length(len as u64)
|
||||||
if !(encoding == ContentEncoding::Identity
|
|
||||||
|| encoding == ContentEncoding::Auto)
|
|
||||||
{
|
|
||||||
let mut tmp = BytesMut::new();
|
|
||||||
let mut transfer = TransferEncoding::eof(tmp);
|
|
||||||
let mut enc = match encoding {
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoding::Deflate => ContentEncoder::Deflate(
|
|
||||||
ZlibEncoder::new(transfer, Compression::fast()),
|
|
||||||
),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoding::Gzip => ContentEncoder::Gzip(
|
|
||||||
GzEncoder::new(transfer, Compression::fast()),
|
|
||||||
),
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoding::Br => {
|
|
||||||
ContentEncoder::Br(BrotliEncoder::new(transfer, 3))
|
|
||||||
}
|
|
||||||
ContentEncoding::Identity | ContentEncoding::Auto => {
|
|
||||||
unreachable!()
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let bin = resp.replace_body(Body::Empty).binary();
|
|
||||||
|
|
||||||
// TODO return error!
|
|
||||||
let _ = enc.write(bin.as_ref());
|
|
||||||
let _ = enc.write_eof();
|
|
||||||
let body = enc.buf_mut().take();
|
|
||||||
len = body.len();
|
|
||||||
resp.replace_body(Binary::from(body));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
info.length = ResponseLength::Length(len);
|
|
||||||
if info.head {
|
|
||||||
*self = Output::Empty(buf);
|
|
||||||
} else {
|
|
||||||
*self = Output::Buffer(buf);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
Body::Streaming(_) => {
|
Body::Streaming(_) => {
|
||||||
if resp.upgrade() {
|
if resp.upgrade() {
|
||||||
if version == Version::HTTP_2 {
|
self.length = ResponseLength::None;
|
||||||
error!("Connection upgrade is forbidden for HTTP/2");
|
TransferEncoding::eof()
|
||||||
}
|
|
||||||
if encoding != ContentEncoding::Identity {
|
|
||||||
encoding = ContentEncoding::Identity;
|
|
||||||
info.content_encoding.take();
|
|
||||||
}
|
|
||||||
TransferEncoding::eof(buf)
|
|
||||||
} else {
|
} else {
|
||||||
if !(encoding == ContentEncoding::Identity
|
self.streaming_encoding(version, resp)
|
||||||
|| encoding == ContentEncoding::Auto)
|
|
||||||
{
|
|
||||||
resp.headers_mut().remove(CONTENT_LENGTH);
|
|
||||||
}
|
|
||||||
Output::streaming_encoding(info, buf, version, resp)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
// check for head response
|
// check for head response
|
||||||
if info.head {
|
if self.head {
|
||||||
resp.set_body(Body::Empty);
|
resp.set_body(Body::Empty);
|
||||||
*self = Output::Empty(transfer.buf.unwrap());
|
} else {
|
||||||
return;
|
self.te = transfer;
|
||||||
}
|
}
|
||||||
|
|
||||||
let enc = match encoding {
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoding::Deflate => {
|
|
||||||
ContentEncoder::Deflate(ZlibEncoder::new(transfer, Compression::fast()))
|
|
||||||
}
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoding::Gzip => {
|
|
||||||
ContentEncoder::Gzip(GzEncoder::new(transfer, Compression::fast()))
|
|
||||||
}
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoding::Br => ContentEncoder::Br(BrotliEncoder::new(transfer, 3)),
|
|
||||||
ContentEncoding::Identity | ContentEncoding::Auto => {
|
|
||||||
*self = Output::TE(transfer);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
*self = Output::Encoder(enc);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn streaming_encoding(
|
fn streaming_encoding(
|
||||||
info: &mut ResponseInfo, buf: BytesMut, version: Version,
|
&mut self, version: Version, resp: &mut HttpResponse,
|
||||||
resp: &mut HttpResponse,
|
|
||||||
) -> TransferEncoding {
|
) -> TransferEncoding {
|
||||||
match resp.chunked() {
|
match resp.chunked() {
|
||||||
Some(true) => {
|
Some(true) => {
|
||||||
// Enable transfer encoding
|
// Enable transfer encoding
|
||||||
if version == Version::HTTP_2 {
|
if version == Version::HTTP_2 {
|
||||||
info.length = ResponseLength::None;
|
self.length = ResponseLength::None;
|
||||||
TransferEncoding::eof(buf)
|
TransferEncoding::eof()
|
||||||
} else {
|
} else {
|
||||||
info.length = ResponseLength::Chunked;
|
self.length = ResponseLength::Chunked;
|
||||||
TransferEncoding::chunked(buf)
|
TransferEncoding::chunked()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Some(false) => TransferEncoding::eof(buf),
|
Some(false) => TransferEncoding::eof(),
|
||||||
None => {
|
None => {
|
||||||
// if Content-Length is specified, then use it as length hint
|
// if Content-Length is specified, then use it as length hint
|
||||||
let (len, chunked) =
|
let (len, chunked) =
|
||||||
|
@ -339,21 +134,21 @@ impl Output {
|
||||||
|
|
||||||
if !chunked {
|
if !chunked {
|
||||||
if let Some(len) = len {
|
if let Some(len) = len {
|
||||||
info.length = ResponseLength::Length64(len);
|
self.length = ResponseLength::Length64(len);
|
||||||
TransferEncoding::length(len, buf)
|
TransferEncoding::length(len)
|
||||||
} else {
|
} else {
|
||||||
TransferEncoding::eof(buf)
|
TransferEncoding::eof()
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Enable transfer encoding
|
// Enable transfer encoding
|
||||||
match version {
|
match version {
|
||||||
Version::HTTP_11 => {
|
Version::HTTP_11 => {
|
||||||
info.length = ResponseLength::Chunked;
|
self.length = ResponseLength::Chunked;
|
||||||
TransferEncoding::chunked(buf)
|
TransferEncoding::chunked()
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
info.length = ResponseLength::None;
|
self.length = ResponseLength::None;
|
||||||
TransferEncoding::eof(buf)
|
TransferEncoding::eof()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -362,178 +157,9 @@ impl Output {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) enum ContentEncoder {
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
Deflate(ZlibEncoder<TransferEncoding>),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
Gzip(GzEncoder<TransferEncoding>),
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
Br(BrotliEncoder<TransferEncoding>),
|
|
||||||
Identity(TransferEncoding),
|
|
||||||
}
|
|
||||||
|
|
||||||
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(feature = "flate2")]
|
|
||||||
ContentEncoder::Deflate(_) => writeln!(f, "ContentEncoder(Deflate)"),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Gzip(_) => writeln!(f, "ContentEncoder(Gzip)"),
|
|
||||||
ContentEncoder::Identity(_) => writeln!(f, "ContentEncoder(Identity)"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ContentEncoder {
|
|
||||||
#[inline]
|
|
||||||
pub fn len(&self) -> usize {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(ref encoder) => encoder.get_ref().len(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Deflate(ref encoder) => encoder.get_ref().len(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Gzip(ref encoder) => encoder.get_ref().len(),
|
|
||||||
ContentEncoder::Identity(ref encoder) => encoder.len(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn is_empty(&self) -> bool {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(ref encoder) => encoder.get_ref().is_empty(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Deflate(ref encoder) => encoder.get_ref().is_empty(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Gzip(ref encoder) => encoder.get_ref().is_empty(),
|
|
||||||
ContentEncoder::Identity(ref encoder) => encoder.is_empty(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub(crate) fn take_buf(&mut self) -> BytesMut {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(ref mut encoder) => encoder.get_mut().take(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Deflate(ref mut encoder) => encoder.get_mut().take(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Gzip(ref mut encoder) => encoder.get_mut().take(),
|
|
||||||
ContentEncoder::Identity(ref mut encoder) => encoder.take(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub(crate) fn buf_mut(&mut self) -> &mut BytesMut {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(ref mut encoder) => encoder.get_mut().buf_mut(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Deflate(ref mut encoder) => encoder.get_mut().buf_mut(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Gzip(ref mut encoder) => encoder.get_mut().buf_mut(),
|
|
||||||
ContentEncoder::Identity(ref mut encoder) => encoder.buf_mut(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub(crate) fn buf_ref(&mut self) -> &BytesMut {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(ref mut encoder) => encoder.get_mut().buf_ref(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Deflate(ref mut encoder) => encoder.get_mut().buf_ref(),
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Gzip(ref mut encoder) => encoder.get_mut().buf_ref(),
|
|
||||||
ContentEncoder::Identity(ref mut encoder) => encoder.buf_ref(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg_attr(feature = "cargo-clippy", allow(clippy::inline_always))]
|
|
||||||
#[inline(always)]
|
|
||||||
pub fn write_eof(&mut self) -> Result<bool, io::Error> {
|
|
||||||
let encoder =
|
|
||||||
mem::replace(self, ContentEncoder::Identity(TransferEncoding::empty()));
|
|
||||||
|
|
||||||
match encoder {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(encoder) => match encoder.finish() {
|
|
||||||
Ok(mut writer) => {
|
|
||||||
writer.encode_eof();
|
|
||||||
*self = ContentEncoder::Identity(writer);
|
|
||||||
Ok(true)
|
|
||||||
}
|
|
||||||
Err(err) => Err(err),
|
|
||||||
},
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Gzip(encoder) => match encoder.finish() {
|
|
||||||
Ok(mut writer) => {
|
|
||||||
writer.encode_eof();
|
|
||||||
*self = ContentEncoder::Identity(writer);
|
|
||||||
Ok(true)
|
|
||||||
}
|
|
||||||
Err(err) => Err(err),
|
|
||||||
},
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Deflate(encoder) => match encoder.finish() {
|
|
||||||
Ok(mut writer) => {
|
|
||||||
writer.encode_eof();
|
|
||||||
*self = ContentEncoder::Identity(writer);
|
|
||||||
Ok(true)
|
|
||||||
}
|
|
||||||
Err(err) => Err(err),
|
|
||||||
},
|
|
||||||
ContentEncoder::Identity(mut writer) => {
|
|
||||||
let res = writer.encode_eof();
|
|
||||||
*self = ContentEncoder::Identity(writer);
|
|
||||||
Ok(res)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg_attr(feature = "cargo-clippy", allow(clippy::inline_always))]
|
|
||||||
#[inline(always)]
|
|
||||||
pub fn write(&mut self, data: &[u8]) -> Result<(), io::Error> {
|
|
||||||
match *self {
|
|
||||||
#[cfg(feature = "brotli")]
|
|
||||||
ContentEncoder::Br(ref mut encoder) => match encoder.write_all(data) {
|
|
||||||
Ok(_) => Ok(()),
|
|
||||||
Err(err) => {
|
|
||||||
trace!("Error decoding br encoding: {}", err);
|
|
||||||
Err(err)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Gzip(ref mut encoder) => match encoder.write_all(data) {
|
|
||||||
Ok(_) => Ok(()),
|
|
||||||
Err(err) => {
|
|
||||||
trace!("Error decoding gzip encoding: {}", err);
|
|
||||||
Err(err)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
#[cfg(feature = "flate2")]
|
|
||||||
ContentEncoder::Deflate(ref mut encoder) => match encoder.write_all(data) {
|
|
||||||
Ok(_) => Ok(()),
|
|
||||||
Err(err) => {
|
|
||||||
trace!("Error decoding deflate encoding: {}", err);
|
|
||||||
Err(err)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
ContentEncoder::Identity(ref mut encoder) => {
|
|
||||||
encoder.encode(data)?;
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Encoders to handle different Transfer-Encodings.
|
/// Encoders to handle different Transfer-Encodings.
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub(crate) struct TransferEncoding {
|
pub(crate) struct TransferEncoding {
|
||||||
buf: Option<BytesMut>,
|
|
||||||
kind: TransferEncodingKind,
|
kind: TransferEncodingKind,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -552,65 +178,41 @@ enum TransferEncodingKind {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl TransferEncoding {
|
impl TransferEncoding {
|
||||||
fn take(&mut self) -> BytesMut {
|
|
||||||
self.buf.take().unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn buf_ref(&mut self) -> &BytesMut {
|
|
||||||
self.buf.as_ref().unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn len(&self) -> usize {
|
|
||||||
self.buf.as_ref().unwrap().len()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_empty(&self) -> bool {
|
|
||||||
self.buf.as_ref().unwrap().is_empty()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn buf_mut(&mut self) -> &mut BytesMut {
|
|
||||||
self.buf.as_mut().unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn empty() -> TransferEncoding {
|
pub fn empty() -> TransferEncoding {
|
||||||
TransferEncoding {
|
TransferEncoding {
|
||||||
buf: None,
|
|
||||||
kind: TransferEncodingKind::Eof,
|
kind: TransferEncodingKind::Eof,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn eof(buf: BytesMut) -> TransferEncoding {
|
pub fn eof() -> TransferEncoding {
|
||||||
TransferEncoding {
|
TransferEncoding {
|
||||||
buf: Some(buf),
|
|
||||||
kind: TransferEncodingKind::Eof,
|
kind: TransferEncodingKind::Eof,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn chunked(buf: BytesMut) -> TransferEncoding {
|
pub fn chunked() -> TransferEncoding {
|
||||||
TransferEncoding {
|
TransferEncoding {
|
||||||
buf: Some(buf),
|
|
||||||
kind: TransferEncodingKind::Chunked(false),
|
kind: TransferEncodingKind::Chunked(false),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn length(len: u64, buf: BytesMut) -> TransferEncoding {
|
pub fn length(len: u64) -> TransferEncoding {
|
||||||
TransferEncoding {
|
TransferEncoding {
|
||||||
buf: Some(buf),
|
|
||||||
kind: TransferEncodingKind::Length(len),
|
kind: TransferEncodingKind::Length(len),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Encode message. Return `EOF` state of encoder
|
/// Encode message. Return `EOF` state of encoder
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn encode(&mut self, msg: &[u8]) -> io::Result<bool> {
|
pub fn encode(&mut self, msg: &[u8], buf: &mut BytesMut) -> io::Result<bool> {
|
||||||
match self.kind {
|
match self.kind {
|
||||||
TransferEncodingKind::Eof => {
|
TransferEncodingKind::Eof => {
|
||||||
let eof = msg.is_empty();
|
let eof = msg.is_empty();
|
||||||
self.buf.as_mut().unwrap().extend_from_slice(msg);
|
buf.extend_from_slice(msg);
|
||||||
Ok(eof)
|
Ok(eof)
|
||||||
}
|
}
|
||||||
TransferEncodingKind::Chunked(ref mut eof) => {
|
TransferEncodingKind::Chunked(ref mut eof) => {
|
||||||
|
@ -620,17 +222,14 @@ impl TransferEncoding {
|
||||||
|
|
||||||
if msg.is_empty() {
|
if msg.is_empty() {
|
||||||
*eof = true;
|
*eof = true;
|
||||||
self.buf.as_mut().unwrap().extend_from_slice(b"0\r\n\r\n");
|
buf.extend_from_slice(b"0\r\n\r\n");
|
||||||
} else {
|
} else {
|
||||||
let mut buf = BytesMut::new();
|
writeln!(buf.as_mut(), "{:X}\r", msg.len())
|
||||||
writeln!(&mut buf, "{:X}\r", msg.len())
|
|
||||||
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;
|
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;
|
||||||
|
|
||||||
let b = self.buf.as_mut().unwrap();
|
buf.reserve(msg.len() + 2);
|
||||||
b.reserve(buf.len() + msg.len() + 2);
|
buf.extend_from_slice(msg);
|
||||||
b.extend_from_slice(buf.as_ref());
|
buf.extend_from_slice(b"\r\n");
|
||||||
b.extend_from_slice(msg);
|
|
||||||
b.extend_from_slice(b"\r\n");
|
|
||||||
}
|
}
|
||||||
Ok(*eof)
|
Ok(*eof)
|
||||||
}
|
}
|
||||||
|
@ -641,10 +240,7 @@ impl TransferEncoding {
|
||||||
}
|
}
|
||||||
let len = cmp::min(*remaining, msg.len() as u64);
|
let len = cmp::min(*remaining, msg.len() as u64);
|
||||||
|
|
||||||
self.buf
|
buf.extend_from_slice(&msg[..len as usize]);
|
||||||
.as_mut()
|
|
||||||
.unwrap()
|
|
||||||
.extend_from_slice(&msg[..len as usize]);
|
|
||||||
|
|
||||||
*remaining -= len as u64;
|
*remaining -= len as u64;
|
||||||
Ok(*remaining == 0)
|
Ok(*remaining == 0)
|
||||||
|
@ -657,14 +253,14 @@ impl TransferEncoding {
|
||||||
|
|
||||||
/// Encode eof. Return `EOF` state of encoder
|
/// Encode eof. Return `EOF` state of encoder
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn encode_eof(&mut self) -> bool {
|
pub fn encode_eof(&mut self, buf: &mut BytesMut) -> bool {
|
||||||
match self.kind {
|
match self.kind {
|
||||||
TransferEncodingKind::Eof => true,
|
TransferEncodingKind::Eof => true,
|
||||||
TransferEncodingKind::Length(rem) => rem == 0,
|
TransferEncodingKind::Length(rem) => rem == 0,
|
||||||
TransferEncodingKind::Chunked(ref mut eof) => {
|
TransferEncodingKind::Chunked(ref mut eof) => {
|
||||||
if !*eof {
|
if !*eof {
|
||||||
*eof = true;
|
*eof = true;
|
||||||
self.buf.as_mut().unwrap().extend_from_slice(b"0\r\n\r\n");
|
buf.extend_from_slice(b"0\r\n\r\n");
|
||||||
}
|
}
|
||||||
true
|
true
|
||||||
}
|
}
|
||||||
|
@ -675,9 +271,9 @@ impl TransferEncoding {
|
||||||
impl io::Write for TransferEncoding {
|
impl io::Write for TransferEncoding {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
||||||
if self.buf.is_some() {
|
// if self.buf.is_some() {
|
||||||
self.encode(buf)?;
|
// self.encode(buf)?;
|
||||||
}
|
// }
|
||||||
Ok(buf.len())
|
Ok(buf.len())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue