use serde::{Deserialize, Serialize};
use super::WireUsage;
use meerkat_core::{ExtractionError, RunResult, SchemaWarning, SessionId, TurnTerminalCauseKind};
#[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, Eq, Hash)]
#[allow(dead_code)]
#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
#[serde(rename_all = "snake_case")]
pub enum WireToolErrorClass {
NotFound,
AccessDenied,
InvalidArguments,
Timeout,
Internal,
}
impl WireToolErrorClass {
#[allow(dead_code)]
pub const fn as_str(self) -> &'static str {
match self {
Self::NotFound => "not_found",
Self::AccessDenied => "access_denied",
Self::InvalidArguments => "invalid_arguments",
Self::Timeout => "timeout",
Self::Internal => "internal",
}
}
}
impl From<&meerkat_core::error::ToolError> for WireToolErrorClass {
fn from(value: &meerkat_core::error::ToolError) -> Self {
use meerkat_core::error::ToolError;
match value {
ToolError::NotFound { .. } => Self::NotFound,
ToolError::AccessDenied { .. } => Self::AccessDenied,
ToolError::InvalidArguments { .. } => Self::InvalidArguments,
ToolError::Timeout { .. } => Self::Timeout,
_ => Self::Internal,
}
}
}
#[derive(Debug, Clone, Serialize, Deserialize)]
#[cfg_attr(feature = "schema", derive(schemars::JsonSchema))]
pub struct WireRunResult {
pub session_id: SessionId,
#[serde(skip_serializing_if = "Option::is_none")]
pub session_ref: Option<String>,
pub text: String,
pub turns: u32,
pub tool_calls: u32,
pub usage: WireUsage,
#[serde(skip_serializing_if = "Option::is_none")]
pub terminal_cause_kind: Option<TurnTerminalCauseKind>,
#[serde(skip_serializing_if = "Option::is_none")]
pub structured_output: Option<serde_json::Value>,
#[serde(skip_serializing_if = "Option::is_none")]
pub extraction_error: Option<ExtractionError>,
#[serde(skip_serializing_if = "Option::is_none")]
pub schema_warnings: Option<Vec<SchemaWarning>>,
#[serde(skip_serializing_if = "Option::is_none")]
pub skill_diagnostics: Option<meerkat_core::skills::SkillRuntimeDiagnostics>,
}
impl From<RunResult> for WireRunResult {
fn from(r: RunResult) -> Self {
Self {
session_id: r.session_id,
session_ref: None,
text: r.text,
turns: r.turns,
tool_calls: r.tool_calls,
usage: r.usage.into(),
terminal_cause_kind: r.terminal_cause_kind,
structured_output: r.structured_output,
extraction_error: r.extraction_error,
schema_warnings: r.schema_warnings,
skill_diagnostics: r.skill_diagnostics,
}
}
}
#[cfg(test)]
mod tests {
use super::*;
use meerkat_core::skills::{SkillRuntimeDiagnostics, SourceHealthSnapshot, SourceHealthState};
#[test]
fn wire_run_result_preserves_unhealthy_skill_diagnostics()
-> Result<(), Box<dyn std::error::Error>> {
let run = RunResult {
text: "ok".to_string(),
session_id: SessionId::new(),
usage: Default::default(),
turns: 1,
tool_calls: 0,
terminal_cause_kind: None,
structured_output: None,
extraction_error: None,
schema_warnings: None,
skill_diagnostics: Some(SkillRuntimeDiagnostics {
source_health: SourceHealthSnapshot {
state: SourceHealthState::Unhealthy,
invalid_ratio: 0.8,
invalid_count: 8,
total_count: 10,
failure_streak: 10,
handshake_failed: true,
},
quarantined: vec![],
}),
};
let wire: WireRunResult = run.into();
let state = wire
.skill_diagnostics
.as_ref()
.map(|value| value.source_health.state);
assert_eq!(state, Some(SourceHealthState::Unhealthy));
let json = serde_json::to_value(&wire)?;
assert_eq!(
json["skill_diagnostics"]["source_health"]["state"],
"unhealthy"
);
Ok(())
}
}