Document Header and HeaderMap.

This commit is contained in:
Sergio Benitez 2016-12-19 18:04:31 -08:00
parent 8d8d504b59
commit 3414266a8a

View File

@ -4,13 +4,43 @@ use std::fmt;
use http::hyper::header as hyper;
/// Simple representation of an HTTP header.
#[derive(Debug, Clone, Hash, PartialEq, Eq)]
pub struct Header<'h> {
/// The name of the header.
pub name: Cow<'h, str>,
/// The value of the header.
pub value: Cow<'h, str>,
}
impl<'h> Header<'h> {
/// Constructs a new header. This method should be used rarely and only for
/// non-standard headers. Instead, prefer to use the `Into<Header>`
/// implementations of many types, including
/// [ContentType](struct.ContentType.html) and all of the headers in
/// [http::hyper::header](hyper/header/index.html).
///
/// # Examples
///
/// Create a custom header with name `X-Custom-Header` and value `custom
/// value`.
///
/// ```rust
/// use rocket::http::Header;
///
/// let header = Header::new("X-Custom-Header", "custom value");
/// assert_eq!(header.to_string(), "X-Custom-Header: custom value");
/// ```
///
/// Use a `String` as a value to do the same.
///
/// ```rust
/// use rocket::http::Header;
///
/// let value = format!("{} value", "custom");
/// let header = Header::new("X-Custom-Header", value);
/// assert_eq!(header.to_string(), "X-Custom-Header: custom value");
/// ```
#[inline(always)]
pub fn new<'a: 'h, 'b: 'h, N, V>(name: N, value: V) -> Header<'h>
where N: Into<Cow<'a, str>>, V: Into<Cow<'b, str>>
@ -36,32 +66,96 @@ impl<T> From<T> for Header<'static> where T: hyper::Header + hyper::HeaderFormat
}
}
/// A collection of headers, mapping many ordered values to a header names.
#[derive(Clone, Debug, PartialEq, Default)]
pub struct HeaderMap<'h> {
headers: HashMap<Cow<'h, str>, Vec<Cow<'h, str>>>
}
impl<'h> HeaderMap<'h> {
/// Returns an empty collection.
#[inline(always)]
pub fn new() -> HeaderMap<'h> {
HeaderMap { headers: HashMap::new() }
}
/// Returns true if `self` contains a header with the name `name`.
///
/// # Example
///
/// ```rust
/// use rocket::http::{HeaderMap, ContentType};
///
/// let mut map = HeaderMap::new();
/// map.add(ContentType::HTML);
///
/// assert!(map.contains("Content-Type"));
/// assert!(!map.contains("Accepts"));
/// ```
#[inline]
pub fn contains(&self, name: &str) -> bool {
self.headers.get(name).is_some()
}
/// Returns the number of _values_ stored in the map.
///
/// # Example
///
/// ```rust
/// use rocket::http::HeaderMap;
///
/// let mut map = HeaderMap::new();
/// assert_eq!(map.len(), 0);
///
/// map.add_raw("X-Custom", "value_1");
/// assert_eq!(map.len(), 1);
///
/// map.replace_raw("X-Custom", "value_2");
/// assert_eq!(map.len(), 1);
///
/// map.add_raw("X-Custom", "value_1");
/// assert_eq!(map.len(), 2);
/// ```
#[inline]
pub fn len(&self) -> usize {
self.headers.iter().flat_map(|(_, values)| values.iter()).count()
}
/// Returns `true` if there are no headers stored in the map. Otherwise
/// returns `false`.
///
/// # Example
///
/// ```rust
/// use rocket::http::{HeaderMap, ContentType};
///
/// let map = HeaderMap::new();
/// assert!(map.is_empty());
/// ```
#[inline]
pub fn is_empty(&self) -> bool {
self.headers.is_empty()
}
/// Returns an iterator over all of the values stored in `self` for the
/// header with name `name`. The headers are returned in FIFO order.
///
/// # Example
///
/// ```rust
/// use rocket::http::{HeaderMap, ContentType};
///
/// let mut map = HeaderMap::new();
/// map.add_raw("X-Custom", "value_1");
/// map.add_raw("X-Custom", "value_2");
///
/// assert_eq!(map.len(), 2);
///
/// let mut values = map.get("X-Custom");
/// assert_eq!(values.next(), Some("value_1"));
/// assert_eq!(values.next(), Some("value_2"));
/// assert_eq!(values.next(), None);
/// ```
#[inline]
pub fn get<'a>(&'a self, name: &str) -> impl Iterator<Item=&'a str> {
self.headers.get(name).into_iter().flat_map(|values| {
@ -69,6 +163,37 @@ impl<'h> HeaderMap<'h> {
})
}
/// Returns the _first_ value stored for the header with name `name` if
/// there is one.
///
/// # Examples
///
/// Retrieve the first value when one exists:
///
/// ```rust
/// use rocket::http::{HeaderMap, ContentType};
///
/// let mut map = HeaderMap::new();
/// map.add_raw("X-Custom", "value_1");
/// map.add_raw("X-Custom", "value_2");
///
/// assert_eq!(map.len(), 2);
///
/// let mut first_value = map.get_one("X-Custom");
/// assert_eq!(first_value, Some("value_1"));
/// ```
///
/// Attempt to retrieve a value that doesn't exist:
///
/// ```rust
/// use rocket::http::{HeaderMap, ContentType};
///
/// let mut map = HeaderMap::new();
/// map.add_raw("X-Custom", "value_1");
///
/// let mut first_value = map.get_one("X-Other");
/// assert_eq!(first_value, None);
/// ```
#[inline]
pub fn get_one<'a>(&'a self, name: &str) -> Option<&'a str> {
self.headers.get(name).and_then(|values| {
@ -77,12 +202,61 @@ impl<'h> HeaderMap<'h> {
})
}
/// Replace any header that matches the name of `header.name` with `header`.
/// If there is no such header in `self`, add `header`. If the matching
/// header had multiple values, all of the values are removed, and only the
/// value in `header` will remain.
///
/// # Example
///
/// Replace a header that doesn't yet exist:
///
/// ```rust
/// use rocket::http::{HeaderMap, ContentType};
///
/// let mut map = HeaderMap::new();
/// map.replace(ContentType::JSON);
///
/// assert!(map.get_one("Content-Type").is_some());
/// ```
///
/// Replace a header that already exists:
///
/// ```rust
/// use rocket::http::{HeaderMap, ContentType};
///
/// let mut map = HeaderMap::new();
///
/// map.replace(ContentType::JSON);
/// assert_eq!(map.get_one("Content-Type"), Some("application/json"));
///
/// map.replace(ContentType::GIF);
/// assert_eq!(map.get_one("Content-Type"), Some("image/gif"));
/// assert_eq!(map.len(), 1);
/// ```
#[inline(always)]
pub fn replace<'p: 'h, H: Into<Header<'p>>>(&mut self, header: H) -> bool {
let header = header.into();
self.headers.insert(header.name, vec![header.value]).is_some()
}
/// A convenience method to replace a header using a raw name and value.
/// Aliases `replace(Header::new(name, value))`. Should be used rarely.
///
/// # Example
///
/// ```rust
/// use rocket::http::HeaderMap;
///
/// let mut map = HeaderMap::new();
///
/// map.replace_raw("X-Custom", "value_1");
/// assert_eq!(map.get_one("X-Custom"), Some("value_1"));
///
/// map.replace_raw("X-Custom", "value_2");
/// assert_eq!(map.get_one("X-Custom"), Some("value_2"));
/// assert_eq!(map.len(), 1);
/// ```
#[inline(always)]
pub fn replace_raw<'a: 'h, 'b: 'h, N, V>(&mut self, name: N, value: V) -> bool
where N: Into<Cow<'a, str>>, V: Into<Cow<'b, str>>
@ -90,6 +264,25 @@ impl<'h> HeaderMap<'h> {
self.replace(Header::new(name, value))
}
/// Replaces all of the values for a header with name `name` with `values`.
/// This a low-level method and should rarely be used.
///
/// # Example
///
/// ```rust
/// use rocket::http::HeaderMap;
///
/// let mut map = HeaderMap::new();
/// map.add_raw("X-Custom", "value_1");
/// map.add_raw("X-Custom", "value_2");
///
/// let vals: Vec<_> = map.get("X-Custom").map(|s| s.to_string()).collect();
/// assert_eq!(vals, vec!["value_1", "value_2"]);
///
/// map.replace_all("X-Custom", vec!["value_3".into(), "value_4".into()]);
/// let vals: Vec<_> = map.get("X-Custom").collect();
/// assert_eq!(vals, vec!["value_3", "value_4"]);
/// ```
#[inline(always)]
pub fn replace_all<'n, 'v: 'h, H>(&mut self, name: H, values: Vec<Cow<'v, str>>)
where 'n: 'h, H: Into<Cow<'n, str>>
@ -97,12 +290,46 @@ impl<'h> HeaderMap<'h> {
self.headers.insert(name.into(), values);
}
/// Adds `header` into the map. If a header with `header.name` was
/// previously added, that header will have one more value.
///
/// ```rust
/// use rocket::http::{Cookie, HeaderMap};
/// use rocket::http::hyper::header;
///
/// let mut map = HeaderMap::new();
///
/// let cookie = vec![Cookie::new("a".to_string(), "b".to_string())];
/// map.add(header::SetCookie(cookie));
/// assert_eq!(map.get("Set-Cookie").count(), 1);
///
/// let cookie = vec![Cookie::new("c".to_string(), "d".to_string())];
/// map.add(header::SetCookie(cookie));
/// assert_eq!(map.get("Set-Cookie").count(), 2);
/// ```
#[inline(always)]
pub fn add<'p: 'h, H: Into<Header<'p>>>(&mut self, header: H) {
let header = header.into();
self.headers.entry(header.name).or_insert(vec![]).push(header.value);
}
/// A convenience method to add a header using a raw name and value.
/// Aliases `add(Header::new(name, value))`. Should be used rarely.
///
/// # Example
///
/// ```rust
/// use rocket::http::HeaderMap;
///
/// let mut map = HeaderMap::new();
///
/// map.add_raw("X-Custom", "value_1");
/// assert_eq!(map.get("X-Custom").count(), 1);
///
/// map.add_raw("X-Custom", "value_2");
/// let values: Vec<_> = map.get("X-Custom").collect();
/// assert_eq!(values, vec!["value_1", "value_2"]);
/// ```
#[inline(always)]
pub fn add_raw<'a: 'h, 'b: 'h, N, V>(&mut self, name: N, value: V)
where N: Into<Cow<'a, str>>, V: Into<Cow<'b, str>>
@ -110,6 +337,30 @@ impl<'h> HeaderMap<'h> {
self.add(Header::new(name, value))
}
/// Adds all of the values to a header with name `name`. This a low-level
/// method and should rarely be used. `values` will be empty when this
/// method returns.
///
/// # Example
///
/// ```rust
/// use rocket::http::HeaderMap;
///
/// let mut map = HeaderMap::new();
///
/// let mut values = vec!["value_1".into(), "value_2".into()];
/// map.add_all("X-Custom", &mut values);
/// assert_eq!(map.get("X-Custom").count(), 2);
/// assert_eq!(values.len(), 0);
///
/// let mut values = vec!["value_3".into(), "value_4".into()];
/// map.add_all("X-Custom", &mut values);
/// assert_eq!(map.get("X-Custom").count(), 4);
/// assert_eq!(values.len(), 0);
///
/// let values: Vec<_> = map.get("X-Custom").collect();
/// assert_eq!(values, vec!["value_1", "value_2", "value_3", "value_4"]);
/// ```
#[inline(always)]
pub fn add_all<'n, H>(&mut self, name: H, values: &mut Vec<Cow<'h, str>>)
where 'n:'h, H: Into<Cow<'n, str>>
@ -117,17 +368,71 @@ impl<'h> HeaderMap<'h> {
self.headers.entry(name.into()).or_insert(vec![]).append(values)
}
/// Remove all of the values for header with name `name`.
///
/// # Example
///
/// ```rust
/// use rocket::http::HeaderMap;
///
/// let mut map = HeaderMap::new();
/// map.add_raw("X-Custom", "value_1");
/// map.add_raw("X-Custom", "value_2");
/// map.add_raw("X-Other", "other");
///
/// assert_eq!(map.len(), 3);
///
/// map.remove("X-Custom");
/// assert_eq!(map.len(), 1);
#[inline(always)]
pub fn remove(&mut self, name: &str) {
self.headers.remove(name);
}
/// Removes all of the headers stored in this map and returns a vector
/// containing them. Header names are returned in no specific order, but all
/// values for a given header name are grouped together, and values are in
/// FIFO order.
///
/// # Example
///
/// ```rust
/// use rocket::http::{HeaderMap, Header};
/// use std::collections::HashSet;
///
/// // The headers we'll be storing.
/// let all_headers = vec![
/// Header::new("X-Custom", "value_1"),
/// Header::new("X-Custom", "value_2"),
/// Header::new("X-Other", "other")
/// ];
///
/// // Create a map, store all of the headers.
/// let mut map = HeaderMap::new();
/// for header in all_headers.clone() {
/// map.add(header)
/// }
///
/// assert_eq!(map.len(), 3);
///
/// // Now remove them all, ensure the map is empty.
/// let removed_headers = map.remove_all();
/// assert!(map.is_empty());
///
/// // Create two sets: what we expect and got. Ensure they're equal.
/// let expected_set: HashSet<_> = all_headers.into_iter().collect();
/// let actual_set: HashSet<_> = removed_headers.into_iter().collect();
/// assert_eq!(expected_set, actual_set);
/// ```
#[inline(always)]
pub fn remove_all(&mut self) -> Vec<Header<'h>> {
let old_map = ::std::mem::replace(self, HeaderMap::new());
old_map.into_iter().collect()
}
/// Returns an iterator over all of the `Header`s stored in the map. Header
/// names are returned in no specific order, but all values for a given
/// header name are grouped together, and values are in FIFO order.
pub fn iter<'s>(&'s self) -> impl Iterator<Item=Header<'s>> {
self.headers.iter().flat_map(|(key, values)| {
values.iter().map(move |val| {
@ -136,6 +441,10 @@ impl<'h> HeaderMap<'h> {
})
}
/// Consumes `self` and returns an iterator over all of the `Header`s stored
/// in the map. Header names are returned in no specific order, but all
/// values for a given header name are grouped together, and values are in
/// FIFO order.
// TODO: Figure out what the return type is to implement IntoIterator.
#[inline(always)]
pub fn into_iter(self) -> impl Iterator<Item=Header<'h>> {
@ -149,6 +458,9 @@ impl<'h> HeaderMap<'h> {
})
}
/// Consumes `self` and returns an iterator over all of the headers stored
/// in the map in the way they are stored. This is a low-level machinism and
/// should likely not be used.
#[doc(hidden)]
#[inline(always)]
pub fn into_iter_raw(self)