1use std::fs;
2use std::io::Write;
3use std::path::{Path, PathBuf};
4
5use log::LevelFilter;
6use simplelog::{
7 ColorChoice, CombinedLogger, ConfigBuilder, SharedLogger, TermLogger, TerminalMode, WriteLogger,
8};
9
10const MAX_LOG_SIZE: u64 = 5 * 1024 * 1024; pub fn log_path(paths: Option<&crate::runtime::env::Paths>) -> Option<PathBuf> {
19 paths.map(crate::runtime::env::Paths::log_file)
20}
21
22fn rotate_if_needed(path: &Path) {
25 if let Ok(meta) = fs::metadata(path) {
26 if meta.len() > MAX_LOG_SIZE {
27 let backup = path.with_file_name("purple.log.1");
28 let _ = fs::rename(path, backup);
29 }
30 }
31}
32
33fn resolve_level(verbose: bool, env_override: Option<&str>) -> LevelFilter {
35 if let Some(val) = env_override {
36 match val.to_lowercase().as_str() {
37 "trace" => return LevelFilter::Trace,
38 "debug" => return LevelFilter::Debug,
39 "info" => return LevelFilter::Info,
40 "warn" => return LevelFilter::Warn,
41 "error" => return LevelFilter::Error,
42 "off" => return LevelFilter::Off,
43 _ => {}
44 }
45 }
46 if verbose {
47 LevelFilter::Debug
48 } else {
49 LevelFilter::Warn
50 }
51}
52
53pub fn init(verbose: bool, cli_stderr: bool, env: &crate::runtime::env::Env) {
58 let Some(path) = log_path(env.paths()) else {
59 return;
60 };
61
62 if let Some(parent) = path.parent() {
63 let _ = fs::create_dir_all(parent);
64 }
65
66 rotate_if_needed(&path);
67
68 let level = resolve_level(verbose, env.var("PURPLE_LOG"));
69 let config = ConfigBuilder::new()
70 .set_time_format_rfc3339()
71 .set_target_level(LevelFilter::Off)
72 .set_thread_level(LevelFilter::Off)
73 .build();
74
75 let mut loggers: Vec<Box<dyn SharedLogger>> = Vec::with_capacity(2);
76
77 if let Ok(file) = fs::OpenOptions::new().create(true).append(true).open(&path) {
78 loggers.push(WriteLogger::new(level, config.clone(), file));
79 }
80
81 if cli_stderr {
82 loggers.push(TermLogger::new(
83 level,
84 config,
85 TerminalMode::Stderr,
86 ColorChoice::Auto,
87 ));
88 }
89
90 if !loggers.is_empty() {
91 if let Err(e) = CombinedLogger::init(loggers) {
92 eprintln!("{}", crate::messages::logging::init_failed(&e));
93 }
94 }
95}
96
97fn format_now_utc() -> String {
99 let now = std::time::SystemTime::now();
100 let secs = now
101 .duration_since(std::time::UNIX_EPOCH)
102 .unwrap_or_default()
103 .as_secs();
104 let days = secs / 86400;
105 let time_of_day = secs % 86400;
106 let hours = time_of_day / 3600;
107 let minutes = (time_of_day % 3600) / 60;
108 let seconds = time_of_day % 60;
109
110 let (year, month, day) = epoch_days_to_date(days);
111 format!("{year:04}-{month:02}-{day:02} {hours:02}:{minutes:02}:{seconds:02}Z")
112}
113
114fn epoch_days_to_date(days: u64) -> (u64, u64, u64) {
117 let z = days + 719_468;
118 let era = z / 146_097;
119 let doe = z % 146_097;
120 let yoe = (doe - doe / 1460 + doe / 36524 - doe / 146_096) / 365;
121 let y = yoe + era * 400;
122 let doy = doe - (365 * yoe + yoe / 4 - yoe / 100);
123 let mp = (5 * doy + 2) / 153;
124 let d = doy - (153 * mp + 2) / 5 + 1;
125 let m = if mp < 10 { mp + 3 } else { mp - 9 };
126 let y = if m <= 2 { y + 1 } else { y };
127 (y, m, d)
128}
129
130pub struct BannerInfo<'a> {
132 pub version: &'a str,
133 pub config_path: &'a str,
134 pub providers: &'a [String],
135 pub askpass_sources: &'a [String],
136 pub vault_ssh_info: Option<&'a str>,
137 pub ssh_version: &'a str,
138 pub term: &'a str,
139 pub colorterm: &'a str,
140 pub level: &'a str,
141 pub theme: &'a str,
143 pub hosts: usize,
145 pub patterns: usize,
147 pub snippets: usize,
149 pub proxy_env: &'a str,
152}
153
154pub fn write_banner(info: &BannerInfo<'_>, paths: Option<&crate::runtime::env::Paths>) {
161 let Some(path) = log_path(paths) else { return };
162 let Ok(mut file) = fs::OpenOptions::new().create(true).append(true).open(&path) else {
163 return;
164 };
165
166 let now = format_now_utc();
167 let os = std::env::consts::OS;
168 let arch = std::env::consts::ARCH;
169 let providers_joined = if info.providers.is_empty() {
170 "none".to_string()
171 } else {
172 info.providers.join(",")
173 };
174 let askpass_joined = if info.askpass_sources.is_empty() {
175 "none".to_string()
176 } else {
177 info.askpass_sources.join(",")
178 };
179
180 let mut banner = format!(
181 "--- purple v{} started at {now} ---\n\
182 \x20 os={os} arch={arch} config={}\n\
183 \x20 ssh={}\n\
184 \x20 term={} colorterm={}\n\
185 \x20 theme={}\n\
186 \x20 hosts={} patterns={} snippets={}\n\
187 \x20 providers={providers_joined}\n\
188 \x20 askpass={askpass_joined}\n\
189 \x20 proxy_env={}\n",
190 info.version,
191 info.config_path,
192 info.ssh_version,
193 info.term,
194 info.colorterm,
195 info.theme,
196 info.hosts,
197 info.patterns,
198 info.snippets,
199 info.proxy_env,
200 );
201 if let Some(vault_info) = info.vault_ssh_info {
202 banner.push_str(&format!(" vault_ssh={vault_info}\n"));
203 }
204 banner.push_str(&format!(" log_level={}\n", info.level));
205
206 let _ = file.write_all(banner.as_bytes());
212}
213
214pub fn level_name(verbose: bool, env: &crate::runtime::env::Env) -> String {
216 resolve_level(verbose, env.var("PURPLE_LOG"))
217 .as_str()
218 .to_lowercase()
219}
220
221pub fn detect_ssh_version() -> String {
229 use std::sync::mpsc;
230 use std::time::Duration;
231
232 let child = std::process::Command::new("ssh")
233 .arg("-V")
234 .stdout(std::process::Stdio::piped())
235 .stderr(std::process::Stdio::piped())
236 .spawn();
237
238 let Ok(child) = child else {
239 eprintln!("{}", crate::messages::logging::SSH_VERSION_FAILED);
240 return "unknown".to_string();
241 };
242
243 let (tx, rx) = mpsc::channel();
244 std::thread::spawn(move || {
245 let _ = tx.send(child.wait_with_output());
246 });
247
248 match rx.recv_timeout(Duration::from_secs(2)) {
249 Ok(Ok(output)) => {
250 let out = if output.stderr.is_empty() {
251 output.stdout
252 } else {
253 output.stderr
254 };
255 String::from_utf8(out)
256 .map(|s| s.trim().to_string())
257 .unwrap_or_else(|_| "unknown".to_string())
258 }
259 _ => "unknown".to_string(),
260 }
261}
262
263#[cfg(test)]
267pub(crate) mod capture {
268 use std::cell::RefCell;
269 use std::sync::Once;
270
271 use log::{Level, Log, Metadata, Record};
272
273 thread_local! {
274 static SINK: RefCell<Option<Vec<(Level, String)>>> = const { RefCell::new(None) };
275 }
276
277 struct CaptureLogger;
278
279 impl Log for CaptureLogger {
280 fn enabled(&self, _: &Metadata) -> bool {
281 true
282 }
283 fn log(&self, record: &Record) {
284 SINK.with(|s| {
285 if let Some(buf) = s.borrow_mut().as_mut() {
286 buf.push((record.level(), record.args().to_string()));
287 }
288 });
289 }
290 fn flush(&self) {}
291 }
292
293 static LOGGER: CaptureLogger = CaptureLogger;
294 static INSTALL: Once = Once::new();
295
296 pub(crate) fn capture<F: FnOnce()>(f: F) -> Vec<(Level, String)> {
298 INSTALL.call_once(|| {
299 if log::set_logger(&LOGGER).is_ok() {
302 log::set_max_level(log::LevelFilter::Trace);
303 }
304 });
305 SINK.with(|s| *s.borrow_mut() = Some(Vec::new()));
306 f();
307 SINK.with(|s| s.borrow_mut().take().unwrap_or_default())
308 }
309
310 pub(crate) fn has(records: &[(Level, String)], level: Level, needle: &str) -> bool {
312 records
313 .iter()
314 .any(|(lvl, msg)| *lvl == level && msg.contains(needle))
315 }
316}
317
318#[cfg(test)]
319mod tests {
320 use super::*;
321 use std::io::Write;
322
323 #[test]
324 fn rotate_if_needed_renames_large_file() {
325 let dir = tempfile::tempdir().unwrap();
326 let log = dir.path().join("purple.log");
327 let backup = dir.path().join("purple.log.1");
328
329 let mut f = fs::File::create(&log).unwrap();
330 let data = vec![0u8; (MAX_LOG_SIZE + 1) as usize];
331 f.write_all(&data).unwrap();
332 drop(f);
333
334 rotate_if_needed(&log);
335
336 assert!(!log.exists());
337 assert!(backup.exists());
338 assert!(fs::metadata(&backup).unwrap().len() > MAX_LOG_SIZE);
339 }
340
341 #[test]
342 fn rotate_if_needed_leaves_small_file() {
343 let dir = tempfile::tempdir().unwrap();
344 let log = dir.path().join("purple.log");
345
346 fs::write(&log, "small content").unwrap();
347
348 rotate_if_needed(&log);
349
350 assert!(log.exists());
351 assert!(!dir.path().join("purple.log.1").exists());
352 }
353
354 #[test]
355 fn rotate_if_needed_handles_missing_file() {
356 let dir = tempfile::tempdir().unwrap();
357 let log = dir.path().join("purple.log");
358
359 rotate_if_needed(&log);
361 }
362
363 #[test]
364 fn resolve_level_defaults_to_warn() {
365 assert_eq!(resolve_level(false, None), LevelFilter::Warn);
366 }
367
368 #[test]
369 fn resolve_level_verbose_returns_debug() {
370 assert_eq!(resolve_level(true, None), LevelFilter::Debug);
371 }
372
373 #[test]
374 fn resolve_level_env_overrides_verbose() {
375 assert_eq!(resolve_level(false, Some("trace")), LevelFilter::Trace);
376 assert_eq!(resolve_level(true, Some("error")), LevelFilter::Error);
377 }
378
379 #[test]
380 fn resolve_level_ignores_unknown_env_value() {
381 assert_eq!(resolve_level(false, Some("bogus")), LevelFilter::Warn);
382 assert_eq!(resolve_level(true, Some("bogus")), LevelFilter::Debug);
383 }
384
385 #[test]
386 fn epoch_days_to_date_unix_epoch() {
387 assert_eq!(epoch_days_to_date(0), (1970, 1, 1));
389 }
390
391 #[test]
392 fn epoch_days_to_date_known_date() {
393 assert_eq!(epoch_days_to_date(20553), (2026, 4, 10));
395 }
396
397 #[test]
398 fn epoch_days_to_date_leap_year() {
399 assert_eq!(epoch_days_to_date(11016), (2000, 2, 29));
401 }
402
403 #[test]
404 fn format_now_utc_returns_valid_timestamp() {
405 let ts = format_now_utc();
406 assert_eq!(ts.len(), 20);
408 assert_eq!(&ts[4..5], "-");
409 assert_eq!(&ts[7..8], "-");
410 assert_eq!(&ts[10..11], " ");
411 assert_eq!(&ts[13..14], ":");
412 assert_eq!(&ts[16..17], ":");
413 assert!(ts.ends_with('Z'));
414 }
415
416 #[test]
417 fn log_path_ends_with_purple_log() {
418 let paths = crate::runtime::env::Paths::new("/home/u");
419 let path = log_path(Some(&paths)).expect("paths present");
420 assert!(path.ends_with(".purple/purple.log"));
421 }
422
423 #[test]
424 fn level_name_defaults_to_warn() {
425 let env = crate::runtime::env::Env::empty();
427 let name = level_name(false, &env);
428 assert_eq!(name, "warn");
429 }
430
431 #[test]
432 fn write_banner_creates_output() {
433 let info = BannerInfo {
436 version: "0.0.0-test",
437 config_path: "/tmp/config",
438 providers: &["testprov".to_string()],
439 askpass_sources: &["keychain:".to_string()],
440 vault_ssh_info: Some("enabled (addr=https://vault:8200)"),
441 ssh_version: "OpenSSH_9.0",
442 term: "xterm-256color",
443 colorterm: "truecolor",
444 level: "warn",
445 theme: "Purple",
446 hosts: 42,
447 patterns: 3,
448 snippets: 7,
449 proxy_env: "none",
450 };
451
452 assert_eq!(info.version, "0.0.0-test");
454 assert_eq!(info.providers.len(), 1);
455 assert!(info.vault_ssh_info.is_some());
456 assert_eq!(info.theme, "Purple");
457 assert_eq!(info.hosts, 42);
458 assert_eq!(info.snippets, 7);
459 assert_eq!(info.proxy_env, "none");
460
461 let ts = format_now_utc();
464 assert!(ts.ends_with('Z'));
465 assert_eq!(ts.len(), 20);
466 }
467}