rama_http/matcher/
method.rsuse crate::{Method, Request};
use rama_core::{context::Extensions, Context};
use std::{
fmt,
fmt::{Debug, Formatter},
};
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
pub struct MethodMatcher(u16);
impl MethodMatcher {
pub const CONNECT: Self = Self::from_bits(0b0_0000_0001);
pub const DELETE: Self = Self::from_bits(0b0_0000_0010);
pub const GET: Self = Self::from_bits(0b0_0000_0100);
pub const HEAD: Self = Self::from_bits(0b0_0000_1000);
pub const OPTIONS: Self = Self::from_bits(0b0_0001_0000);
pub const PATCH: Self = Self::from_bits(0b0_0010_0000);
pub const POST: Self = Self::from_bits(0b0_0100_0000);
pub const PUT: Self = Self::from_bits(0b0_1000_0000);
pub const TRACE: Self = Self::from_bits(0b1_0000_0000);
const fn bits(&self) -> u16 {
let bits = self;
bits.0
}
const fn from_bits(bits: u16) -> Self {
Self(bits)
}
pub(crate) const fn contains(&self, other: Self) -> bool {
self.bits() & other.bits() == other.bits()
}
pub const fn or(self, other: Self) -> Self {
Self(self.0 | other.0)
}
}
impl<State, Body> rama_core::matcher::Matcher<State, Request<Body>> for MethodMatcher {
fn matches(
&self,
_ext: Option<&mut Extensions>,
_ctx: &Context<State>,
req: &Request<Body>,
) -> bool {
MethodMatcher::try_from(req.method())
.ok()
.map(|method| self.contains(method))
.unwrap_or_default()
}
}
#[derive(Debug)]
pub struct NoMatchingMethodMatcher {
method: Method,
}
impl NoMatchingMethodMatcher {
pub fn method(&self) -> &Method {
&self.method
}
}
impl fmt::Display for NoMatchingMethodMatcher {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "no `MethodMatcher` for `{}`", self.method.as_str())
}
}
impl std::error::Error for NoMatchingMethodMatcher {}
impl TryFrom<&Method> for MethodMatcher {
type Error = NoMatchingMethodMatcher;
fn try_from(m: &Method) -> Result<Self, Self::Error> {
match m {
&Method::CONNECT => Ok(MethodMatcher::CONNECT),
&Method::DELETE => Ok(MethodMatcher::DELETE),
&Method::GET => Ok(MethodMatcher::GET),
&Method::HEAD => Ok(MethodMatcher::HEAD),
&Method::OPTIONS => Ok(MethodMatcher::OPTIONS),
&Method::PATCH => Ok(MethodMatcher::PATCH),
&Method::POST => Ok(MethodMatcher::POST),
&Method::PUT => Ok(MethodMatcher::PUT),
&Method::TRACE => Ok(MethodMatcher::TRACE),
other => Err(Self::Error {
method: other.clone(),
}),
}
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn from_http_method() {
assert_eq!(
MethodMatcher::try_from(&Method::CONNECT).unwrap(),
MethodMatcher::CONNECT
);
assert_eq!(
MethodMatcher::try_from(&Method::DELETE).unwrap(),
MethodMatcher::DELETE
);
assert_eq!(
MethodMatcher::try_from(&Method::GET).unwrap(),
MethodMatcher::GET
);
assert_eq!(
MethodMatcher::try_from(&Method::HEAD).unwrap(),
MethodMatcher::HEAD
);
assert_eq!(
MethodMatcher::try_from(&Method::OPTIONS).unwrap(),
MethodMatcher::OPTIONS
);
assert_eq!(
MethodMatcher::try_from(&Method::PATCH).unwrap(),
MethodMatcher::PATCH
);
assert_eq!(
MethodMatcher::try_from(&Method::POST).unwrap(),
MethodMatcher::POST
);
assert_eq!(
MethodMatcher::try_from(&Method::PUT).unwrap(),
MethodMatcher::PUT
);
assert_eq!(
MethodMatcher::try_from(&Method::TRACE).unwrap(),
MethodMatcher::TRACE
);
}
}