1
0
mirror of https://github.com/actix/actix-extras.git synced 2024-11-25 16:32:43 +01:00
actix-extras/src/header/mod.rs

249 lines
6.1 KiB
Rust
Raw Normal View History

2018-03-06 04:28:42 +01:00
//! Various http headers
2018-03-06 09:43:25 +01:00
// This is mostly copy of [hyper](https://github.com/hyperium/hyper/tree/master/src/header)
2018-03-06 09:43:25 +01:00
use std::fmt;
use std::str::FromStr;
use bytes::{Bytes, BytesMut};
2018-03-31 02:31:18 +02:00
use modhttp::{Error as HttpError};
use modhttp::header::GetAll;
2018-03-06 09:43:25 +01:00
use mime::Mime;
2018-03-31 02:31:18 +02:00
pub use modhttp::header::*;
use error::ParseError;
2018-03-06 04:28:42 +01:00
use httpmessage::HttpMessage;
mod common;
2018-03-06 09:43:25 +01:00
mod shared;
#[doc(hidden)]
2018-03-06 04:28:42 +01:00
pub use self::common::*;
2018-03-06 09:43:25 +01:00
#[doc(hidden)]
pub use self::shared::*;
2018-03-06 04:28:42 +01:00
2018-03-06 04:28:42 +01:00
#[doc(hidden)]
/// A trait for any object that will represent a header field and value.
pub trait Header where Self: IntoHeaderValue {
2018-03-06 04:28:42 +01:00
/// Returns the name of the header field
2018-03-31 02:31:18 +02:00
fn name() -> HeaderName;
2018-03-06 04:28:42 +01:00
/// Parse a header
fn parse<T: HttpMessage>(msg: &T) -> Result<Self, ParseError>;
}
#[doc(hidden)]
/// A trait for any object that can be Converted to a `HeaderValue`
pub trait IntoHeaderValue: Sized {
/// The type returned in the event of a conversion error.
type Error: Into<HttpError>;
/// Cast from PyObject to a concrete Python object type.
2018-03-31 02:31:18 +02:00
fn try_into(self) -> Result<HeaderValue, Self::Error>;
}
2018-03-31 02:31:18 +02:00
impl IntoHeaderValue for HeaderValue {
type Error = InvalidHeaderValue;
#[inline]
2018-03-31 02:31:18 +02:00
fn try_into(self) -> Result<HeaderValue, Self::Error> {
Ok(self)
}
}
impl<'a> IntoHeaderValue for &'a str {
2018-03-31 02:31:18 +02:00
type Error = InvalidHeaderValue;
#[inline]
2018-03-31 02:31:18 +02:00
fn try_into(self) -> Result<HeaderValue, Self::Error> {
self.parse()
}
}
impl<'a> IntoHeaderValue for &'a [u8] {
2018-03-31 02:31:18 +02:00
type Error = InvalidHeaderValue;
#[inline]
2018-03-31 02:31:18 +02:00
fn try_into(self) -> Result<HeaderValue, Self::Error> {
HeaderValue::from_bytes(self)
}
}
impl IntoHeaderValue for Bytes {
2018-03-31 02:31:18 +02:00
type Error = InvalidHeaderValueBytes;
#[inline]
2018-03-31 02:31:18 +02:00
fn try_into(self) -> Result<HeaderValue, Self::Error> {
HeaderValue::from_shared(self)
2018-03-06 09:43:25 +01:00
}
}
impl IntoHeaderValue for Vec<u8> {
2018-03-31 02:31:18 +02:00
type Error = InvalidHeaderValueBytes;
2018-03-06 09:43:25 +01:00
#[inline]
2018-03-31 02:31:18 +02:00
fn try_into(self) -> Result<HeaderValue, Self::Error> {
HeaderValue::from_shared(Bytes::from(self))
2018-03-06 09:43:25 +01:00
}
}
impl IntoHeaderValue for String {
2018-03-31 02:31:18 +02:00
type Error = InvalidHeaderValueBytes;
2018-03-06 09:43:25 +01:00
#[inline]
2018-03-31 02:31:18 +02:00
fn try_into(self) -> Result<HeaderValue, Self::Error> {
HeaderValue::from_shared(Bytes::from(self))
2018-03-06 09:43:25 +01:00
}
}
impl IntoHeaderValue for Mime {
2018-03-31 02:31:18 +02:00
type Error = InvalidHeaderValueBytes;
2018-03-06 09:43:25 +01:00
#[inline]
2018-03-31 02:31:18 +02:00
fn try_into(self) -> Result<HeaderValue, Self::Error> {
HeaderValue::from_shared(Bytes::from(format!("{}", self)))
}
}
/// Represents supported types of content encodings
#[derive(Copy, Clone, PartialEq, Debug)]
pub enum ContentEncoding {
/// Automatically select encoding based on encoding negotiation
Auto,
/// A format using the Brotli algorithm
2018-03-14 01:21:22 +01:00
#[cfg(feature="brotli")]
Br,
/// A format using the zlib structure with deflate algorithm
Deflate,
/// Gzip algorithm
Gzip,
/// Indicates the identity function (i.e. no compression, nor modification)
Identity,
}
impl ContentEncoding {
#[inline]
pub fn is_compression(&self) -> bool {
match *self {
ContentEncoding::Identity | ContentEncoding::Auto => false,
_ => true
}
}
#[inline]
pub fn as_str(&self) -> &'static str {
match *self {
2018-03-14 01:21:22 +01:00
#[cfg(feature="brotli")]
ContentEncoding::Br => "br",
ContentEncoding::Gzip => "gzip",
ContentEncoding::Deflate => "deflate",
ContentEncoding::Identity | ContentEncoding::Auto => "identity",
}
}
2018-03-18 19:05:44 +01:00
#[inline]
/// default quality value
pub fn quality(&self) -> f64 {
match *self {
2018-03-14 01:21:22 +01:00
#[cfg(feature="brotli")]
ContentEncoding::Br => 1.1,
ContentEncoding::Gzip => 1.0,
ContentEncoding::Deflate => 0.9,
ContentEncoding::Identity | ContentEncoding::Auto => 0.1,
}
}
}
// TODO: remove memory allocation
impl<'a> From<&'a str> for ContentEncoding {
fn from(s: &'a str) -> ContentEncoding {
match s.trim().to_lowercase().as_ref() {
2018-03-14 01:21:22 +01:00
#[cfg(feature="brotli")]
"br" => ContentEncoding::Br,
"gzip" => ContentEncoding::Gzip,
"deflate" => ContentEncoding::Deflate,
2018-03-11 22:50:13 +01:00
_ => ContentEncoding::Identity,
}
}
}
2018-03-06 09:43:25 +01:00
#[doc(hidden)]
pub(crate) struct Writer {
buf: BytesMut,
}
impl Writer {
fn new() -> Writer {
Writer{buf: BytesMut::new()}
}
fn take(&mut self) -> Bytes {
self.buf.take().freeze()
}
}
impl fmt::Write for Writer {
#[inline]
fn write_str(&mut self, s: &str) -> fmt::Result {
self.buf.extend_from_slice(s.as_bytes());
Ok(())
}
#[inline]
fn write_fmt(&mut self, args: fmt::Arguments) -> fmt::Result {
fmt::write(self, args)
}
}
#[inline]
#[doc(hidden)]
/// Reads a comma-delimited raw header into a Vec.
2018-03-31 02:31:18 +02:00
pub fn from_comma_delimited<T: FromStr>(all: GetAll<HeaderValue>)
2018-03-06 09:43:25 +01:00
-> Result<Vec<T>, ParseError>
{
let mut result = Vec::new();
for h in all {
let s = h.to_str().map_err(|_| ParseError::Header)?;
result.extend(s.split(',')
.filter_map(|x| match x.trim() {
"" => None,
y => Some(y)
})
.filter_map(|x| x.trim().parse().ok()))
}
Ok(result)
}
#[inline]
#[doc(hidden)]
/// Reads a single string when parsing a header.
2018-03-31 02:31:18 +02:00
pub fn from_one_raw_str<T: FromStr>(val: Option<&HeaderValue>)
2018-03-06 09:43:25 +01:00
-> Result<T, ParseError>
{
if let Some(line) = val {
let line = line.to_str().map_err(|_| ParseError::Header)?;
if !line.is_empty() {
return T::from_str(line).or(Err(ParseError::Header))
}
}
Err(ParseError::Header)
}
#[inline]
#[doc(hidden)]
/// Format an array into a comma-delimited string.
pub fn fmt_comma_delimited<T>(f: &mut fmt::Formatter, parts: &[T]) -> fmt::Result
where T: fmt::Display
{
let mut iter = parts.iter();
if let Some(part) = iter.next() {
fmt::Display::fmt(part, f)?;
}
for part in iter {
f.write_str(", ")?;
fmt::Display::fmt(part, f)?;
}
Ok(())
}