//! Lightweight verbose logging helpers for the CLI. use std::sync::atomic::{AtomicBool, Ordering}; use colored::Colorize; use crate::palette; static VERBOSE: AtomicBool = AtomicBool::new(false); /// Enable or disable verbose logging output. pub fn set_verbose(enabled: bool) { VERBOSE.store(enabled, Ordering::SeqCst); } /// Return true when supported env logging knobs request verbose output. #[must_use] pub fn env_requests_verbose_logging() -> bool { std::env::var("DEEPSEEK_LOG_LEVEL") .ok() .is_some_and(|value| log_value_enables_verbose(&value)) || std::env::var("RUST_LOG") .ok() .is_some_and(|value| log_value_enables_verbose(&value)) } fn log_value_enables_verbose(value: &str) -> bool { value.split(',').any(|directive| { let level = directive .rsplit('=') .next() .unwrap_or(directive) .trim() .to_ascii_lowercase(); matches!(level.as_str(), "trace" | "debug" | "info") }) } /// Check whether verbose logging is enabled. #[must_use] pub fn is_verbose() -> bool { VERBOSE.load(Ordering::SeqCst) } /// Emit a verbose info message (no-op when verbosity is disabled). pub fn info(message: impl AsRef) { if is_verbose() { let (r, g, b) = palette::DEEPSEEK_SKY_RGB; eprintln!("{} {}", "info".truecolor(r, g, b).bold(), message.as_ref()); } } /// Emit a verbose warning message (no-op when verbosity is disabled). pub fn warn(message: impl AsRef) { if is_verbose() { let (r, g, b) = palette::DEEPSEEK_SKY_RGB; eprintln!("{} {}", "warn".truecolor(r, g, b).bold(), message.as_ref()); } } #[cfg(test)] mod tests { use super::*; #[test] fn log_value_parser_accepts_common_rust_log_directives() { assert!(log_value_enables_verbose("debug")); assert!(log_value_enables_verbose("deepseek_cli=debug")); assert!(log_value_enables_verbose("warn,deepseek_tui::client=trace")); assert!(!log_value_enables_verbose("warn")); assert!(!log_value_enables_verbose("deepseek_tui=off")); } }