use std::fmt;
use std::sync::Arc;
use serde::Deserialize;
use camel_api::metrics::MetricsCollector;
#[derive(Clone, Deserialize, Default)]
pub struct TracerConfig {
#[serde(default)]
pub enabled: bool,
#[serde(default = "default_detail_level")]
pub detail_level: DetailLevel,
#[serde(default)]
pub outputs: TracerOutputs,
#[serde(skip)]
pub metrics_collector: Option<Arc<dyn MetricsCollector>>,
}
impl fmt::Debug for TracerConfig {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
f.debug_struct("TracerConfig")
.field("enabled", &self.enabled)
.field("detail_level", &self.detail_level)
.field("outputs", &self.outputs)
.field(
"metrics_collector",
&self.metrics_collector.as_ref().map(|_| "MetricsCollector"),
)
.finish()
}
}
#[derive(Debug, Clone, Deserialize, Default)]
pub struct TracerOutputs {
#[serde(default)]
pub stdout: StdoutOutput,
#[serde(default)]
pub file: Option<FileOutput>,
}
#[derive(Debug, Clone, Deserialize)]
pub struct StdoutOutput {
#[serde(default = "default_true")]
pub enabled: bool,
#[serde(default = "default_format")]
pub format: OutputFormat,
}
impl Default for StdoutOutput {
fn default() -> Self {
Self {
enabled: true,
format: OutputFormat::Json,
}
}
}
#[derive(Debug, Clone, Deserialize)]
pub struct FileOutput {
pub enabled: bool,
pub path: String,
#[serde(default = "default_format")]
pub format: OutputFormat,
}
#[derive(Debug, Clone, Deserialize, Default, PartialEq, Eq, PartialOrd, Ord)]
#[serde(rename_all = "lowercase")]
pub enum DetailLevel {
#[default]
Minimal,
Medium,
Full,
}
#[derive(Debug, Clone, Deserialize, Default)]
#[serde(rename_all = "lowercase")]
pub enum OutputFormat {
#[default]
Json,
Plain,
}
fn default_detail_level() -> DetailLevel {
DetailLevel::Minimal
}
fn default_format() -> OutputFormat {
OutputFormat::Json
}
fn default_true() -> bool {
true
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn tracer_config_defaults_are_stable() {
let cfg = TracerConfig::default();
assert!(!cfg.enabled);
assert_eq!(cfg.detail_level, DetailLevel::Minimal);
assert!(cfg.outputs.stdout.enabled);
assert!(matches!(cfg.outputs.stdout.format, OutputFormat::Json));
assert!(cfg.outputs.file.is_none());
assert!(cfg.metrics_collector.is_none());
}
#[test]
fn tracer_config_deserializes_lowercase_enums() {
let cfg: TracerConfig = serde_json::from_str(
r#"{
"enabled": true,
"detail_level": "full",
"outputs": {
"stdout": { "enabled": false, "format": "plain" },
"file": { "enabled": true, "path": "/tmp/trace.log", "format": "json" }
}
}"#,
)
.unwrap();
assert!(cfg.enabled);
assert_eq!(cfg.detail_level, DetailLevel::Full);
assert!(!cfg.outputs.stdout.enabled);
assert!(matches!(cfg.outputs.stdout.format, OutputFormat::Plain));
assert_eq!(cfg.outputs.file.as_ref().unwrap().path, "/tmp/trace.log");
}
#[test]
fn tracer_config_debug_redacts_metrics_collector() {
let dbg = format!("{:?}", TracerConfig::default());
assert!(dbg.contains("TracerConfig"));
assert!(dbg.contains("metrics_collector"));
}
}