Boost.Log:支持文件名和行号

时间:2015-07-01 06:53:22

标签: c++ boost log4cxx boost-log

我正在尝试让我的团队离开log4cxx并尝试使用Boost.Log v2。我们当前的log4cxx pattern非常简单:

log4cxx::helpers::Properties prop;
prop.setProperty("log4j.rootLogger","DEBUG, A1");
prop.setProperty("log4j.appender.A1","org.apache.log4j.ConsoleAppender");
prop.setProperty("log4j.appender.A1.layout","org.apache.log4j.PatternLayout");
prop.setProperty("log4j.appender.A1.layout.ConversionPattern","%d{ABSOLUTE} %-5p [%c] %m%n");
log4cxx::PropertyConfigurator::configure(prop);

但是我找不到支持文件名和行号打印的解决方案。到目前为止,我发现了old post,但没有明确的解决方案(没有公认的解决方案)。我已经研究过使用那些BOOST_LOG_NAMED_SCOPE,但是他们觉得非常难看,因为当在同一个函数中使用多个这样的行号时,无法打印正确的行号。

我还找到了一个更简单的直接解决方案,here。但是这也觉得很难看,因为它会打印完整路径,而不仅仅是基本名称(它不可配置)。因此,完整路径和行号始终打印在固定位置(expr::smessage之前)。

我也发现了这个post,这看起来像是一个旧解决方案。

最后,我发现最有希望的解决方案是here。但是它甚至都没有为我编译。

所以我的问题很简单:如何使用Boost.Log v2打印文件名(不是完整路径)和最小formatter flexibility的行号(没有MACRO__FILE__ /的解决方案__LINE__已接受,请)。我感谢一个不涉及BOOST_LOG_NAMED_SCOPE的解决方案,如here所述:

  

如果您希望查看特定日志记录的行号   最好的方法是定义一个用于编写的自定义宏   日志。在该宏中,您可以添加文件名和行号   属性到记录(你可以使用操纵器)。注意   在这种情况下,您无法使用这些属性   过滤器,但你可能不需要它。

2 个答案:

答案 0 :(得分:9)

我终于找到了一个基于add_value的简单解决方案。这是完整的源代码:

#include <ostream>
#include <fstream>
#include <boost/log/core.hpp>
#include <boost/log/trivial.hpp>
#include <boost/log/sinks/sync_frontend.hpp>
#include <boost/log/sinks/text_ostream_backend.hpp>
#include <boost/log/utility/setup/common_attributes.hpp>
#include <boost/log/utility/manipulators/add_value.hpp>
#include <boost/filesystem.hpp>

namespace logging = boost::log;
namespace src = boost::log::sources;
namespace expr = boost::log::expressions;
namespace sinks = boost::log::sinks;

void my_formatter(logging::record_view const& rec, logging::formatting_ostream& strm)
{
    // Get the LineID attribute value and put it into the stream
    strm << logging::extract< unsigned int >("LineID", rec) << ": ";
    strm << logging::extract< int >("Line", rec) << ": ";
    logging::value_ref< std::string > fullpath = logging::extract< std::string >("File", rec);
    strm << boost::filesystem::path(fullpath.get()).filename().string() << ": ";

    // The same for the severity level.
    // The simplified syntax is possible if attribute keywords are used.
    strm << "<" << rec[logging::trivial::severity] << "> ";

    // Finally, put the record message to the stream
    strm << rec[expr::smessage];
}

void init()
{
    typedef sinks::synchronous_sink< sinks::text_ostream_backend > text_sink;
    boost::shared_ptr< text_sink > sink = boost::make_shared< text_sink >();

    sink->locked_backend()->add_stream(
        boost::make_shared< std::ofstream >("sample.log"));

    sink->set_formatter(&my_formatter);

    logging::core::get()->add_sink(sink);
}

#define MY_GLOBAL_LOGGER(log_,sv) BOOST_LOG_SEV( log_, sv) \
  << boost::log::add_value("Line", __LINE__)      \
  << boost::log::add_value("File", __FILE__)       \
  << boost::log::add_value("Function", BOOST_CURRENT_FUNCTION)

int main(int, char*[])
{
    init();
    logging::add_common_attributes();

    using namespace logging::trivial;
    src::severity_logger< severity_level > lg;

    MY_GLOBAL_LOGGER(lg,debug) << "Keep";
    MY_GLOBAL_LOGGER(lg,info) << "It";
    MY_GLOBAL_LOGGER(lg,warning) << "Simple";
    MY_GLOBAL_LOGGER(lg,error) << "Stupid";

    return 0;
}

在我的Linux机器上,我使用:

编译它
$ c++ -otutorial_fmt_custom -DBOOST_LOG_DYN_LINK tutorial_fmt_custom.cpp -lboost_log -lboost_log_setup -lboost_thread -lpthread -lboost_filesystem

如果你运行并检查这里生成的日志文件是你得到的:

$ ./tutorial_fmt_custom && cat sample.log 
1: 61: tutorial_fmt_custom.cpp: <debug> Keep
2: 62: tutorial_fmt_custom.cpp: <info> It
3: 63: tutorial_fmt_custom.cpp: <warning> Simple
4: 64: tutorial_fmt_custom.cpp: <error> Stupid

我的解决方案基于两个输入(谢谢!):

答案 1 :(得分:2)

要输出文件名和行,您需要注册Scopes

logging::core::get()->add_global_attribute("Scopes", attributes::named_scope());

并添加自定义格式化程序:

void my_formatter(logging::record_view const& rec, logging::formatting_ostream& strm)
{
    const auto cont = logging::extract< attributes::named_scope::value_type >("Scopes", rec);
    if(cont.empty())
        return;

    auto it = cont->begin();
    boost::filesystem::path path(it->file_name.c_str());
    strm << path.filename().string() << ":" << it->line << "\t" << rec[expr::smessage];
}

注意:为了使其正常工作,需要在每次记录之前清除作用域容器: if(!attributes::named_scope::get_scopes().empty()) attributes::named_scope::pop_scope();