use bytes::Buf;
use bytes::Bytes;
use http::Response;
use http::StatusCode;
use quick_xml::de;
use serde::Deserialize;
use crate::raw::*;
use crate::*;
#[allow(dead_code)]
#[derive(Debug, Deserialize)]
struct ErrorResponse {
h1: String,
p: String,
}
pub(super) fn parse_error(resp: Response<Buffer>) -> Error {
let (parts, body) = resp.into_parts();
let bs = body.to_bytes();
let (kind, retryable) = match parts.status {
StatusCode::NOT_FOUND => (ErrorKind::NotFound, false),
StatusCode::UNAUTHORIZED | StatusCode::FORBIDDEN => (ErrorKind::PermissionDenied, false),
StatusCode::PRECONDITION_FAILED => (ErrorKind::ConditionNotMatch, false),
StatusCode::INTERNAL_SERVER_ERROR
| StatusCode::BAD_GATEWAY
| StatusCode::SERVICE_UNAVAILABLE
| StatusCode::GATEWAY_TIMEOUT => (ErrorKind::Unexpected, true),
_ => (ErrorKind::Unexpected, false),
};
let message = parse_error_response(&bs);
let mut err = Error::new(kind, message);
err = with_error_response_context(err, parts);
if retryable {
err = err.set_temporary();
}
err
}
fn parse_error_response(resp: &Bytes) -> String {
match de::from_reader::<_, ErrorResponse>(resp.clone().reader()) {
Ok(swift_err) => swift_err.p,
Err(_) => String::from_utf8_lossy(resp).into_owned(),
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn parse_error_response_test() -> Result<()> {
let resp = Bytes::from(
r#"
<html>
<h1>Not Found</h1>
<p>The resource could not be found.</p>
</html>
"#,
);
let msg = parse_error_response(&resp);
assert_eq!(msg, "The resource could not be found.".to_string(),);
Ok(())
}
}