2016-10-01 15:35:06 +00:00
|
|
|
//
|
|
|
|
// imag - the personal information management suite for the commandline
|
2018-02-07 01:48:53 +00:00
|
|
|
// Copyright (C) 2015-2018 Matthias Beyer <mail@beyermatthias.de> and contributors
|
2016-10-01 15:35:06 +00:00
|
|
|
//
|
|
|
|
// This library is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
|
|
// License as published by the Free Software Foundation; version
|
|
|
|
// 2.1 of the License.
|
|
|
|
//
|
|
|
|
// This library is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
// Lesser General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
|
|
// License along with this library; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
//
|
|
|
|
|
2016-02-04 17:02:50 +00:00
|
|
|
use std::io::Write;
|
|
|
|
use std::io::stderr;
|
2017-08-26 13:27:13 +00:00
|
|
|
use std::collections::BTreeMap;
|
2017-10-08 16:06:43 +00:00
|
|
|
use std::sync::Arc;
|
|
|
|
use std::sync::Mutex;
|
|
|
|
use std::ops::Deref;
|
2016-02-04 17:02:50 +00:00
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
use error::RuntimeErrorKind as EK;
|
2017-09-04 21:02:45 +00:00
|
|
|
use error::RuntimeError as RE;
|
|
|
|
use error::ResultExt;
|
2017-08-26 13:27:13 +00:00
|
|
|
use runtime::Runtime;
|
|
|
|
|
|
|
|
use clap::ArgMatches;
|
2017-12-06 17:08:25 +00:00
|
|
|
use log::{Log, Level, Record, Metadata};
|
2017-08-26 13:27:13 +00:00
|
|
|
use toml::Value;
|
|
|
|
use toml_query::read::TomlValueReadExt;
|
2018-01-12 15:31:13 +00:00
|
|
|
use toml_query::read::TomlValueReadTypeExt;
|
2017-08-26 20:49:30 +00:00
|
|
|
use handlebars::Handlebars;
|
2016-01-10 14:04:06 +00:00
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
type ModuleName = String;
|
2017-09-04 21:02:45 +00:00
|
|
|
type Result<T> = ::std::result::Result<T, RE>;
|
2017-08-26 13:27:13 +00:00
|
|
|
|
|
|
|
enum LogDestination {
|
|
|
|
Stderr,
|
2017-10-08 16:06:43 +00:00
|
|
|
File(Arc<Mutex<::std::fs::File>>),
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Default for LogDestination {
|
|
|
|
fn default() -> LogDestination {
|
|
|
|
LogDestination::Stderr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ModuleSettings {
|
2017-08-28 09:40:50 +00:00
|
|
|
enabled: bool,
|
2017-12-06 17:08:25 +00:00
|
|
|
level: Option<Level>,
|
2016-01-10 14:04:06 +00:00
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
#[allow(unused)]
|
2017-08-28 09:40:50 +00:00
|
|
|
destinations: Option<Vec<LogDestination>>,
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
2016-08-02 09:38:38 +00:00
|
|
|
|
2017-02-21 14:26:10 +00:00
|
|
|
/// Logger implementation for `log` crate.
|
2016-01-10 14:04:06 +00:00
|
|
|
pub struct ImagLogger {
|
2017-12-06 17:08:25 +00:00
|
|
|
global_loglevel : Level,
|
2017-08-26 13:27:13 +00:00
|
|
|
|
|
|
|
#[allow(unused)]
|
|
|
|
global_destinations : Vec<LogDestination>,
|
|
|
|
// global_format_trace : ,
|
|
|
|
// global_format_debug : ,
|
|
|
|
// global_format_info : ,
|
|
|
|
// global_format_warn : ,
|
|
|
|
// global_format_error : ,
|
|
|
|
module_settings : BTreeMap<ModuleName, ModuleSettings>,
|
2017-08-26 20:49:30 +00:00
|
|
|
|
|
|
|
handlebars: Handlebars,
|
2016-01-10 14:04:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl ImagLogger {
|
|
|
|
|
2017-02-21 14:26:10 +00:00
|
|
|
/// Create a new ImagLogger object with a certain level
|
2017-10-28 10:50:22 +00:00
|
|
|
pub fn new(matches: &ArgMatches, config: Option<&Value>) -> Result<ImagLogger> {
|
2017-08-26 20:49:30 +00:00
|
|
|
let mut handlebars = Handlebars::new();
|
|
|
|
|
2017-09-20 17:12:06 +00:00
|
|
|
handlebars.register_escape_fn(::handlebars::no_escape);
|
|
|
|
|
2017-10-12 14:55:03 +00:00
|
|
|
::libimaginteraction::format::register_all_color_helpers(&mut handlebars);
|
|
|
|
::libimaginteraction::format::register_all_format_helpers(&mut handlebars);
|
2017-08-30 19:16:31 +00:00
|
|
|
|
2017-08-26 20:49:30 +00:00
|
|
|
{
|
2017-10-21 14:17:35 +00:00
|
|
|
let fmt = aggregate_global_format_trace(config)?;
|
|
|
|
handlebars.register_template_string("TRACE", fmt)?; // name must be uppercase
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
|
|
|
{
|
2017-10-21 14:17:35 +00:00
|
|
|
let fmt = aggregate_global_format_debug(config)?;
|
|
|
|
handlebars.register_template_string("DEBUG", fmt)?; // name must be uppercase
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
|
|
|
{
|
2017-10-21 14:17:35 +00:00
|
|
|
let fmt = aggregate_global_format_info(config)?;
|
|
|
|
handlebars.register_template_string("INFO", fmt)?; // name must be uppercase
|
2016-08-02 09:38:38 +00:00
|
|
|
}
|
2017-08-26 20:49:30 +00:00
|
|
|
{
|
2017-10-21 14:17:35 +00:00
|
|
|
let fmt = aggregate_global_format_warn(config)?;
|
|
|
|
handlebars.register_template_string("WARN", fmt)?; // name must be uppercase
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
|
|
|
{
|
2017-10-21 14:17:35 +00:00
|
|
|
let fmt = aggregate_global_format_error(config)?;
|
|
|
|
handlebars.register_template_string("ERROR", fmt)?; // name must be uppercase
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
Ok(ImagLogger {
|
2017-10-21 14:17:35 +00:00
|
|
|
global_loglevel : aggregate_global_loglevel(matches, config)?,
|
|
|
|
global_destinations : aggregate_global_destinations(matches, config)?,
|
|
|
|
module_settings : aggregate_module_settings(matches, config)?,
|
2017-08-26 20:49:30 +00:00
|
|
|
handlebars : handlebars,
|
2017-08-26 13:27:13 +00:00
|
|
|
})
|
2016-08-02 09:38:38 +00:00
|
|
|
}
|
|
|
|
|
2017-12-06 17:08:25 +00:00
|
|
|
pub fn global_loglevel(&self) -> Level {
|
2017-08-26 13:27:13 +00:00
|
|
|
self.global_loglevel
|
2016-08-02 09:38:38 +00:00
|
|
|
}
|
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Log for ImagLogger {
|
|
|
|
|
2017-12-06 17:08:25 +00:00
|
|
|
fn enabled(&self, metadata: &Metadata) -> bool {
|
2017-08-26 13:27:13 +00:00
|
|
|
metadata.level() <= self.global_loglevel
|
2017-01-21 11:56:59 +00:00
|
|
|
}
|
|
|
|
|
2017-12-06 17:08:25 +00:00
|
|
|
fn flush(&self) {
|
|
|
|
// nothing?
|
|
|
|
}
|
|
|
|
|
|
|
|
fn log(&self, record: &Record) {
|
|
|
|
if record.module_path().map(|m| m.starts_with("handlebars")).unwrap_or(false) {
|
2017-08-30 18:56:24 +00:00
|
|
|
// This is a ugly, yet necessary hack. When logging, we use handlebars for templating.
|
|
|
|
// But as the handlebars library itselfs logs via a normal logging macro ("debug!()"),
|
|
|
|
// we have a recursion in our chain.
|
|
|
|
//
|
|
|
|
// To prevent this recursion, we return here.
|
|
|
|
//
|
|
|
|
// (As of handlebars 0.29.0 - please check whether you can update handlebars if you see
|
|
|
|
// this. Hopefully the next version has a compiletime flag to disable logging)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-08-26 20:49:30 +00:00
|
|
|
let mut data = BTreeMap::new();
|
2017-08-28 10:09:23 +00:00
|
|
|
|
2017-08-26 20:49:30 +00:00
|
|
|
{
|
2017-08-28 10:09:23 +00:00
|
|
|
data.insert("level", format!("{}", record.level()));
|
2017-12-06 17:08:25 +00:00
|
|
|
data.insert("module_path", String::from(record.module_path().unwrap_or("<modulepath unknown>")));
|
|
|
|
data.insert("file", String::from(record.file().unwrap_or("<file unknown>")));
|
|
|
|
data.insert("line", format!("{}", record.line().unwrap_or(0)));
|
2017-08-28 10:09:23 +00:00
|
|
|
data.insert("target", String::from(record.target()));
|
|
|
|
data.insert("message", format!("{}", record.args()));
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
2017-08-28 10:09:23 +00:00
|
|
|
|
2017-08-26 20:49:30 +00:00
|
|
|
let logtext = self
|
|
|
|
.handlebars
|
2017-08-28 10:09:23 +00:00
|
|
|
.render(&format!("{}", record.level()), &data)
|
2017-08-26 20:49:30 +00:00
|
|
|
.unwrap_or_else(|e| format!("Failed rendering logging data: {:?}\n", e));
|
|
|
|
|
2017-10-08 16:06:43 +00:00
|
|
|
let log_to_destination = |d: &LogDestination| match d {
|
|
|
|
&LogDestination::Stderr => {
|
|
|
|
let _ = write!(stderr(), "{}\n", logtext);
|
|
|
|
},
|
|
|
|
&LogDestination::File(ref arc_mutex_logdest) => {
|
|
|
|
// if there is an error in the lock, we cannot do anything. So we ignore it here.
|
|
|
|
let _ = arc_mutex_logdest
|
|
|
|
.deref()
|
|
|
|
.lock()
|
|
|
|
.map(|mut logdest| {
|
|
|
|
write!(logdest, "{}\n", logtext)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// hack to get the right target configuration.
|
|
|
|
// If there is no element here, we use the empty string which automatically drops through
|
|
|
|
// to the unwrap_or_else() case
|
|
|
|
let record_target = record
|
|
|
|
.target()
|
|
|
|
.split("::")
|
|
|
|
.next()
|
|
|
|
.unwrap_or("");
|
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
self.module_settings
|
2017-10-08 16:06:43 +00:00
|
|
|
.get(record_target)
|
2017-08-26 13:27:13 +00:00
|
|
|
.map(|module_setting| {
|
2017-08-28 10:09:23 +00:00
|
|
|
let set = module_setting.enabled &&
|
|
|
|
module_setting.level.unwrap_or(self.global_loglevel) >= record.level();
|
|
|
|
|
|
|
|
if set {
|
2017-10-08 16:06:43 +00:00
|
|
|
module_setting.destinations.as_ref().map(|destinations| for d in destinations {
|
|
|
|
// If there's an error, we cannot do anything, can we?
|
|
|
|
let _ = log_to_destination(&d);
|
|
|
|
});
|
|
|
|
|
|
|
|
for d in self.global_destinations.iter() {
|
|
|
|
// If there's an error, we cannot do anything, can we?
|
|
|
|
let _ = log_to_destination(&d);
|
|
|
|
}
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
|
|
|
})
|
2017-10-08 16:06:43 +00:00
|
|
|
.unwrap_or_else(|| {
|
|
|
|
if self.global_loglevel >= record.level() {
|
|
|
|
// Yes, we log
|
|
|
|
for d in self.global_destinations.iter() {
|
|
|
|
// If there's an error, we cannot do anything, can we?
|
|
|
|
let _ = log_to_destination(&d);
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
2017-10-08 16:06:43 +00:00
|
|
|
}
|
|
|
|
});
|
2017-01-21 11:37:11 +00:00
|
|
|
}
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
2017-01-21 11:37:11 +00:00
|
|
|
|
2017-12-06 17:08:25 +00:00
|
|
|
fn match_log_level_str(s: &str) -> Result<Level> {
|
2017-08-26 13:27:13 +00:00
|
|
|
match s {
|
2017-12-06 17:08:25 +00:00
|
|
|
"trace" => Ok(Level::Trace),
|
|
|
|
"debug" => Ok(Level::Debug),
|
|
|
|
"info" => Ok(Level::Info),
|
|
|
|
"warn" => Ok(Level::Warn),
|
|
|
|
"error" => Ok(Level::Error),
|
2017-09-04 21:02:45 +00:00
|
|
|
_ => return Err(RE::from_kind(EK::InvalidLogLevelSpec)),
|
2016-08-02 09:38:38 +00:00
|
|
|
}
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
2016-08-02 09:38:38 +00:00
|
|
|
|
2017-12-06 17:08:25 +00:00
|
|
|
fn aggregate_global_loglevel(matches: &ArgMatches, config: Option<&Value>) -> Result<Level>
|
2017-08-26 13:27:13 +00:00
|
|
|
{
|
2017-12-06 17:08:25 +00:00
|
|
|
fn get_arg_loglevel(matches: &ArgMatches) -> Result<Option<Level>> {
|
2017-10-15 17:45:34 +00:00
|
|
|
if matches.is_present(Runtime::arg_debugging_name()) {
|
2017-12-06 17:08:25 +00:00
|
|
|
return Ok(Some(Level::Debug))
|
2017-10-15 17:45:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if matches.is_present(Runtime::arg_verbosity_name()) {
|
2017-12-06 17:08:25 +00:00
|
|
|
return Ok(Some(Level::Info))
|
2017-10-15 17:45:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
match matches.value_of(Runtime::arg_verbosity_name()) {
|
|
|
|
Some(v) => match_log_level_str(v).map(Some),
|
|
|
|
None => Ok(None),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(cfg) = config {
|
2018-01-04 22:08:14 +00:00
|
|
|
let cfg_loglevel = cfg
|
2018-01-12 15:31:13 +00:00
|
|
|
.read_string("imag.logging.level")?
|
|
|
|
.ok_or(RE::from_kind(EK::GlobalLogLevelConfigMissing))
|
|
|
|
.and_then(|s| match_log_level_str(&s))?;
|
2016-08-02 09:38:38 +00:00
|
|
|
|
2017-10-15 17:45:34 +00:00
|
|
|
if let Some(cli_loglevel) = get_arg_loglevel(matches)? {
|
|
|
|
if cli_loglevel > cfg_loglevel {
|
|
|
|
return Ok(cli_loglevel)
|
|
|
|
}
|
2016-08-02 09:38:38 +00:00
|
|
|
}
|
2017-10-15 17:45:34 +00:00
|
|
|
|
|
|
|
Ok(cfg_loglevel)
|
|
|
|
|
|
|
|
} else {
|
2017-12-06 17:08:25 +00:00
|
|
|
get_arg_loglevel(matches).map(|o| o.unwrap_or(Level::Info))
|
2016-08-02 09:38:38 +00:00
|
|
|
}
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn translate_destination(raw: &str) -> Result<LogDestination> {
|
|
|
|
use std::fs::OpenOptions;
|
2016-08-02 09:38:38 +00:00
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
match raw {
|
|
|
|
"-" => Ok(LogDestination::Stderr),
|
|
|
|
other => {
|
|
|
|
OpenOptions::new()
|
|
|
|
.append(true)
|
|
|
|
.create(true)
|
|
|
|
.open(other)
|
2017-10-08 16:06:43 +00:00
|
|
|
.map(Mutex::new)
|
|
|
|
.map(Arc::new)
|
2017-08-26 13:27:13 +00:00
|
|
|
.map(LogDestination::File)
|
2017-09-04 21:02:45 +00:00
|
|
|
.chain_err(|| EK::IOLogFileOpenError)
|
2016-01-10 14:04:06 +00:00
|
|
|
}
|
|
|
|
}
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
|
|
|
|
2016-01-10 14:04:06 +00:00
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
fn translate_destinations(raw: &Vec<Value>) -> Result<Vec<LogDestination>> {
|
|
|
|
raw.iter()
|
|
|
|
.fold(Ok(vec![]), |acc, val| {
|
|
|
|
acc.and_then(|mut v| {
|
2018-01-04 22:08:14 +00:00
|
|
|
let dest = val.as_str()
|
|
|
|
.ok_or_else(|| {
|
2017-09-09 20:24:58 +00:00
|
|
|
let path = "imag.logging.modules.<mod>.destinations".to_owned();
|
|
|
|
let ty = "Array<String>";
|
2018-01-04 22:08:14 +00:00
|
|
|
RE::from_kind(EK::ConfigTypeError(path, ty))
|
|
|
|
})
|
|
|
|
.and_then(translate_destination)?;
|
2017-08-26 13:27:13 +00:00
|
|
|
v.push(dest);
|
|
|
|
Ok(v)
|
|
|
|
})
|
|
|
|
})
|
2016-01-10 14:04:06 +00:00
|
|
|
}
|
|
|
|
|
2017-10-28 10:50:22 +00:00
|
|
|
fn aggregate_global_destinations(matches: &ArgMatches, config: Option<&Value>)
|
2017-08-26 13:27:13 +00:00
|
|
|
-> Result<Vec<LogDestination>>
|
|
|
|
{
|
2016-01-10 14:04:06 +00:00
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
match config {
|
2018-01-04 22:08:14 +00:00
|
|
|
Some(cfg) => cfg
|
|
|
|
.read("imag.logging.destinations")?
|
|
|
|
.ok_or_else(|| RE::from_kind(EK::GlobalDestinationConfigMissing))?
|
|
|
|
.as_array()
|
|
|
|
.ok_or_else(|| {
|
2017-09-09 20:24:58 +00:00
|
|
|
let path = "imag.logging.destinations".to_owned();
|
|
|
|
let ty = "Array";
|
2018-01-04 22:08:14 +00:00
|
|
|
RE::from_kind(EK::ConfigTypeError(path, ty))
|
|
|
|
})
|
|
|
|
.and_then(translate_destinations),
|
2017-08-26 13:27:13 +00:00
|
|
|
None => {
|
|
|
|
if let Some(values) = matches.value_of(Runtime::arg_logdest_name()) {
|
|
|
|
// parse logdest specification from commandline
|
|
|
|
|
|
|
|
values.split(",")
|
|
|
|
.fold(Ok(vec![]), move |acc, dest| {
|
|
|
|
acc.and_then(|mut v| {
|
2017-10-21 14:17:35 +00:00
|
|
|
v.push(translate_destination(dest)?);
|
2017-08-26 13:27:13 +00:00
|
|
|
Ok(v)
|
|
|
|
})
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
Ok(vec![ LogDestination::default() ])
|
|
|
|
}
|
|
|
|
}
|
2016-01-10 14:04:06 +00:00
|
|
|
}
|
2017-08-26 13:27:13 +00:00
|
|
|
}
|
2016-01-10 14:04:06 +00:00
|
|
|
|
2017-10-15 17:27:56 +00:00
|
|
|
macro_rules! aggregate_global_format {
|
|
|
|
($read_str:expr, $error_kind_if_missing:expr, $config:expr) => {
|
2018-01-04 22:08:14 +00:00
|
|
|
try!($config.ok_or(RE::from_kind($error_kind_if_missing)))
|
2018-01-12 15:31:13 +00:00
|
|
|
.read_string($read_str)?
|
|
|
|
.ok_or_else(|| RE::from_kind($error_kind_if_missing))
|
2017-10-15 17:27:56 +00:00
|
|
|
};
|
2017-08-26 21:00:48 +00:00
|
|
|
}
|
2016-01-10 14:04:06 +00:00
|
|
|
|
2017-10-28 10:50:22 +00:00
|
|
|
fn aggregate_global_format_trace(config: Option<&Value>)
|
2017-08-26 20:49:30 +00:00
|
|
|
-> Result<String>
|
|
|
|
{
|
2017-10-15 17:27:56 +00:00
|
|
|
aggregate_global_format!("imag.logging.format.trace",
|
2017-08-26 21:00:48 +00:00
|
|
|
EK::ConfigMissingLoggingFormatTrace,
|
|
|
|
config)
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
2016-05-15 15:38:02 +00:00
|
|
|
|
2017-10-28 10:50:22 +00:00
|
|
|
fn aggregate_global_format_debug(config: Option<&Value>)
|
2017-08-26 20:49:30 +00:00
|
|
|
-> Result<String>
|
|
|
|
{
|
2017-10-15 17:27:56 +00:00
|
|
|
aggregate_global_format!("imag.logging.format.debug",
|
2017-08-26 21:00:48 +00:00
|
|
|
EK::ConfigMissingLoggingFormatDebug,
|
|
|
|
config)
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
|
|
|
|
2017-10-28 10:50:22 +00:00
|
|
|
fn aggregate_global_format_info(config: Option<&Value>)
|
2017-08-26 20:49:30 +00:00
|
|
|
-> Result<String>
|
|
|
|
{
|
2017-10-15 17:27:56 +00:00
|
|
|
aggregate_global_format!("imag.logging.format.info",
|
2017-08-26 21:00:48 +00:00
|
|
|
EK::ConfigMissingLoggingFormatInfo,
|
|
|
|
config)
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
|
|
|
|
2017-10-28 10:50:22 +00:00
|
|
|
fn aggregate_global_format_warn(config: Option<&Value>)
|
2017-08-26 20:49:30 +00:00
|
|
|
-> Result<String>
|
|
|
|
{
|
2017-10-15 17:27:56 +00:00
|
|
|
aggregate_global_format!("imag.logging.format.warn",
|
2017-08-26 21:00:48 +00:00
|
|
|
EK::ConfigMissingLoggingFormatWarn,
|
|
|
|
config)
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
|
|
|
|
2017-10-28 10:50:22 +00:00
|
|
|
fn aggregate_global_format_error(config: Option<&Value>)
|
2017-08-26 20:49:30 +00:00
|
|
|
-> Result<String>
|
|
|
|
{
|
2017-10-15 17:27:56 +00:00
|
|
|
aggregate_global_format!("imag.logging.format.error",
|
2017-08-26 21:00:48 +00:00
|
|
|
EK::ConfigMissingLoggingFormatError,
|
|
|
|
config)
|
2017-08-26 20:49:30 +00:00
|
|
|
}
|
2016-05-15 15:38:02 +00:00
|
|
|
|
2017-10-28 10:50:22 +00:00
|
|
|
fn aggregate_module_settings(_matches: &ArgMatches, config: Option<&Value>)
|
2017-08-26 13:27:13 +00:00
|
|
|
-> Result<BTreeMap<ModuleName, ModuleSettings>>
|
|
|
|
{
|
2018-01-04 22:08:14 +00:00
|
|
|
// Helper macro to return the error from Some(Err(_)) and map everything else to an
|
|
|
|
// Option<_>
|
|
|
|
macro_rules! inner_try {
|
|
|
|
($v:expr) => {
|
|
|
|
match $v {
|
|
|
|
Some(Ok(v)) => Some(v),
|
|
|
|
Some(Err(e)) => return Err(e),
|
|
|
|
None => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
match config {
|
2017-09-09 20:15:18 +00:00
|
|
|
Some(cfg) => match cfg.read("imag.logging.modules") {
|
|
|
|
Ok(Some(&Value::Table(ref t))) => {
|
|
|
|
// translate the module settings from the table `t`
|
|
|
|
let mut settings = BTreeMap::new();
|
|
|
|
|
|
|
|
for (module_name, v) in t {
|
2018-01-04 22:08:14 +00:00
|
|
|
let destinations = inner_try! {
|
|
|
|
v.read("destinations")?
|
|
|
|
.map(|val| {
|
|
|
|
val.as_array()
|
|
|
|
.ok_or_else(|| {
|
|
|
|
let path = "imag.logging.modules.<mod>.destinations".to_owned();
|
|
|
|
let ty = "Array";
|
|
|
|
RE::from_kind(EK::ConfigTypeError(path, ty))
|
|
|
|
})
|
|
|
|
.and_then(translate_destinations)
|
|
|
|
})
|
2017-10-21 14:45:57 +00:00
|
|
|
};
|
2017-09-09 20:15:18 +00:00
|
|
|
|
2018-01-04 22:08:14 +00:00
|
|
|
let level = inner_try! {
|
2018-01-12 15:31:13 +00:00
|
|
|
v.read_string("level")?.map(|s| match_log_level_str(&s))
|
2017-10-21 14:45:57 +00:00
|
|
|
};
|
2017-09-09 20:15:18 +00:00
|
|
|
|
2018-01-04 22:08:14 +00:00
|
|
|
let enabled = v.read("enabled")?
|
|
|
|
.map(|v| v.as_bool().unwrap_or(false))
|
|
|
|
.ok_or_else(|| {
|
2017-09-09 20:24:58 +00:00
|
|
|
let path = "imag.logging.modules.<mod>.enabled".to_owned();
|
|
|
|
let ty = "Boolean";
|
2018-01-04 22:08:14 +00:00
|
|
|
RE::from_kind(EK::ConfigTypeError(path, ty))
|
|
|
|
})?;
|
2017-09-09 20:15:18 +00:00
|
|
|
|
|
|
|
let module_settings = ModuleSettings {
|
|
|
|
enabled: enabled,
|
|
|
|
level: level,
|
|
|
|
destinations: destinations,
|
|
|
|
};
|
|
|
|
|
|
|
|
// We don't care whether there was a value, we override it.
|
|
|
|
let _ = settings.insert(module_name.to_owned(), module_settings);
|
|
|
|
}
|
|
|
|
|
|
|
|
Ok(settings)
|
|
|
|
},
|
2017-09-09 20:24:58 +00:00
|
|
|
Ok(Some(_)) => {
|
|
|
|
let path = "imag.logging.modules".to_owned();
|
|
|
|
let ty = "Table";
|
|
|
|
Err(RE::from_kind(EK::ConfigTypeError(path, ty)))
|
|
|
|
},
|
2017-09-09 20:15:18 +00:00
|
|
|
Ok(None) => {
|
|
|
|
// No modules configured. This is okay!
|
|
|
|
Ok(BTreeMap::new())
|
2017-08-26 13:27:13 +00:00
|
|
|
},
|
2017-09-09 20:15:18 +00:00
|
|
|
Err(e) => Err(e).map_err(From::from),
|
|
|
|
},
|
2017-08-26 13:27:13 +00:00
|
|
|
None => {
|
|
|
|
write!(stderr(), "No Configuration.").ok();
|
|
|
|
write!(stderr(), "cannot find module-settings for logging.").ok();
|
|
|
|
write!(stderr(), "Will use global defaults").ok();
|
2016-05-15 15:38:02 +00:00
|
|
|
|
2017-08-26 13:27:13 +00:00
|
|
|
Ok(BTreeMap::new())
|
2016-01-10 14:04:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|