http_types/headers/
headers.rs

1//! HTTP headers.
2
3use std::collections::HashMap;
4use std::convert::Into;
5use std::fmt::{self, Debug};
6use std::iter::IntoIterator;
7use std::ops::Index;
8use std::str::FromStr;
9
10use crate::headers::{
11    HeaderName, HeaderValues, IntoIter, Iter, IterMut, Names, ToHeaderValues, Values,
12};
13
14/// A collection of HTTP Headers.
15///
16/// Headers are never manually constructed, but are part of `Request`,
17/// `Response`, and `Trailers`. Each of these types implements `AsRef<Headers>`
18/// and `AsMut<Headers>` so functions that want to modify headers can be generic
19/// over either of these traits.
20///
21/// # Examples
22///
23/// ```
24/// use http_types::{Response, StatusCode};
25///
26/// let mut res = Response::new(StatusCode::Ok);
27/// res.insert_header("hello", "foo0");
28/// assert_eq!(res["hello"], "foo0");
29/// ```
30#[derive(Clone)]
31pub struct Headers {
32    pub(crate) headers: HashMap<HeaderName, HeaderValues>,
33}
34
35impl Headers {
36    /// Create a new instance.
37    pub(crate) fn new() -> Self {
38        Self {
39            headers: HashMap::new(),
40        }
41    }
42
43    /// Insert a header into the headers.
44    ///
45    /// Not that this will replace all header values for a given header name.
46    /// If you wish to add header values for a header name that already exists
47    /// use `Headers::append`
48    pub fn insert(
49        &mut self,
50        name: impl Into<HeaderName>,
51        values: impl ToHeaderValues,
52    ) -> Option<HeaderValues> {
53        let name = name.into();
54        let values: HeaderValues = values.to_header_values().unwrap().collect();
55        self.headers.insert(name, values)
56    }
57
58    /// Append a header to the headers.
59    ///
60    /// Unlike `insert` this function will not override the contents of a header, but insert a
61    /// header if there aren't any. Or else append to the existing list of headers.
62    pub fn append(&mut self, name: impl Into<HeaderName>, values: impl ToHeaderValues) {
63        let name = name.into();
64        match self.get_mut(&name) {
65            Some(headers) => {
66                let mut values: HeaderValues = values.to_header_values().unwrap().collect();
67                headers.append(&mut values);
68            }
69            None => {
70                self.insert(name, values);
71            }
72        }
73    }
74
75    /// Get a reference to a header.
76    pub fn get(&self, name: impl Into<HeaderName>) -> Option<&HeaderValues> {
77        self.headers.get(&name.into())
78    }
79
80    /// Get a mutable reference to a header.
81    pub fn get_mut(&mut self, name: impl Into<HeaderName>) -> Option<&mut HeaderValues> {
82        self.headers.get_mut(&name.into())
83    }
84
85    /// Remove a header.
86    pub fn remove(&mut self, name: impl Into<HeaderName>) -> Option<HeaderValues> {
87        self.headers.remove(&name.into())
88    }
89
90    /// An iterator visiting all header pairs in arbitrary order.
91    pub fn iter(&self) -> Iter<'_> {
92        Iter {
93            inner: self.headers.iter(),
94        }
95    }
96
97    /// An iterator visiting all header pairs in arbitrary order, with mutable references to the
98    /// values.
99    pub fn iter_mut(&mut self) -> IterMut<'_> {
100        IterMut {
101            inner: self.headers.iter_mut(),
102        }
103    }
104
105    /// An iterator visiting all header names in arbitrary order.
106    pub fn names(&self) -> Names<'_> {
107        Names {
108            inner: self.headers.keys(),
109        }
110    }
111
112    /// An iterator visiting all header values in arbitrary order.
113    pub fn values(&self) -> Values<'_> {
114        Values::new(self.headers.values())
115    }
116}
117
118impl Index<HeaderName> for Headers {
119    type Output = HeaderValues;
120
121    /// Returns a reference to the value corresponding to the supplied name.
122    ///
123    /// # Panics
124    ///
125    /// Panics if the name is not present in `Headers`.
126    #[inline]
127    fn index(&self, name: HeaderName) -> &HeaderValues {
128        self.get(name).expect("no entry found for name")
129    }
130}
131
132impl Index<&str> for Headers {
133    type Output = HeaderValues;
134
135    /// Returns a reference to the value corresponding to the supplied name.
136    ///
137    /// # Panics
138    ///
139    /// Panics if the name is not present in `Headers`.
140    #[inline]
141    fn index(&self, name: &str) -> &HeaderValues {
142        let name = HeaderName::from_str(name).expect("string slice needs to be valid ASCII");
143        self.get(name).expect("no entry found for name")
144    }
145}
146
147impl IntoIterator for Headers {
148    type Item = (HeaderName, HeaderValues);
149    type IntoIter = IntoIter;
150
151    /// Returns a iterator of references over the remaining items.
152    #[inline]
153    fn into_iter(self) -> Self::IntoIter {
154        IntoIter {
155            inner: self.headers.into_iter(),
156        }
157    }
158}
159
160impl<'a> IntoIterator for &'a Headers {
161    type Item = (&'a HeaderName, &'a HeaderValues);
162    type IntoIter = Iter<'a>;
163
164    #[inline]
165    fn into_iter(self) -> Self::IntoIter {
166        self.iter()
167    }
168}
169
170impl<'a> IntoIterator for &'a mut Headers {
171    type Item = (&'a HeaderName, &'a mut HeaderValues);
172    type IntoIter = IterMut<'a>;
173
174    #[inline]
175    fn into_iter(self) -> Self::IntoIter {
176        self.iter_mut()
177    }
178}
179
180impl Debug for Headers {
181    fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
182        f.debug_map().entries(self.headers.iter()).finish()
183    }
184}
185
186impl AsRef<Headers> for Headers {
187    fn as_ref(&self) -> &Headers {
188        self
189    }
190}
191
192impl AsMut<Headers> for Headers {
193    fn as_mut(&mut self) -> &mut Headers {
194        self
195    }
196}
197
198#[cfg(test)]
199mod tests {
200    use super::*;
201    use std::str::FromStr;
202
203    const STATIC_HEADER: HeaderName = HeaderName::from_lowercase_str("hello");
204
205    #[test]
206    fn test_header_name_static_non_static() -> crate::Result<()> {
207        let static_header = HeaderName::from_lowercase_str("hello");
208        let non_static_header = HeaderName::from_str("hello")?;
209
210        let mut headers = Headers::new();
211        headers.append(STATIC_HEADER, "foo0");
212        headers.append(static_header.clone(), "foo1");
213        headers.append(non_static_header.clone(), "foo2");
214
215        assert_eq!(headers[STATIC_HEADER], ["foo0", "foo1", "foo2",][..]);
216        assert_eq!(headers[static_header], ["foo0", "foo1", "foo2",][..]);
217        assert_eq!(headers[non_static_header], ["foo0", "foo1", "foo2",][..]);
218
219        Ok(())
220    }
221
222    #[test]
223    fn index_into_headers() {
224        let mut headers = Headers::new();
225        headers.insert("hello", "foo0");
226        assert_eq!(headers["hello"], "foo0");
227        assert_eq!(headers.get("hello").unwrap(), "foo0");
228    }
229
230    #[test]
231    fn test_debug_single() {
232        let mut headers = Headers::new();
233        headers.insert("single", "foo0");
234        assert_eq!(format!("{:?}", headers), r#"{"single": "foo0"}"#);
235    }
236
237    #[test]
238    fn test_debug_multiple() {
239        let mut headers = Headers::new();
240        headers.append("multi", "foo0");
241        headers.append("multi", "foo1");
242        assert_eq!(format!("{:?}", headers), r#"{"multi": ["foo0", "foo1"]}"#);
243    }
244}