use log::{LogLevel, LogLevelFilter, LogMetadata, LogRecord, SetLoggerError, set_logger, Log};
use term;
use term::{StderrTerminal, StdoutTerminal, Terminal, color};
use std::error;
use std::fmt;
use std::sync::{Mutex, MutexGuard};
use std::io::{Write, Error};
use self::TermLogError::{SetLogger, Term};
use super::logging::*;
use ::{Config, SharedLogger};
#[derive(Debug)]
pub enum TermLogError {
SetLogger(SetLoggerError),
Term,
}
impl fmt::Display for TermLogError {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
use std::error::Error as FmtError;
write!(f, "{}", self.description())
}
}
impl error::Error for TermLogError {
fn description(&self) -> &str {
match * self {
SetLogger(ref err) => err.description(),
Term => "A terminal could not be opened",
}
}
fn cause(&self) -> Option<&error::Error> {
match *self {
SetLogger(ref err) => Some(err),
Term => None,
}
}
}
impl From<SetLoggerError> for TermLogError {
fn from(error: SetLoggerError) -> Self {
SetLogger(error)
}
}
pub struct TermLogger {
level: LogLevelFilter,
config: Config,
stderr: Mutex<Box<StderrTerminal>>,
stdout: Mutex<Box<StdoutTerminal>>,
}
impl TermLogger
{
pub fn init(log_level: LogLevelFilter, config: Config) -> Result<(), TermLogError> {
let logger = try!(TermLogger::new(log_level, config).ok_or(Term));
try!(set_logger(|max_log_level| {
max_log_level.set(log_level.clone());
logger
}));
Ok(())
}
pub fn new(log_level: LogLevelFilter, config: Config) -> Option<Box<TermLogger>> {
term::stderr().and_then(|stderr|
term::stdout().map(|stdout| {
Box::new(TermLogger { level: log_level, config: config, stderr: Mutex::new(stderr), stdout: Mutex::new(stdout) })
})
)
}
fn try_log_term<W>(&self, record: &LogRecord, mut term_lock: MutexGuard<Box<Terminal<Output=W> + Send>>) -> Result<(), Error>
where W: Write + Sized
{
let color = match record.level() {
LogLevel::Error => color::RED,
LogLevel::Warn => color::YELLOW,
LogLevel::Info => color::BLUE,
LogLevel::Debug => color::CYAN,
LogLevel::Trace => color::WHITE
};
if self.config.time.is_some() || self.config.time.unwrap() <= record.level() {
try!(write_time(&mut *term_lock));
}
if self.config.level.is_some() || self.config.level.unwrap() <= record.level() {
try!(term_lock.fg(color));
try!(write_level(record, &mut *term_lock));
try!(term_lock.reset());
}
if self.config.target.is_some() || self.config.target.unwrap() <= record.level() {
try!(write_target(record, &mut *term_lock));
}
if self.config.location.is_some() || self.config.location.unwrap() <= record.level() {
try!(write_location(record, &mut *term_lock));
}
try!(write_args(record, &mut *term_lock));
try!(term_lock.flush());
Ok(())
}
fn try_log(&self, record: &LogRecord) -> Result<(), Error> {
if self.enabled(record.metadata()) {
if record.level() == LogLevel::Error {
self.try_log_term(record, self.stderr.lock().unwrap())
} else {
self.try_log_term(record, self.stdout.lock().unwrap())
}
} else {
Ok(())
}
}
}
impl Log for TermLogger
{
fn enabled(&self, metadata: &LogMetadata) -> bool {
metadata.level() <= self.level
}
fn log(&self, record: &LogRecord) {
let _ = self.try_log(record);
}
}
impl SharedLogger for TermLogger
{
fn level(&self) -> LogLevelFilter {
self.level
}
fn config(&self) -> Option<&Config>
{
Some(&self.config)
}
fn as_log(self: Box<Self>) -> Box<Log> {
Box::new(*self)
}
}