1
0
mirror of https://github.com/actix/actix-extras.git synced 2025-01-23 23:34:35 +01:00
actix-extras/src/server/encoding.rs

930 lines
30 KiB
Rust
Raw Normal View History

2017-11-08 16:44:23 -08:00
use std::fmt::Write as FmtWrite;
2018-04-13 16:02:01 -07:00
use std::io::{Read, Write};
2017-11-08 16:44:23 -08:00
use std::str::FromStr;
use std::{cmp, io, mem, ptr};
2017-11-08 16:44:23 -08:00
2018-04-13 16:02:01 -07:00
#[cfg(feature = "brotli")]
use brotli2::write::{BrotliDecoder, BrotliEncoder};
use bytes::{BufMut, Bytes, BytesMut};
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-01-12 12:31:33 -08:00
use flate2::read::GzDecoder;
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-04-13 16:02:01 -07:00
use flate2::write::{DeflateDecoder, DeflateEncoder, GzEncoder};
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
use flate2::Compression;
2018-05-15 16:41:46 -07:00
use http::header::{
HeaderMap, HeaderValue, ACCEPT_ENCODING, CONTENT_ENCODING, CONTENT_LENGTH,
TRANSFER_ENCODING,
};
2018-04-13 16:02:01 -07:00
use http::{HttpTryFrom, Method, Version};
2017-11-06 16:23:58 -08:00
2018-04-13 16:02:01 -07:00
use body::{Binary, Body};
2017-11-15 20:06:28 -10:00
use error::PayloadError;
2018-04-13 16:02:01 -07:00
use header::ContentEncoding;
use httprequest::HttpInnerMessage;
2017-11-08 16:44:23 -08:00
use httpresponse::HttpResponse;
2018-04-13 16:02:01 -07:00
use payload::{PayloadSender, PayloadStatus, PayloadWriter};
2017-11-06 16:23:58 -08:00
pub(crate) enum PayloadType {
Sender(PayloadSender),
2017-11-26 17:30:35 -08:00
Encoding(Box<EncodedPayload>),
2017-11-06 16:23:58 -08:00
}
impl PayloadType {
#[cfg(any(feature = "brotli", feature = "flate2"))]
2017-11-06 16:23:58 -08:00
pub fn new(headers: &HeaderMap, sender: PayloadSender) -> PayloadType {
// check content-encoding
let enc = if let Some(enc) = headers.get(CONTENT_ENCODING) {
if let Ok(enc) = enc.to_str() {
ContentEncoding::from(enc)
} else {
ContentEncoding::Auto
}
} else {
ContentEncoding::Auto
};
match enc {
2018-04-13 16:02:01 -07:00
ContentEncoding::Auto | ContentEncoding::Identity => {
PayloadType::Sender(sender)
}
2017-11-26 17:30:35 -08:00
_ => PayloadType::Encoding(Box::new(EncodedPayload::new(sender, enc))),
2017-11-06 16:23:58 -08:00
}
}
#[cfg(not(any(feature = "brotli", feature = "flate2")))]
pub fn new(headers: &HeaderMap, sender: PayloadSender) -> PayloadType {
PayloadType::Sender(sender)
}
2017-11-06 16:23:58 -08:00
}
impl PayloadWriter for PayloadType {
2018-02-09 16:20:10 -08:00
#[inline]
2017-11-06 16:23:58 -08:00
fn set_error(&mut self, err: PayloadError) {
match *self {
PayloadType::Sender(ref mut sender) => sender.set_error(err),
PayloadType::Encoding(ref mut enc) => enc.set_error(err),
}
}
2018-02-09 16:20:10 -08:00
#[inline]
2017-11-06 16:23:58 -08:00
fn feed_eof(&mut self) {
match *self {
PayloadType::Sender(ref mut sender) => sender.feed_eof(),
PayloadType::Encoding(ref mut enc) => enc.feed_eof(),
}
}
2018-02-09 16:20:10 -08:00
#[inline]
2017-11-06 16:23:58 -08:00
fn feed_data(&mut self, data: Bytes) {
match *self {
PayloadType::Sender(ref mut sender) => sender.feed_data(data),
PayloadType::Encoding(ref mut enc) => enc.feed_data(data),
}
}
2018-02-09 16:20:10 -08:00
#[inline]
fn need_read(&self) -> PayloadStatus {
2017-11-06 16:23:58 -08:00
match *self {
2018-02-26 20:07:22 -08:00
PayloadType::Sender(ref sender) => sender.need_read(),
PayloadType::Encoding(ref enc) => enc.need_read(),
2017-11-06 16:23:58 -08:00
}
}
}
/// Payload wrapper with content decompression support
pub(crate) struct EncodedPayload {
inner: PayloadSender,
error: bool,
payload: PayloadStream,
}
impl EncodedPayload {
pub fn new(inner: PayloadSender, enc: ContentEncoding) -> EncodedPayload {
2018-04-13 16:02:01 -07:00
EncodedPayload {
inner,
error: false,
payload: PayloadStream::new(enc),
}
}
}
impl PayloadWriter for EncodedPayload {
fn set_error(&mut self, err: PayloadError) {
self.inner.set_error(err)
}
fn feed_eof(&mut self) {
if !self.error {
match self.payload.feed_eof() {
Err(err) => {
self.error = true;
self.set_error(PayloadError::Io(err));
2018-04-13 16:02:01 -07:00
}
Ok(value) => {
if let Some(b) = value {
self.inner.feed_data(b);
}
self.inner.feed_eof();
}
}
}
}
fn feed_data(&mut self, data: Bytes) {
if self.error {
2018-04-13 16:02:01 -07:00
return;
}
match self.payload.feed_data(data) {
Ok(Some(b)) => self.inner.feed_data(b),
Ok(None) => (),
Err(e) => {
self.error = true;
self.set_error(e.into());
}
}
}
#[inline]
fn need_read(&self) -> PayloadStatus {
self.inner.need_read()
}
}
2018-02-24 07:29:35 +03:00
pub(crate) enum Decoder {
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
Deflate(Box<DeflateDecoder<Writer>>),
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-01-12 12:31:33 -08:00
Gzip(Option<Box<GzDecoder<Wrapper>>>),
2018-04-13 16:02:01 -07:00
#[cfg(feature = "brotli")]
Br(Box<BrotliDecoder<Writer>>),
2017-11-06 16:23:58 -08:00
Identity,
}
2018-01-12 12:31:33 -08:00
// should go after write::GzDecoder get implemented
#[derive(Debug)]
2018-02-24 07:29:35 +03:00
pub(crate) struct Wrapper {
pub buf: BytesMut,
pub eof: bool,
2018-01-12 12:31:33 -08:00
}
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 {
buf: BytesMut,
}
impl Writer {
fn new() -> Writer {
2018-04-13 16:02:01 -07:00
Writer {
buf: BytesMut::with_capacity(8192),
}
}
fn take(&mut self) -> Bytes {
self.buf.take().freeze()
}
}
impl io::Write for Writer {
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(())
}
}
2018-02-25 11:43:00 +03:00
/// Payload stream with decompression support
pub(crate) struct PayloadStream {
2017-11-06 16:23:58 -08:00
decoder: Decoder,
2018-01-12 12:31:33 -08:00
dst: BytesMut,
2017-11-06 16:23:58 -08:00
}
2018-02-25 11:43:00 +03:00
impl PayloadStream {
pub fn new(enc: ContentEncoding) -> PayloadStream {
2017-11-06 16:23:58 -08:00
let dec = match enc {
2018-04-13 16:02:01 -07:00
#[cfg(feature = "brotli")]
ContentEncoding::Br => {
Decoder::Br(Box::new(BrotliDecoder::new(Writer::new())))
}
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-04-13 16:02:01 -07:00
ContentEncoding::Deflate => {
Decoder::Deflate(Box::new(DeflateDecoder::new(Writer::new())))
}
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-01-12 12:31:33 -08:00
ContentEncoding::Gzip => Decoder::Gzip(None),
2017-11-06 16:23:58 -08:00
_ => Decoder::Identity,
};
2018-04-13 16:02:01 -07:00
PayloadStream {
decoder: dec,
dst: BytesMut::new(),
}
2017-11-06 16:23:58 -08:00
}
}
2018-02-25 11:43:00 +03:00
impl PayloadStream {
pub fn feed_eof(&mut self) -> io::Result<Option<Bytes>> {
match self.decoder {
2018-04-13 16:02:01 -07:00
#[cfg(feature = "brotli")]
Decoder::Br(ref mut decoder) => match decoder.finish() {
Ok(mut writer) => {
let b = writer.take();
if !b.is_empty() {
Ok(Some(b))
} else {
Ok(None)
}
2017-11-06 16:23:58 -08:00
}
2018-04-13 16:02:01 -07:00
Err(e) => Err(e),
2017-11-07 15:59:37 -08:00
},
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2017-11-06 16:23:58 -08:00
Decoder::Gzip(ref mut decoder) => {
2018-01-12 12:31:33 -08:00
if let Some(ref mut decoder) = *decoder {
decoder.as_mut().get_mut().eof = true;
2018-03-06 11:02:03 -08:00
self.dst.reserve(8192);
2018-04-13 16:02:01 -07:00
match decoder.read(unsafe { self.dst.bytes_mut() }) {
Ok(n) => {
unsafe { self.dst.advance_mut(n) };
return Ok(Some(self.dst.take().freeze()));
2017-11-06 16:23:58 -08:00
}
2018-04-13 16:02:01 -07:00
Err(e) => return Err(e),
2018-01-12 12:31:33 -08:00
}
} else {
2018-02-25 11:43:00 +03:00
Ok(None)
2017-11-06 16:23:58 -08:00
}
2018-04-13 16:02:01 -07:00
}
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-04-13 16:02:01 -07:00
Decoder::Deflate(ref mut decoder) => match decoder.try_finish() {
Ok(_) => {
let b = decoder.get_mut().take();
if !b.is_empty() {
Ok(Some(b))
} else {
Ok(None)
}
2017-11-06 16:23:58 -08:00
}
2018-04-13 16:02:01 -07:00
Err(e) => Err(e),
2017-11-06 16:23:58 -08:00
},
2018-02-25 11:43:00 +03:00
Decoder::Identity => Ok(None),
2017-11-06 16:23:58 -08:00
}
}
2018-02-25 11:43:00 +03:00
pub fn feed_data(&mut self, data: Bytes) -> io::Result<Option<Bytes>> {
2017-11-06 16:23:58 -08:00
match self.decoder {
2018-04-13 16:02:01 -07:00
#[cfg(feature = "brotli")]
Decoder::Br(ref mut decoder) => match decoder.write_all(&data) {
Ok(_) => {
decoder.flush()?;
let b = decoder.get_mut().take();
if !b.is_empty() {
Ok(Some(b))
} else {
Ok(None)
}
2017-11-06 16:23:58 -08:00
}
2018-04-13 16:02:01 -07:00
Err(e) => Err(e),
2018-02-25 11:43:00 +03:00
},
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2017-11-06 16:23:58 -08:00
Decoder::Gzip(ref mut decoder) => {
2018-01-12 12:31:33 -08:00
if decoder.is_none() {
2018-04-13 16:02:01 -07:00
*decoder = Some(Box::new(GzDecoder::new(Wrapper {
buf: BytesMut::from(data),
eof: false,
})));
2018-01-12 12:31:33 -08:00
} else {
let _ = decoder.as_mut().unwrap().write(&data);
}
loop {
self.dst.reserve(8192);
2018-04-13 16:02:01 -07:00
match decoder
.as_mut()
.as_mut()
.unwrap()
.read(unsafe { self.dst.bytes_mut() })
2018-03-06 11:02:03 -08:00
{
2018-04-13 16:02:01 -07:00
Ok(n) => {
2018-03-06 11:02:03 -08:00
if n != 0 {
2018-04-13 16:02:01 -07:00
unsafe { self.dst.advance_mut(n) };
2018-03-06 11:02:03 -08:00
}
2018-01-12 12:31:33 -08:00
if n == 0 {
return Ok(Some(self.dst.take().freeze()));
2018-01-12 12:31:33 -08:00
}
}
Err(e) => {
2018-04-13 16:02:01 -07:00
if e.kind() == io::ErrorKind::WouldBlock
&& !self.dst.is_empty()
2018-03-06 11:02:03 -08:00
{
return Ok(Some(self.dst.take().freeze()));
}
2018-04-13 16:02:01 -07:00
return Err(e);
}
2017-11-06 16:23:58 -08:00
}
}
2018-04-13 16:02:01 -07:00
}
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-04-13 16:02:01 -07:00
Decoder::Deflate(ref mut decoder) => match decoder.write_all(&data) {
Ok(_) => {
decoder.flush()?;
let b = decoder.get_mut().take();
if !b.is_empty() {
Ok(Some(b))
} else {
Ok(None)
}
2018-02-25 11:43:00 +03:00
}
2018-04-13 16:02:01 -07:00
Err(e) => Err(e),
2018-02-25 11:43:00 +03:00
},
Decoder::Identity => Ok(Some(data)),
}
}
}
pub(crate) enum ContentEncoder {
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-02-25 11:43:00 +03:00
Deflate(DeflateEncoder<TransferEncoding>),
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-02-25 11:43:00 +03:00
Gzip(GzEncoder<TransferEncoding>),
2018-04-13 16:02:01 -07:00
#[cfg(feature = "brotli")]
2018-02-25 11:43:00 +03:00
Br(BrotliEncoder<TransferEncoding>),
Identity(TransferEncoding),
}
2017-11-08 16:44:23 -08:00
2018-02-25 11:43:00 +03:00
impl ContentEncoder {
pub fn empty() -> ContentEncoder {
ContentEncoder::Identity(TransferEncoding::eof())
2017-12-14 19:34:31 -08:00
}
2018-04-13 16:02:01 -07:00
pub fn for_server(
buf: &mut BytesMut, req: &HttpInnerMessage, resp: &mut HttpResponse,
2018-04-13 16:02:01 -07:00
response_encoding: ContentEncoding,
) -> ContentEncoder {
2017-12-09 04:33:40 -08:00
let version = resp.version().unwrap_or_else(|| req.version);
2018-03-18 11:05:44 -07:00
let is_head = req.method == Method::HEAD;
2018-05-15 16:41:46 -07:00
let mut len = 0;
2018-05-17 12:23:37 -07:00
#[cfg_attr(feature = "cargo-clippy", allow(match_ref_pats))]
2018-05-15 16:41:46 -07:00
let has_body = match resp.body() {
2018-05-15 19:07:43 -07:00
&Body::Empty => false,
&Body::Binary(ref bin) => {
2018-05-15 16:41:46 -07:00
len = bin.len();
!(response_encoding == ContentEncoding::Auto && len < 96)
2018-04-13 16:02:01 -07:00
}
_ => true,
};
2017-11-08 16:44:23 -08:00
2018-04-13 16:02:01 -07:00
// Enable content encoding only if response does not contain Content-Encoding
// header
#[cfg(any(feature = "brotli", feature = "flate2"))]
2018-01-10 22:42:26 -08:00
let mut encoding = if has_body {
let encoding = match response_encoding {
2017-11-08 16:44:23 -08:00
ContentEncoding::Auto => {
// negotiate content-encoding
2017-12-09 04:33:40 -08:00
if let Some(val) = req.headers.get(ACCEPT_ENCODING) {
2017-11-08 16:44:23 -08:00
if let Ok(enc) = val.to_str() {
AcceptEncoding::parse(enc)
} else {
ContentEncoding::Identity
}
} else {
ContentEncoding::Identity
}
}
encoding => encoding,
};
2017-12-13 16:44:35 -08:00
if encoding.is_compression() {
resp.headers_mut().insert(
2018-04-13 16:02:01 -07:00
CONTENT_ENCODING,
HeaderValue::from_static(encoding.as_str()),
);
2017-12-13 16:44:35 -08:00
}
2017-11-08 16:44:23 -08:00
encoding
} else {
ContentEncoding::Identity
};
#[cfg(not(any(feature = "brotli", feature = "flate2")))]
let mut encoding = ContentEncoding::Identity;
2017-11-08 16:44:23 -08:00
2018-05-17 12:23:37 -07:00
#[cfg_attr(feature = "cargo-clippy", allow(match_ref_pats))]
2018-05-15 16:41:46 -07:00
let mut transfer = match resp.body() {
2018-05-15 19:07:43 -07:00
&Body::Empty => {
2018-01-20 16:12:38 -08:00
if req.method != Method::HEAD {
resp.headers_mut().remove(CONTENT_LENGTH);
}
TransferEncoding::length(0)
2018-04-13 16:02:01 -07:00
}
2018-05-15 19:07:43 -07:00
&Body::Binary(_) => {
#[cfg(any(feature = "brotli", feature = "flate2"))]
2018-03-18 11:05:44 -07:00
{
if !(encoding == ContentEncoding::Identity
|| encoding == ContentEncoding::Auto)
{
let mut tmp = BytesMut::default();
let mut transfer = TransferEncoding::eof();
transfer.set_buffer(&mut tmp);
let mut enc = match encoding {
#[cfg(feature = "flate2")]
ContentEncoding::Deflate => ContentEncoder::Deflate(
DeflateEncoder::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!()
}
};
2018-05-15 16:41:46 -07:00
let bin = resp.replace_body(Body::Empty).binary();
2018-05-15 16:41:46 -07:00
// TODO return error!
let _ = enc.write(bin.as_ref());
let _ = enc.write_eof();
let body = tmp.take();
len = body.len();
encoding = ContentEncoding::Identity;
resp.replace_body(Binary::from(body));
}
2017-11-08 16:44:23 -08:00
}
2018-05-15 16:41:46 -07:00
2018-03-18 11:05:44 -07:00
if is_head {
2018-01-20 16:12:38 -08:00
let mut b = BytesMut::new();
2018-05-15 16:41:46 -07:00
let _ = write!(b, "{}", len);
2018-01-20 16:12:38 -08:00
resp.headers_mut().insert(
2018-04-13 16:02:01 -07:00
CONTENT_LENGTH,
HeaderValue::try_from(b.freeze()).unwrap(),
);
2018-01-20 16:12:38 -08:00
} else {
2018-03-18 11:05:44 -07:00
// resp.headers_mut().remove(CONTENT_LENGTH);
2018-01-20 16:12:38 -08:00
}
TransferEncoding::eof()
2017-11-08 16:44:23 -08:00
}
2018-05-15 19:07:43 -07:00
&Body::Streaming(_) | &Body::Actor(_) => {
2017-12-31 17:26:32 -08:00
if resp.upgrade() {
if version == Version::HTTP_2 {
error!("Connection upgrade is forbidden for HTTP/2");
}
if encoding != ContentEncoding::Identity {
encoding = ContentEncoding::Identity;
resp.headers_mut().remove(CONTENT_ENCODING);
}
TransferEncoding::eof()
2017-11-08 16:44:23 -08:00
} else {
if !(encoding == ContentEncoding::Identity
|| encoding == ContentEncoding::Auto)
{
resp.headers_mut().remove(CONTENT_LENGTH);
}
ContentEncoder::streaming_encoding(version, resp)
2017-11-08 16:44:23 -08:00
}
}
};
2018-05-15 16:41:46 -07:00
// check for head response
2018-03-18 11:05:44 -07:00
if is_head {
2018-05-15 16:41:46 -07:00
resp.set_body(Body::Empty);
2018-01-20 16:12:38 -08:00
transfer.kind = TransferEncodingKind::Length(0);
}
transfer.set_buffer(buf);
2017-11-08 16:44:23 -08:00
2018-02-25 11:43:00 +03:00
match encoding {
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-04-13 16:02:01 -07:00
ContentEncoding::Deflate => ContentEncoder::Deflate(DeflateEncoder::new(
transfer,
Compression::fast(),
)),
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-04-13 16:02:01 -07:00
ContentEncoding::Gzip => {
ContentEncoder::Gzip(GzEncoder::new(transfer, Compression::fast()))
}
#[cfg(feature = "brotli")]
ContentEncoding::Br => ContentEncoder::Br(BrotliEncoder::new(transfer, 3)),
ContentEncoding::Identity | ContentEncoding::Auto => {
ContentEncoder::Identity(transfer)
}
2018-02-25 11:43:00 +03:00
}
2017-11-08 16:44:23 -08:00
}
2018-01-11 15:26:46 -08:00
2018-04-13 16:02:01 -07:00
fn streaming_encoding(
version: Version, resp: &mut HttpResponse,
2018-04-13 16:02:01 -07:00
) -> TransferEncoding {
match resp.chunked() {
Some(true) => {
// Enable transfer encoding
resp.headers_mut().remove(CONTENT_LENGTH);
if version == Version::HTTP_2 {
resp.headers_mut().remove(TRANSFER_ENCODING);
TransferEncoding::eof()
} else {
2018-04-13 16:02:01 -07:00
resp.headers_mut()
.insert(TRANSFER_ENCODING, HeaderValue::from_static("chunked"));
TransferEncoding::chunked()
}
2018-04-13 16:02:01 -07:00
}
Some(false) => TransferEncoding::eof(),
None => {
// if Content-Length is specified, then use it as length hint
let (len, chunked) =
if let Some(len) = resp.headers().get(CONTENT_LENGTH) {
// Content-Length
if let Ok(s) = len.to_str() {
if let Ok(len) = s.parse::<u64>() {
(Some(len), false)
} else {
error!("illegal Content-Length: {:?}", len);
(None, false)
}
2018-01-11 15:26:46 -08:00
} else {
error!("illegal Content-Length: {:?}", len);
(None, false)
}
} else {
(None, true)
};
2018-01-11 15:26:46 -08:00
if !chunked {
if let Some(len) = len {
TransferEncoding::length(len)
} else {
TransferEncoding::eof()
}
} else {
// Enable transfer encoding
match version {
Version::HTTP_11 => {
resp.headers_mut().insert(
2018-04-13 16:02:01 -07:00
TRANSFER_ENCODING,
HeaderValue::from_static("chunked"),
);
TransferEncoding::chunked()
2018-04-13 16:02:01 -07:00
}
_ => {
resp.headers_mut().remove(TRANSFER_ENCODING);
TransferEncoding::eof()
}
}
2018-01-11 15:26:46 -08:00
}
}
}
}
2017-11-08 16:44:23 -08:00
}
impl ContentEncoder {
2017-12-12 21:32:58 -08:00
#[inline]
2017-11-08 16:44:23 -08:00
pub fn is_eof(&self) -> bool {
match *self {
2018-04-13 16:02:01 -07:00
#[cfg(feature = "brotli")]
2018-01-14 19:28:34 -08:00
ContentEncoder::Br(ref encoder) => encoder.get_ref().is_eof(),
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-01-14 19:28:34 -08:00
ContentEncoder::Deflate(ref encoder) => encoder.get_ref().is_eof(),
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-01-14 19:28:34 -08:00
ContentEncoder::Gzip(ref encoder) => encoder.get_ref().is_eof(),
ContentEncoder::Identity(ref encoder) => encoder.is_eof(),
2017-11-08 16:44:23 -08:00
}
}
2017-12-13 16:44:35 -08:00
#[cfg_attr(feature = "cargo-clippy", allow(inline_always))]
2017-12-13 12:47:07 -08:00
#[inline(always)]
2018-05-15 16:41:46 -07:00
pub fn write_eof(&mut self) -> Result<bool, io::Error> {
let encoder =
mem::replace(self, ContentEncoder::Identity(TransferEncoding::eof()));
2017-11-08 16:44:23 -08:00
match encoder {
2018-04-13 16:02:01 -07:00
#[cfg(feature = "brotli")]
ContentEncoder::Br(encoder) => match encoder.finish() {
Ok(mut writer) => {
writer.encode_eof();
*self = ContentEncoder::Identity(writer);
2018-05-15 16:41:46 -07:00
Ok(true)
2017-11-08 16:44:23 -08:00
}
2018-04-13 16:02:01 -07:00
Err(err) => Err(err),
},
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-04-13 16:02:01 -07:00
ContentEncoder::Gzip(encoder) => match encoder.finish() {
Ok(mut writer) => {
writer.encode_eof();
*self = ContentEncoder::Identity(writer);
2018-05-15 16:41:46 -07:00
Ok(true)
2017-11-08 16:44:23 -08:00
}
2018-04-13 16:02:01 -07:00
Err(err) => Err(err),
2017-11-08 16:44:23 -08:00
},
2018-04-24 12:24:04 -07:00
#[cfg(feature = "flate2")]
2018-04-13 16:02:01 -07:00
ContentEncoder::Deflate(encoder) => match encoder.finish() {
Ok(mut writer) => {
writer.encode_eof();
*self = ContentEncoder::Identity(writer);
2018-05-15 16:41:46 -07:00
Ok(true)
2017-11-08 16:44:23 -08:00
}
2018-04-13 16:02:01 -07:00
Err(err) => Err(err),
2017-11-08 16:44:23 -08:00
},
ContentEncoder::Identity(mut writer) => {
2018-05-15 16:41:46 -07:00
let res = writer.encode_eof();
2018-01-02 14:53:51 -08:00
*self = ContentEncoder::Identity(writer);
2018-05-15 16:41:46 -07:00
Ok(res)
2017-11-08 16:44:23 -08:00
}
}
}
2017-12-13 16:44:35 -08:00
#[cfg_attr(feature = "cargo-clippy", allow(inline_always))]
2017-12-13 12:47:07 -08:00
#[inline(always)]
pub fn write(&mut self, data: &[u8]) -> Result<(), io::Error> {
2017-11-08 16:44:23 -08:00
match *self {
2018-04-13 16:02:01 -07:00
#[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)
2017-11-08 16:44:23 -08:00
}
},
2018-04-24 12:24:04 -07:00
#[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)
2017-11-08 16:44:23 -08:00
}
},
2018-04-24 12:24:04 -07:00
#[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)
2017-11-08 16:44:23 -08:00
}
},
2017-11-08 16:44:23 -08:00
ContentEncoder::Identity(ref mut encoder) => {
2018-01-04 09:32:15 -08:00
encoder.encode(data)?;
2017-11-08 16:44:23 -08:00
Ok(())
}
}
}
}
/// Encoders to handle different Transfer-Encodings.
#[derive(Debug, Clone)]
pub(crate) struct TransferEncoding {
buf: *mut BytesMut,
2017-11-08 16:44:23 -08:00
kind: TransferEncodingKind,
}
unsafe impl Send for TransferEncoding {}
2017-11-08 16:44:23 -08:00
#[derive(Debug, PartialEq, Clone)]
enum TransferEncodingKind {
/// An Encoder for when Transfer-Encoding includes `chunked`.
Chunked(bool),
/// An Encoder for when Content-Length is set.
///
/// Enforces that the body is not longer than the Content-Length header.
Length(u64),
/// An Encoder for when Content-Length is not known.
///
2018-01-16 00:47:25 +03:00
/// Application decides when to stop writing.
2017-11-08 16:44:23 -08:00
Eof,
}
impl TransferEncoding {
pub(crate) fn set_buffer(&mut self, buf: *mut BytesMut) {
self.buf = buf;
}
2017-12-12 21:32:58 -08:00
#[inline]
pub fn eof() -> TransferEncoding {
2017-11-08 16:44:23 -08:00
TransferEncoding {
kind: TransferEncodingKind::Eof,
buf: ptr::null_mut(),
2017-11-08 16:44:23 -08:00
}
}
2017-12-12 21:32:58 -08:00
#[inline]
pub fn chunked() -> TransferEncoding {
2017-11-08 16:44:23 -08:00
TransferEncoding {
kind: TransferEncodingKind::Chunked(false),
buf: ptr::null_mut(),
2017-11-08 16:44:23 -08:00
}
}
2017-12-12 21:32:58 -08:00
#[inline]
pub fn length(len: u64) -> TransferEncoding {
2017-11-08 16:44:23 -08:00
TransferEncoding {
kind: TransferEncodingKind::Length(len),
buf: ptr::null_mut(),
2017-11-08 16:44:23 -08:00
}
}
2017-12-12 21:32:58 -08:00
#[inline]
2017-11-08 16:44:23 -08:00
pub fn is_eof(&self) -> bool {
match self.kind {
TransferEncodingKind::Eof => true,
2018-01-10 21:02:28 -08:00
TransferEncodingKind::Chunked(ref eof) => *eof,
TransferEncodingKind::Length(ref remaining) => *remaining == 0,
2017-11-08 16:44:23 -08:00
}
}
/// Encode message. Return `EOF` state of encoder
2017-12-13 16:44:35 -08:00
#[inline]
pub fn encode(&mut self, msg: &[u8]) -> io::Result<bool> {
2017-11-08 16:44:23 -08:00
match self.kind {
TransferEncodingKind::Eof => {
2018-01-14 17:00:28 -08:00
let eof = msg.is_empty();
debug_assert!(!self.buf.is_null());
let buf = unsafe { &mut *self.buf };
buf.extend(msg);
2018-01-14 17:00:28 -08:00
Ok(eof)
2018-04-13 16:02:01 -07:00
}
2017-11-08 16:44:23 -08:00
TransferEncodingKind::Chunked(ref mut eof) => {
if *eof {
2018-01-04 09:32:15 -08:00
return Ok(true);
2017-11-08 16:44:23 -08:00
}
if msg.is_empty() {
*eof = true;
debug_assert!(!self.buf.is_null());
let buf = unsafe { &mut *self.buf };
buf.extend_from_slice(b"0\r\n\r\n");
2017-11-08 16:44:23 -08:00
} else {
2018-01-14 17:00:28 -08:00
let mut buf = BytesMut::new();
2018-04-09 14:25:30 -07:00
writeln!(&mut buf, "{:X}\r", msg.len())
2018-01-04 09:32:15 -08:00
.map_err(|e| io::Error::new(io::ErrorKind::Other, e))?;
debug_assert!(!self.buf.is_null());
let b = unsafe { &mut *self.buf };
b.reserve(buf.len() + msg.len() + 2);
b.extend_from_slice(buf.as_ref());
b.extend_from_slice(msg);
b.extend_from_slice(b"\r\n");
2017-11-08 16:44:23 -08:00
}
2018-01-04 09:32:15 -08:00
Ok(*eof)
2018-04-13 16:02:01 -07:00
}
2017-11-08 16:44:23 -08:00
TransferEncodingKind::Length(ref mut remaining) => {
2018-01-20 16:12:38 -08:00
if *remaining > 0 {
if msg.is_empty() {
2018-04-13 16:02:01 -07:00
return Ok(*remaining == 0);
2018-01-20 16:12:38 -08:00
}
let len = cmp::min(*remaining, msg.len() as u64);
debug_assert!(!self.buf.is_null());
unsafe { &mut *self.buf }.extend(&msg[..len as usize]);
2017-11-08 16:44:23 -08:00
2018-01-20 16:12:38 -08:00
*remaining -= len as u64;
Ok(*remaining == 0)
} else {
Ok(true)
}
2018-04-13 16:02:01 -07:00
}
2017-11-08 16:44:23 -08:00
}
}
/// Encode eof. Return `EOF` state of encoder
2017-12-13 16:44:35 -08:00
#[inline]
2018-05-15 16:41:46 -07:00
pub fn encode_eof(&mut self) -> bool {
2017-11-08 16:44:23 -08:00
match self.kind {
2018-05-15 16:41:46 -07:00
TransferEncodingKind::Eof => true,
TransferEncodingKind::Length(rem) => rem == 0,
2017-11-08 16:44:23 -08:00
TransferEncodingKind::Chunked(ref mut eof) => {
if !*eof {
*eof = true;
debug_assert!(!self.buf.is_null());
let buf = unsafe { &mut *self.buf };
buf.extend_from_slice(b"0\r\n\r\n");
2017-11-08 16:44:23 -08:00
}
2018-05-15 16:41:46 -07:00
true
2018-04-13 16:02:01 -07:00
}
2017-11-08 16:44:23 -08:00
}
}
}
impl io::Write for TransferEncoding {
2017-12-12 21:32:58 -08:00
#[inline]
2017-11-08 16:44:23 -08:00
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
self.encode(buf)?;
2017-11-08 16:44:23 -08:00
Ok(buf.len())
}
2017-12-12 21:32:58 -08:00
#[inline]
2017-11-08 16:44:23 -08:00
fn flush(&mut self) -> io::Result<()> {
Ok(())
}
}
struct AcceptEncoding {
encoding: ContentEncoding,
quality: f64,
}
impl Eq for AcceptEncoding {}
impl Ord for AcceptEncoding {
fn cmp(&self, other: &AcceptEncoding) -> cmp::Ordering {
if self.quality > other.quality {
cmp::Ordering::Less
} else if self.quality < other.quality {
cmp::Ordering::Greater
} else {
cmp::Ordering::Equal
}
}
}
impl PartialOrd for AcceptEncoding {
fn partial_cmp(&self, other: &AcceptEncoding) -> Option<cmp::Ordering> {
Some(self.cmp(other))
}
}
impl PartialEq for AcceptEncoding {
fn eq(&self, other: &AcceptEncoding) -> bool {
self.quality == other.quality
}
}
impl AcceptEncoding {
fn new(tag: &str) -> Option<AcceptEncoding> {
let parts: Vec<&str> = tag.split(';').collect();
let encoding = match parts.len() {
0 => return None,
_ => ContentEncoding::from(parts[0]),
};
let quality = match parts.len() {
2017-11-08 19:42:13 -08:00
1 => encoding.quality(),
2017-11-08 16:44:23 -08:00
_ => match f64::from_str(parts[1]) {
Ok(q) => q,
Err(_) => 0.0,
2018-04-13 16:02:01 -07:00
},
2017-11-08 16:44:23 -08:00
};
2018-05-15 16:41:46 -07:00
Some(AcceptEncoding { encoding, quality })
2017-11-08 16:44:23 -08:00
}
/// Parse a raw Accept-Encoding header value into an ordered list.
pub fn parse(raw: &str) -> ContentEncoding {
2018-05-17 12:20:20 -07:00
let mut encodings: Vec<_> = raw
.replace(' ', "")
2018-04-29 09:09:08 -07:00
.split(',')
.map(|l| AcceptEncoding::new(l))
.collect();
2017-11-08 16:44:23 -08:00
encodings.sort();
for enc in encodings {
if let Some(enc) = enc {
2018-04-13 16:02:01 -07:00
return enc.encoding;
2017-11-08 16:44:23 -08:00
}
}
ContentEncoding::Identity
}
}
2018-01-04 09:32:15 -08:00
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_chunked_te() {
let mut bytes = BytesMut::new();
let mut enc = TransferEncoding::chunked();
{
enc.set_buffer(&mut bytes);
assert!(!enc.encode(b"test").ok().unwrap());
assert!(enc.encode(b"").ok().unwrap());
}
2018-04-13 16:02:01 -07:00
assert_eq!(
bytes.take().freeze(),
2018-04-13 16:02:01 -07:00
Bytes::from_static(b"4\r\ntest\r\n0\r\n\r\n")
);
2018-01-04 09:32:15 -08:00
}
}