2020-09-18 16:12:53 -04:00
|
|
|
#include "Logger.hpp"
|
|
|
|
|
|
|
|
#include <boost/algorithm/string/predicate.hpp>
|
|
|
|
#include <boost/date_time/posix_time/posix_time.hpp>
|
|
|
|
#include <boost/log/core.hpp>
|
|
|
|
#include <boost/log/common.hpp>
|
|
|
|
#include <boost/log/sinks.hpp>
|
|
|
|
#include <boost/log/expressions.hpp>
|
|
|
|
#include <boost/log/expressions/keyword.hpp>
|
|
|
|
#include <boost/log/attributes.hpp>
|
|
|
|
#include <boost/log/utility/setup/console.hpp>
|
|
|
|
#include <boost/log/utility/setup/common_attributes.hpp>
|
|
|
|
#include <boost/log/utility/setup/filter_parser.hpp>
|
|
|
|
#include <boost/log/utility/setup/from_stream.hpp>
|
|
|
|
#include <boost/log/utility/setup/settings.hpp>
|
|
|
|
#include <boost/log/sinks/sync_frontend.hpp>
|
|
|
|
#include <boost/log/sinks/text_ostream_backend.hpp>
|
|
|
|
#include <boost/log/sinks/debug_output_backend.hpp>
|
|
|
|
#include <boost/log/support/date_time.hpp>
|
|
|
|
#include <boost/shared_ptr.hpp>
|
|
|
|
#include <boost/make_shared.hpp>
|
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
#include <string>
|
|
|
|
|
2020-09-24 13:55:00 -04:00
|
|
|
namespace logging = boost::log;
|
|
|
|
namespace srcs = logging::sources;
|
|
|
|
namespace sinks = logging::sinks;
|
|
|
|
namespace keywords = logging::keywords;
|
|
|
|
namespace expr = logging::expressions;
|
|
|
|
namespace attrs = logging::attributes;
|
2020-09-18 16:12:53 -04:00
|
|
|
namespace ptime = boost::posix_time;
|
|
|
|
|
2020-10-01 18:32:33 -04:00
|
|
|
BOOST_LOG_GLOBAL_LOGGER_CTOR_ARGS (sys,
|
|
|
|
srcs::severity_channel_logger_mt<logging::trivial::severity_level>,
|
|
|
|
(keywords::channel = "SYSLOG"));
|
|
|
|
BOOST_LOG_GLOBAL_LOGGER_CTOR_ARGS (data,
|
|
|
|
srcs::severity_channel_logger_mt<logging::trivial::severity_level>,
|
|
|
|
(keywords::channel = "DATALOG"));
|
|
|
|
|
2020-09-18 16:12:53 -04:00
|
|
|
namespace Logger
|
|
|
|
{
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
// Custom formatter factory to add TimeStamp format support in config ini file.
|
|
|
|
// Allows %TimeStamp(format=\"%Y.%m.%d %H:%M:%S.%f\")% to be used in ini config file for property Format.
|
|
|
|
class TimeStampFormatterFactory
|
2020-09-24 13:55:00 -04:00
|
|
|
: public logging::basic_formatter_factory<char, ptime::ptime>
|
2020-09-18 16:12:53 -04:00
|
|
|
{
|
|
|
|
public:
|
2020-09-24 13:55:00 -04:00
|
|
|
formatter_type create_formatter (logging::attribute_name const& name, args_map const& args)
|
2020-09-18 16:12:53 -04:00
|
|
|
{
|
|
|
|
args_map::const_iterator it = args.find ("format");
|
|
|
|
if (it != args.end ())
|
|
|
|
{
|
|
|
|
return expr::stream
|
|
|
|
<< expr::format_date_time<ptime::ptime>
|
|
|
|
(
|
|
|
|
expr::attr<ptime::ptime> (name), it->second
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return expr::stream
|
|
|
|
<< expr::attr<ptime::ptime> (name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Custom formatter factory to add Uptime format support in config ini file.
|
|
|
|
// Allows %Uptime(format=\"%O:%M:%S.%f\")% to be used in ini config file for property Format.
|
|
|
|
// attrs::timer value type is ptime::time_duration
|
|
|
|
class UptimeFormatterFactory
|
2020-09-24 13:55:00 -04:00
|
|
|
: public logging::basic_formatter_factory<char, ptime::time_duration>
|
2020-09-18 16:12:53 -04:00
|
|
|
{
|
|
|
|
public:
|
2020-09-24 13:55:00 -04:00
|
|
|
formatter_type create_formatter (logging::attribute_name const& name, args_map const& args)
|
2020-09-18 16:12:53 -04:00
|
|
|
{
|
|
|
|
args_map::const_iterator it = args.find ("format");
|
|
|
|
if (it != args.end ())
|
|
|
|
{
|
|
|
|
return expr::stream
|
|
|
|
<< expr::format_date_time<ptime::time_duration>
|
|
|
|
(
|
|
|
|
expr::attr<ptime::time_duration> (name), it->second
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return expr::stream
|
|
|
|
<< expr::attr<ptime::time_duration> (name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class CommonInitialization
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CommonInitialization ()
|
|
|
|
{
|
|
|
|
// Add common attributes: LineID, TimeStamp, ProcessID, ThreadID
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::add_common_attributes ();
|
2020-09-18 16:12:53 -04:00
|
|
|
// Add boost log timer as global attribute Uptime
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::core::get ()->add_global_attribute ("Uptime", attrs::timer ());
|
2020-09-18 16:12:53 -04:00
|
|
|
// Allows %Severity% to be used in ini config file for property Filter.
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::register_simple_filter_factory<logging::trivial::severity_level, char> ("Severity");
|
2020-09-18 16:12:53 -04:00
|
|
|
// Allows %Severity% to be used in ini config file for property Format.
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::register_simple_formatter_factory<logging::trivial::severity_level, char> ("Severity");
|
2020-09-18 16:12:53 -04:00
|
|
|
// Allows %TimeStamp(format=\"%Y.%m.%d %H:%M:%S.%f\")% to be used in ini config file for property Format.
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::register_formatter_factory ("TimeStamp", boost::make_shared<TimeStampFormatterFactory> ());
|
2020-09-18 16:12:53 -04:00
|
|
|
// Allows %Uptime(format=\"%O:%M:%S.%f\")% to be used in ini config file for property Format.
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::register_formatter_factory ("Uptime", boost::make_shared<UptimeFormatterFactory> ());
|
2020-09-18 16:12:53 -04:00
|
|
|
}
|
|
|
|
~CommonInitialization ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
void init ()
|
|
|
|
{
|
|
|
|
CommonInitialization ci;
|
|
|
|
}
|
|
|
|
|
|
|
|
void init_from_config (std::istream& stream)
|
|
|
|
{
|
|
|
|
CommonInitialization ci;
|
|
|
|
try
|
|
|
|
{
|
|
|
|
// Still can throw even with the exception suppressor above.
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::init_from_stream (stream);
|
2020-09-18 16:12:53 -04:00
|
|
|
}
|
|
|
|
catch (std::exception& e)
|
|
|
|
{
|
|
|
|
std::string err = "Caught exception initializing boost logging: ";
|
|
|
|
err += e.what ();
|
|
|
|
// Since we cannot be sure of boost log state, output to cerr and cout.
|
|
|
|
std::cerr << "ERROR: " << err << std::endl;
|
|
|
|
std::cout << "ERROR: " << err << std::endl;
|
|
|
|
LOG_ERROR (err);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void disable ()
|
|
|
|
{
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::core::get ()->set_logging_enabled (false);
|
2020-09-18 16:12:53 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void add_datafile_log (std::string const& log_file_name)
|
|
|
|
{
|
|
|
|
// Create a text file sink
|
2020-09-27 11:52:19 -04:00
|
|
|
boost::shared_ptr<sinks::text_ostream_backend> backend
|
2020-09-18 16:12:53 -04:00
|
|
|
(
|
2020-09-27 11:52:19 -04:00
|
|
|
new sinks::text_ostream_backend()
|
2020-09-18 16:12:53 -04:00
|
|
|
);
|
2020-09-27 11:52:19 -04:00
|
|
|
backend->add_stream (boost::shared_ptr<std::ostream> (new std::ofstream (log_file_name)));
|
2020-09-18 16:12:53 -04:00
|
|
|
|
|
|
|
// Flush after each log record
|
|
|
|
backend->auto_flush (true);
|
|
|
|
|
|
|
|
// Create a sink for the backend
|
2020-09-27 11:52:19 -04:00
|
|
|
typedef sinks::synchronous_sink<sinks::text_ostream_backend> sink_t;
|
2020-09-18 16:12:53 -04:00
|
|
|
boost::shared_ptr<sink_t> sink (new sink_t (backend));
|
|
|
|
|
|
|
|
// The log output formatter
|
2020-09-27 11:52:19 -04:00
|
|
|
sink->set_formatter (expr::format ("[%1%][%2%] %3%")
|
2020-09-18 16:12:53 -04:00
|
|
|
% expr::attr<ptime::ptime> ("TimeStamp")
|
2020-09-24 13:55:00 -04:00
|
|
|
% logging::trivial::severity
|
2020-09-26 09:15:44 -04:00
|
|
|
% expr::message
|
2020-09-18 16:12:53 -04:00
|
|
|
);
|
|
|
|
|
|
|
|
// Filter by severity and by DATALOG channel
|
2020-09-24 13:55:00 -04:00
|
|
|
sink->set_filter (logging::trivial::severity >= logging::trivial::info &&
|
2020-09-18 16:12:53 -04:00
|
|
|
expr::attr<std::string> ("Channel") == "DATALOG");
|
|
|
|
|
|
|
|
// Add it to the core
|
2020-09-24 13:55:00 -04:00
|
|
|
logging::core::get ()->add_sink (sink);
|
2020-09-18 16:12:53 -04:00
|
|
|
}
|
|
|
|
}
|