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/h1/client.rs

240 lines
6.8 KiB
Rust
Raw Normal View History

2022-01-31 17:30:34 +00:00
use std::{fmt, io};
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::{Bytes, BytesMut};
2018-12-06 22:32:52 +00:00
use http::{Method, Version};
use super::{
decoder::{self, PayloadDecoder, PayloadItem, PayloadType},
encoder, reserve_readbuf, Message, MessageType,
};
use crate::{
body::BodySize,
error::{ParseError, PayloadError},
ConnectionType, RequestHeadType, ResponseHead, ServiceConfig,
};
bitflags! {
struct Flags: u8 {
2022-01-31 17:30:34 +00:00
const HEAD = 0b0000_0001;
const KEEP_ALIVE_ENABLED = 0b0000_1000;
const STREAM = 0b0001_0000;
}
}
/// HTTP/1 Codec
pub struct ClientCodec {
inner: ClientCodecInner,
}
/// HTTP/1 Payload Codec
pub struct ClientPayloadCodec {
inner: ClientCodecInner,
}
struct ClientCodecInner {
config: ServiceConfig,
decoder: decoder::MessageDecoder<ResponseHead>,
payload: Option<PayloadDecoder>,
version: Version,
2022-01-31 17:30:34 +00:00
conn_type: ConnectionType,
// encoder part
flags: Flags,
encoder: encoder::MessageEncoder<RequestHeadType>,
}
impl Default for ClientCodec {
fn default() -> Self {
ClientCodec::new(ServiceConfig::default())
}
}
2022-01-31 17:30:34 +00:00
impl fmt::Debug for ClientCodec {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("h1::ClientCodec")
.field("flags", &self.inner.flags)
.finish_non_exhaustive()
}
}
impl ClientCodec {
/// Create HTTP/1 codec.
///
/// `keepalive_enabled` how response `connection` header get generated.
pub fn new(config: ServiceConfig) -> Self {
2022-01-31 17:30:34 +00:00
let flags = if config.keep_alive().enabled() {
Flags::KEEP_ALIVE_ENABLED
} else {
Flags::empty()
};
2022-01-31 17:30:34 +00:00
ClientCodec {
inner: ClientCodecInner {
config,
2018-11-19 22:57:12 +00:00
decoder: decoder::MessageDecoder::default(),
payload: None,
version: Version::HTTP_11,
2022-01-31 17:30:34 +00:00
conn_type: ConnectionType::Close,
flags,
2018-11-19 22:57:12 +00:00
encoder: encoder::MessageEncoder::default(),
},
}
}
/// Check if request is upgrade
pub fn upgrade(&self) -> bool {
2022-01-31 17:30:34 +00:00
self.inner.conn_type == ConnectionType::Upgrade
}
/// Check if last response is keep-alive
2022-01-31 17:30:34 +00:00
pub fn keep_alive(&self) -> bool {
self.inner.conn_type == ConnectionType::KeepAlive
}
/// Check last request's message type
pub fn message_type(&self) -> MessageType {
if self.inner.flags.contains(Flags::STREAM) {
2018-11-14 06:53:30 +00:00
MessageType::Stream
} else if self.inner.payload.is_none() {
MessageType::None
} else {
MessageType::Payload
}
}
/// Convert message codec to a payload codec
pub fn into_payload_codec(self) -> ClientPayloadCodec {
ClientPayloadCodec { inner: self.inner }
}
}
impl ClientPayloadCodec {
2018-11-19 22:57:12 +00:00
/// Check if last response is keep-alive
2022-01-31 17:30:34 +00:00
pub fn keep_alive(&self) -> bool {
self.inner.conn_type == ConnectionType::KeepAlive
2018-11-19 22:57:12 +00:00
}
/// Transform payload codec to a message codec
pub fn into_message_codec(self) -> ClientCodec {
ClientCodec { inner: self.inner }
}
}
impl Decoder for ClientCodec {
type Item = ResponseHead;
type Error = ParseError;
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
2022-02-08 08:00:47 +00:00
debug_assert!(
self.inner.payload.is_none(),
"Payload decoder should not be set"
);
2018-11-14 18:52:40 +00:00
if let Some((req, payload)) = self.inner.decoder.decode(src)? {
2022-01-31 17:30:34 +00:00
if let Some(conn_type) = req.conn_type() {
2018-11-19 22:57:12 +00:00
// do not use peer's keep-alive
2022-01-31 17:30:34 +00:00
self.inner.conn_type = if conn_type == ConnectionType::KeepAlive {
self.inner.conn_type
2018-11-19 22:57:12 +00:00
} else {
2022-01-31 17:30:34 +00:00
conn_type
2018-11-19 22:57:12 +00:00
};
}
2018-11-19 22:57:12 +00:00
if !self.inner.flags.contains(Flags::HEAD) {
match payload {
PayloadType::None => self.inner.payload = None,
PayloadType::Payload(pl) => self.inner.payload = Some(pl),
PayloadType::Stream(pl) => {
self.inner.payload = Some(pl);
self.inner.flags.insert(Flags::STREAM);
}
}
2018-11-19 22:57:12 +00:00
} else {
self.inner.payload = None;
}
2019-04-02 18:11:32 +00:00
reserve_readbuf(src);
2018-11-14 18:52:40 +00:00
Ok(Some(req))
} else {
Ok(None)
}
}
}
impl Decoder for ClientPayloadCodec {
type Item = Option<Bytes>;
type Error = PayloadError;
fn decode(&mut self, src: &mut BytesMut) -> Result<Option<Self::Item>, Self::Error> {
2018-11-14 18:52:40 +00:00
debug_assert!(
self.inner.payload.is_some(),
"Payload decoder is not specified"
);
Ok(match self.inner.payload.as_mut().unwrap().decode(src)? {
2019-04-02 18:11:32 +00:00
Some(PayloadItem::Chunk(chunk)) => {
reserve_readbuf(src);
Some(Some(chunk))
}
Some(PayloadItem::Eof) => {
self.inner.payload.take();
Some(None)
}
None => None,
})
}
}
impl Encoder<Message<(RequestHeadType, BodySize)>> for ClientCodec {
type Error = io::Error;
fn encode(
2018-10-29 23:39:46 +00:00
&mut self,
item: Message<(RequestHeadType, BodySize)>,
2018-10-29 23:39:46 +00:00
dst: &mut BytesMut,
) -> Result<(), Self::Error> {
match item {
Message::Item((mut head, length)) => {
2018-11-19 22:57:12 +00:00
let inner = &mut self.inner;
inner.version = head.as_ref().version;
2019-09-12 15:52:46 +00:00
inner
.flags
.set(Flags::HEAD, head.as_ref().method == Method::HEAD);
2018-11-19 22:57:12 +00:00
// connection status
2022-01-31 17:30:34 +00:00
inner.conn_type = match head.as_ref().connection_type() {
2018-11-19 22:57:12 +00:00
ConnectionType::KeepAlive => {
2022-01-31 17:30:34 +00:00
if inner.flags.contains(Flags::KEEP_ALIVE_ENABLED) {
2018-11-19 22:57:12 +00:00
ConnectionType::KeepAlive
} else {
ConnectionType::Close
}
}
ConnectionType::Upgrade => ConnectionType::Upgrade,
ConnectionType::Close => ConnectionType::Close,
};
inner.encoder.encode(
dst,
&mut head,
2018-11-19 22:57:12 +00:00
false,
2019-03-18 04:57:53 +00:00
false,
2018-11-19 22:57:12 +00:00
inner.version,
length,
2022-01-31 17:30:34 +00:00
inner.conn_type,
2018-11-19 22:57:12 +00:00
&inner.config,
)?;
}
Message::Chunk(Some(bytes)) => {
2018-11-19 22:57:12 +00:00
self.inner.encoder.encode_chunk(bytes.as_ref(), dst)?;
}
Message::Chunk(None) => {
2018-11-19 22:57:12 +00:00
self.inner.encoder.encode_eof(dst)?;
}
}
Ok(())
}
}