static-web-server 2.42.0

A cross-platform, high-performance and asynchronous web server for static files-serving.
Documentation
// SPDX-License-Identifier: MIT OR Apache-2.0
// Original code by Parker Timmerman
// Original code sourced from https://github.com/hyperium/headers/pull/70

// Derives an enum to represent content codings and some helpful impls
macro_rules! define_content_coding {
    ($($coding:ident; $str:expr,)+) => {
        use hyper::header::HeaderValue;
        use std::str::FromStr;

        #[derive(Copy, Clone, Debug, Eq, PartialEq)]
        /// Values that are used with headers like [`Content-Encoding`](headers::ContentEncoding) or
        /// [`Accept-Encoding`](super::AcceptEncoding)
        ///
        /// [RFC7231](https://www.iana.org/assignments/http-parameters/http-parameters.xhtml)
        pub enum ContentCoding {
            $(
                #[allow(clippy::upper_case_acronyms)]
                #[doc = $str]
                $coding,
            )+
        }

        impl ContentCoding {
            /// Returns a `&'static str` for a `ContentCoding`
            #[inline]
            pub(crate) fn as_str(&self) -> &'static str {
                match *self {
                    $(ContentCoding::$coding => $str,)+
                }
            }
        }

        impl From<&str> for ContentCoding {
            /// Given a `&str` returns a `ContentCoding`
            ///
            /// Note this will never fail, in the case of `&str` being an invalid content coding,
            /// will return `ContentCoding::IDENTITY` because `'identity'` is generally always an
            /// accepted coding.
            #[inline]
            fn from(s: &str) -> Self {
                ContentCoding::from_str(s).unwrap_or_else(|_| ContentCoding::IDENTITY)
            }
        }

        impl FromStr for ContentCoding {
            type Err = ();

            /// Given a `&str` will try to return a `ContentCoding`
            ///
            /// Different from `ContentCoding::from(&str)`, if `&str` is an invalid content
            /// coding, it will return `Err(())`
            #[inline]
            fn from_str(s: &str) -> Result<Self, Self::Err> {
                match s {
                    $(
                        stringify!($coding)
                        | $str => Ok(Self::$coding),
                    )+
                    _ => Err(())
                }
            }
        }

        impl std::fmt::Display for ContentCoding {
            #[inline]
            fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> Result<(), std::fmt::Error> {
                write!(f, "{}", match *self {
                    $(ContentCoding::$coding => $str.to_string(),)+
                })
            }
        }

        impl From<ContentCoding> for HeaderValue {
            fn from(coding: ContentCoding) -> HeaderValue {
                match coding {
                    $(ContentCoding::$coding => HeaderValue::from_static($str),)+
                }
            }
        }
    }
}

impl ContentCoding {
    pub fn priority(&self) -> u8 {
        match self {
            Self::IDENTITY => 1,
            Self::COMPRESS => 2,
            Self::DEFLATE => 3,
            Self::GZIP => 4,
            Self::BROTLI => 5,
            Self::ZSTD => 6,
            _ => 0,
        }
    }
}

define_content_coding! {
    BROTLI; "br",
    COMPRESS; "compress",
    DEFLATE; "deflate",
    GZIP; "gzip",
    IDENTITY; "identity",
    ZSTD; "zstd",
}

#[cfg(test)]
mod tests {
    use super::ContentCoding;
    use std::str::FromStr;

    #[test]
    fn as_str() {
        assert_eq!(ContentCoding::GZIP.as_str(), "gzip");
    }

    #[test]
    fn to_string() {
        assert_eq!(ContentCoding::DEFLATE.to_string(), "deflate".to_string());
    }

    #[test]
    fn from() {
        assert_eq!(ContentCoding::from("br"), ContentCoding::BROTLI);
        assert_eq!(ContentCoding::from("GZIP"), ContentCoding::GZIP);
        assert_eq!(ContentCoding::from("zstd"), ContentCoding::ZSTD);
        assert_eq!(ContentCoding::from("blah blah"), ContentCoding::IDENTITY);
    }

    #[test]
    fn from_str() {
        assert_eq!(ContentCoding::from_str("br"), Ok(ContentCoding::BROTLI));
        assert_eq!(ContentCoding::from_str("zstd"), Ok(ContentCoding::ZSTD));
        assert_eq!(ContentCoding::from_str("blah blah"), Err(()));
    }
}