1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2024-05-20 01:08:10 +00:00
actix-web/actix-http/src/encoding/encoder.rs

429 lines
13 KiB
Rust
Raw Normal View History

2021-02-11 22:39:54 +00:00
//! Stream encoders.
2021-02-12 00:15:25 +00:00
use std::{
error::Error as StdError,
2021-02-12 00:15:25 +00:00
future::Future,
io::{self, Write as _},
pin::Pin,
task::{Context, Poll},
};
2019-03-26 22:14:32 +00:00
use actix_rt::task::{spawn_blocking, JoinHandle};
use bytes::Bytes;
use derive_more::Display;
#[cfg(feature = "compress-gzip")]
use flate2::write::{GzEncoder, ZlibEncoder};
2023-07-17 01:38:12 +00:00
use futures_core::ready;
use pin_project_lite::pin_project;
2022-03-10 03:12:29 +00:00
use tracing::trace;
#[cfg(feature = "compress-zstd")]
use zstd::stream::write::Encoder as ZstdEncoder;
2019-03-26 22:14:32 +00:00
2021-12-04 19:40:47 +00:00
use super::Writer;
2021-02-12 00:15:25 +00:00
use crate::{
2021-12-17 20:56:54 +00:00
body::{self, BodySize, MessageBody},
header::{self, ContentEncoding, HeaderValue, CONTENT_ENCODING},
ResponseHead, StatusCode,
2021-02-12 00:15:25 +00:00
};
2019-03-26 22:14:32 +00:00
2021-02-12 00:15:25 +00:00
const MAX_CHUNK_SIZE_ENCODE_IN_PLACE: usize = 1024;
2021-12-04 19:40:47 +00:00
pin_project! {
pub struct Encoder<B> {
#[pin]
body: EncoderBody<B>,
encoder: Option<ContentEncoder>,
fut: Option<JoinHandle<Result<ContentEncoder, io::Error>>>,
eof: bool,
}
2019-03-26 22:14:32 +00:00
}
impl<B: MessageBody> Encoder<B> {
2021-12-04 19:40:47 +00:00
fn none() -> Self {
Encoder {
2021-12-17 20:56:54 +00:00
body: EncoderBody::None {
body: body::None::new(),
},
2021-12-04 19:40:47 +00:00
encoder: None,
fut: None,
eof: true,
}
}
pub fn response(encoding: ContentEncoding, head: &mut ResponseHead, body: B) -> Self {
// no need to compress an empty body
if matches!(body.size(), BodySize::None) {
return Self::none();
2021-12-04 19:40:47 +00:00
}
2022-01-03 14:05:08 +00:00
let should_encode = !(head.headers().contains_key(&CONTENT_ENCODING)
|| head.status == StatusCode::SWITCHING_PROTOCOLS
|| head.status == StatusCode::NO_CONTENT
|| encoding == ContentEncoding::Identity);
let body = match body.try_into_bytes() {
Ok(body) => EncoderBody::Full { body },
Err(body) => EncoderBody::Stream { body },
};
2022-01-03 13:17:57 +00:00
if should_encode {
// wrap body only if encoder is feature-enabled
2022-01-03 14:59:01 +00:00
if let Some(enc) = ContentEncoder::select(encoding) {
update_head(encoding, head);
2021-12-04 19:40:47 +00:00
return Encoder {
body,
encoder: Some(enc),
2021-12-04 19:40:47 +00:00
fut: None,
eof: false,
};
}
2019-03-26 22:14:32 +00:00
}
2021-02-15 11:24:46 +00:00
2021-12-04 19:40:47 +00:00
Encoder {
body,
encoder: None,
2021-12-04 19:40:47 +00:00
fut: None,
eof: false,
}
2019-03-26 22:14:32 +00:00
}
}
2021-12-04 19:40:47 +00:00
pin_project! {
#[project = EncoderBodyProj]
enum EncoderBody<B> {
2021-12-17 20:56:54 +00:00
None { body: body::None },
Full { body: Bytes },
2021-12-04 19:40:47 +00:00
Stream { #[pin] body: B },
}
2019-03-26 22:14:32 +00:00
}
impl<B> MessageBody for EncoderBody<B>
where
B: MessageBody,
{
2021-12-04 19:40:47 +00:00
type Error = EncoderError;
2021-12-17 19:19:21 +00:00
#[inline]
fn size(&self) -> BodySize {
match self {
2021-12-17 20:56:54 +00:00
EncoderBody::None { body } => body.size(),
EncoderBody::Full { body } => body.size(),
2021-12-04 19:40:47 +00:00
EncoderBody::Stream { body } => body.size(),
}
}
2020-02-27 02:10:55 +00:00
fn poll_next(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<Option<Result<Bytes, Self::Error>>> {
match self.project() {
2021-12-17 20:56:54 +00:00
EncoderBodyProj::None { body } => {
Pin::new(body).poll_next(cx).map_err(|err| match err {})
}
EncoderBodyProj::Full { body } => {
Pin::new(body).poll_next(cx).map_err(|err| match err {})
}
2021-12-04 19:40:47 +00:00
EncoderBodyProj::Stream { body } => body
.poll_next(cx)
.map_err(|err| EncoderError::Body(err.into())),
}
}
2021-12-17 19:19:21 +00:00
#[inline]
fn try_into_bytes(self) -> Result<Bytes, Self>
where
Self: Sized,
{
match self {
2021-12-17 20:56:54 +00:00
EncoderBody::None { body } => Ok(body.try_into_bytes().unwrap()),
EncoderBody::Full { body } => Ok(body.try_into_bytes().unwrap()),
_ => Err(self),
}
}
}
impl<B> MessageBody for Encoder<B>
where
B: MessageBody,
{
2021-12-04 19:40:47 +00:00
type Error = EncoderError;
2021-12-17 19:19:21 +00:00
#[inline]
fn size(&self) -> BodySize {
if self.encoder.is_some() {
2019-03-27 16:24:55 +00:00
BodySize::Stream
} else {
self.body.size()
2019-03-26 22:14:32 +00:00
}
}
2020-02-27 02:10:55 +00:00
fn poll_next(
self: Pin<&mut Self>,
cx: &mut Context<'_>,
) -> Poll<Option<Result<Bytes, Self::Error>>> {
let mut this = self.project();
2022-01-03 14:59:01 +00:00
2019-03-26 22:14:32 +00:00
loop {
if *this.eof {
return Poll::Ready(None);
}
if let Some(ref mut fut) = this.fut {
let mut encoder = ready!(Pin::new(fut).poll(cx))
2022-02-22 08:45:28 +00:00
.map_err(|_| {
EncoderError::Io(io::Error::new(
io::ErrorKind::Other,
"Blocking task was cancelled unexpectedly",
))
})?
.map_err(EncoderError::Io)?;
2021-02-12 00:15:25 +00:00
let chunk = encoder.take();
*this.encoder = Some(encoder);
this.fut.take();
2021-02-12 00:15:25 +00:00
if !chunk.is_empty() {
return Poll::Ready(Some(Ok(chunk)));
}
}
2021-02-12 00:15:25 +00:00
let result = ready!(this.body.as_mut().poll_next(cx));
2019-03-26 22:14:32 +00:00
match result {
2021-02-12 00:15:25 +00:00
Some(Err(err)) => return Poll::Ready(Some(Err(err))),
Some(Ok(chunk)) => {
if let Some(mut encoder) = this.encoder.take() {
2021-02-12 00:15:25 +00:00
if chunk.len() < MAX_CHUNK_SIZE_ENCODE_IN_PLACE {
encoder.write(&chunk).map_err(EncoderError::Io)?;
let chunk = encoder.take();
*this.encoder = Some(encoder);
2021-02-12 00:15:25 +00:00
if !chunk.is_empty() {
return Poll::Ready(Some(Ok(chunk)));
}
} else {
*this.fut = Some(spawn_blocking(move || {
encoder.write(&chunk)?;
Ok(encoder)
}));
2019-03-26 22:14:32 +00:00
}
} else {
return Poll::Ready(Some(Ok(chunk)));
2019-03-26 22:14:32 +00:00
}
}
2021-02-12 00:15:25 +00:00
None => {
if let Some(encoder) = this.encoder.take() {
let chunk = encoder.finish().map_err(EncoderError::Io)?;
2021-12-04 19:40:47 +00:00
2019-03-26 22:14:32 +00:00
if chunk.is_empty() {
return Poll::Ready(None);
2019-03-26 22:14:32 +00:00
} else {
*this.eof = true;
return Poll::Ready(Some(Ok(chunk)));
2019-03-26 22:14:32 +00:00
}
} else {
return Poll::Ready(None);
2019-03-26 22:14:32 +00:00
}
}
}
}
}
2021-12-17 19:19:21 +00:00
#[inline]
fn try_into_bytes(mut self) -> Result<Bytes, Self>
where
Self: Sized,
{
if self.encoder.is_some() {
Err(self)
} else {
match self.body.try_into_bytes() {
Ok(body) => Ok(body),
Err(body) => {
self.body = body;
Err(self)
}
}
}
}
2019-03-26 22:14:32 +00:00
}
fn update_head(encoding: ContentEncoding, head: &mut ResponseHead) {
2022-01-03 13:17:57 +00:00
head.headers_mut()
.insert(header::CONTENT_ENCODING, encoding.to_header_value());
head.headers_mut()
.append(header::VARY, HeaderValue::from_static("accept-encoding"));
head.no_chunking(false);
2019-03-26 22:14:32 +00:00
}
enum ContentEncoder {
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
Deflate(ZlibEncoder<Writer>),
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
Gzip(GzEncoder<Writer>),
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-brotli")]
2022-01-15 14:03:16 +00:00
Brotli(Box<brotli::CompressorWriter<Writer>>),
2021-12-04 19:40:47 +00:00
// Wwe need explicit 'static lifetime here because ZstdEncoder needs a lifetime argument and we
// use `spawn_blocking` in `Encoder::poll_next` that requires `FnOnce() -> R + Send + 'static`.
#[cfg(feature = "compress-zstd")]
Zstd(ZstdEncoder<'static, Writer>),
2019-03-26 22:14:32 +00:00
}
impl ContentEncoder {
2022-01-03 14:59:01 +00:00
fn select(encoding: ContentEncoding) -> Option<Self> {
2019-03-26 22:14:32 +00:00
match encoding {
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
ContentEncoding::Deflate => Some(ContentEncoder::Deflate(ZlibEncoder::new(
Writer::new(),
flate2::Compression::fast(),
))),
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
ContentEncoding::Gzip => Some(ContentEncoder::Gzip(GzEncoder::new(
Writer::new(),
flate2::Compression::fast(),
))),
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-brotli")]
2022-01-15 14:03:16 +00:00
ContentEncoding::Brotli => Some(ContentEncoder::Brotli(new_brotli_compressor())),
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-zstd")]
ContentEncoding::Zstd => {
let encoder = ZstdEncoder::new(Writer::new(), 3).ok()?;
Some(ContentEncoder::Zstd(encoder))
}
2021-12-04 19:40:47 +00:00
2022-01-03 14:05:08 +00:00
_ => None,
2019-03-26 22:14:32 +00:00
}
}
#[inline]
pub(crate) fn take(&mut self) -> Bytes {
match *self {
#[cfg(feature = "compress-brotli")]
2022-01-03 13:17:57 +00:00
ContentEncoder::Brotli(ref mut encoder) => encoder.get_mut().take(),
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
ContentEncoder::Deflate(ref mut encoder) => encoder.get_mut().take(),
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
ContentEncoder::Gzip(ref mut encoder) => encoder.get_mut().take(),
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-zstd")]
ContentEncoder::Zstd(ref mut encoder) => encoder.get_mut().take(),
2019-03-26 22:14:32 +00:00
}
}
fn finish(self) -> Result<Bytes, io::Error> {
match self {
#[cfg(feature = "compress-brotli")]
2022-01-15 14:03:16 +00:00
ContentEncoder::Brotli(mut encoder) => match encoder.flush() {
Ok(()) => Ok(encoder.into_inner().buf.freeze()),
2019-12-20 07:50:07 +00:00
Err(err) => Err(err),
},
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
ContentEncoder::Gzip(encoder) => match encoder.finish() {
Ok(writer) => Ok(writer.buf.freeze()),
Err(err) => Err(err),
},
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
ContentEncoder::Deflate(encoder) => match encoder.finish() {
Ok(writer) => Ok(writer.buf.freeze()),
Err(err) => Err(err),
},
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-zstd")]
ContentEncoder::Zstd(encoder) => match encoder.finish() {
Ok(writer) => Ok(writer.buf.freeze()),
Err(err) => Err(err),
},
2019-03-26 22:14:32 +00:00
}
}
fn write(&mut self, data: &[u8]) -> Result<(), io::Error> {
2019-03-26 22:14:32 +00:00
match *self {
#[cfg(feature = "compress-brotli")]
2022-01-03 13:17:57 +00:00
ContentEncoder::Brotli(ref mut encoder) => match encoder.write_all(data) {
Ok(_) => Ok(()),
2019-03-26 22:14:32 +00:00
Err(err) => {
2022-03-10 03:12:29 +00:00
trace!("Error decoding br encoding: {}", err);
2019-03-26 22:14:32 +00:00
Err(err)
}
},
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
ContentEncoder::Gzip(ref mut encoder) => match encoder.write_all(data) {
Ok(_) => Ok(()),
2019-03-26 22:14:32 +00:00
Err(err) => {
2022-03-10 03:12:29 +00:00
trace!("Error decoding gzip encoding: {}", err);
2019-03-26 22:14:32 +00:00
Err(err)
}
},
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-gzip")]
2019-03-26 22:14:32 +00:00
ContentEncoder::Deflate(ref mut encoder) => match encoder.write_all(data) {
Ok(_) => Ok(()),
2019-03-26 22:14:32 +00:00
Err(err) => {
2022-03-10 03:12:29 +00:00
trace!("Error decoding deflate encoding: {}", err);
2019-03-26 22:14:32 +00:00
Err(err)
}
},
2021-12-04 19:40:47 +00:00
#[cfg(feature = "compress-zstd")]
ContentEncoder::Zstd(ref mut encoder) => match encoder.write_all(data) {
Ok(_) => Ok(()),
Err(err) => {
2022-03-10 03:12:29 +00:00
trace!("Error decoding ztsd encoding: {}", err);
Err(err)
}
},
2019-03-26 22:14:32 +00:00
}
}
}
2022-01-15 14:03:16 +00:00
#[cfg(feature = "compress-brotli")]
fn new_brotli_compressor() -> Box<brotli::CompressorWriter<Writer>> {
Box::new(brotli::CompressorWriter::new(
Writer::new(),
2022-01-21 21:17:07 +00:00
32 * 1024, // 32 KiB buffer
3, // BROTLI_PARAM_QUALITY
22, // BROTLI_PARAM_LGWIN
2022-01-15 14:03:16 +00:00
))
}
#[derive(Debug, Display)]
#[non_exhaustive]
2021-12-04 19:40:47 +00:00
pub enum EncoderError {
2022-02-22 08:45:28 +00:00
/// Wrapped body stream error.
#[display(fmt = "body")]
2021-12-04 19:40:47 +00:00
Body(Box<dyn StdError>),
2022-02-22 08:45:28 +00:00
/// Generic I/O error.
#[display(fmt = "io")]
Io(io::Error),
}
2021-12-04 19:40:47 +00:00
impl StdError for EncoderError {
fn source(&self) -> Option<&(dyn StdError + 'static)> {
2021-06-17 16:57:58 +00:00
match self {
2021-12-04 19:40:47 +00:00
EncoderError::Body(err) => Some(&**err),
2021-06-17 16:57:58 +00:00
EncoderError::Io(err) => Some(err),
}
}
}
2021-12-04 19:40:47 +00:00
impl From<EncoderError> for crate::Error {
fn from(err: EncoderError) -> Self {
2021-06-17 16:57:58 +00:00
crate::Error::new_encoder().with_cause(err)
}
}