#![deny(missing_docs)]
#![deny(missing_debug_implementations)]
#![cfg_attr(test, deny(warnings))]
extern crate bytes;
extern crate http;
use std::fmt;
pub use http::header::{self, HeaderName, HeaderValue};
pub mod decode;
pub mod encode;
pub trait Header {
const NAME: &'static HeaderName;
fn decode(values: &mut Values) -> Option<Self>
where
Self: Sized;
fn encode(&self, values: &mut ToValues);
}
#[derive(Debug)]
pub struct Values<'a> {
inner: http::header::ValueIter<'a, http::header::HeaderValue>,
should_exhaust: bool,
}
impl<'a> Values<'a> {
pub fn skip_exhaustive_iter_check(&mut self) {
self.should_exhaust = false;
}
}
impl<'a> Iterator for Values<'a> {
type Item = &'a HeaderValue;
#[inline]
fn next(&mut self) -> Option<Self::Item> {
self.inner.next()
}
#[inline]
fn size_hint(&self) -> (usize, Option<usize>) {
self.inner.size_hint()
}
}
impl<'a> DoubleEndedIterator for Values<'a> {
#[inline]
fn next_back(&mut self) -> Option<&'a HeaderValue> {
self.inner.next_back()
}
}
#[derive(Debug)]
pub struct ToValues<'a> {
state: State<'a>,
}
#[derive(Debug)]
enum State<'a> {
First(http::header::Entry<'a, HeaderValue>),
Latter(http::header::OccupiedEntry<'a, HeaderValue>),
Tmp,
}
impl<'a> ToValues<'a> {
pub fn append(&mut self, value: HeaderValue) {
let entry = match ::std::mem::replace(&mut self.state, State::Tmp) {
State::First(http::header::Entry::Occupied(mut e)) => {
e.insert(value);
e
},
State::First(http::header::Entry::Vacant(e)) => e.insert_entry(value),
State::Latter(mut e) => {
e.append(value);
e
},
State::Tmp => unreachable!("ToValues State::Tmp"),
};
self.state = State::Latter(entry);
}
pub fn append_fmt<T: fmt::Display>(&mut self, fmt: T) {
let s = fmt.to_string();
let value = match HeaderValue::from_shared(s.into()) {
Ok(val) => val,
Err(err) => panic!("illegal HeaderValue; error = {:?}, fmt = \"{}\"", err, fmt),
};
self.append(value);
}
}
pub trait HeaderMapExt: self::sealed::Sealed {
fn typed_insert<H>(&mut self, header: H)
where
H: Header;
fn typed_get<H>(&self) -> Option<H>
where
H: Header;
}
impl HeaderMapExt for http::HeaderMap {
fn typed_insert<H>(&mut self, header: H)
where
H: Header,
{
let entry = self
.entry(H::NAME)
.expect("HeaderName is always valid");
let mut values = ToValues {
state: State::First(entry),
};
header.encode(&mut values);
}
fn typed_get<H>(&self) -> Option<H>
where
H: Header,
{
let mut values = Values {
inner: self.get_all(H::NAME).iter(),
should_exhaust: true,
};
let header = H::decode(&mut values)?;
if !values.should_exhaust || values.next().is_none() {
Some(header)
} else {
None
}
}
}
mod sealed {
pub trait Sealed {}
impl Sealed for ::http::HeaderMap {}
}