mirror of
https://github.com/rwf2/Rocket.git
synced 2025-02-16 13:42:05 +00:00
Document Header and HeaderMap.
This commit is contained in:
parent
8d8d504b59
commit
3414266a8a
@ -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)
|
||||
|
Loading…
Reference in New Issue
Block a user