use std::cell::{Ref, RefCell, RefMut}; use std::collections::VecDeque; use std::rc::Rc; use bitflags::bitflags; use crate::extensions::Extensions; use crate::http::{header, HeaderMap, Method, StatusCode, Uri, Version}; /// Represents various types of connection #[derive(Copy, Clone, PartialEq, Debug)] pub enum ConnectionType { /// Close connection after response Close, /// Keep connection alive after response KeepAlive, /// Connection is upgraded to different type Upgrade, } bitflags! { pub(crate) struct Flags: u8 { const CLOSE = 0b0000_0001; const KEEP_ALIVE = 0b0000_0010; const UPGRADE = 0b0000_0100; const NO_CHUNKING = 0b0000_1000; } } #[doc(hidden)] pub trait Head: Default + 'static { fn clear(&mut self); fn pool() -> &'static MessagePool; } #[derive(Debug)] pub struct RequestHead { pub uri: Uri, pub method: Method, pub version: Version, pub headers: HeaderMap, pub extensions: RefCell, flags: Flags, } impl Default for RequestHead { fn default() -> RequestHead { RequestHead { uri: Uri::default(), method: Method::default(), version: Version::HTTP_11, headers: HeaderMap::with_capacity(16), flags: Flags::empty(), extensions: RefCell::new(Extensions::new()), } } } impl Head for RequestHead { fn clear(&mut self) { self.flags = Flags::empty(); self.headers.clear(); self.extensions.borrow_mut().clear(); } fn pool() -> &'static MessagePool { REQUEST_POOL.with(|p| *p) } } impl RequestHead { /// Message extensions #[inline] pub fn extensions(&self) -> Ref { self.extensions.borrow() } /// Mutable reference to a the message's extensions #[inline] pub fn extensions_mut(&self) -> RefMut { self.extensions.borrow_mut() } /// Read the message headers. pub fn headers(&self) -> &HeaderMap { &self.headers } /// Mutable reference to the message headers. pub fn headers_mut(&mut self) -> &mut HeaderMap { &mut self.headers } #[inline] /// Set connection type of the message pub fn set_connection_type(&mut self, ctype: ConnectionType) { match ctype { ConnectionType::Close => self.flags.insert(Flags::CLOSE), ConnectionType::KeepAlive => self.flags.insert(Flags::KEEP_ALIVE), ConnectionType::Upgrade => self.flags.insert(Flags::UPGRADE), } } #[inline] /// Connection type pub fn connection_type(&self) -> ConnectionType { if self.flags.contains(Flags::CLOSE) { ConnectionType::Close } else if self.flags.contains(Flags::KEEP_ALIVE) { ConnectionType::KeepAlive } else if self.flags.contains(Flags::UPGRADE) { ConnectionType::Upgrade } else if self.version < Version::HTTP_11 { ConnectionType::Close } else { ConnectionType::KeepAlive } } /// Connection upgrade status pub fn upgrade(&self) -> bool { if let Some(hdr) = self.headers().get(header::CONNECTION) { if let Ok(s) = hdr.to_str() { s.to_ascii_lowercase().contains("upgrade") } else { false } } else { false } } #[inline] /// Get response body chunking state pub fn chunked(&self) -> bool { !self.flags.contains(Flags::NO_CHUNKING) } #[inline] pub fn no_chunking(&mut self, val: bool) { if val { self.flags.insert(Flags::NO_CHUNKING); } else { self.flags.remove(Flags::NO_CHUNKING); } } } #[derive(Debug)] pub struct ResponseHead { pub version: Version, pub status: StatusCode, pub headers: HeaderMap, pub reason: Option<&'static str>, pub(crate) extensions: RefCell, flags: Flags, } impl Default for ResponseHead { fn default() -> ResponseHead { ResponseHead { version: Version::default(), status: StatusCode::OK, headers: HeaderMap::with_capacity(16), reason: None, flags: Flags::empty(), extensions: RefCell::new(Extensions::new()), } } } impl Head for ResponseHead { fn clear(&mut self) { self.reason = None; self.flags = Flags::empty(); self.headers.clear(); } fn pool() -> &'static MessagePool { RESPONSE_POOL.with(|p| *p) } } impl ResponseHead { /// Message extensions #[inline] pub fn extensions(&self) -> Ref { self.extensions.borrow() } /// Mutable reference to a the message's extensions #[inline] pub fn extensions_mut(&self) -> RefMut { self.extensions.borrow_mut() } #[inline] /// Read the message headers. pub fn headers(&self) -> &HeaderMap { &self.headers } #[inline] /// Mutable reference to the message headers. pub fn headers_mut(&mut self) -> &mut HeaderMap { &mut self.headers } #[inline] /// Set connection type of the message pub fn set_connection_type(&mut self, ctype: ConnectionType) { match ctype { ConnectionType::Close => self.flags.insert(Flags::CLOSE), ConnectionType::KeepAlive => self.flags.insert(Flags::KEEP_ALIVE), ConnectionType::Upgrade => self.flags.insert(Flags::UPGRADE), } } #[inline] pub fn connection_type(&self) -> ConnectionType { if self.flags.contains(Flags::CLOSE) { ConnectionType::Close } else if self.flags.contains(Flags::KEEP_ALIVE) { ConnectionType::KeepAlive } else if self.flags.contains(Flags::UPGRADE) { ConnectionType::Upgrade } else if self.version < Version::HTTP_11 { ConnectionType::Close } else { ConnectionType::KeepAlive } } #[inline] /// Check if keep-alive is enabled pub fn keep_alive(&self) -> bool { self.connection_type() == ConnectionType::KeepAlive } #[inline] /// Check upgrade status of this message pub fn upgrade(&self) -> bool { self.connection_type() == ConnectionType::Upgrade } /// Get custom reason for the response #[inline] pub fn reason(&self) -> &str { if let Some(reason) = self.reason { reason } else { self.status .canonical_reason() .unwrap_or("") } } #[inline] pub(crate) fn ctype(&self) -> Option { if self.flags.contains(Flags::CLOSE) { Some(ConnectionType::Close) } else if self.flags.contains(Flags::KEEP_ALIVE) { Some(ConnectionType::KeepAlive) } else if self.flags.contains(Flags::UPGRADE) { Some(ConnectionType::Upgrade) } else { None } } #[inline] /// Get response body chunking state pub fn chunked(&self) -> bool { !self.flags.contains(Flags::NO_CHUNKING) } #[inline] /// Set no chunking for payload pub fn no_chunking(&mut self, val: bool) { if val { self.flags.insert(Flags::NO_CHUNKING); } else { self.flags.remove(Flags::NO_CHUNKING); } } } pub struct Message { head: Rc, pool: &'static MessagePool, } impl Message { /// Get new message from the pool of objects pub fn new() -> Self { T::pool().get_message() } } impl Clone for Message { fn clone(&self) -> Self { Message { head: self.head.clone(), pool: self.pool, } } } impl std::ops::Deref for Message { type Target = T; fn deref(&self) -> &Self::Target { &self.head.as_ref() } } impl std::ops::DerefMut for Message { fn deref_mut(&mut self) -> &mut Self::Target { Rc::get_mut(&mut self.head).expect("Multiple copies exist") } } impl Drop for Message { fn drop(&mut self) { if Rc::strong_count(&self.head) == 1 { self.pool.release(self.head.clone()); } } } #[doc(hidden)] /// Request's objects pool pub struct MessagePool(RefCell>>); thread_local!(static REQUEST_POOL: &'static MessagePool = MessagePool::::create()); thread_local!(static RESPONSE_POOL: &'static MessagePool = MessagePool::::create()); impl MessagePool { fn create() -> &'static MessagePool { let pool = MessagePool(RefCell::new(VecDeque::with_capacity(128))); Box::leak(Box::new(pool)) } /// Get message from the pool #[inline] fn get_message(&'static self) -> Message { if let Some(mut msg) = self.0.borrow_mut().pop_front() { if let Some(r) = Rc::get_mut(&mut msg) { r.clear(); } Message { head: msg, pool: self, } } else { Message { head: Rc::new(T::default()), pool: self, } } } #[inline] /// Release request instance fn release(&self, msg: Rc) { let v = &mut self.0.borrow_mut(); if v.len() < 128 { v.push_front(msg); } } }