Rocket/lib/src/http/header.rs

539 lines
17 KiB
Rust
Raw Normal View History

use std::collections::HashMap;
use std::borrow::{Borrow, Cow};
use std::fmt;
use http::ascii::{UncasedAscii, UncasedAsciiRef};
2016-12-20 02:04:31 +00:00
/// Simple representation of an HTTP header.
#[derive(Debug, Clone, Hash, PartialEq, Eq)]
pub struct Header<'h> {
2016-12-20 02:04:31 +00:00
/// The name of the header.
pub name: UncasedAscii<'h>,
2016-12-20 02:04:31 +00:00
/// The value of the header.
pub value: Cow<'h, str>,
}
impl<'h> Header<'h> {
2016-12-20 02:04:31 +00:00
/// 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>>
{
Header {
name: UncasedAscii::new(name),
value: value.into()
}
}
/// Returns the name of this header with casing preserved. To do a
/// case-insensitive equality check, use `.name` directly.
///
/// # Example
///
/// A case-sensitive equality check:
///
/// ```rust
/// use rocket::http::Header;
///
/// let value = format!("{} value", "custom");
/// let header = Header::new("X-Custom-Header", value);
/// assert_eq!(header.name(), "X-Custom-Header");
/// assert!(header.name() != "X-CUSTOM-HEADER");
/// ```
///
/// A case-insensitive equality check via `.name`:
///
/// ```rust
/// use rocket::http::Header;
///
/// let header = Header::new("X-Custom-Header", "custom value");
/// assert_eq!(header.name, "X-Custom-Header");
/// assert_eq!(header.name, "X-CUSTOM-HEADER");
/// ```
#[inline(always)]
pub fn name(&self) -> &str {
self.name.as_str()
}
/// Returns the value of this header.
///
/// # Example
///
/// A case-sensitive equality check:
///
/// ```rust
/// use rocket::http::Header;
///
/// let header = Header::new("X-Custom-Header", "custom value");
/// assert_eq!(header.value(), "custom value");
/// ```
#[inline(always)]
pub fn value(&self) -> &str {
&self.value
}
}
impl<'h> fmt::Display for Header<'h> {
#[inline(always)]
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "{}: {}", self.name, self.value)
}
}
2016-12-21 09:30:45 +00:00
/// A collection of headers, mapping a header name to its many ordered values.
2016-12-17 17:18:30 +00:00
#[derive(Clone, Debug, PartialEq, Default)]
pub struct HeaderMap<'h> {
headers: HashMap<UncasedAscii<'h>, Vec<Cow<'h, str>>>
}
impl<'h> HeaderMap<'h> {
2016-12-20 02:04:31 +00:00
/// Returns an empty collection.
#[inline(always)]
pub fn new() -> HeaderMap<'h> {
HeaderMap { headers: HashMap::new() }
}
2016-12-20 02:04:31 +00:00
/// 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.into() : &UncasedAsciiRef).is_some()
}
2016-12-20 02:04:31 +00:00
/// 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()
}
2016-12-20 02:04:31 +00:00
/// Returns `true` if there are no headers stored in the map. Otherwise
/// returns `false`.
///
/// # Example
///
/// ```rust
2017-02-02 10:16:21 +00:00
/// use rocket::http::HeaderMap;
2016-12-20 02:04:31 +00:00
///
/// let map = HeaderMap::new();
/// assert!(map.is_empty());
/// ```
2016-12-17 17:18:30 +00:00
#[inline]
pub fn is_empty(&self) -> bool {
self.headers.is_empty()
}
2016-12-20 02:04:31 +00:00
/// 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
2017-02-02 10:16:21 +00:00
/// use rocket::http::HeaderMap;
2016-12-20 02:04:31 +00:00
///
/// 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() : &UncasedAsciiRef)
.into_iter()
.flat_map(|values| values.iter().map(|val| val.borrow()))
}
2016-12-20 02:04:31 +00:00
/// Returns the _first_ value stored for the header with name `name` if
/// there is one.
///
/// # Examples
///
/// Retrieve the first value when one exists:
///
/// ```rust
2017-02-02 10:16:21 +00:00
/// use rocket::http::HeaderMap;
2016-12-20 02:04:31 +00:00
///
/// let mut map = HeaderMap::new();
/// map.add_raw("X-Custom", "value_1");
/// map.add_raw("X-Custom", "value_2");
///
/// assert_eq!(map.len(), 2);
///
2017-02-02 10:16:21 +00:00
/// let first_value = map.get_one("X-Custom");
2016-12-20 02:04:31 +00:00
/// assert_eq!(first_value, Some("value_1"));
/// ```
///
/// Attempt to retrieve a value that doesn't exist:
///
/// ```rust
2017-02-02 10:16:21 +00:00
/// use rocket::http::HeaderMap;
2016-12-20 02:04:31 +00:00
///
/// let mut map = HeaderMap::new();
/// map.add_raw("X-Custom", "value_1");
///
2017-02-02 10:16:21 +00:00
/// let first_value = map.get_one("X-Other");
2016-12-20 02:04:31 +00:00
/// assert_eq!(first_value, None);
/// ```
#[inline]
pub fn get_one<'a>(&'a self, name: &str) -> Option<&'a str> {
self.headers.get(name.into() : &UncasedAsciiRef)
.and_then(|values| {
if values.len() >= 1 { Some(values[0].borrow()) }
else { None }
})
}
2016-12-20 02:04:31 +00:00
/// 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()
}
2016-12-20 02:04:31 +00:00
/// 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>>
{
self.replace(Header::new(name, value))
}
2016-12-20 02:04:31 +00:00
/// 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>>
{
self.headers.insert(UncasedAscii::new(name), values);
}
2016-12-20 02:04:31 +00:00
/// 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};
///
/// let mut map = HeaderMap::new();
///
/// map.add(&Cookie::new("a", "b"));
2016-12-20 02:04:31 +00:00
/// assert_eq!(map.get("Set-Cookie").count(), 1);
///
/// map.add(&Cookie::new("c", "d"));
2016-12-20 02:04:31 +00:00
/// 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);
}
2016-12-20 02:04:31 +00:00
/// 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>>
{
self.add(Header::new(name, value))
}
2016-12-20 02:04:31 +00:00
/// 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>>
{
self.headers.entry(UncasedAscii::new(name))
.or_insert(vec![])
.append(values)
}
2016-12-20 02:04:31 +00:00
/// 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.into() : &UncasedAsciiRef);
}
2016-12-20 02:04:31 +00:00
/// 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()
}
2016-12-20 02:04:31 +00:00
/// 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| {
Header::new(key.as_str(), val.borrow())
})
})
}
2016-12-20 02:04:31 +00:00
/// 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.
2016-12-17 17:18:30 +00:00
// TODO: Figure out what the return type is to implement IntoIterator.
#[inline(always)]
pub fn into_iter(self) -> impl Iterator<Item=Header<'h>> {
self.headers.into_iter().flat_map(|(name, value)| {
value.into_iter().map(move |value| {
Header {
name: name.clone(),
value: value
}
})
})
}
2016-12-20 02:04:31 +00:00
/// 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.
2017-02-03 10:16:46 +00:00
#[inline]
pub(crate) fn into_iter_raw(self)
-> impl Iterator<Item=(UncasedAscii<'h>, Vec<Cow<'h, str>>)> {
self.headers.into_iter()
}
}
#[cfg(test)]
mod tests {
use super::HeaderMap;
#[test]
fn case_insensitive_add_get() {
let mut map = HeaderMap::new();
map.add_raw("content-type", "application/json");
let ct = map.get_one("Content-Type");
assert_eq!(ct, Some("application/json"));
let ct2 = map.get_one("CONTENT-TYPE");
assert_eq!(ct2, Some("application/json"))
}
#[test]
fn case_insensitive_multiadd() {
let mut map = HeaderMap::new();
map.add_raw("x-custom", "a");
map.add_raw("X-Custom", "b");
map.add_raw("x-CUSTOM", "c");
let vals: Vec<_> = map.get("x-CuStOm").collect();
assert_eq!(vals, vec!["a", "b", "c"]);
}
}