use axum_core::{
body::Body,
extract::{FromRequest, Request},
response::{IntoResponse, Response},
};
use core::fmt;
use core::ops::{Deref, DerefMut};
use facet_core::Facet;
use http::{HeaderValue, StatusCode, header};
use http_body_util::BodyExt;
use crate::DeserializeError;
#[derive(Debug, Clone, Copy, Default)]
pub struct Toml<T>(pub T);
impl<T> Toml<T> {
pub fn into_inner(self) -> T {
self.0
}
}
impl<T> Deref for Toml<T> {
type Target = T;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl<T> DerefMut for Toml<T> {
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.0
}
}
impl<T> From<T> for Toml<T> {
fn from(inner: T) -> Self {
Self(inner)
}
}
#[derive(Debug)]
pub struct TomlRejection {
kind: TomlRejectionKind,
}
#[derive(Debug)]
enum TomlRejectionKind {
Body(axum_core::Error),
Deserialize(DeserializeError),
}
impl TomlRejection {
pub fn status(&self) -> StatusCode {
match &self.kind {
TomlRejectionKind::Body(_) => StatusCode::BAD_REQUEST,
TomlRejectionKind::Deserialize(_) => StatusCode::UNPROCESSABLE_ENTITY,
}
}
pub fn is_body_error(&self) -> bool {
matches!(&self.kind, TomlRejectionKind::Body(_))
}
pub fn is_deserialize_error(&self) -> bool {
matches!(&self.kind, TomlRejectionKind::Deserialize(_))
}
}
impl fmt::Display for TomlRejection {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match &self.kind {
TomlRejectionKind::Body(err) => {
write!(f, "Failed to read request body: {err}")
}
TomlRejectionKind::Deserialize(err) => {
write!(f, "Failed to deserialize TOML: {err}")
}
}
}
}
impl std::error::Error for TomlRejection {
fn source(&self) -> Option<&(dyn std::error::Error + 'static)> {
match &self.kind {
TomlRejectionKind::Body(err) => Some(err),
TomlRejectionKind::Deserialize(err) => Some(err),
}
}
}
impl IntoResponse for TomlRejection {
fn into_response(self) -> Response {
(self.status(), self.to_string()).into_response()
}
}
impl<T, S> FromRequest<S> for Toml<T>
where
T: Facet<'static>,
S: Send + Sync,
{
type Rejection = TomlRejection;
async fn from_request(req: Request, _state: &S) -> Result<Self, Self::Rejection> {
let bytes = req
.into_body()
.collect()
.await
.map_err(|e| TomlRejection {
kind: TomlRejectionKind::Body(axum_core::Error::new(e)),
})?
.to_bytes();
let value: T = crate::from_slice(&bytes).map_err(|e| TomlRejection {
kind: TomlRejectionKind::Deserialize(e),
})?;
Ok(Toml(value))
}
}
impl<T> IntoResponse for Toml<T>
where
T: Facet<'static>,
{
fn into_response(self) -> Response {
match crate::to_string(&self.0) {
Ok(s) => {
let mut res = Response::new(Body::from(s));
res.headers_mut().insert(
header::CONTENT_TYPE,
HeaderValue::from_static("application/toml"),
);
res
}
Err(err) => {
let body = format!("Failed to serialize response: {err}");
(StatusCode::INTERNAL_SERVER_ERROR, body).into_response()
}
}
}
}