#![deny(missing_docs)]
#![deny(missing_debug_implementations)]
#![cfg_attr(test, deny(warnings))]
extern crate bytes;
extern crate http;
pub use http::header::{self, HeaderName, HeaderValue};
pub trait Header {
const NAME: &'static HeaderName;
fn decode<'i, I>(values: &mut I) -> Result<Self, Error>
where
Self: Sized,
I: Iterator<Item = &'i HeaderValue>;
fn encode<E: Extend<HeaderValue>>(&self, values: &mut E);
}
#[derive(Debug)]
pub struct Error {
kind: Kind,
}
#[derive(Debug)]
enum Kind {
Invalid,
}
impl Error {
pub fn invalid() -> Error {
Error {
kind: Kind::Invalid,
}
}
}
#[derive(Debug)]
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> Extend<HeaderValue> for ToValues<'a> {
fn extend<T: IntoIterator<Item=HeaderValue>>(&mut self, iter: T) {
for value in iter {
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 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;
fn typed_try_get<H>(&self) -> Result<Option<H>, Error>
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,
{
HeaderMapExt::typed_try_get(self)
.unwrap_or(None)
}
fn typed_try_get<H>(&self) -> Result<Option<H>, Error>
where
H: Header,
{
let mut values = self.get_all(H::NAME).iter();
if values.size_hint() == (0, Some(0)) {
Ok(None)
} else {
H::decode(&mut values).map(Some)
}
}
}
mod sealed {
pub trait Sealed {}
impl Sealed for ::http::HeaderMap {}
}