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
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
use bytes::Bytes;
use http::header::HeaderValue;
use http::{header, Request, Response};
use hyperx::header::ContentType;
use mime;
use serde::de::DeserializeOwned;
use serde::ser::Serialize;
use serde_json;
use std::ops::Deref;
use context::Context;
use error::handler::ErrorHandler;
use error::{CritError, Error, HttpError};
use input::body::FromData;
use input::RequestExt;
use output::{HttpResponse, Output, Responder, ResponseBody};
#[derive(Debug)]
pub struct Json<T>(pub T);
impl<T> Json<T> {
#[allow(missing_docs)]
pub fn into_inner(self) -> T {
self.0
}
}
impl<T> From<T> for Json<T> {
fn from(val: T) -> Self {
Json(val)
}
}
impl<T> Deref for Json<T> {
type Target = T;
fn deref(&self) -> &Self::Target {
&self.0
}
}
impl<T: DeserializeOwned> FromData for Json<T> {
fn from_data<U>(data: Bytes, request: &Request<U>) -> Result<Json<T>, Error> {
if let Some(ContentType(mime)) = request.header()? {
if mime != mime::APPLICATION_JSON {
return Err(Error::bad_request(format_err!(
"The value of Content-type is not equal to application/json"
)));
}
}
serde_json::from_slice(&*data).map_err(Error::bad_request).map(Json)
}
}
impl<T: Serialize + HttpResponse> Responder for Json<T> {
fn respond_to(self, _: &Context) -> Result<Output, Error> {
let body = serde_json::to_vec(&self.0).map_err(Error::internal_server_error)?;
let mut response = json_response(body);
*response.status_mut() = self.0.status_code();
self.0.append_headers(response.headers_mut());
Ok(response.into())
}
}
#[derive(Debug)]
pub struct JsonValue(serde_json::Value);
impl From<serde_json::Value> for JsonValue {
fn from(val: serde_json::Value) -> JsonValue {
JsonValue(val)
}
}
impl Responder for JsonValue {
fn respond_to(self, _: &Context) -> Result<Output, Error> {
Ok(json_response(self.0.to_string()).into())
}
}
#[derive(Debug, Default)]
pub struct JsonErrorHandler {
_priv: (),
}
impl JsonErrorHandler {
#[allow(missing_docs)]
pub fn new() -> JsonErrorHandler {
Default::default()
}
}
impl ErrorHandler for JsonErrorHandler {
fn handle_error(&self, e: &HttpError, _: &Request<()>) -> Result<Response<ResponseBody>, CritError> {
let body = json!({
"code": e.status_code().as_u16(),
"description": e.to_string(),
}).to_string();
Response::builder()
.status(e.status_code())
.header(header::CONNECTION, "close")
.header(header::CACHE_CONTROL, "no-cache")
.body(body.into())
.map_err(Into::into)
}
}
fn json_response<T: Into<ResponseBody>>(body: T) -> Response<ResponseBody> {
let mut response = Response::new(body.into());
response
.headers_mut()
.insert(header::CONTENT_TYPE, HeaderValue::from_static("application/json"));
response
}