在 Golang 中阅读 os.OpenFile,同时还在编写?
Reading os.OpenFile in Golang while still being written?
我的代码在执行系统命令时正在写入日志文件。例如
logfile, err := os.OpenFile(THIS_LOG_FILE, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
if err != nil {
return err
}
cmd.Stderr = logfile
cmd.Stdout = logfile
go func() {
err := cmd.Run()
if err != nil {
// WANT TO LOG ERROR HERE
}
}()
在“// WANT TO LOG”行,除了之前分配的日志文件目标之外,我还想将内容输出到标准记录器。有没有办法在内存中捕获它?或者我应该只将所有内容写入内存缓冲区并在最后刷新?
澄清一下,在内存中捕获命令的输出时,我可以解析它并在 运行 程序中采取行动(处理 errors/etc)。当我写入日志文件时,该信息丢失了。
我的问题是,理论上,我 可以 从我刚写的文件中读回,但这似乎很浪费(如果命令失败则容易失败)。
如果我没理解错的话,你想在执行shell命令的同时将stdout/stderror的内容写入文件。
由于 stdout 和 stderror 实现了 ReadCloser 接口,您可以通过 io.MultiReader
合并它们并执行 io.Copy
从源到目标。
以下代码片段实现了管道
package main
import (
"io"
"log"
"os"
"os/exec"
)
func main() {
// prepare the command
cmd := exec.Command("your-shell-command.sh")
// get the stdout and stderr stream
erc, err := cmd.StderrPipe()
if err != nil {
log.Fatalln("Failed to get stderr reader: ", err)
}
orc, err := cmd.StdoutPipe()
if err != nil {
log.Fatalln("Failed to get stdout reader: ", err)
}
// combine stdout and stderror ReadCloser
rc := io.MultiReader(erc, orc)
// Prepare the writer
f, err := os.Create("output.log")
if err != nil {
log.Fatalln("Failed to create file")
}
defer f.Close()
// Command.Start starts a new go routine
if err := cmd.Start(); err != nil {
log.Println("Failed to start the command")
}
// add the TeeReader.
var buf bytes.Buffer
tr := io.TeeReader(rc, &buf)
if _, err := io.Copy(f, tr); err != nil {
logger.Fatalf("Failed to stream to file: %s", err)
}
if err := cmd.Wait(); err != nil {
log.Println("Failed to wait the command to execute: ", err)
}
}
我的代码在执行系统命令时正在写入日志文件。例如
logfile, err := os.OpenFile(THIS_LOG_FILE, os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
if err != nil {
return err
}
cmd.Stderr = logfile
cmd.Stdout = logfile
go func() {
err := cmd.Run()
if err != nil {
// WANT TO LOG ERROR HERE
}
}()
在“// WANT TO LOG”行,除了之前分配的日志文件目标之外,我还想将内容输出到标准记录器。有没有办法在内存中捕获它?或者我应该只将所有内容写入内存缓冲区并在最后刷新?
澄清一下,在内存中捕获命令的输出时,我可以解析它并在 运行 程序中采取行动(处理 errors/etc)。当我写入日志文件时,该信息丢失了。
我的问题是,理论上,我 可以 从我刚写的文件中读回,但这似乎很浪费(如果命令失败则容易失败)。
如果我没理解错的话,你想在执行shell命令的同时将stdout/stderror的内容写入文件。
由于 stdout 和 stderror 实现了 ReadCloser 接口,您可以通过 io.MultiReader
合并它们并执行 io.Copy
从源到目标。
以下代码片段实现了管道
package main
import (
"io"
"log"
"os"
"os/exec"
)
func main() {
// prepare the command
cmd := exec.Command("your-shell-command.sh")
// get the stdout and stderr stream
erc, err := cmd.StderrPipe()
if err != nil {
log.Fatalln("Failed to get stderr reader: ", err)
}
orc, err := cmd.StdoutPipe()
if err != nil {
log.Fatalln("Failed to get stdout reader: ", err)
}
// combine stdout and stderror ReadCloser
rc := io.MultiReader(erc, orc)
// Prepare the writer
f, err := os.Create("output.log")
if err != nil {
log.Fatalln("Failed to create file")
}
defer f.Close()
// Command.Start starts a new go routine
if err := cmd.Start(); err != nil {
log.Println("Failed to start the command")
}
// add the TeeReader.
var buf bytes.Buffer
tr := io.TeeReader(rc, &buf)
if _, err := io.Copy(f, tr); err != nil {
logger.Fatalf("Failed to stream to file: %s", err)
}
if err := cmd.Wait(); err != nil {
log.Println("Failed to wait the command to execute: ", err)
}
}