[英]How to understand if exec.cmd was canceled
当命令被上下文取消时,我试图返回特定错误。 在调查 ProcessState 后了解到,如果在 exitCode 中得到 -1,则进程得到终止信号https://golang.org/pkg/os/#ProcessState.ExitCode但也许我们有更优雅的方法? 也许我可以将这个错误从取消功能中删除? 也许 exitCode 不足以理解命令是否被取消?
var (
CmdParamsErr = errors.New("failed to get params for execution command")
ExecutionCanceled = errors.New("command canceled")
)
func execute(m My) error {
filePath, args, err := cmdParams(m)
err = nil
if err != nil {
log.Infof("cmdParams: err: %v\n, m: %v\n", err, m)
return CmdParamsErr
}
var out bytes.Buffer
var errStd bytes.Buffer
ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
defer cancel()
cmd := exec.CommandContext(ctx, filePath, args...)
cmd.Stdout = &out
cmd.Stderr = &errStd
err = cmd.Run()
if err != nil {
if cmd.ProcessState.ExitCode() == -1 {
log.Warnf("execution was canceled by signal, err: %v\n", err)
err = ExecutionCanceled
return err
} else {
log.Errorf("run failed, err: %v, filePath: %v, args: %v\n", err, filePath, args)
return err
}
}
return err
}
exec.ExitError不提供任何退出代码的原因(没有相关的结构字段,也没有 Unwrap 方法),因此您必须直接检查上下文:
if ctx.Err() != nil {
log.Println("canceled")
}
请注意,这是一个轻微的竞争,因为上下文可能会在命令因其他原因失败后立即取消,但您对此无能为力。
没有直接或优雅的方法来确定进程是否因为上下文被取消而被终止。 最接近你的是:
func run() error {
ctx, cancel := context.WithTimeout(context.Background(), 1*time.Second)
defer cancel()
cmd := exec.CommandContext(ctx, "bash", "-c", "exit 1")
// Start() returns an error if the process can't be started. It will return
// ctx.Err() if the context is expired before starting the process.
if err := cmd.Start(); err != nil {
return err
}
if err := cmd.Wait(); err != nil {
if e, ok := err.(*exec.ExitError); ok {
// If the process exited by itself, just return the error to the
// caller.
if e.Exited() {
return e
}
// We know now that the process could be started, but didn't exit
// by itself. Something must have killed it. If the context is done,
// we can *assume* that it has been killed by the exec.Command.
// Let's return ctx.Err() so our user knows that this *might* be
// the case.
select {
case <-ctx.Done():
return ctx.Err()
default:
return e
}
}
return err
}
return nil
}
这里的问题是可能存在竞争条件,因此返回ctx.Err()
可能会产生误导。 例如,想象以下场景:
此时,上面的函数将返回ctx.Err()
,但这可能会产生误导,因为进程被ctx.Err()
的原因不是因为上下文被取消。 如果您决定使用类似于上述函数的代码,请记住这个近似值。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.