我使用升压日志2.0,其中有从版本1的一些差异,我也很难输出“严重性”属性。
我使用的是“Boost.Format库式”格式化
"%TimeStamp% [%Uptime%] (%LineID%) <%Severity%>: %Message%"
TimeStamp
, LineID
,并且Message
是common_attributes
。 Uptime
是我添加使用属性attrs::timer()
我认为, Severity
使用时自动添加severity_logger
,但显然它不是,这就是我的问题。 我得到空的严重性,例如:
2013-Apr-06 19:21:52.408974 [00:00:00.001337] (3) <>: A warning severity message
注意空<>
我试着用添加严重性register_simple_formatter_factory
但后来我得到的编译器错误:
error: no matching function for call to ‘register_simple_formatter_factory(const char [9])’
我不知道为什么。
这里是我的代码:
#include <iostream>
#include <boost/log/common.hpp>
#include <boost/log/core.hpp>
#include <boost/log/trivial.hpp>
#include <boost/log/expressions.hpp>
#include <boost/log/sinks/text_file_backend.hpp>
#include <boost/log/sinks/sync_frontend.hpp>
#include <boost/log/sinks/text_ostream_backend.hpp>
#include <boost/log/utility/setup/file.hpp>
#include <boost/log/utility/setup/common_attributes.hpp>
#include <boost/log/utility/setup/formatter_parser.hpp>
#include <boost/log/sources/severity_logger.hpp>
#include <boost/log/sources/severity_feature.hpp>
#include <boost/log/sources/record_ostream.hpp>
#include <boost/log/attributes.hpp>
using namespace std;
namespace logging = boost::log;
namespace expr = boost::log::expressions;
namespace attrs = boost::log::attributes;
namespace src = boost::log::sources;
namespace keywords = boost::log::keywords;
enum severity_level
{
DEBUG,
INFO,
WARNING,
ERROR,
CRITICAL
};
BOOST_LOG_INLINE_GLOBAL_LOGGER_DEFAULT(my_logger, src::severity_logger_mt< severity_level> )
// The formatting logic for the severity level
template< typename CharT, typename TraitsT >
inline std::basic_ostream< CharT, TraitsT >& operator<< (
std::basic_ostream< CharT, TraitsT >& strm, severity_level lvl)
{
static const char* const str[] =
{
"DEBUG",
"INFO",
"WARNING",
"ERROR",
"CRITICAL"
};
if (static_cast< std::size_t >(lvl) < (sizeof(str) / sizeof(*str)))
strm << str[lvl];
else
strm << static_cast< int >(lvl);
return strm;
}
void init() {
// logging::register_simple_formatter_factory< severity_level >("Severity");
logging::add_file_log(
keywords::file_name = "blop.log",
keywords::auto_flush = true,
keywords::open_mode = (std::ios::out | std::ios::app),
keywords::format = "%TimeStamp% [%Uptime%] (%LineID%) <%Severity%>: %Message%"
);
logging::add_common_attributes();
logging::core::get()->add_global_attribute("Uptime", attrs::timer());
}
int main(int argc, const char *argv[]) {
init();
src::severity_logger_mt< severity_level > lg = my_logger::get();
BOOST_LOG_SEV(lg, DEBUG) << "A debug severity message";
BOOST_LOG_SEV(lg, INFO) << "An informational severity message";
BOOST_LOG_SEV(lg, WARNING) << "A warning severity message";
BOOST_LOG_SEV(lg, ERROR) << "An error severity message";
BOOST_LOG_SEV(lg, CRITICAL) << "A critical severity message";
return 0;
}
注意注释掉的行:
// logging::register_simple_formatter_factory< severity_level >("Severity");
这产生所提到的错误。
我编译:
g++ main.cpp -Wall -DBOOST_ALL_DYN_LINK -lboost_system -lboost_log_setup -lboost_log -lboost_filesystem -lboost_date_time -lboost_thread -o main
遇到同样的问题(空白%严重性%的日志文件,并试图增加寄存器功能时相同的编译器错误)。 我发现,调用register_simple_formatter_factory应该是:
boost::log::register_simple_formatter_factory< boost::log::trivial::severity_level, char >("Severity");
这里的原型register_simple_formatter_factory
函数模板:
template<typename AttributeValueT, typename CharT>
void register_simple_formatter_factory(attribute_name const &);
这意味着你忘记typename CharT
,所以,这里是正确的:
logging::register_simple_formatter_factory<severity_level, char>("Severity");
或其它字符类型,如wchar_t
。
你可以让你自己的后端,并呼吁set_formatter就可以了。 这里是我的代码展示了如何做到这一点再加上一些补充,如日志文件的清理和自动清空:
BOOST_LOG_DECLARE_GLOBAL_LOGGER(db_logger, src::severity_logger_mt< dreambridge::log::LogLevel >);
#define LOGGER(level) BOOST_LOG_STREAM_SEV(db_logger::get(), level)
void InitializeFileLog(const std::string & logDir)
{
boost::shared_ptr< logging::core > loggingCore = logging::core::get();
db_logger::get().add_attribute(
"TimeStamp",
boost::make_shared< attrs::local_clock >());
string logPath = logDir + "/DreamBridgeSapi2MrcpTts_%N.log";
boost::shared_ptr< sinks::text_file_backend > backend =
boost::make_shared< sinks::text_file_backend >(
// file name pattern
keywords::file_name = logPath,
// rotate the file upon reaching 5 MiB size...
keywords::rotation_size = 5 * 1024 * 1024,
// ...or at midnight, whichever comes first
keywords::time_based_rotation = sinks::file::rotation_at_time_point(0, 0, 0)
);
backend->auto_flush(true);
// Wrap it into the frontend and register in the core.
// The backend requires synchronization in the frontend.
typedef sinks::synchronous_sink< sinks::text_file_backend > sink_t;
boost::shared_ptr< sink_t > sink(new sink_t(backend));
loggingCore->add_sink(sink);
backend->set_formatter(fmt::stream
<< fmt::date_time< boost::posix_time::ptime >("TimeStamp")
<< " : [" << fmt::attr< dreambridge::log::LogLevel >("Severity")
<< "] " << fmt::message()
);
backend->set_file_collector(sinks::file::make_collector(
// rotated logs will be moved here
keywords::target = logDir + "/old_logs",
// oldest log files will be removed if the total size reaches 100 MiB...
keywords::max_size = 100 * 1024 * 1024,
// ...or the free space in the target directory comes down to 50 MiB
keywords::min_free_space = 50 * 1024 * 1024
));
try
{
backend->scan_for_files(sinks::file::scan_all);
}
catch(std::exception & e)
{
LOGGER(emergency) << "exception during scanning : " << e.what();
}
}
你也可以设置日志记录级别为:
void SetLogLevel(dreambridge::log::LogLevel logLevel)
{
boost::shared_ptr< logging::core > loggingCore = logging::core::get();
loggingCore->set_filter
(
flt::attr< dreambridge::log::LogLevel >("Severity") <= logLevel
);
}
注意:此代码是从我自己的仓库。 dreamdridge是产品名称,因此命名空间。
其原因可能是,你不应该将数据写入到std :: ostream的。 boost.log使用boost.log.formatting_ostream输出日志,没有标准:: ostream的。 请参阅从网页中提取的这种代码http://www.boost.org/doc/libs/1_54_0/libs/log/doc/html/log/detailed/expressions.html
// The operator is used when putting the severity level to log
logging::formatting_ostream& operator<<
(
logging::formatting_ostream& strm,
logging::to_log_manip< severity_level, severity_tag > const& manip
)
{
static const char* strings[] =
{
"NORM",
"NTFY",
"WARN",
"ERRR",
"CRIT"
};
severity_level level = manip.get();
if (static_cast< std::size_t >(level) < sizeof(strings) / sizeof(*strings))
strm << strings[level];
else
strm << static_cast< int >(level);
return strm;
}
更正登记格式化的呼叫(即operator<<
通过从答案中指出) W. Merrt和马特·杨是所有你需要让你的当前代码编译和正常运行。
然而,为了充分利用升压日志的过滤功能,你还需要定义operator>>
您的自定义严重级别,然后注册与过滤器工厂的方法。
template< typename CharT, typename TraitsT>
inline std::basic_istream< CharT, TraitsT >& operator>> (
std::basic_istream< CharT, TraitsT >& strm, severity_level &lvl)
{
std::string tmp;
strm >> tmp;
// can make it case insensitive to allow 'warning' instead of only 'WARNING'
// std::transform(tmp.begin(), tmp.end(), tmp.begin(), ::toupper);
// if you have a lot of levels you may want use a map instead
if (tmp.compare("DEBUG") == 0) lvl = DEBUG;
else if (tmp.compare("INFO") == 0) lvl = INFO;
else if (tmp.compare("WARNING") == 0) lvl = WARNING;
else if (tmp.compare("ERROR") == 0) lvl = ERROR;
else if (tmp.compare("CRITICAL") == 0) lvl = CRITICAL;
// provide a default value for invalid strings
else lvl = DEBUG;
// or you could throw an exception
// else tmp.append(" is not a valid severity level."), throw std::invalid_argument(tmp);
return strm;
}
void init() {
// register operator<< for severity_level
logging::register_simple_formatter_factory< severity_level, char >("Severity");
// register operator>> for severity_level
logging::register_simple_filter_factory< severity_level >("Severity");
logging::add_file_log(
keywords::file_name = "blop.log",
keywords::auto_flush = true,
keywords::open_mode = (std::ios::out | std::ios::app),
keywords::format = "%TimeStamp% [%Uptime%] (%LineID%) <%Severity%>: %Message%",
// now you can filter using the name of the severity level
keywords::filter = "%Severity% >= INFO" );
logging::add_common_attributes();
logging::core::get()->add_global_attribute("Uptime", attrs::timer());
}