use serde::{Deserialize, Serialize};
use std::fmt;
#[derive(Debug, Clone, Serialize, Deserialize)]
pub struct UserFacingError {
pub summary: String,
pub message: String,
pub suggestion: String,
pub category: ErrorCategory,
pub recoverable: bool,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Serialize, Deserialize)]
pub enum ErrorCategory {
Connection,
Auth,
Config,
NotFound,
Temporary,
Internal,
}
#[derive(Debug)]
pub enum ModelError {
Backend(BackendError),
Config(ConfigError),
ModelNotFound {
model: String,
searched: Vec<String>,
},
Timeout {
operation: String,
duration_secs: u64,
},
RateLimit { retry_after: Option<u64> },
InvalidRequest(String),
ParseError {
message: String,
raw: Option<String>,
},
StreamError(String),
Authentication(String),
Unsupported { feature: String },
Cancelled,
}
impl fmt::Display for ModelError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
ModelError::Backend(e) => write!(f, "Backend error: {}", e),
ModelError::Config(e) => write!(f, "Configuration error: {}", e),
ModelError::ModelNotFound { model, searched } => {
write!(
f,
"Model '{}' not found. Searched: {}",
model,
searched.join(", ")
)
},
ModelError::Timeout {
operation,
duration_secs,
} => {
if *duration_secs == 0 {
write!(f, "Operation '{}' timed out", operation)
} else {
write!(
f,
"Operation '{}' timed out after {} seconds",
operation, duration_secs
)
}
},
ModelError::RateLimit { retry_after } => {
if let Some(secs) = retry_after {
write!(f, "Rate limit exceeded. Retry after {} seconds", secs)
} else {
write!(f, "Rate limit exceeded")
}
},
ModelError::InvalidRequest(msg) => write!(f, "Invalid request: {}", msg),
ModelError::ParseError { message, raw } => {
if let Some(r) = raw {
write!(f, "Parse error: {} (raw: {})", message, r)
} else {
write!(f, "Parse error: {}", message)
}
},
ModelError::StreamError(msg) => write!(f, "Stream error: {}", msg),
ModelError::Authentication(msg) => write!(f, "Authentication error: {}", msg),
ModelError::Unsupported { feature } => {
write!(f, "Feature not supported by this adapter: {}", feature)
},
ModelError::Cancelled => write!(f, "Cancelled by user"),
}
}
}
impl std::error::Error for ModelError {}
impl ModelError {
pub fn to_user_facing(&self) -> UserFacingError {
match self {
ModelError::Backend(BackendError::ConnectionFailed { backend, url, .. }) => {
UserFacingError {
summary: format!("{} connection failed", backend),
message: format!("Could not connect to {} at {}", backend, url),
suggestion: if backend == "ollama" {
"Run 'ollama serve' to start Ollama, or check if it's running on the correct port".to_string()
} else {
format!("Check if {} is running and accessible", backend)
},
category: ErrorCategory::Connection,
recoverable: true,
}
},
ModelError::Backend(BackendError::NotAvailable { backend, reason }) => {
UserFacingError {
summary: format!("{} unavailable", backend),
message: format!("{} is not available: {}", backend, reason),
suggestion: if backend == "ollama" {
"Start Ollama with 'ollama serve' or pull the model with 'ollama pull <model>'".to_string()
} else {
format!("Ensure {} service is running and healthy", backend)
},
category: ErrorCategory::Connection,
recoverable: true,
}
},
ModelError::Backend(BackendError::HttpError { status, message }) => {
let (summary, suggestion) = match status {
401 | 403 => (
"Authentication failed",
"Check your API key in ~/.config/mermaid/config.toml",
),
404 => (
"Model not found",
"Use /model <name> to switch models (auto-pulls if needed), or pull manually with 'ollama pull <name>'",
),
429 => (
"Rate limited",
"Wait a moment before retrying, or switch to a local model",
),
500..=599 => (
"Server error",
"The backend service is experiencing issues - try again later",
),
_ => (
"Request failed",
"Check your network connection and backend configuration",
),
};
let rendered = match try_extract_error_message(message) {
Some(clean) => format!("HTTP {}: {}", status, clean),
None => format!("HTTP {}: {}", status, message),
};
UserFacingError {
summary: summary.to_string(),
message: rendered,
suggestion: suggestion.to_string(),
category: if *status == 401 || *status == 403 {
ErrorCategory::Auth
} else if *status == 429 || (500..=599).contains(status) {
ErrorCategory::Temporary
} else {
ErrorCategory::Internal
},
recoverable: *status == 429 || *status >= 500,
}
},
ModelError::Backend(BackendError::UnexpectedResponse { backend, message }) => {
UserFacingError {
summary: "Unexpected response".to_string(),
message: format!("Received unexpected response from {}: {}", backend, message),
suggestion: "This might be a version mismatch - try updating the backend"
.to_string(),
category: ErrorCategory::Internal,
recoverable: false,
}
},
ModelError::Backend(BackendError::ProviderError {
provider,
code,
message,
}) => {
let code_str = code.as_deref().unwrap_or("unknown");
UserFacingError {
summary: format!("{} error", provider),
message: format!("{} returned error {}: {}", provider, code_str, message),
suggestion: format!(
"Check {} documentation for error code {}",
provider, code_str
),
category: ErrorCategory::Internal,
recoverable: false,
}
},
ModelError::Config(ConfigError::MissingRequired(field)) => UserFacingError {
summary: "Missing configuration".to_string(),
message: format!("Required configuration '{}' is missing", field),
suggestion: format!("Add '{}' to ~/.config/mermaid/config.toml", field),
category: ErrorCategory::Config,
recoverable: false,
},
ModelError::Config(ConfigError::InvalidValue {
field,
value,
reason,
}) => UserFacingError {
summary: "Invalid configuration".to_string(),
message: format!("Invalid value '{}' for '{}': {}", value, field, reason),
suggestion: format!("Fix '{}' in ~/.config/mermaid/config.toml", field),
category: ErrorCategory::Config,
recoverable: false,
},
ModelError::Config(ConfigError::FileError { path, reason }) => UserFacingError {
summary: "Config file error".to_string(),
message: format!("Cannot read config file '{}': {}", path, reason),
suggestion: "Check file permissions and syntax".to_string(),
category: ErrorCategory::Config,
recoverable: false,
},
ModelError::ModelNotFound { model, searched } => UserFacingError {
summary: "Model not found".to_string(),
message: format!("Model '{}' not found in: {}", model, searched.join(", ")),
suggestion: format!(
"Pull the model with 'ollama pull {}' or check if the model name is correct",
model
),
category: ErrorCategory::NotFound,
recoverable: false,
},
ModelError::Timeout {
operation,
duration_secs,
} => UserFacingError {
summary: "Request timed out".to_string(),
message: if *duration_secs == 0 {
format!("'{}' timed out", operation)
} else {
format!("'{}' timed out after {} seconds", operation, duration_secs)
},
suggestion: "The model might be overloaded - try a smaller model or wait and retry"
.to_string(),
category: ErrorCategory::Temporary,
recoverable: true,
},
ModelError::RateLimit { retry_after } => {
let wait_msg = retry_after
.map(|s| format!("Wait {} seconds", s))
.unwrap_or_else(|| "Wait a moment".to_string());
UserFacingError {
summary: "Rate limited".to_string(),
message: "Too many requests - rate limit exceeded".to_string(),
suggestion: format!(
"{}. Consider using a local Ollama model to avoid rate limits",
wait_msg
),
category: ErrorCategory::Temporary,
recoverable: true,
}
},
ModelError::InvalidRequest(msg) => UserFacingError {
summary: "Invalid request".to_string(),
message: format!("The request was invalid: {}", msg),
suggestion: "Check your message format or try rephrasing".to_string(),
category: ErrorCategory::Internal,
recoverable: false,
},
ModelError::ParseError { message, .. } => UserFacingError {
summary: "Parse error".to_string(),
message: format!("Failed to parse response: {}", message),
suggestion:
"The model returned an unexpected format - try sending the message again"
.to_string(),
category: ErrorCategory::Internal,
recoverable: true,
},
ModelError::StreamError(msg) => UserFacingError {
summary: "Stream interrupted".to_string(),
message: format!("Connection lost during streaming: {}", msg),
suggestion: "Check your network connection and try again".to_string(),
category: ErrorCategory::Connection,
recoverable: true,
},
ModelError::Authentication(msg) => UserFacingError {
summary: "Authentication failed".to_string(),
message: format!("Authentication error: {}", msg),
suggestion:
"Check your API key in ~/.config/mermaid/config.toml or environment variables"
.to_string(),
category: ErrorCategory::Auth,
recoverable: false,
},
ModelError::Unsupported { feature } => UserFacingError {
summary: "Unsupported feature".to_string(),
message: format!("The current model adapter does not support '{}'.", feature),
suggestion: format!(
"Switch to a provider/model that supports '{}', or omit this operation.",
feature
),
category: ErrorCategory::Internal,
recoverable: false,
},
ModelError::Cancelled => UserFacingError {
summary: "Cancelled".to_string(),
message: "The request was cancelled.".to_string(),
suggestion: String::new(),
category: ErrorCategory::Temporary,
recoverable: true,
},
}
}
}
#[derive(Debug)]
pub enum BackendError {
ConnectionFailed {
backend: String,
url: String,
reason: String,
},
NotAvailable { backend: String, reason: String },
HttpError { status: u16, message: String },
UnexpectedResponse { backend: String, message: String },
ProviderError {
provider: String,
code: Option<String>,
message: String,
},
}
impl fmt::Display for BackendError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
BackendError::ConnectionFailed {
backend,
url,
reason,
} => {
write!(f, "Failed to connect to {} at {}: {}", backend, url, reason)
},
BackendError::NotAvailable { backend, reason } => {
write!(f, "Backend '{}' not available: {}", backend, reason)
},
BackendError::HttpError { status, message } => {
write!(f, "HTTP error {}: {}", status, message)
},
BackendError::UnexpectedResponse { backend, message } => {
write!(f, "Unexpected response from {}: {}", backend, message)
},
BackendError::ProviderError {
provider,
code,
message,
} => {
if let Some(c) = code {
write!(f, "{} error {}: {}", provider, c, message)
} else {
write!(f, "{} error: {}", provider, message)
}
},
}
}
}
impl std::error::Error for BackendError {}
#[derive(Debug)]
pub enum ConfigError {
MissingRequired(String),
InvalidValue {
field: String,
value: String,
reason: String,
},
FileError { path: String, reason: String },
}
impl fmt::Display for ConfigError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
ConfigError::MissingRequired(field) => {
write!(f, "Missing required configuration: {}", field)
},
ConfigError::InvalidValue {
field,
value,
reason,
} => {
write!(f, "Invalid value for '{}': '{}' ({})", field, value, reason)
},
ConfigError::FileError { path, reason } => {
write!(f, "Error reading config file '{}': {}", path, reason)
},
}
}
}
impl std::error::Error for ConfigError {}
pub type Result<T> = std::result::Result<T, ModelError>;
impl From<anyhow::Error> for ModelError {
fn from(err: anyhow::Error) -> Self {
ModelError::InvalidRequest(err.to_string())
}
}
impl From<reqwest::Error> for ModelError {
fn from(err: reqwest::Error) -> Self {
if err.is_timeout() {
ModelError::Timeout {
operation: "HTTP request".to_string(),
duration_secs: 0,
}
} else if err.is_connect() {
ModelError::Backend(BackendError::ConnectionFailed {
backend: "unknown".to_string(),
url: err
.url()
.map(|u| u.to_string())
.unwrap_or_else(|| "unknown".to_string()),
reason: err.to_string(),
})
} else if err.is_status() {
let status = err.status().map(|s| s.as_u16()).unwrap_or(500);
ModelError::Backend(BackendError::HttpError {
status,
message: err.to_string(),
})
} else {
ModelError::Backend(BackendError::UnexpectedResponse {
backend: "unknown".to_string(),
message: err.to_string(),
})
}
}
}
impl From<serde_json::Error> for ModelError {
fn from(err: serde_json::Error) -> Self {
ModelError::ParseError {
message: err.to_string(),
raw: None,
}
}
}
fn try_extract_error_message(body: &str) -> Option<String> {
let trimmed = body.trim();
if !trimmed.starts_with('{') {
return None;
}
let value: serde_json::Value = serde_json::from_str(trimmed).ok()?;
let error = value.get("error")?;
if let Some(s) = error.as_str() {
return Some(s.trim().to_string());
}
if let Some(obj) = error.as_object() {
let message = obj.get("message").and_then(|v| v.as_str())?;
let kind = obj
.get("type")
.and_then(|v| v.as_str())
.or_else(|| obj.get("code").and_then(|v| v.as_str()));
let out = match kind {
Some(k) if !k.is_empty() => format!("{}: {}", k, message),
_ => message.to_string(),
};
return Some(out.trim().to_string());
}
None
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn timeout_display_omits_zero_duration() {
let err = ModelError::Timeout {
operation: "HTTP request".to_string(),
duration_secs: 0,
};
let rendered = err.to_string();
assert_eq!(rendered, "Operation 'HTTP request' timed out");
assert!(!rendered.contains("0 seconds"));
}
#[test]
fn timeout_display_shows_nonzero_duration() {
let err = ModelError::Timeout {
operation: "HTTP request".to_string(),
duration_secs: 45,
};
let rendered = err.to_string();
assert_eq!(
rendered,
"Operation 'HTTP request' timed out after 45 seconds"
);
}
#[test]
fn timeout_user_facing_omits_zero_duration() {
let err = ModelError::Timeout {
operation: "HTTP request".to_string(),
duration_secs: 0,
};
let ufe = err.to_user_facing();
assert_eq!(ufe.message, "'HTTP request' timed out");
assert!(!ufe.message.contains("0 seconds"));
}
#[test]
fn extract_error_handles_ollama_string_shape() {
let body = r#"{"error":"Internal Server Error (ref: 6e8ae4c7)"}"#;
assert_eq!(
try_extract_error_message(body).as_deref(),
Some("Internal Server Error (ref: 6e8ae4c7)")
);
}
#[test]
fn extract_error_handles_openai_object_shape_with_type() {
let body = r#"{"error":{"message":"Rate limit","type":"rate_limit_error","code":null}}"#;
assert_eq!(
try_extract_error_message(body).as_deref(),
Some("rate_limit_error: Rate limit")
);
}
#[test]
fn extract_error_handles_openrouter_numeric_code() {
let body = r#"{"error":{"message":"upstream timeout","code":504,"metadata":{}}}"#;
assert_eq!(
try_extract_error_message(body).as_deref(),
Some("upstream timeout")
);
}
#[test]
fn extract_error_returns_none_for_non_json() {
assert_eq!(try_extract_error_message("<html>bad gateway</html>"), None);
assert_eq!(try_extract_error_message(""), None);
assert_eq!(try_extract_error_message("plain text error"), None);
}
#[test]
fn extract_error_returns_none_for_missing_error_field() {
let body = r#"{"status":"ok","message":"nothing here"}"#;
assert_eq!(try_extract_error_message(body), None);
}
#[test]
fn http_500_renders_clean_message_and_temporary_category() {
let err = ModelError::Backend(BackendError::HttpError {
status: 500,
message: r#"{"error":"Internal Server Error (ref: abc-123)"}"#.to_string(),
});
let ufe = err.to_user_facing();
assert_eq!(ufe.summary, "Server error");
assert_eq!(
ufe.message,
"HTTP 500: Internal Server Error (ref: abc-123)"
);
assert!(ufe.recoverable);
assert_eq!(ufe.category, ErrorCategory::Temporary);
}
#[test]
fn http_500_falls_back_to_raw_body_for_html() {
let err = ModelError::Backend(BackendError::HttpError {
status: 502,
message: "<html>Bad Gateway</html>".to_string(),
});
let ufe = err.to_user_facing();
assert_eq!(ufe.message, "HTTP 502: <html>Bad Gateway</html>");
}
}