尝试第一个日志语句时Boost Log会导致崩溃(当不是管理员时)

时间:2015-07-29 23:37:24

标签: c++ boost boost-log

我正在尝试部署我的应用程序,它使用Boost Log(Boost 1.58)。它是一个简单的控制台应用程序,在Windows 7中运行。日志在我的个人桌面上运行得非常好。

但是,当我将应用程序部署到Win7虚拟机时,它会在我的第一个日志语句中崩溃:

boost::log::sources::severity_logger<SeverityLevel> slg;
BOOST_LOG_SEV(slg, SeverityLevel::Notification) << L"Application loaded"; // <-- Crash here

创建了日志目录,但永远不会创建日志文件并且应用程序崩溃。

我在我的%APPDATA%目录中尝试了一个日志文件目录,也在我的My Documents目录中。

奇怪的是:当我以管理员身份运行应用程序时,它可以正常运行

所以这必须是权限,但我对这些文件夹有权限,所以......

有什么想法吗?

* MORE *

以下是设置记录器的代码:

    wstring appData = GetMyAppDataPath();
    boost::shared_ptr< boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend > > textFileSink(new boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend >(to store rotated files
        boost::log::keywords::file_name = "log_%7N.log",                // file name pattern
        boost::log::keywords::rotation_size = 16384                     // rotation size, in characters
        ));
    // Set up where the rotated files will be stored
    textFileSink->locked_backend()->set_file_collector(boost::log::sinks::file::make_collector(
        boost::log::keywords::target = appData + L"\\logs",         // where to store rotated files
        boost::log::keywords::max_size = 64 * 1024 * 1024,              // maximum total size of the stored files, in bytes (64 MiB)
        boost::log::keywords::min_free_space = 100 * 1024 * 1024        // minimum free space on the drive, in bytes
        ));
    // Upon restart, scan the target directory for files matching the file_name pattern
    textFileSink->locked_backend()->scan_for_files();
    // Set up the format for output to the text file.
    textFileSink->set_formatter(boost::log::expressions::stream
        << "[" << boost::log::expressions::attr< boost::posix_time::ptime >("TimeStamp") 
        << " " << boost::log::expressions::attr< SeverityLevel, severity_tag >("Severity") << "] "
        << boost::log::expressions::message
        );
    // Add it to the core
    boost::log::core::get()->add_sink(textFileSink);

主要取自此处:http://www.boost.org/doc/libs/1_58_0/libs/log/example/rotating_file/main.cpp

* ALSO *

我通过添加:

为Boost Logger添加了一个异常处理程序
boost::log::core::get()->set_exception_handler(boost::log::make_exception_handler<
        std::runtime_error,
        std::logic_error,
        std::exception
    >(pbb_boost_log_exception_handler()));

然后添加处理程序。当我运行时,我能够在崩溃之前捕获以下异常:

std::exception: Failed to open file for writing: Input/output error: "C:\Program Files\My App\log_0000000.log"

WTF?我肯定将日志文件位置设置为appData值,我已验证该值是正确的。此外,如果我以管理员身份运行此应用程序,则日志文件最终会出现在我期望的位置(appdata文件夹)。所以它必须只是在可执行文件的位置创建一个临时文件。这是正常的行为吗?我无法想象它是......所以我做了什么?

1 个答案:

答案 0 :(得分:4)

好的,我明白了。

问题在于此处,设置文本后端:

boost::shared_ptr< boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend > > textFileSink(new boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend >
    boost::log::keywords::file_name = "log_%7N.log",                // file name pattern
    boost::log::keywords::rotation_size = 16384                     // rotation size, in characters
    ));

肯定会有一个临时日志文件写入本地目录(Program Files目录,这是坏的)。基于documentation here,我看到写入的临时文件,然后传递给文件收集器。

因此,对于解决方案,我将代码更改为:

boost::shared_ptr< boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend > > textFileSink(new boost::log::sinks::synchronous_sink< boost::log::sinks::text_file_backend >(
    boost::log::keywords::file_name = appData + L"\\log_%7N.log",   // file name pattern
    boost::log::keywords::rotation_size = 16384                         // rotation size, in characters
    ));

请注意,我现在将file_name指定为位于AppData目录中。

这解决了问题

我很难相信我是第一个遇到这个问题的人,但我无法在网络上找到它。对于Windows开发人员来说,这将是一个经常出现的问题,所以希望这对其他人有帮助。