1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
use rocket::http::{ContentType, Method};
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)
} 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::Failure(status),
}
}
}
}