teaql_runtime/
log_formatter.rs1use crate::event::RawAuditEvent;
2use teaql_core::TraceNode;
3
4pub use crate::context::SqlLogEntry;
6
7pub trait LogFormatter: Send + Sync {
9 fn format_sql_log(&self, trace_chain: &[TraceNode], entry: &SqlLogEntry) -> String;
11
12 fn format_audit_log(&self, event: &RawAuditEvent) -> String;
14}
15
16pub struct HumanReaderFormatter;
19
20impl HumanReaderFormatter {
21 fn format_trace_chain(&self, trace_chain: &[TraceNode]) -> String {
22 if trace_chain.is_empty() {
23 "".to_string()
24 } else {
25 trace_chain.iter().map(|n| n.comment.clone()).collect::<Vec<_>>().join(" -> ")
26 }
27 }
28}
29
30impl LogFormatter for HumanReaderFormatter {
31 fn format_sql_log(&self, trace_chain: &[TraceNode], entry: &SqlLogEntry) -> String {
32 let ts = chrono::Utc::now().format("%Y-%m-%d %H:%M:%S%.3f");
33 let trace_str = self.format_trace_chain(trace_chain);
34 let trace_display = if trace_str.is_empty() {
35 "".to_string()
36 } else {
37 format!(" - [{}]", trace_str)
38 };
39
40 let elapsed_us = (entry.elapsed.as_secs_f64() * 1_000_000.0).round() as u64;
41 format!("[{}]-[{:>5}µs]-[DEBUG]-SqlLogEntry{} - [{}]\n {}",
42 ts, elapsed_us, trace_display, entry.result_summary, entry.pretty_sql.replace('\n', " "))
43 }
44
45 fn format_audit_log(&self, event: &RawAuditEvent) -> String {
46 let ts = chrono::Utc::now().format("%Y-%m-%d %H:%M:%S%.3f");
47 let trace_str = self.format_trace_chain(&event.trace_chain);
48 let trace_display = if trace_str.is_empty() {
49 String::new()
50 } else {
51 format!(" (Trace: {})", trace_str)
52 };
53
54 let mut field_changes = Vec::new();
55 for change in &event.changes {
56 if change.field.starts_with('_') {
57 continue;
58 }
59 let val = change.new_value.as_ref().map(|v| format!("{:?}", v)).unwrap_or_else(|| "null".to_string());
60 field_changes.push(format!("{}: {}", change.field, val));
61 }
62 let fields_part = if field_changes.is_empty() {
63 String::new()
64 } else {
65 format!(" {{{}}}", field_changes.join(", "))
66 };
67
68 let mut entity_id = "Unknown".to_string();
69 if let Some(vals) = &event.new_values {
70 if let Some(id_val) = vals.get("id") {
71 entity_id = format!("{:?}", id_val);
72 }
73 }
74
75 format!("[{}]-[AUDIT]-Entity [{}:{}] {:?}{}{}", ts, event.entity, entity_id, event.kind, trace_display, fields_part)
76 }
77}
78
79pub struct DebugReaderFormatter;
81
82impl DebugReaderFormatter {
83 fn format_trace_chain(&self, trace_chain: &[TraceNode]) -> String {
84 if trace_chain.is_empty() {
85 "(Trace: None)".to_string()
86 } else {
87 format!("(Trace: {})", trace_chain.iter().map(|n| n.comment.clone()).collect::<Vec<_>>().join(" -> "))
88 }
89 }
90}
91
92impl LogFormatter for DebugReaderFormatter {
93 fn format_sql_log(&self, trace_chain: &[TraceNode], entry: &SqlLogEntry) -> String {
94 let trace_str = self.format_trace_chain(trace_chain);
95 format!("[SQL_LOG] {} - Event: {:?}", trace_str, entry)
96 }
97
98 fn format_audit_log(&self, event: &RawAuditEvent) -> String {
99 let trace_str = self.format_trace_chain(&event.trace_chain);
100 format!("[AUDIT_LOG] {} - Event: {:?}", trace_str, event)
101 }
102}
103
104pub struct LogFormatterFactory;
106
107impl LogFormatterFactory {
108 pub fn get_formatter() -> &'static (dyn LogFormatter + Send + Sync) {
111 static FORMATTER: std::sync::OnceLock<Box<dyn LogFormatter + Send + Sync>> = std::sync::OnceLock::new();
112 FORMATTER.get_or_init(|| {
113 let format = std::env::var("TEAQL_LOG_FORMAT").unwrap_or_else(|_| "human".to_string());
114 if format == "json" || format == "debug" {
115 Box::new(DebugReaderFormatter)
116 } else {
117 Box::new(HumanReaderFormatter)
118 }
119 }).as_ref()
120 }
121}
122
123pub struct LogManager;
125
126static LOG_ENDPOINT: std::sync::OnceLock<Option<String>> = std::sync::OnceLock::new();
127
128impl LogManager {
129 fn get_log_endpoint() -> Option<&'static str> {
130 LOG_ENDPOINT.get_or_init(|| {
131 std::env::var("TEAQL_LOG_ENDPOINT").ok().filter(|s| !s.is_empty())
132 }).as_deref()
133 }
134
135 fn write_to_file(content: &str) {
136 if let Some(endpoint) = Self::get_log_endpoint() {
137 if let Ok(mut file) = std::fs::OpenOptions::new().create(true).append(true).open(endpoint) {
138 use std::io::Write;
139 let _ = writeln!(file, "{}", content);
140 }
141 }
142 }
143
144 pub fn write_sql_log(trace_chain: &[TraceNode], entry: &SqlLogEntry) {
145 if Self::get_log_endpoint().is_none() {
146 return;
147 }
148 let content = LogFormatterFactory::get_formatter().format_sql_log(trace_chain, entry);
149 Self::write_to_file(&content);
150 }
151
152 pub fn write_audit_log(event: &RawAuditEvent) {
153 if Self::get_log_endpoint().is_none() {
154 return;
155 }
156 let content = LogFormatterFactory::get_formatter().format_audit_log(event);
157 Self::write_to_file(&content);
158 }
159}