use bytes::Buf;
use http::Response;
use serde::Deserialize;
use crate::raw::*;
use crate::*;
#[derive(Default, Debug, Deserialize)]
#[serde(rename_all = "camelCase")]
#[allow(dead_code)]
struct AlluxioError {
status_code: String,
message: String,
}
pub(super) fn parse_error(resp: Response<Buffer>) -> Error {
let (parts, body) = resp.into_parts();
let bs = body.to_bytes();
let mut kind = match parts.status.as_u16() {
500 => ErrorKind::Unexpected,
_ => ErrorKind::Unexpected,
};
let (message, alluxio_err) = serde_json::from_reader::<_, AlluxioError>(bs.clone().reader())
.map(|alluxio_err| (format!("{alluxio_err:?}"), Some(alluxio_err)))
.unwrap_or_else(|_| (String::from_utf8_lossy(&bs).into_owned(), None));
if let Some(alluxio_err) = alluxio_err {
kind = match alluxio_err.status_code.as_str() {
"ALREADY_EXISTS" => ErrorKind::AlreadyExists,
"NOT_FOUND" => ErrorKind::NotFound,
_ => ErrorKind::Unexpected,
}
}
let mut err = Error::new(kind, message);
err = with_error_response_context(err, parts);
err
}
#[cfg(test)]
mod tests {
use http::StatusCode;
use super::*;
#[test]
fn test_parse_error() {
let err_res = vec![
(
r#"{"statusCode":"ALREADY_EXISTS","message":"The resource you requested already exist"}"#,
ErrorKind::AlreadyExists,
),
(
r#"{"statusCode":"NOT_FOUND","message":"The resource you requested does not exist"}"#,
ErrorKind::NotFound,
),
(
r#"{"statusCode":"INTERNAL_SERVER_ERROR","message":"Internal server error"}"#,
ErrorKind::Unexpected,
),
];
for res in err_res {
let bs = bytes::Bytes::from(res.0);
let body = Buffer::from(bs);
let resp = Response::builder()
.status(StatusCode::INTERNAL_SERVER_ERROR)
.body(body)
.unwrap();
let err = parse_error(resp);
assert_eq!(err.kind(), res.1);
}
}
}