1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2024-06-02 21:39:26 +00:00
actix-web/actix-http/src/encoding/encoder.rs

252 lines
8.4 KiB
Rust
Raw Normal View History

2019-03-26 22:14:32 +00:00
//! Stream encoder
use std::future::Future;
2019-03-26 22:14:32 +00:00
use std::io::{self, Write};
use std::pin::Pin;
use std::task::{Context, Poll};
2019-03-26 22:14:32 +00:00
use actix_threadpool::{run, CpuFuture};
use brotli::CompressorWriter;
use bytes::Bytes;
2019-03-26 22:14:32 +00:00
use flate2::write::{GzEncoder, ZlibEncoder};
2019-12-13 05:24:57 +00:00
use futures_core::ready;
2019-03-26 22:14:32 +00:00
2019-03-27 16:24:55 +00:00
use crate::body::{Body, BodySize, MessageBody, ResponseBody};
2019-03-26 22:14:32 +00:00
use crate::http::header::{ContentEncoding, CONTENT_ENCODING};
2019-12-05 17:35:43 +00:00
use crate::http::{HeaderValue, StatusCode};
2019-03-27 17:38:01 +00:00
use crate::{Error, ResponseHead};
2019-03-26 22:14:32 +00:00
use super::Writer;
const INPLACE: usize = 2049;
2019-03-26 22:14:32 +00:00
pub struct Encoder<B> {
eof: bool,
2019-03-26 22:14:32 +00:00
body: EncoderBody<B>,
encoder: Option<ContentEncoder>,
2019-12-02 17:33:39 +00:00
fut: Option<CpuFuture<ContentEncoder, io::Error>>,
2019-03-26 22:14:32 +00:00
}
impl<B: MessageBody> Encoder<B> {
pub fn response(
encoding: ContentEncoding,
head: &mut ResponseHead,
body: ResponseBody<B>,
) -> ResponseBody<Encoder<B>> {
let can_encode = !(head.headers().contains_key(&CONTENT_ENCODING)
|| head.status == StatusCode::SWITCHING_PROTOCOLS
|| head.status == StatusCode::NO_CONTENT
|| encoding == ContentEncoding::Identity
|| encoding == ContentEncoding::Auto);
let body = match body {
2019-03-26 22:14:32 +00:00
ResponseBody::Other(b) => match b {
Body::None => return ResponseBody::Other(Body::None),
Body::Empty => return ResponseBody::Other(Body::Empty),
2019-03-26 22:14:32 +00:00
Body::Bytes(buf) => {
if can_encode {
EncoderBody::Bytes(buf)
2019-03-26 22:14:32 +00:00
} else {
return ResponseBody::Other(Body::Bytes(buf));
2019-03-26 22:14:32 +00:00
}
}
Body::Message(stream) => EncoderBody::BoxedStream(stream),
2019-03-26 22:14:32 +00:00
},
ResponseBody::Body(stream) => EncoderBody::Stream(stream),
};
if can_encode {
// Modify response body only if encoder is not None
if let Some(enc) = ContentEncoder::encoder(encoding) {
update_head(encoding, head);
head.no_chunking(false);
return ResponseBody::Body(Encoder {
body,
eof: false,
fut: None,
encoder: Some(enc),
});
}
2019-03-26 22:14:32 +00:00
}
ResponseBody::Body(Encoder {
body,
eof: false,
fut: None,
encoder: None,
})
2019-03-26 22:14:32 +00:00
}
}
enum EncoderBody<B> {
Bytes(Bytes),
Stream(B),
BoxedStream(Box<dyn MessageBody>),
2019-03-26 22:14:32 +00:00
}
impl<B: MessageBody> MessageBody for Encoder<B> {
fn size(&self) -> BodySize {
2019-03-26 22:14:32 +00:00
if self.encoder.is_none() {
match self.body {
EncoderBody::Bytes(ref b) => b.size(),
EncoderBody::Stream(ref b) => b.size(),
EncoderBody::BoxedStream(ref b) => b.size(),
2019-03-26 22:14:32 +00:00
}
} else {
2019-03-27 16:24:55 +00:00
BodySize::Stream
2019-03-26 22:14:32 +00:00
}
}
2019-12-07 18:46:51 +00:00
fn poll_next(&mut self, cx: &mut Context<'_>) -> Poll<Option<Result<Bytes, Error>>> {
2019-03-26 22:14:32 +00:00
loop {
if self.eof {
return Poll::Ready(None);
}
if let Some(ref mut fut) = self.fut {
2019-12-13 05:24:57 +00:00
let mut encoder = match ready!(Pin::new(fut).poll(cx)) {
2019-12-02 17:33:39 +00:00
Ok(item) => item,
Err(e) => return Poll::Ready(Some(Err(e.into()))),
};
let chunk = encoder.take();
self.encoder = Some(encoder);
self.fut.take();
if !chunk.is_empty() {
return Poll::Ready(Some(Ok(chunk)));
}
}
2019-03-26 22:14:32 +00:00
let result = match self.body {
EncoderBody::Bytes(ref mut b) => {
if b.is_empty() {
Poll::Ready(None)
} else {
Poll::Ready(Some(Ok(std::mem::replace(b, Bytes::new()))))
}
}
EncoderBody::Stream(ref mut b) => b.poll_next(cx),
EncoderBody::BoxedStream(ref mut b) => b.poll_next(cx),
2019-03-26 22:14:32 +00:00
};
match result {
Poll::Ready(Some(Ok(chunk))) => {
if let Some(mut encoder) = self.encoder.take() {
if chunk.len() < INPLACE {
encoder.write(&chunk)?;
let chunk = encoder.take();
self.encoder = Some(encoder);
if !chunk.is_empty() {
return Poll::Ready(Some(Ok(chunk)));
}
} else {
self.fut = Some(run(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
}
}
Poll::Ready(None) => {
2019-03-26 22:14:32 +00:00
if let Some(encoder) = self.encoder.take() {
let chunk = encoder.finish()?;
if chunk.is_empty() {
return Poll::Ready(None);
2019-03-26 22:14:32 +00:00
} else {
self.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
}
}
val => return val,
2019-03-26 22:14:32 +00:00
}
}
}
}
fn update_head(encoding: ContentEncoding, head: &mut ResponseHead) {
head.headers_mut().insert(
CONTENT_ENCODING,
2019-12-05 17:35:43 +00:00
HeaderValue::from_static(encoding.as_str()),
2019-03-26 22:14:32 +00:00
);
}
enum ContentEncoder {
Deflate(ZlibEncoder<Writer>),
Gzip(GzEncoder<Writer>),
2019-12-07 18:46:51 +00:00
Br(Box<CompressorWriter<Writer>>),
2019-03-26 22:14:32 +00:00
}
impl ContentEncoder {
fn encoder(encoding: ContentEncoding) -> Option<Self> {
match encoding {
ContentEncoding::Deflate => Some(ContentEncoder::Deflate(ZlibEncoder::new(
Writer::new(),
flate2::Compression::fast(),
))),
ContentEncoding::Gzip => Some(ContentEncoder::Gzip(GzEncoder::new(
Writer::new(),
flate2::Compression::fast(),
))),
2019-12-07 18:46:51 +00:00
ContentEncoding::Br => Some(ContentEncoder::Br(Box::new(
CompressorWriter::new(Writer::new(), 0, 3, 0),
))),
2019-03-26 22:14:32 +00:00
_ => None,
}
}
#[inline]
pub(crate) fn take(&mut self) -> Bytes {
match *self {
ContentEncoder::Br(ref mut encoder) => {
2019-12-07 18:46:51 +00:00
let mut encoder_new =
Box::new(CompressorWriter::new(Writer::new(), 0, 3, 0));
std::mem::swap(encoder, &mut encoder_new);
encoder_new.into_inner().freeze()
}
2019-03-26 22:14:32 +00:00
ContentEncoder::Deflate(ref mut encoder) => encoder.get_mut().take(),
ContentEncoder::Gzip(ref mut encoder) => encoder.get_mut().take(),
}
}
fn finish(self) -> Result<Bytes, io::Error> {
match self {
ContentEncoder::Br(encoder) => Ok(encoder.into_inner().buf.freeze()),
2019-03-26 22:14:32 +00:00
ContentEncoder::Gzip(encoder) => match encoder.finish() {
Ok(writer) => Ok(writer.buf.freeze()),
Err(err) => Err(err),
},
ContentEncoder::Deflate(encoder) => match encoder.finish() {
Ok(writer) => Ok(writer.buf.freeze()),
Err(err) => Err(err),
},
}
}
fn write(&mut self, data: &[u8]) -> Result<(), io::Error> {
2019-03-26 22:14:32 +00:00
match *self {
ContentEncoder::Br(ref mut encoder) => match encoder.write_all(data) {
Ok(_) => Ok(()),
2019-03-26 22:14:32 +00:00
Err(err) => {
trace!("Error decoding br encoding: {}", err);
Err(err)
}
},
ContentEncoder::Gzip(ref mut encoder) => match encoder.write_all(data) {
Ok(_) => Ok(()),
2019-03-26 22:14:32 +00:00
Err(err) => {
trace!("Error decoding gzip encoding: {}", err);
Err(err)
}
},
ContentEncoder::Deflate(ref mut encoder) => match encoder.write_all(data) {
Ok(_) => Ok(()),
2019-03-26 22:14:32 +00:00
Err(err) => {
trace!("Error decoding deflate encoding: {}", err);
Err(err)
}
},
}
}
}