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
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
extern crate chrono;
extern crate colored;
use self::chrono::{DateTime, Local};
use self::colored::{Color, Colorize};
use super::ffi;
use std::ffi::CString;
use std::fmt::Write;
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
pub enum LogLevel {
Fatal = 1,
Err = 2,
Warn = 3,
Info = 4,
Verb = 5,
Huge = 6,
Dbg = 7,
}
impl LogLevel {
fn color(&self) -> Option<Color> {
match *self {
LogLevel::Fatal => Some(Color::Magenta),
LogLevel::Err => Some(Color::Red),
LogLevel::Warn => Some(Color::Yellow),
_ => None,
}
}
}
#[derive(Debug, Clone)]
pub struct LogParameters {
pub max_log_level: i32,
pub log_timestamps: bool,
pub log_colors: bool,
pub clock: fn() -> DateTime<Local>,
}
impl Default for LogParameters {
fn default() -> Self {
unsafe {
Self {
max_log_level: ffi::janus_log_level,
log_timestamps: ffi::janus_log_timestamps == 1,
log_colors: ffi::janus_log_colors == 1,
clock: Local::now,
}
}
}
}
pub fn write_log(level: LogLevel, message: &str, params: LogParameters) {
if level as i32 <= params.max_log_level {
let output = CString::new(print_log(level, message, params)).expect("Null character in log message :(");
unsafe { ffi::janus_vprintf(output.as_ptr()) }
}
}
pub fn print_log(level: LogLevel, message: &str, params: LogParameters) -> String {
let mut output = String::with_capacity(message.len() + 40);
if params.log_timestamps {
write!(output, "{} ", (params.clock)().format("[%a %b %e %T %Y]")).unwrap();
}
if level <= LogLevel::Warn {
let name = format!("[{:?}] ", level).to_uppercase();
match level.color() {
Some(c) if params.log_colors => output.push_str(&name.color(c)),
_ => output.push_str(&name),
}
}
output.push_str(message);
output.push('\n');
output
}
pub fn log(level: LogLevel, message: &str) {
write_log(level, message, LogParameters::default());
}
#[cfg(test)]
mod tests {
use super::*;
use super::chrono::TimeZone;
fn fixed_clock(year: i32, month: u32, day: u32, hour: u32, min: u32, sec: u32) -> DateTime<Local> {
Local.ymd(year, month, day).and_hms(hour, min, sec)
}
#[test]
fn log_format_correctness() {
assert_eq!(
"[Tue Oct 10 01:37:46 2017] [WARN] Test message.\n",
print_log(
LogLevel::Warn,
"Test message.",
LogParameters {
max_log_level: 6,
log_timestamps: true,
log_colors: false,
clock: || fixed_clock(2017, 10, 10, 1, 37, 46),
}
)
)
}
}