use http::Response;
use http::StatusCode;
use serde::Deserialize;
use crate::raw::*;
use crate::*;
#[derive(Default, Debug, Deserialize)]
#[serde(default)]
pub struct DropboxErrorResponse {
pub error_summary: String,
}
pub(super) fn parse_error(resp: Response<Buffer>) -> Error {
let (parts, body) = resp.into_parts();
let bs = body.to_bytes();
let (mut kind, mut retryable) = match parts.status {
StatusCode::NOT_FOUND => (ErrorKind::NotFound, false),
StatusCode::FORBIDDEN => (ErrorKind::PermissionDenied, false),
StatusCode::TOO_MANY_REQUESTS => (ErrorKind::RateLimited, true),
StatusCode::INTERNAL_SERVER_ERROR
| StatusCode::BAD_GATEWAY
| StatusCode::SERVICE_UNAVAILABLE
| StatusCode::GATEWAY_TIMEOUT => (ErrorKind::Unexpected, true),
_ => (ErrorKind::Unexpected, false),
};
let (message, dropbox_err) = serde_json::from_slice::<DropboxErrorResponse>(&bs)
.map(|dropbox_err| (format!("{dropbox_err:?}"), Some(dropbox_err)))
.unwrap_or_else(|_| (String::from_utf8_lossy(&bs).into_owned(), None));
if let Some(dropbox_err) = dropbox_err {
(kind, retryable) =
parse_dropbox_error_summary(&dropbox_err.error_summary).unwrap_or((kind, retryable));
}
let mut err = Error::new(kind, message);
err = with_error_response_context(err, parts);
if retryable {
err = err.set_temporary();
}
err
}
pub fn parse_dropbox_error_summary(summary: &str) -> Option<(ErrorKind, bool)> {
if summary.starts_with("path/not_found")
|| summary.starts_with("path_lookup/not_found")
|| summary.starts_with("from_lookup/not_found")
{
Some((ErrorKind::NotFound, false))
} else if summary.starts_with("path/conflict") {
Some((ErrorKind::AlreadyExists, false))
} else if summary.starts_with("too_many_write_operations") {
Some((ErrorKind::RateLimited, true))
} else {
None
}
}