Line data Source code
1 : use anyhow::Context;
2 : use bytes::Buf;
3 : use hyper::{header, Body, Request, Response, StatusCode};
4 : use serde::{Deserialize, Serialize};
5 :
6 : use super::error::ApiError;
7 :
8 5557 : pub async fn json_request<T: for<'de> Deserialize<'de>>(
9 5557 : request: &mut Request<Body>,
10 5557 : ) -> Result<T, ApiError> {
11 5557 : json_request_or_empty_body(request)
12 104 : .await?
13 5557 : .context("missing request body")
14 5557 : .map_err(ApiError::BadRequest)
15 5557 : }
16 :
17 : /// Will be removed as part of <https://github.com/neondatabase/neon/issues/4282>
18 5602 : pub async fn json_request_or_empty_body<T: for<'de> Deserialize<'de>>(
19 5602 : request: &mut Request<Body>,
20 5602 : ) -> Result<Option<T>, ApiError> {
21 5602 : let body = hyper::body::aggregate(request.body_mut())
22 107 : .await
23 5602 : .context("Failed to read request body")
24 5602 : .map_err(ApiError::BadRequest)?;
25 5602 : if body.remaining() == 0 {
26 0 : return Ok(None);
27 5602 : }
28 5602 :
29 5602 : let mut deser = serde_json::de::Deserializer::from_reader(body.reader());
30 5602 :
31 5602 : serde_path_to_error::deserialize(&mut deser)
32 5602 : // intentionally stringify because the debug version is not helpful in python logs
33 5602 : .map_err(|e| anyhow::anyhow!("Failed to parse json request: {e}"))
34 5602 : .map(Some)
35 5602 : .map_err(ApiError::BadRequest)
36 5602 : }
37 :
38 15104 : pub fn json_response<T: Serialize>(
39 15104 : status: StatusCode,
40 15104 : data: T,
41 15104 : ) -> Result<Response<Body>, ApiError> {
42 15104 : let json = serde_json::to_string(&data)
43 15104 : .context("Failed to serialize JSON response")
44 15104 : .map_err(ApiError::InternalServerError)?;
45 15104 : let response = Response::builder()
46 15104 : .status(status)
47 15104 : .header(header::CONTENT_TYPE, "application/json")
48 15104 : .body(Body::from(json))
49 15104 : .map_err(|e| ApiError::InternalServerError(e.into()))?;
50 15104 : Ok(response)
51 15104 : }
|