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
use std::error::Error;
use std::io::{ self, BufWriter, Write };
use std::fs;
use std::fs::OpenOptions;
use chrono::prelude::*;
use output::USIStdErrorWriter;
use string::AddIndent;
pub trait Logger {
fn logging(&mut self, msg:&String) -> bool;
fn logging_error<E: Error>(&mut self, e:&E) -> bool {
let mut messages:Vec<String> = vec![];
let mut indent:u32 = 0;
messages.push(format!("{}", e).add_indent(indent*2));
let mut e:&(dyn Error) = e;
while let Some(cause) = e.source() {
indent += 1;
messages.push(format!("{}", cause).add_indent(indent*2));
e = cause;
}
self.logging(&messages.join("\n"))
}
}
#[derive(Debug)]
pub struct FileLogger {
writer:BufWriter<fs::File>,
}
impl FileLogger {
pub fn new(file:String) -> Result<FileLogger,io::Error> {
Ok(FileLogger {
writer:BufWriter::new(OpenOptions::new().append(true).create(true).open(file)?),
})
}
}
impl Logger for FileLogger {
fn logging(&mut self, msg:&String) -> bool {
let dt = Local::now();
let msg = format!("{}\n{}\n", dt.format("%Y-%m-%d %H:%M:%S").to_string(), msg.add_indent(2));
match self.writer.write(msg.as_bytes()) {
Ok(_) => !self.writer.flush().is_err(),
Err(_)=> {
USIStdErrorWriter::write("The log could not be written to the file.").unwrap();
let _ = self.writer.flush();
false
}
}
}
}