use axum_core::response::{IntoResponse, Response};
use pin_project_lite::pin_project;
use std::{
future::Future,
ops::Deref,
pin::Pin,
sync::Arc,
task::{ready, Context, Poll},
};
use tower::Service;
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
pub(crate) struct PercentDecodedStr(Arc<str>);
impl PercentDecodedStr {
pub(crate) fn new<S>(s: S) -> Option<Self>
where
S: AsRef<str>,
{
percent_encoding::percent_decode(s.as_ref().as_bytes())
.decode_utf8()
.ok()
.map(|decoded| Self(decoded.as_ref().into()))
}
pub(crate) fn as_str(&self) -> &str {
&self.0
}
}
impl Deref for PercentDecodedStr {
type Target = str;
#[inline]
fn deref(&self) -> &Self::Target {
self.as_str()
}
}
pin_project! {
#[project = EitherProj]
pub(crate) enum Either<A, B> {
A { #[pin] inner: A },
B { #[pin] inner: B },
}
}
#[derive(Clone)]
pub(crate) struct MapIntoResponse<S> {
inner: S,
}
impl<S> MapIntoResponse<S> {
pub(crate) fn new(inner: S) -> Self {
Self { inner }
}
}
impl<B, S> Service<http::Request<B>> for MapIntoResponse<S>
where
S: Service<http::Request<B>>,
S::Response: IntoResponse,
{
type Response = Response;
type Error = S::Error;
type Future = MapIntoResponseFuture<S::Future>;
fn poll_ready(&mut self, cx: &mut Context<'_>) -> Poll<Result<(), Self::Error>> {
self.inner.poll_ready(cx)
}
fn call(&mut self, req: http::Request<B>) -> Self::Future {
MapIntoResponseFuture {
inner: self.inner.call(req),
}
}
}
pin_project! {
pub(crate) struct MapIntoResponseFuture<F> {
#[pin]
inner: F,
}
}
impl<F, T, E> Future for MapIntoResponseFuture<F>
where
F: Future<Output = Result<T, E>>,
T: IntoResponse,
{
type Output = Result<Response, E>;
fn poll(self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
let res = ready!(self.project().inner.poll(cx)?);
Poll::Ready(Ok(res.into_response()))
}
}
pub(crate) fn try_downcast<T, K>(k: K) -> Result<T, K>
where
T: 'static,
K: Send + 'static,
{
let mut k = Some(k);
if let Some(k) = <dyn std::any::Any>::downcast_mut::<Option<T>>(&mut k) {
Ok(k.take().unwrap())
} else {
Err(k.unwrap())
}
}
#[test]
fn test_try_downcast() {
assert_eq!(try_downcast::<i32, _>(5_u32), Err(5_u32));
assert_eq!(try_downcast::<i32, _>(5_i32), Ok(5_i32));
}