2018-03-05 22:33:44 +03:00
|
|
|
// Copyright 2018 The Grin Developers
|
2017-10-12 19:56:44 +03:00
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
//! Logging wrapper to be used throughout all crates in the workspace
|
2018-12-08 02:59:40 +03:00
|
|
|
use crate::Mutex;
|
2018-06-22 11:08:06 +03:00
|
|
|
use std::ops::Deref;
|
2017-10-12 19:56:44 +03:00
|
|
|
|
2018-03-14 21:22:18 +03:00
|
|
|
use backtrace::Backtrace;
|
|
|
|
use std::{panic, thread};
|
|
|
|
|
2018-12-08 02:59:40 +03:00
|
|
|
use crate::types::{LogLevel, LoggingConfig};
|
2017-10-12 19:56:44 +03:00
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
use log::{LevelFilter, Record};
|
|
|
|
use log4rs;
|
|
|
|
use log4rs::append::console::ConsoleAppender;
|
|
|
|
use log4rs::append::file::FileAppender;
|
|
|
|
use log4rs::append::rolling_file::{
|
|
|
|
policy::compound::roll::fixed_window::FixedWindowRoller,
|
|
|
|
policy::compound::trigger::size::SizeTrigger, policy::compound::CompoundPolicy,
|
|
|
|
RollingFileAppender,
|
|
|
|
};
|
|
|
|
use log4rs::append::Append;
|
|
|
|
use log4rs::config::{Appender, Config, Root};
|
|
|
|
use log4rs::encode::pattern::PatternEncoder;
|
|
|
|
use log4rs::filter::{threshold::ThresholdFilter, Filter, Response};
|
|
|
|
|
|
|
|
fn convert_log_level(in_level: &LogLevel) -> LevelFilter {
|
2017-10-12 19:56:44 +03:00
|
|
|
match *in_level {
|
2018-10-21 23:30:56 +03:00
|
|
|
LogLevel::Info => LevelFilter::Info,
|
|
|
|
LogLevel::Warning => LevelFilter::Warn,
|
|
|
|
LogLevel::Debug => LevelFilter::Debug,
|
|
|
|
LogLevel::Trace => LevelFilter::Trace,
|
|
|
|
LogLevel::Error => LevelFilter::Error,
|
2017-10-12 19:56:44 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lazy_static! {
|
2017-11-19 21:34:43 +03:00
|
|
|
/// Flag to observe whether logging was explicitly initialised (don't output otherwise)
|
2017-10-13 19:42:04 +03:00
|
|
|
static ref WAS_INIT: Mutex<bool> = Mutex::new(false);
|
2018-03-19 22:23:58 +03:00
|
|
|
/// Flag to observe whether tui is running, and we therefore don't want to attempt to write
|
|
|
|
/// panics to stdout
|
|
|
|
static ref TUI_RUNNING: Mutex<bool> = Mutex::new(false);
|
2017-10-12 19:56:44 +03:00
|
|
|
/// Static Logging configuration, should only be set once, before first logging call
|
|
|
|
static ref LOGGING_CONFIG: Mutex<LoggingConfig> = Mutex::new(LoggingConfig::default());
|
2018-10-21 23:30:56 +03:00
|
|
|
}
|
|
|
|
|
2018-11-02 18:13:12 +03:00
|
|
|
const LOGGING_PATTERN: &str = "{d(%Y%m%d %H:%M:%S%.3f)} {h({l})} {M} - {m}{n}";
|
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
/// This filter is rejecting messages that doesn't start with "grin"
|
|
|
|
/// in order to save log space for only Grin-related records
|
|
|
|
#[derive(Debug)]
|
|
|
|
struct GrinFilter;
|
2017-11-19 21:34:43 +03:00
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
impl Filter for GrinFilter {
|
2018-12-08 02:59:40 +03:00
|
|
|
fn filter(&self, record: &Record<'_>) -> Response {
|
2018-10-21 23:30:56 +03:00
|
|
|
if let Some(module_path) = record.module_path() {
|
|
|
|
if module_path.starts_with("grin") {
|
|
|
|
return Response::Neutral;
|
|
|
|
}
|
2018-03-19 22:23:58 +03:00
|
|
|
}
|
2017-10-12 19:56:44 +03:00
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
Response::Reject
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Initialize the logger with the given configuration
|
|
|
|
pub fn init_logger(config: Option<LoggingConfig>) {
|
|
|
|
if let Some(c) = config {
|
2018-10-27 20:37:03 +03:00
|
|
|
let tui_running = c.tui_running.unwrap_or(false);
|
|
|
|
if tui_running {
|
|
|
|
let mut tui_running_ref = TUI_RUNNING.lock();
|
|
|
|
*tui_running_ref = true;
|
|
|
|
}
|
|
|
|
|
2018-10-24 19:13:57 +03:00
|
|
|
// Save current logging configuration
|
|
|
|
let mut config_ref = LOGGING_CONFIG.lock();
|
|
|
|
*config_ref = c.clone();
|
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
let level_stdout = convert_log_level(&c.stdout_log_level);
|
|
|
|
let level_file = convert_log_level(&c.file_log_level);
|
|
|
|
let level_minimum;
|
|
|
|
|
|
|
|
// Determine minimum logging level for Root logger
|
|
|
|
if level_stdout > level_file {
|
|
|
|
level_minimum = level_stdout;
|
|
|
|
} else {
|
|
|
|
level_minimum = level_file;
|
2017-10-12 19:56:44 +03:00
|
|
|
}
|
2017-11-01 02:32:33 +03:00
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
// Start logger
|
|
|
|
let stdout = ConsoleAppender::builder()
|
2018-11-02 18:13:12 +03:00
|
|
|
.encoder(Box::new(PatternEncoder::new(&LOGGING_PATTERN)))
|
2018-10-21 23:30:56 +03:00
|
|
|
.build();
|
2017-10-12 19:56:44 +03:00
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
let mut root = Root::builder();
|
2017-10-12 19:56:44 +03:00
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
let mut appenders = vec![];
|
2017-10-12 19:56:44 +03:00
|
|
|
|
2018-10-27 20:37:03 +03:00
|
|
|
if c.log_to_stdout && !tui_running {
|
2018-10-21 23:30:56 +03:00
|
|
|
let filter = Box::new(ThresholdFilter::new(level_stdout));
|
|
|
|
appenders.push(
|
|
|
|
Appender::builder()
|
|
|
|
.filter(filter)
|
|
|
|
.filter(Box::new(GrinFilter))
|
|
|
|
.build("stdout", Box::new(stdout)),
|
|
|
|
);
|
|
|
|
|
|
|
|
root = root.appender("stdout");
|
2018-04-24 13:02:49 +03:00
|
|
|
}
|
2017-10-12 19:56:44 +03:00
|
|
|
|
2018-10-21 23:30:56 +03:00
|
|
|
if c.log_to_file {
|
|
|
|
// If maximum log size is specified, use rolling file appender
|
|
|
|
// or use basic one otherwise
|
|
|
|
let filter = Box::new(ThresholdFilter::new(level_file));
|
2018-12-08 02:59:40 +03:00
|
|
|
let file: Box<dyn Append> = {
|
2018-10-21 23:30:56 +03:00
|
|
|
if let Some(size) = c.log_max_size {
|
|
|
|
let roller = FixedWindowRoller::builder()
|
|
|
|
.build(&format!("{}.{{}}.gz", c.log_file_path), 32)
|
|
|
|
.unwrap();
|
|
|
|
let trigger = SizeTrigger::new(size);
|
|
|
|
|
|
|
|
let policy = CompoundPolicy::new(Box::new(trigger), Box::new(roller));
|
|
|
|
|
|
|
|
Box::new(
|
|
|
|
RollingFileAppender::builder()
|
|
|
|
.append(c.log_file_append)
|
2018-11-02 18:13:12 +03:00
|
|
|
.encoder(Box::new(PatternEncoder::new(&LOGGING_PATTERN)))
|
2018-10-21 23:30:56 +03:00
|
|
|
.build(c.log_file_path, Box::new(policy))
|
|
|
|
.unwrap(),
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
Box::new(
|
|
|
|
FileAppender::builder()
|
|
|
|
.append(c.log_file_append)
|
2018-11-02 18:13:12 +03:00
|
|
|
.encoder(Box::new(PatternEncoder::new(&LOGGING_PATTERN)))
|
2018-10-21 23:30:56 +03:00
|
|
|
.build(c.log_file_path)
|
|
|
|
.unwrap(),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
appenders.push(
|
|
|
|
Appender::builder()
|
|
|
|
.filter(filter)
|
|
|
|
.filter(Box::new(GrinFilter))
|
|
|
|
.build("file", file),
|
|
|
|
);
|
|
|
|
root = root.appender("file");
|
|
|
|
}
|
|
|
|
|
|
|
|
let config = Config::builder()
|
|
|
|
.appenders(appenders)
|
|
|
|
.build(root.build(level_minimum))
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let _ = log4rs::init_config(config).unwrap();
|
|
|
|
|
|
|
|
info!(
|
|
|
|
"log4rs is initialized, file level: {:?}, stdout level: {:?}, min. level: {:?}",
|
|
|
|
level_file, level_stdout, level_minimum
|
|
|
|
);
|
|
|
|
|
2018-10-24 19:13:57 +03:00
|
|
|
// Mark logger as initialized
|
2018-10-20 03:13:07 +03:00
|
|
|
let mut was_init_ref = WAS_INIT.lock();
|
2017-10-13 19:42:04 +03:00
|
|
|
*was_init_ref = true;
|
2017-10-12 19:56:44 +03:00
|
|
|
}
|
2018-10-24 19:13:57 +03:00
|
|
|
|
2018-03-14 21:22:18 +03:00
|
|
|
send_panic_to_log();
|
2017-10-12 19:56:44 +03:00
|
|
|
}
|
2017-10-22 10:09:40 +03:00
|
|
|
|
|
|
|
/// Initializes the logger for unit and integration tests
|
|
|
|
pub fn init_test_logger() {
|
2018-10-20 03:13:07 +03:00
|
|
|
let mut was_init_ref = WAS_INIT.lock();
|
2017-10-22 11:09:02 +03:00
|
|
|
if *was_init_ref.deref() {
|
2017-10-22 10:09:40 +03:00
|
|
|
return;
|
|
|
|
}
|
2018-08-16 21:32:03 +03:00
|
|
|
let mut logger = LoggingConfig::default();
|
|
|
|
logger.log_to_file = false;
|
|
|
|
logger.stdout_log_level = LogLevel::Debug;
|
2018-11-05 06:57:59 +03:00
|
|
|
|
|
|
|
// Save current logging configuration
|
2018-10-20 03:13:07 +03:00
|
|
|
let mut config_ref = LOGGING_CONFIG.lock();
|
2018-08-16 21:32:03 +03:00
|
|
|
*config_ref = logger;
|
2018-11-05 06:57:59 +03:00
|
|
|
|
|
|
|
let level_stdout = convert_log_level(&config_ref.stdout_log_level);
|
|
|
|
let level_minimum = level_stdout; // minimum logging level for Root logger
|
|
|
|
|
|
|
|
// Start logger
|
|
|
|
let stdout = ConsoleAppender::builder()
|
|
|
|
.encoder(Box::new(PatternEncoder::default()))
|
|
|
|
.build();
|
|
|
|
|
|
|
|
let mut root = Root::builder();
|
|
|
|
|
|
|
|
let mut appenders = vec![];
|
|
|
|
|
|
|
|
{
|
|
|
|
let filter = Box::new(ThresholdFilter::new(level_stdout));
|
|
|
|
appenders.push(
|
|
|
|
Appender::builder()
|
|
|
|
.filter(filter)
|
|
|
|
.filter(Box::new(GrinFilter))
|
|
|
|
.build("stdout", Box::new(stdout)),
|
|
|
|
);
|
|
|
|
|
|
|
|
root = root.appender("stdout");
|
|
|
|
}
|
|
|
|
|
|
|
|
let config = Config::builder()
|
|
|
|
.appenders(appenders)
|
|
|
|
.build(root.build(level_minimum))
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let _ = log4rs::init_config(config).unwrap();
|
|
|
|
|
|
|
|
info!(
|
|
|
|
"log4rs is initialized, stdout level: {:?}, min. level: {:?}",
|
|
|
|
level_stdout, level_minimum
|
|
|
|
);
|
|
|
|
|
2017-11-01 02:32:33 +03:00
|
|
|
*was_init_ref = true;
|
2018-03-14 21:22:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
/// hook to send panics to logs as well as stderr
|
|
|
|
fn send_panic_to_log() {
|
|
|
|
panic::set_hook(Box::new(|info| {
|
|
|
|
let backtrace = Backtrace::new();
|
|
|
|
|
|
|
|
let thread = thread::current();
|
|
|
|
let thread = thread.name().unwrap_or("unnamed");
|
|
|
|
|
|
|
|
let msg = match info.payload().downcast_ref::<&'static str>() {
|
|
|
|
Some(s) => *s,
|
|
|
|
None => match info.payload().downcast_ref::<String>() {
|
|
|
|
Some(s) => &**s,
|
|
|
|
None => "Box<Any>",
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
match info.location() {
|
|
|
|
Some(location) => {
|
|
|
|
error!(
|
2018-03-26 04:35:28 +03:00
|
|
|
"\nthread '{}' panicked at '{}': {}:{}{:?}\n\n",
|
2018-03-14 21:22:18 +03:00
|
|
|
thread,
|
|
|
|
msg,
|
|
|
|
location.file(),
|
|
|
|
location.line(),
|
|
|
|
backtrace
|
|
|
|
);
|
|
|
|
}
|
2018-10-21 23:30:56 +03:00
|
|
|
None => error!("thread '{}' panicked at '{}'{:?}", thread, msg, backtrace),
|
2018-03-14 21:22:18 +03:00
|
|
|
}
|
|
|
|
//also print to stderr
|
2018-10-20 03:13:07 +03:00
|
|
|
let tui_running = TUI_RUNNING.lock().clone();
|
2018-03-19 22:23:58 +03:00
|
|
|
if !tui_running {
|
2018-10-24 19:13:57 +03:00
|
|
|
let config = LOGGING_CONFIG.lock();
|
|
|
|
|
2018-03-19 22:23:58 +03:00
|
|
|
eprintln!(
|
2018-10-24 19:13:57 +03:00
|
|
|
"Thread '{}' panicked with message:\n\"{}\"\nSee {} for further details.",
|
|
|
|
thread, msg, config.log_file_path
|
2018-03-19 22:23:58 +03:00
|
|
|
);
|
|
|
|
}
|
2018-03-14 21:22:18 +03:00
|
|
|
}));
|
2017-10-22 10:09:40 +03:00
|
|
|
}
|