use rocket::http::{ContentType, Method, Status};
use rocket::response::Responder;
use rocket::route::{Handler, Outcome};
use rocket::{Data, Request, Route};
#[derive(Clone)]
pub struct ContentHandler<R: AsRef<[u8]> + Clone + Send + Sync> {
content: (ContentType, R),
}
impl ContentHandler<String> {
pub fn json(content: &impl serde::Serialize) -> Self {
let json =
serde_json::to_string_pretty(content).expect("Could not serialize content as JSON.");
ContentHandler {
content: (ContentType::JSON, json),
}
}
}
impl ContentHandler<&'static [u8]> {
pub fn bytes(content_type: ContentType, content: &'static [u8]) -> Self {
ContentHandler {
content: (content_type, content),
}
}
}
impl ContentHandler<Vec<u8>> {
pub fn bytes_owned(content_type: ContentType, content: Vec<u8>) -> Self {
ContentHandler {
content: (content_type, content),
}
}
}
impl<R: AsRef<[u8]> + Clone + Send + Sync + 'static> ContentHandler<R> {
pub fn into_route(self, path: impl AsRef<str>) -> Route {
Route::new(Method::Get, path.as_ref(), self)
}
}
#[rocket::async_trait]
impl<R> Handler for ContentHandler<R>
where
R: AsRef<[u8]> + Clone + Send + Sync + 'static,
{
async fn handle<'r>(&self, req: &'r Request<'_>, data: Data<'r>) -> Outcome<'r> {
if req.uri().path().ends_with('/') {
Outcome::forward(data, Status::PermanentRedirect)
} else {
let content: (_, Vec<u8>) = (self.content.0.clone(), self.content.1.as_ref().into());
match content.respond_to(req) {
Ok(response) => Outcome::Success(response),
Err(status) => Outcome::Error(status),
}
}
}
}