diff --git a/Cargo.lock b/Cargo.lock index 9834641..87828e4 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -805,6 +805,7 @@ dependencies = [ "diesel-crud-derive", "diesel-crud-trait", "into-response-derive", + "mime", "nom", "read-files", "serde", diff --git a/Cargo.toml b/Cargo.toml index 4dcbc30..3c96b1f 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -24,6 +24,7 @@ homepage = { workspace = true } axum = { version = "0.7", optional = true, features = ["multipart"] } tower = { version = "0.5", optional = true } tower-http = { version = "0.5", optional = true, features = ["trace", "cors", "normalize-path"] } +mime = { version = "0.3.17", optional = true } # Async tokio = { version = "1.39", optional = true, features = ["fs"] } tokio-util = { version = "0.7", optional = true, features = ["io"] } @@ -57,7 +58,7 @@ diesel-async = "0.5" derive_more = "1.0" [features] -axum = ["dep:axum", "dep:tower", "dep:tower-http", "dep:thiserror", "dep:tracing", "dep:tracing-subscriber", "dep:tokio"] +axum = ["dep:axum", "dep:tower", "dep:tower-http", "dep:thiserror", "dep:tracing", "dep:tracing-subscriber", "dep:tokio", "dep:mime"] diesel = ["dep:diesel-crud-trait"] io = ["dep:tokio", "dep:tokio-util"] iter = [] diff --git a/examples/multipart_file/Cargo.lock b/examples/multipart_file/Cargo.lock index b3e8f65..141d9a8 100644 --- a/examples/multipart_file/Cargo.lock +++ b/examples/multipart_file/Cargo.lock @@ -310,6 +310,7 @@ version = "1.4.3" dependencies = [ "axum", "derive_more", + "mime", "thiserror", "tokio", "tower 0.5.0", diff --git a/src/axum/extractor.rs b/src/axum/extractor.rs index aeca22a..271d827 100644 --- a/src/axum/extractor.rs +++ b/src/axum/extractor.rs @@ -6,57 +6,22 @@ use axum::{ }, response::IntoResponse, }; +use mime::Mime; +use std::str::FromStr; use thiserror::Error; -#[derive(PartialEq, Eq, Ord, PartialOrd, Hash, Debug, Clone, Copy)] -pub enum ContentType { - Json, - Form, - Multipart, - Pdf, - Html, - Unknown, -} - -impl From<&str> for ContentType { - fn from(content_type: &str) -> Self { - match content_type { - "application/json" => ContentType::Json, - "application/x-www-form-urlencoded" => ContentType::Form, - "multipart/form-data" => ContentType::Multipart, - "application/pdf" => ContentType::Pdf, - "text/html" => ContentType::Html, - _ => ContentType::Unknown, - } - } -} - -impl From for ContentType { - fn from(content_type: String) -> Self { - ContentType::from(content_type.as_str()) - } -} - -impl From> for ContentType { - fn from(content_type: Option<&str>) -> Self { - content_type - .map(ContentType::from) - .unwrap_or(ContentType::Unknown) - } -} - #[derive(Debug, Clone, PartialEq)] pub struct File { pub filename: String, pub bytes: Vec, - pub content_type: ContentType, + pub content_type: Mime, } impl File { pub fn new( filename: impl Into, bytes: impl Into>, - content_type: impl Into, + content_type: impl Into, ) -> Self { Self { filename: filename.into(), @@ -70,7 +35,9 @@ impl File { .file_name() .ok_or(MultipartFileRejection::MissingFilename)? .to_string(); - let content_type: ContentType = field.content_type().into(); + let content_type = Mime::from_str(field.content_type().ok_or_else(|| { + MultipartFileRejection::FieldError("Missing or illegal content type".to_string()) + })?)?; let bytes = field.bytes().await?; Ok(File::new(filename, bytes, content_type)) } @@ -93,6 +60,8 @@ pub enum MultipartFileRejection { MultipartRejection(#[from] MultipartRejection), #[error("Field error: {0}")] FieldError(String), + #[error(transparent)] + FromStrError(#[from] mime::FromStrError), #[error("No files found")] NoFiles, #[error("Expected one file, got several")] @@ -130,6 +99,9 @@ impl IntoResponse for MultipartFileRejection { MultipartFileRejection::BodyError(error) => { (axum::http::StatusCode::BAD_REQUEST, error).into_response() } + MultipartFileRejection::FromStrError(error) => { + (axum::http::StatusCode::BAD_REQUEST, error.to_string()).into_response() + } } } }