include!(concat!(env!("OUT_DIR"), "/codegen.rs"));
use alien_error::{AlienError, GenericError, HumanLayerPresentation};
pub trait SdkResultExt<T> {
fn into_sdk_error(self) -> Result<T, AlienError<GenericError>>;
}
impl<T> SdkResultExt<ResponseValue<T>> for Result<ResponseValue<T>, Error<types::ApiError>> {
fn into_sdk_error(self) -> Result<ResponseValue<T>, AlienError<GenericError>> {
self.map_err(convert_sdk_error)
}
}
pub fn convert_sdk_error(err: Error<types::ApiError>) -> AlienError<GenericError> {
match err {
Error::ErrorResponse(response) => {
let status = response.status().as_u16();
let api_error = response.into_inner();
AlienError {
code: api_error.code.to_string(),
message: api_error.message.to_string(),
context: api_error.context,
hint: None,
retryable: api_error.retryable,
internal: false, http_status_code: Some(status),
source: api_error.source.and_then(parse_source_error),
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError {
message: api_error.message.to_string(),
}),
}
}
Error::CommunicationError(reqwest_err) => {
let retryable =
reqwest_err.is_connect() || reqwest_err.is_timeout() || reqwest_err.is_request();
AlienError {
code: "COMMUNICATION_ERROR".to_string(),
message: format!("Communication Error: {}", reqwest_err),
context: None,
hint: None,
retryable,
internal: false,
http_status_code: reqwest_err.status().map(|s| s.as_u16()),
source: build_reqwest_source(&reqwest_err),
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError {
message: format!("Communication Error: {}", reqwest_err),
}),
}
}
Error::InvalidRequest(msg) => AlienError {
code: "INVALID_REQUEST".to_string(),
message: format!("Invalid Request: {}", msg),
context: None,
hint: None,
retryable: false,
internal: false,
http_status_code: Some(400),
source: None,
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError {
message: format!("Invalid Request: {}", msg),
}),
},
Error::ResponseBodyError(reqwest_err) => AlienError {
code: "RESPONSE_BODY_ERROR".to_string(),
message: format!("Error reading response body: {}", reqwest_err),
context: None,
hint: None,
retryable: true, internal: false,
http_status_code: reqwest_err.status().map(|s| s.as_u16()),
source: build_reqwest_source(&reqwest_err),
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError {
message: format!("Error reading response body: {}", reqwest_err),
}),
},
Error::InvalidResponsePayload(bytes, json_err) => {
let raw_body = String::from_utf8_lossy(&bytes);
let truncated = if raw_body.len() > 1000 {
format!(
"{}...(truncated {} bytes)",
&raw_body[..1000],
raw_body.len() - 1000
)
} else {
raw_body.to_string()
};
AlienError {
code: "INVALID_RESPONSE_PAYLOAD".to_string(),
message: format!("Failed to parse response: {}", json_err),
context: Some(serde_json::json!({
"parseError": json_err.to_string(),
"responseBody": truncated,
})),
hint: None,
retryable: false,
internal: false,
http_status_code: None,
source: Some(Box::new(AlienError::new(GenericError {
message: json_err.to_string(),
}))),
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError {
message: format!("Failed to parse response: {}", json_err),
}),
}
}
Error::InvalidUpgrade(reqwest_err) => AlienError {
code: "INVALID_UPGRADE".to_string(),
message: format!("Connection upgrade failed: {}", reqwest_err),
context: None,
hint: None,
retryable: false,
internal: false,
http_status_code: reqwest_err.status().map(|s| s.as_u16()),
source: build_reqwest_source(&reqwest_err),
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError {
message: format!("Connection upgrade failed: {}", reqwest_err),
}),
},
Error::UnexpectedResponse(response) => {
let status = response.status().as_u16();
AlienError {
code: "UNEXPECTED_RESPONSE".to_string(),
message: format!(
"Unexpected response: {} {}",
status,
response.status().canonical_reason().unwrap_or("Unknown")
),
context: Some(serde_json::json!({
"status": status,
"url": response.url().to_string(),
})),
hint: None,
retryable: status >= 500, internal: false,
http_status_code: Some(status),
source: None,
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError {
message: format!("Unexpected response status: {}", status),
}),
}
}
Error::Custom(msg) => AlienError {
code: "SDK_HOOK_ERROR".to_string(),
message: msg.clone(),
context: None,
hint: None,
retryable: false,
internal: false,
http_status_code: None,
source: None,
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError { message: msg }),
},
}
}
fn build_reqwest_source(err: &reqwest::Error) -> Option<Box<AlienError<GenericError>>> {
use std::error::Error;
let mut sources = Vec::new();
let mut current: Option<&(dyn Error + 'static)> = err.source();
while let Some(src) = current {
sources.push(src.to_string());
current = src.source();
}
if sources.is_empty() {
return None;
}
let mut result: Option<Box<AlienError<GenericError>>> = None;
for msg in sources.into_iter().rev() {
let error = AlienError {
code: "GENERIC_ERROR".to_string(),
message: msg.clone(),
context: None,
hint: None,
retryable: false,
internal: false,
http_status_code: None,
source: result,
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError { message: msg }),
};
result = Some(Box::new(error));
}
result
}
fn parse_source_error(value: serde_json::Value) -> Option<Box<AlienError<GenericError>>> {
let obj = value.as_object()?;
let code = obj
.get("code")
.and_then(|v| v.as_str())
.unwrap_or("NESTED_ERROR")
.to_string();
let message = obj
.get("message")
.and_then(|v| v.as_str())
.unwrap_or("Nested error")
.to_string();
let context = obj.get("context").cloned();
let retryable = obj
.get("retryable")
.and_then(|v| v.as_bool())
.unwrap_or(false);
let nested_source = obj.get("source").cloned().and_then(parse_source_error);
Some(Box::new(AlienError {
code,
message: message.clone(),
context,
hint: None,
retryable,
internal: false,
http_status_code: None,
source: nested_source,
human_layer_presentation: HumanLayerPresentation::Normal,
error: Some(GenericError { message }),
}))
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_api_error_code_deref() {
let code = types::ApiErrorCode::try_from("TEST_ERROR").unwrap();
assert_eq!(code.as_str(), "TEST_ERROR");
}
}