繁体   English   中英

如何理解 exec.cmd 是否被取消

[英]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()可能会产生误导。 例如,想象以下场景:

  1. 该过程开始。
  2. 该进程被外部参与者杀死。
  3. 上下文被取消。
  4. 你检查上下文。

此时,上面的函数将返回ctx.Err() ,但这可能会产生误导,因为进程被ctx.Err()的原因不是因为上下文被取消。 如果您决定使用类似于上述函数的代码,请记住这个近似值。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM