我有一个使用boost日志的C ++程序,我加载了一个用户提供的动态链接库。我想将stderr重定向到boost日志,以便用户的库随时执行:
std::cerr << "Some stuff";
它产生相同的结果**:
BOOST_LOG_SEV(log,info) << "Some stuff";
这是可能的,如果是的话,我该怎么做?
(另外,我不确定如何处理严重性......因为cerr <<
没有提供严重性信息。我也对这方面的建议持开放态度......)
**“相同的结果”我的意思是它会记录到与其他日志消息相同的日志文件中,并且相同的日志格式化程序将应用于这些行。
答案 0 :(得分:3)
这是我的C ++ 11实现。这个类可以用于任何事情(不仅仅是提升)来逐行捕获stdout / stderr并调用用户函数(可能是一个lambda)来处理它。
警告:如果您重定向stderr
和stdout
并且正在使用Boost,请先重定向stderr
,然后重定向stdout
。否则,Boost会将stderr消息回送到stdout,并在另一个boost日志条目中获得一个boost日志条目。
cout << "testing out before 1 2 3 " << endl;
cerr << "testing err before 1 2 3 " << endl;
{
StdErrHandler err([](const char* line){
BOOST_LOG_TRIVIAL(error) << "ERROR:" << strlen(line) << " " << line;
});
StdOutHandler out([](const char* line){
BOOST_LOG_TRIVIAL(info) << "OUT:" << strlen(line) << " " << line;
});
cout << "cout testing 1 2 3 " << endl;
cerr << "cerr testing 1 2 3 " << endl;
}
cout << "testing out after 1 2 3 " << endl;
cerr << "testing err after 1 2 3 " << endl;
pa-poca$ ./test
testing out before 1 2 3
testing err before 1 2 3
[2014-08-01 12:24:56.468335] [0x000007f89d8990d4] [error] ERROR:19 cerr testing 1 2 3
[2014-08-01 12:24:56.468360] [0x000007f89d8990d4] [info] OUT:19 cout testing 1 2 3
testing out after 1 2 3
testing err after 1 2 3
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <string.h>
#include <signal.h>
#include <sys/wait.h>
class StdioHandler
{
private:
pid_t pid = 0;
int origfd;
int streamid;
int pipefd[2];
public:
enum class Stream
{
stdout = STDOUT_FILENO,
stderr = STDERR_FILENO
};
StdioHandler(Stream stream, std::function<void(const char*)> callback)
:streamid(static_cast<int>(stream))
{
origfd = dup(streamid);
pipe(pipefd); // create pipe
pid = fork(); //spawn a child process to handle output of pipe
if (pid == 0)
{
char line[256];
FILE* output;
close(pipefd[1]);
output = fdopen(pipefd[0], "r");
if (output)
{
while(fgets(line, sizeof(line), output))
{
int n = strlen(line);
if (n > 0)
if (line[n-1] == '\n') line[n-1] = 0;
callback(line);
}
fclose(output);
}
abort();
} else {
// connect input of pipe to
close(pipefd[0]);
dup2(pipefd[1], streamid);
}
}
~StdioHandler()
{
int status;
usleep(10000);
close(pipefd[1]);
kill(pid,SIGINT);
waitpid(pid, &status, 0);
dup2(origfd, streamid);
}
};
class StdOutHandler : public StdioHandler
{
public:
StdOutHandler(std::function<void(const char*)> callback) :
StdioHandler(Stream::stdout, callback)
{
}
};
class StdErrHandler : public StdioHandler
{
public:
StdErrHandler(std::function<void(const char*)> callback) :
StdioHandler(Stream::stderr, callback)
{
}
};
答案 1 :(得分:-2)
我猜,您可以使用File descriptor STDERR
api [它的posix api]将dup/dup2
重定向到您的流文件描述符[您必须获取流的文件desc]