1
0
Fork 0
mirror of https://github.com/actix/actix-web.git synced 2024-06-02 13:29:24 +00:00
actix-web/actix-http/src/h1/codec.rs

247 lines
6.9 KiB
Rust
Raw Normal View History

2019-12-13 04:59:02 +00:00
use std::{fmt, io};
2018-10-04 23:22:00 +00:00
2018-12-11 02:08:33 +00:00
use actix_codec::{Decoder, Encoder};
2018-12-06 22:32:52 +00:00
use bitflags::bitflags;
2019-12-13 04:59:02 +00:00
use bytes::BytesMut;
use http::{Method, Version};
2018-10-04 23:22:00 +00:00
2018-11-19 22:57:12 +00:00
use super::decoder::{PayloadDecoder, PayloadItem, PayloadType};
2019-04-06 07:16:04 +00:00
use super::{decoder, encoder};
use super::{Message, MessageType};
2019-03-27 16:24:55 +00:00
use crate::body::BodySize;
2018-12-06 22:32:52 +00:00
use crate::config::ServiceConfig;
use crate::error::ParseError;
2019-12-13 04:59:02 +00:00
use crate::message::ConnectionType;
2018-12-06 22:32:52 +00:00
use crate::request::Request;
use crate::response::Response;
2018-10-04 23:22:00 +00:00
2018-10-05 17:03:10 +00:00
bitflags! {
struct Flags: u8 {
const HEAD = 0b0000_0001;
const KEEPALIVE_ENABLED = 0b0000_0010;
const STREAM = 0b0000_0100;
2018-10-05 17:03:10 +00:00
}
}
2018-10-05 03:02:10 +00:00
/// HTTP/1 Codec
pub struct Codec {
2019-05-14 15:48:11 +00:00
config: ServiceConfig,
2018-11-19 22:57:12 +00:00
decoder: decoder::MessageDecoder<Request>,
payload: Option<PayloadDecoder>,
2018-10-05 06:39:11 +00:00
version: Version,
2018-11-19 22:57:12 +00:00
ctype: ConnectionType,
2018-10-05 17:03:10 +00:00
// encoder part
flags: Flags,
2018-11-19 22:57:12 +00:00
encoder: encoder::MessageEncoder<Response<()>>,
2018-10-04 23:22:00 +00:00
}
2018-11-06 03:32:03 +00:00
impl Default for Codec {
fn default() -> Self {
Codec::new(ServiceConfig::default())
}
}
2018-10-24 23:48:45 +00:00
impl fmt::Debug for Codec {
2019-12-07 18:46:51 +00:00
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
2018-10-24 23:48:45 +00:00
write!(f, "h1::Codec({:?})", self.flags)
}
}
2018-10-05 03:02:10 +00:00
impl Codec {
2018-10-05 17:03:10 +00:00
/// Create HTTP/1 codec.
///
/// `keepalive_enabled` how response `connection` header get generated.
2018-10-08 17:14:29 +00:00
pub fn new(config: ServiceConfig) -> Self {
let flags = if config.keep_alive_enabled() {
2018-10-05 17:03:10 +00:00
Flags::KEEPALIVE_ENABLED
} else {
Flags::empty()
};
2018-10-05 03:02:10 +00:00
Codec {
2018-10-08 17:14:29 +00:00
config,
flags,
2018-11-19 22:57:12 +00:00
decoder: decoder::MessageDecoder::default(),
payload: None,
2018-10-05 06:39:11 +00:00
version: Version::HTTP_11,
2018-11-19 22:57:12 +00:00
ctype: ConnectionType::Close,
encoder: encoder::MessageEncoder::default(),
2018-10-04 23:22:00 +00:00
}
}
2019-04-06 07:16:04 +00:00
#[inline]
2018-10-07 05:36:57 +00:00
/// Check if request is upgrade
2018-10-04 23:22:00 +00:00
pub fn upgrade(&self) -> bool {
2018-11-19 22:57:12 +00:00
self.ctype == ConnectionType::Upgrade
2018-10-04 23:22:00 +00:00
}
2019-04-06 07:16:04 +00:00
#[inline]
2018-10-07 05:36:57 +00:00
/// Check if last response is keep-alive
2018-10-04 23:22:00 +00:00
pub fn keepalive(&self) -> bool {
2018-11-19 22:57:12 +00:00
self.ctype == ConnectionType::KeepAlive
2018-10-04 23:22:00 +00:00
}
2019-04-06 07:16:04 +00:00
#[inline]
/// Check if keep-alive enabled on server level
pub fn keepalive_enabled(&self) -> bool {
self.flags.contains(Flags::KEEPALIVE_ENABLED)
}
#[inline]
2018-10-15 22:56:47 +00:00
/// Check last request's message type
pub fn message_type(&self) -> MessageType {
2018-11-14 06:53:30 +00:00
if self.flags.contains(Flags::STREAM) {
MessageType::Stream
2018-10-15 22:56:47 +00:00
} else if self.payload.is_none() {
MessageType::None
2018-10-15 22:56:47 +00:00
} else {
MessageType::Payload
2018-10-15 22:56:47 +00:00
}
}
2019-05-14 15:48:11 +00:00
#[inline]
pub fn config(&self) -> &ServiceConfig {
&self.config
}
2018-10-04 23:22:00 +00:00
}
2018-10-05 17:03:10 +00:00
impl Decoder for Codec {
type Item = Message<Request>;
2018-10-05 17:03:10 +00:00
type Error = ParseError;
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
if self.payload.is_some() {
Ok(match self.payload.as_mut().unwrap().decode(src)? {
2019-04-06 07:16:04 +00:00
Some(PayloadItem::Chunk(chunk)) => Some(Message::Chunk(Some(chunk))),
2018-11-06 03:32:03 +00:00
Some(PayloadItem::Eof) => {
self.payload.take();
Some(Message::Chunk(None))
}
None => None,
})
} else if let Some((req, payload)) = self.decoder.decode(src)? {
2019-02-08 05:16:46 +00:00
let head = req.head();
self.flags.set(Flags::HEAD, head.method == Method::HEAD);
self.version = head.version;
self.ctype = head.connection_type();
2018-11-19 22:57:12 +00:00
if self.ctype == ConnectionType::KeepAlive
&& !self.flags.contains(Flags::KEEPALIVE_ENABLED)
{
self.ctype = ConnectionType::Close
}
match payload {
PayloadType::None => self.payload = None,
PayloadType::Payload(pl) => self.payload = Some(pl),
2018-11-14 06:53:30 +00:00
PayloadType::Stream(pl) => {
self.payload = Some(pl);
self.flags.insert(Flags::STREAM);
}
}
Ok(Some(Message::Item(req)))
} else {
Ok(None)
2018-10-05 17:03:10 +00:00
}
}
}
impl Encoder for Codec {
2019-03-27 16:24:55 +00:00
type Item = Message<(Response<()>, BodySize)>;
2018-10-05 17:03:10 +00:00
type Error = io::Error;
fn encode(
2018-10-29 23:39:46 +00:00
&mut self,
item: Self::Item,
dst: &mut BytesMut,
2018-10-05 17:03:10 +00:00
) -> Result<(), Self::Error> {
match item {
2018-11-19 01:52:56 +00:00
Message::Item((mut res, length)) => {
// set response version
res.head_mut().version = self.version;
// connection status
self.ctype = if let Some(ct) = res.head().ctype() {
if ct == ConnectionType::KeepAlive {
2019-04-05 23:46:44 +00:00
self.ctype
} else {
ct
}
} else {
self.ctype
};
// encode message
self.encoder.encode(
dst,
&mut res,
self.flags.contains(Flags::HEAD),
self.flags.contains(Flags::STREAM),
self.version,
length,
self.ctype,
&self.config,
)?;
// self.headers_size = (dst.len() - len) as u32;
2018-10-05 17:03:10 +00:00
}
Message::Chunk(Some(bytes)) => {
2018-11-19 22:57:12 +00:00
self.encoder.encode_chunk(bytes.as_ref(), dst)?;
2018-10-08 22:24:51 +00:00
}
Message::Chunk(None) => {
2018-11-19 22:57:12 +00:00
self.encoder.encode_eof(dst)?;
2018-10-05 17:03:10 +00:00
}
}
Ok(())
}
}
2018-11-06 03:32:03 +00:00
#[cfg(test)]
mod tests {
use std::{cmp, io};
2018-12-11 02:08:33 +00:00
use actix_codec::{AsyncRead, AsyncWrite};
2018-11-06 03:32:03 +00:00
use bytes::{Buf, Bytes, BytesMut};
use http::{Method, Version};
use super::*;
2018-12-06 23:03:01 +00:00
use crate::error::ParseError;
use crate::h1::Message;
use crate::httpmessage::HttpMessage;
use crate::request::Request;
2018-11-06 03:32:03 +00:00
#[test]
fn test_http_request_chunked_payload_and_next_message() {
let mut codec = Codec::default();
let mut buf = BytesMut::from(
"GET /test HTTP/1.1\r\n\
transfer-encoding: chunked\r\n\r\n",
);
let item = codec.decode(&mut buf).unwrap().unwrap();
let req = item.message();
assert_eq!(req.method(), Method::GET);
assert!(req.chunked().unwrap());
buf.extend(
b"4\r\ndata\r\n4\r\nline\r\n0\r\n\r\n\
POST /test2 HTTP/1.1\r\n\
transfer-encoding: chunked\r\n\r\n"
.iter(),
);
let msg = codec.decode(&mut buf).unwrap().unwrap();
assert_eq!(msg.chunk().as_ref(), b"data");
let msg = codec.decode(&mut buf).unwrap().unwrap();
assert_eq!(msg.chunk().as_ref(), b"line");
let msg = codec.decode(&mut buf).unwrap().unwrap();
assert!(msg.eof());
// decode next message
let item = codec.decode(&mut buf).unwrap().unwrap();
let req = item.message();
assert_eq!(*req.method(), Method::POST);
assert!(req.chunked().unwrap());
}
}