- Commit
- 036129dd66bd2058b7d1abfe16aa5310d5cae12a
-
- Parent
- 98fc8cba86648d2af6d45ba6fcb4659684d3fc69
-
- Author
- Pablo <pablo-pie@riseup.net>
- Date
Made the log system more responsive
Made the log system automatically reprint the job counter after the posting of logs
This is achieved by keeping track of the counter state via a global static
Also made logs blue instead of green
diff --git a/src/log.rs b/src/log.rs
@@ -1,14 +1,21 @@
//! Macros for logging.
//!
-//! This implementation should be thread safe (unlink an implementation using
+//! This implementation should be thread safe (unlike an implementation using
//! `println!` and `eprintln!`) because access to the global stdout/stderr
//! handle is syncronized using a lock.
-use std::{io::{self, Write}, fmt::Arguments};
+#![allow(static_mut_refs)]
+use std::{io::{self, Write}, fmt::Arguments, sync::RwLock};
-pub const BOLD_GREEN: &str = "\u{001b}[1;32m";
-pub const BOLD_RED: &str = "\u{001b}[1;31m";
-pub const BOLD_YELLOW: &str = "\u{001b}[1;33m";
-pub const RESET: &str = "\u{001b}[0m";
+const BOLD_WHITE: &str = "\u{001b}[1;37m";
+const BOLD_BLUE: &str = "\u{001b}[1;34m";
+const BOLD_RED: &str = "\u{001b}[1;31m";
+const BOLD_YELLOW: &str = "\u{001b}[1;33m";
+const RESET: &str = "\u{001b}[0m";
+
+// TODO: [optimize]: make a thread-unsafe version of this under a compile flag
+static NEEDS_NEWLINE: RwLock<bool> = RwLock::new(false);
+static mut COUNTER_TOTAL: RwLock<usize> = RwLock::new(0);
+static mut COUNTER_STATE: RwLock<Option<CounterState>> = RwLock::new(None);
#[derive(Clone, Copy, Debug)]
pub(crate) enum Level {
@@ -17,39 +24,141 @@ pub(crate) enum Level {
Warn,
}
+struct CounterState {
+ count: usize,
+ current_repo_name: String,
+}
+
pub(crate) fn log(level: Level, args: &Arguments<'_>, newline: bool) {
match level {
Level::Error => {
let mut stderr = io::stderr();
+
+ if needs_newline() {
+ let _ = writeln!(stderr);
+ }
+
let _ = write!(stderr, "{BOLD_RED}ERROR:{RESET} ");
- let _ = if newline {
- writeln!(stderr, "{}", args)
+ if newline {
+ let _ = writeln!(stderr, "{}", args);
+ // shouldn't print the job counter because we are about to die
} else {
- write!(stderr, "{}", args)
- };
- if !newline { let _ = stderr.flush(); }
+ let _ = write!(stderr, "{}", args);
+ let _ = stderr.flush();
+ }
}
Level::Info => {
let mut stdout = io::stdout().lock();
- let _ = write!(stdout, "{BOLD_GREEN}INFO:{RESET} ");
- let _ = if newline {
- writeln!(stdout, "{}", args)
+
+ if needs_newline() {
+ let _ = writeln!(stdout);
+ }
+
+ let _ = write!(stdout, "{BOLD_BLUE}INFO:{RESET} ");
+ if newline {
+ let counter_state = unsafe { COUNTER_STATE.get_mut().unwrap() };
+ let _ = writeln!(stdout, "{}", args);
+ if let Some(ref counter) = counter_state {
+ log_job_counter(counter);
+ }
} else {
- write!(stdout, "{}", args)
- };
- if !newline { let _ = stdout.flush(); }
+ let _ = write!(stdout, "{}", args);
+ let _ = stdout.flush();
+ }
}
Level::Warn => {
let mut stdout = io::stdout().lock();
+
+ if needs_newline() {
+ let _ = writeln!(stdout);
+ }
+
let _ = write!(stdout, "{BOLD_YELLOW}WARNING:{RESET} ");
- let _ = if newline {
- writeln!(stdout, "{}", args)
+ if newline {
+ let counter_state = unsafe { COUNTER_STATE.get_mut().unwrap() };
+ let _ = writeln!(stdout, "{}", args);
+ if let Some(ref counter) = counter_state {
+ log_job_counter(counter);
+ }
} else {
- write!(stdout, "{}", args)
- };
+ let _ = write!(stdout, "{}", args);
+ }
if !newline { let _ = stdout.flush(); }
}
}
+
+ if !newline {
+ set_needs_newline(true);
+ }
+}
+
+pub fn info_done(args: Option<&Arguments<'_>>) {
+ let mut stdout = io::stdout().lock();
+ let _ = match args {
+ Some(args) => {
+ writeln!(stdout, " {}", args)
+ }
+ None => {
+ writeln!(stdout, " done!")
+ }
+ };
+ set_needs_newline(false);
+}
+
+pub fn job_counter_start(total: usize) {
+ unsafe {
+ *COUNTER_TOTAL.write().unwrap() = total;
+ }
+}
+
+pub fn job_counter_increment(repo_name: &str) {
+ let counter_total = unsafe { COUNTER_TOTAL.get_mut().unwrap() };
+ let counter = unsafe { COUNTER_STATE.get_mut().unwrap() };
+
+ if let Some(ref mut inner) = counter {
+ inner.count += 1;
+ inner.current_repo_name = repo_name.to_owned();
+
+ log_job_counter(inner);
+
+ // deinit the counter when we reach the total
+ if inner.count == *counter_total {
+ *counter = None;
+ *counter_total = 0;
+ }
+ } else {
+ let new_counter = CounterState {
+ count: 1,
+ current_repo_name: repo_name.to_owned(),
+ };
+
+ log_job_counter(&new_counter);
+ *counter = Some(new_counter);
+ }
+}
+
+fn log_job_counter(counter: &CounterState) {
+ let mut stdout = io::stdout().lock();
+ let counter_total = unsafe { *COUNTER_TOTAL.read().unwrap() };
+
+ let _ = write!(
+ stdout,
+ "{BOLD_BLUE}->{RESET} {BOLD_WHITE}{count:>padding$}/{total}{RESET} {name}...",
+ count = counter.count,
+ total = counter_total,
+ padding = crate::log_floor(counter_total),
+ name = counter.current_repo_name,
+ );
+ let _ = stdout.flush();
+ set_needs_newline(true);
+}
+
+fn needs_newline() -> bool {
+ *NEEDS_NEWLINE.read().unwrap()
+}
+
+fn set_needs_newline(val: bool) {
+ *NEEDS_NEWLINE.write().unwrap() = val;
}
#[macro_export]
@@ -79,33 +188,26 @@ macro_rules! infoln {
#[macro_export]
macro_rules! info_done {
() => ({
- let _ = writeln!(io::stdout().lock(), " done!");
+ $crate::log::info_done(None);
});
// infoln!("a {} event", "log");
($($arg:tt)+) => ({
- let _ = writeln!(
- io::stdout().lock(),
- " {}",
- &std::format_args!($($arg)+)
- );
+ $crate::log::info_done(Some(&std::format_args!($($arg)+)));
});
}
#[macro_export]
-macro_rules! info_count {
- ($count:expr, $total:expr; $($arg:tt)+) => ({
- use $crate::log::{BOLD_GREEN, RESET};
- let mut stdout = io::stdout().lock();
- let _ = write!(
- stdout,
- "{BOLD_GREEN}=> {count:>padding$}/{total}{RESET} {args}",
- count = $count,
- total = $total,
- padding = $crate::log_floor($total),
- args = std::format_args!($($arg)+)
- );
- let _ = stdout.flush();
+macro_rules! job_counter_start {
+ ($total:expr) => ({
+ $crate::log::job_counter_start($total as usize);
+ });
+}
+
+#[macro_export]
+macro_rules! job_counter_increment {
+ ($repo_name:expr) => ({
+ $crate::log::job_counter_increment(&$repo_name);
});
}