1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
use crate::progress::{MessageLevel, Progress}; use std::time::Duration; pub struct Log { name: String, max: Option<u32>, unit: Option<&'static str>, last_set: Option<std::time::SystemTime>, step: u32, current_level: usize, max_level: usize, } const EMIT_LOG_EVERY_S: f32 = 0.5; impl Log { pub fn new(name: impl Into<String>, max_level: Option<usize>) -> Self { Log { name: name.into(), current_level: 0, max_level: max_level.unwrap_or(usize::MAX), max: None, step: 0, unit: None, last_set: None, } } } impl Progress for Log { type SubProgress = Log; fn add_child(&mut self, name: impl Into<String>) -> Self::SubProgress { Log { name: format!("{}::{}", self.name, Into::<String>::into(name)), current_level: self.current_level + 1, max_level: self.max_level, step: 0, max: None, unit: None, last_set: None, } } fn init(&mut self, max: Option<u32>, unit: Option<&'static str>) { self.max = max; self.unit = unit; } fn set(&mut self, step: u32) { self.step = step; if self.current_level > self.max_level { return; } let now = std::time::SystemTime::now(); if self .last_set .map(|last| { now.duration_since(last) .unwrap_or_else(|_| Duration::default()) .as_secs_f32() }) .unwrap_or_else(|| EMIT_LOG_EVERY_S * 2.0) > EMIT_LOG_EVERY_S { self.last_set = Some(now); match (self.max, self.unit) { (Some(max), Some(unit)) => log::info!("{} → {} / {} {}", self.name, step, max, unit), (None, Some(unit)) => log::info!("{} → {} {}", self.name, step, unit), (Some(max), None) => log::info!("{} → {} / {}", self.name, step, max), (None, None) => log::info!("{} → {}", self.name, step), } } } fn inc_by(&mut self, step: u32) { self.set(self.step + step) } fn message(&mut self, level: MessageLevel, message: impl Into<String>) { let message: String = message.into(); match level { MessageLevel::Info => log::info!("ℹ{} → {}", self.name, message), MessageLevel::Failure => log::error!("𐄂{} → {}", self.name, message), MessageLevel::Success => log::info!("✓{} → {}", self.name, message), } } }