将 g++ 编译错误消息从 TMB 库导出到文本文件
Exporting g++ compilation error messages from TMB library to a text file
我正在尝试在 R 中使用 gcc (c++) 编译模型(使用 TMB 包)。错误如此之多,以至于在 Rstudio 中,我什至无法向上滚动以查看它们的开头。因此,我想将控制台中的所有内容(消息、错误和警告)打印到一个文本文件中。这样我也可以比较不同的模型输出(或者具体来说,它们的失败)。
我尝试了以下操作:
fileConn<-file("Fail.txt")
writeLines(compile("Mymodel.cpp"), fileConn)
close(fileConn)
=> 给我一个空文件(这是我或多或少期望的)
zz <- file("Fail.txt", open = "wt")
sink(zz, type = "message")
compile("Mymodel.cpp")
sink()
=> 只打印最后的错误
我忽略了什么?
你没有提供你的 compile()
函数,但我假设它运行一个系统命令调用 g++
来编译 Mymodel.cpp
.
在这种情况下,g++
进程会将其错误输出打印到 stderr。使用 R 捕获此输出的唯一方法是调用 system2()
with stderr=T
. Note that system()
does not have the capability to capture stderr directly (although it can capture stdout via intern=T
, and you could add 2>&1
to the shell command to capture stderr along with it, so that's another viable option). sink()
不捕获系统命令输出;它只捕获 R 输出。
我建议将可变参数从您的 compile()
函数传递给 system2()
调用,从而参数化您对 compile()
的调用是否会导致 g++
的标准错误终端或 compile()
函数的 return 值。这是如何完成的:
write('error!','test1.cpp'); ## generate a test file with invalid C++
compile <- function(file,...) system2('g++',file,...);
compile('test1.cpp'); ## output lost to the terminal
## test1.cpp:1:1: error: ‘error’ does not name a type
## error!
## ^
output <- compile('test1.cpp',stdout=T,stderr=T); ## capture output
## Warning message:
## running command ''g++' 'test1.cpp' 2>&1' had status 1
output;
## [1] "test1.cpp:1:1: error: ‘error’ does not name a type"
## [2] " error!"
## [3] " ^"
## attr(,"status")
## [1] 1
write(output,'output.txt'); ## write output to a text file
cat(readLines('output.txt'),sep='\n'); ## show it
## test1.cpp:1:1: error: ‘error’ does not name a type
## error!
## ^
如果您真的想捕获在 compile()
函数中生成的 all 输出,那么您可以将上述解决方案与 sink()
结合使用,如下所示: How to save all console output to file in R?.
在这种情况下,我建议放弃可变参数的想法,并为 compile()
添加一个附加参数,这将采用一个输出文件名,所有输出都将写入该文件名。
这将需要对附加参数缺失的几个断言:
write('error!','test1.cpp'); ## generate a test file with invalid C++
compile <- function(file,outputFile) {
if (!missing(outputFile)) {
outputCon <- file(outputFile,'wt'); ## require file name
sink(outputCon);
sink(outputCon,type='message'); ## must sink messages separately
warn.old <- options(warn=1)$warn; ## necessary to capture warnings as they occur
}; ## end if
cat('some random output 1\n');
if (!missing(outputFile)) {
output <- system2('g++',file,stdout=T,stderr=T); ## before flush to get warnings
sink(); ## force flush before appending system command output
sink(type='message');
outputCon <- file(outputFile,'at'); ## must reopen connection for appending
write(output,outputCon);
sink(outputCon);
sink(outputCon,type='message');
} else {
system2('g++',file);
}; ## end if
cat('some random output 2\n');
if (!missing(outputFile)) {
sink();
sink(type='message');
options(warn=warn.old);
}; ## end if
}; ## end compile()
compile('test1.cpp'); ## output lost to the terminal
## some random output 1
## test1.cpp:1:1: error: ‘error’ does not name a type
## error!
## ^
## some random output 2
compile('test1.cpp','output.txt'); ## internally capture all output
cat(readLines('output.txt'),sep='\n'); ## show it
## some random output 1
## Warning: running command ''g++' test1.cpp 2>&1' had status 1
## test1.cpp:1:1: error: ‘error’ does not name a type
## error!
## ^
## some random output 2
这是一个老问题,但我找到了一个简单的解决方案来获取 TMB::compile
的日志文件输出,我在其他任何地方都找不到。
如上所述,sink() 不捕获系统命令输出;它只捕获 R 输出。
但是,库 TMB
中的函数 compile
中未列出的任何参数都作为 Makeconf 变量传递。因此,您只需要以字符串格式传递带有目标日志文件路径的经典输出命令:
TMB::compile(file = "Mymodel.cpp", "&> /tmp/logfile.log")
我正在尝试在 R 中使用 gcc (c++) 编译模型(使用 TMB 包)。错误如此之多,以至于在 Rstudio 中,我什至无法向上滚动以查看它们的开头。因此,我想将控制台中的所有内容(消息、错误和警告)打印到一个文本文件中。这样我也可以比较不同的模型输出(或者具体来说,它们的失败)。
我尝试了以下操作:
fileConn<-file("Fail.txt")
writeLines(compile("Mymodel.cpp"), fileConn)
close(fileConn)
=> 给我一个空文件(这是我或多或少期望的)
zz <- file("Fail.txt", open = "wt")
sink(zz, type = "message")
compile("Mymodel.cpp")
sink()
=> 只打印最后的错误
我忽略了什么?
你没有提供你的 compile()
函数,但我假设它运行一个系统命令调用 g++
来编译 Mymodel.cpp
.
在这种情况下,g++
进程会将其错误输出打印到 stderr。使用 R 捕获此输出的唯一方法是调用 system2()
with stderr=T
. Note that system()
does not have the capability to capture stderr directly (although it can capture stdout via intern=T
, and you could add 2>&1
to the shell command to capture stderr along with it, so that's another viable option). sink()
不捕获系统命令输出;它只捕获 R 输出。
我建议将可变参数从您的 compile()
函数传递给 system2()
调用,从而参数化您对 compile()
的调用是否会导致 g++
的标准错误终端或 compile()
函数的 return 值。这是如何完成的:
write('error!','test1.cpp'); ## generate a test file with invalid C++
compile <- function(file,...) system2('g++',file,...);
compile('test1.cpp'); ## output lost to the terminal
## test1.cpp:1:1: error: ‘error’ does not name a type
## error!
## ^
output <- compile('test1.cpp',stdout=T,stderr=T); ## capture output
## Warning message:
## running command ''g++' 'test1.cpp' 2>&1' had status 1
output;
## [1] "test1.cpp:1:1: error: ‘error’ does not name a type"
## [2] " error!"
## [3] " ^"
## attr(,"status")
## [1] 1
write(output,'output.txt'); ## write output to a text file
cat(readLines('output.txt'),sep='\n'); ## show it
## test1.cpp:1:1: error: ‘error’ does not name a type
## error!
## ^
如果您真的想捕获在 compile()
函数中生成的 all 输出,那么您可以将上述解决方案与 sink()
结合使用,如下所示: How to save all console output to file in R?.
在这种情况下,我建议放弃可变参数的想法,并为 compile()
添加一个附加参数,这将采用一个输出文件名,所有输出都将写入该文件名。
这将需要对附加参数缺失的几个断言:
write('error!','test1.cpp'); ## generate a test file with invalid C++
compile <- function(file,outputFile) {
if (!missing(outputFile)) {
outputCon <- file(outputFile,'wt'); ## require file name
sink(outputCon);
sink(outputCon,type='message'); ## must sink messages separately
warn.old <- options(warn=1)$warn; ## necessary to capture warnings as they occur
}; ## end if
cat('some random output 1\n');
if (!missing(outputFile)) {
output <- system2('g++',file,stdout=T,stderr=T); ## before flush to get warnings
sink(); ## force flush before appending system command output
sink(type='message');
outputCon <- file(outputFile,'at'); ## must reopen connection for appending
write(output,outputCon);
sink(outputCon);
sink(outputCon,type='message');
} else {
system2('g++',file);
}; ## end if
cat('some random output 2\n');
if (!missing(outputFile)) {
sink();
sink(type='message');
options(warn=warn.old);
}; ## end if
}; ## end compile()
compile('test1.cpp'); ## output lost to the terminal
## some random output 1
## test1.cpp:1:1: error: ‘error’ does not name a type
## error!
## ^
## some random output 2
compile('test1.cpp','output.txt'); ## internally capture all output
cat(readLines('output.txt'),sep='\n'); ## show it
## some random output 1
## Warning: running command ''g++' test1.cpp 2>&1' had status 1
## test1.cpp:1:1: error: ‘error’ does not name a type
## error!
## ^
## some random output 2
这是一个老问题,但我找到了一个简单的解决方案来获取 TMB::compile
的日志文件输出,我在其他任何地方都找不到。
如上所述,sink() 不捕获系统命令输出;它只捕获 R 输出。
但是,库 TMB
中的函数 compile
中未列出的任何参数都作为 Makeconf 变量传递。因此,您只需要以字符串格式传递带有目标日志文件路径的经典输出命令:
TMB::compile(file = "Mymodel.cpp", "&> /tmp/logfile.log")