//! HTTP extensions. use bytes::Bytes; #[cfg(any(feature = "http1", feature = "ffi"))] use http::header::HeaderName; #[cfg(feature = "http1")] use http::header::{IntoHeaderName, ValueIter}; use http::HeaderMap; #[cfg(feature = "ffi")] use std::collections::HashMap; #[cfg(feature = "http2")] use std::fmt; #[cfg(any(feature = "http1", feature = "ffi"))] mod h1_reason_phrase; #[cfg(any(feature = "http1", feature = "ffi"))] pub use h1_reason_phrase::ReasonPhrase; #[cfg(feature = "http2")] /// Represents the `:protocol` pseudo-header used by /// the [Extended CONNECT Protocol]. /// /// [Extended CONNECT Protocol]: https://datatracker.ietf.org/doc/html/rfc8441#section-4 #[derive(Clone, Eq, PartialEq)] pub struct Protocol { inner: h2::ext::Protocol, } #[cfg(feature = "http2")] impl Protocol { /// Converts a static string to a protocol name. pub const fn from_static(value: &'static str) -> Self { Self { inner: h2::ext::Protocol::from_static(value), } } /// Returns a str representation of the header. pub fn as_str(&self) -> &str { self.inner.as_str() } #[cfg(feature = "server")] pub(crate) fn from_inner(inner: h2::ext::Protocol) -> Self { Self { inner } } pub(crate) fn into_inner(self) -> h2::ext::Protocol { self.inner } } #[cfg(feature = "http2")] impl<'a> From<&'a str> for Protocol { fn from(value: &'a str) -> Self { Self { inner: h2::ext::Protocol::from(value), } } } #[cfg(feature = "http2")] impl AsRef<[u8]> for Protocol { fn as_ref(&self) -> &[u8] { self.inner.as_ref() } } #[cfg(feature = "http2")] impl fmt::Debug for Protocol { fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result { self.inner.fmt(f) } } /// A map from header names to their original casing as received in an HTTP message. /// /// If an HTTP/1 response `res` is parsed on a connection whose option /// [`http1_preserve_header_case`] was set to true and the response included /// the following headers: /// /// ```ignore /// x-Bread: Baguette /// X-BREAD: Pain /// x-bread: Ficelle /// ``` /// /// Then `res.extensions().get::<HeaderCaseMap>()` will return a map with: /// /// ```ignore /// HeaderCaseMap({ /// "x-bread": ["x-Bread", "X-BREAD", "x-bread"], /// }) /// ``` /// /// [`http1_preserve_header_case`]: /client/struct.Client.html#method.http1_preserve_header_case #[derive(Clone, Debug)] pub(crate) struct HeaderCaseMap(HeaderMap<Bytes>); #[cfg(feature = "http1")] impl HeaderCaseMap { /// Returns a view of all spellings associated with that header name, /// in the order they were found. pub(crate) fn get_all<'a>( &'a self, name: &HeaderName, ) -> impl Iterator<Item = impl AsRef<[u8]> + 'a> + 'a { self.get_all_internal(name).into_iter() } /// Returns a view of all spellings associated with that header name, /// in the order they were found. pub(crate) fn get_all_internal<'a>(&'a self, name: &HeaderName) -> ValueIter<'_, Bytes> { self.0.get_all(name).into_iter() } pub(crate) fn default() -> Self { Self(Default::default()) } #[cfg(any(test, feature = "ffi"))] pub(crate) fn insert(&mut self, name: HeaderName, orig: Bytes) { self.0.insert(name, orig); } pub(crate) fn append<N>(&mut self, name: N, orig: Bytes) where N: IntoHeaderName, { self.0.append(name, orig); } } #[cfg(feature = "ffi")] #[derive(Clone, Debug)] /// Hashmap<Headername, numheaders with that name> pub(crate) struct OriginalHeaderOrder { /// Stores how many entries a Headername maps to. This is used /// for accounting. num_entries: HashMap<HeaderName, usize>, /// Stores the ordering of the headers. ex: `vec[i] = (headerName, idx)`, /// The vector is ordered such that the ith element /// represents the ith header that came in off the line. /// The `HeaderName` and `idx` are then used elsewhere to index into /// the multi map that stores the header values. entry_order: Vec<(HeaderName, usize)>, } #[cfg(all(feature = "http1", feature = "ffi"))] impl OriginalHeaderOrder { pub(crate) fn default() -> Self { OriginalHeaderOrder { num_entries: HashMap::new(), entry_order: Vec::new(), } } pub(crate) fn insert(&mut self, name: HeaderName) { if !self.num_entries.contains_key(&name) { let idx = 0; self.num_entries.insert(name.clone(), 1); self.entry_order.push((name, idx)); } // Replacing an already existing element does not // change ordering, so we only care if its the first // header name encountered } pub(crate) fn append<N>(&mut self, name: N) where N: IntoHeaderName + Into<HeaderName> + Clone, { let name: HeaderName = name.into(); let idx; if self.num_entries.contains_key(&name) { idx = self.num_entries[&name]; *self.num_entries.get_mut(&name).unwrap() += 1; } else { idx = 0; self.num_entries.insert(name.clone(), 1); } self.entry_order.push((name, idx)); } // No doc test is run here because `RUSTFLAGS='--cfg hyper_unstable_ffi'` // is needed to compile. Once ffi is stablized `no_run` should be removed // here. /// This returns an iterator that provides header names and indexes /// in the original order received. /// /// # Examples /// ```no_run /// use hyper::ext::OriginalHeaderOrder; /// use hyper::header::{HeaderName, HeaderValue, HeaderMap}; /// /// let mut h_order = OriginalHeaderOrder::default(); /// let mut h_map = Headermap::new(); /// /// let name1 = b"Set-CookiE"; /// let value1 = b"a=b"; /// h_map.append(name1); /// h_order.append(name1); /// /// let name2 = b"Content-Encoding"; /// let value2 = b"gzip"; /// h_map.append(name2, value2); /// h_order.append(name2); /// /// let name3 = b"SET-COOKIE"; /// let value3 = b"c=d"; /// h_map.append(name3, value3); /// h_order.append(name3) /// /// let mut iter = h_order.get_in_order() /// /// let (name, idx) = iter.next(); /// assert_eq!(b"a=b", h_map.get_all(name).nth(idx).unwrap()); /// /// let (name, idx) = iter.next(); /// assert_eq!(b"gzip", h_map.get_all(name).nth(idx).unwrap()); /// /// let (name, idx) = iter.next(); /// assert_eq!(b"c=d", h_map.get_all(name).nth(idx).unwrap()); /// ``` pub(crate) fn get_in_order(&self) -> impl Iterator<Item = &(HeaderName, usize)> { self.entry_order.iter() } }