Files
grim/src/logger.rs
T
2026-05-21 00:56:28 +03:00

169 lines
4.6 KiB
Rust

// Copyright 2026 The Grim Developers
//
// 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.
use backtrace::Backtrace;
use log::{LevelFilter, Record, error};
use log4rs::Config;
use log4rs::append::Append;
use log4rs::append::console::ConsoleAppender;
use log4rs::append::rolling_file::RollingFileAppender;
use log4rs::append::rolling_file::policy::compound::CompoundPolicy;
use log4rs::append::rolling_file::policy::compound::roll::fixed_window::FixedWindowRoller;
use log4rs::append::rolling_file::policy::compound::trigger::size::SizeTrigger;
use log4rs::config::{Appender, Root};
use log4rs::encode::pattern::PatternEncoder;
use log4rs::filter::threshold::ThresholdFilter;
use log4rs::filter::{Filter, Response};
use std::fs::File;
use std::{panic, thread};
use crate::Settings;
const LOGGING_PATTERN: &str = "{d(%Y%m%d %H:%M:%S%.3f)} {h({l})} {M} - {m}{n}";
/// 32 log files to rotate over by default.
const ROTATE_LOG_FILES: u32 = 32;
/// Size of the log in bytes to rotate over (6 megabytes).
const MAX_FILE_SIZE: u64 = 1024 * 1024 * 6;
/// Include build information.
pub mod built_info {
include!(concat!(env!("OUT_DIR"), "/built.rs"));
}
/// Filter is rejecting messages that doesn't start with "grin" or "grim"
#[derive(Debug)]
struct AppFilter;
impl Filter for AppFilter {
fn filter(&self, record: &Record<'_>) -> Response {
if let Some(module_path) = record.module_path() {
if module_path.starts_with("grin")
|| module_path.starts_with("grim")
|| module_path.starts_with("arti")
{
return Response::Neutral;
}
}
Response::Reject
}
}
/// Initialize the logger.
pub fn init_logger() {
let stdout = ConsoleAppender::builder()
.encoder(Box::new(PatternEncoder::new(&LOGGING_PATTERN)))
.build();
let mut root = Root::builder();
let level_filter = Box::new(ThresholdFilter::new(LevelFilter::Debug));
let mut app = vec![];
app.push(
Appender::builder()
.filter(level_filter.clone())
.filter(Box::new(AppFilter))
.build("stdout", Box::new(stdout)),
);
root = root.appender("stdout");
// Setup file logging.
let file: Box<dyn Append> = {
let path = Settings::log_path();
let roller = FixedWindowRoller::builder()
.build(&format!("{}.{{}}.gz", path), ROTATE_LOG_FILES)
.unwrap();
let trigger = SizeTrigger::new(MAX_FILE_SIZE);
let policy = CompoundPolicy::new(Box::new(trigger), Box::new(roller));
Box::new(
RollingFileAppender::builder()
.append(true)
.encoder(Box::new(PatternEncoder::new(&LOGGING_PATTERN)))
.build(path, Box::new(policy))
.expect("Failed to create logfile"),
)
};
app.push(
Appender::builder()
.filter(level_filter)
.filter(Box::new(AppFilter))
.build("file", file),
);
root = root.appender("file");
let config = Config::builder()
.appenders(app)
.build(root.build(LevelFilter::Debug))
.unwrap();
let _ = log4rs::init_config(config).unwrap();
log::info!("{}", build_info());
send_panic_to_log();
}
/// Get information about application build.
fn build_info() -> String {
format!(
"This is Grim version {}, built for {} by {}.",
built_info::PKG_VERSION,
built_info::TARGET,
built_info::RUSTC_VERSION,
)
}
/// 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!(
"{}\nThread '{}' panicked at '{}': {}:{}{:?}\n\n",
build_info(),
thread,
msg,
location.file(),
location.line(),
backtrace
);
}
None => error!("Thread '{}' panicked at '{}'{:?}", thread, msg, backtrace),
}
// Also print to stderr.
eprintln!(
"Thread '{}' panicked with message:\n\"{}\"\nSee {} for further details.",
thread,
msg,
Settings::log_path()
);
// Create file to show report send on launch.
let log = Settings::crash_check_path();
let _ = File::create(log);
}));
}