Prevent Ctrl+C from interrupting exec.Command in Golang
Asked Answered
M

2

7

I've noticed that processes started with exec.Command get interrupted even when the interrupt call has been intercepted via signal.Notify. I've done the following example to show the issue:

package main

import (
    "log"
    "os"
    "os/exec"
    "os/signal"
    "syscall"
)

func sleep() {
    log.Println("Sleep start")
    cmd := exec.Command("sleep", "60")
    cmd.Run()
    log.Println("Sleep stop")
}

func main() {
    var doneChannel = make(chan bool)

    go sleep()

    c := make(chan os.Signal, 1)
    signal.Notify(c, os.Interrupt)
    signal.Notify(c, syscall.SIGTERM)
    go func() {
        <-c
        log.Println("Receved Ctrl + C")
    }()

    <-doneChannel
}

If Ctrl+C is pressed while this program is running, it's going to print:

2015/10/16 10:05:50 Sleep start
^C2015/10/16 10:05:52 Receved Ctrl + C
2015/10/16 10:05:52 Sleep stop

showing that the sleep commands gets interrupted. Ctrl+C is successfully caught though and the main program doesn't quit, it's just the sleep commands that gets affected.

Any idea how to prevent this from happening?

Magnetochemistry answered 16/10, 2015 at 8:11 Comment(0)
B
14

The shell will signal the entire process group when you press ctrl+c. If you signal the parent process directly, the child process won't receive the signal.

To prevent the shell from signaling the children, you need to start the command in its own process group with with the Setpgid and Pgid fields in syscall.SysProcAttr before starting the processes

cmd := exec.Command("sleep", "60")
cmd.SysProcAttr = &syscall.SysProcAttr{
    Setpgid: true,
}
Belemnite answered 16/10, 2015 at 13:7 Comment(2)
This doesn't work for windows as there is no Setpgid in SysProcAttr under windows. Any alternatives?Teasley
@Teasley Yes. On Windows you need to set CreationFlags in the syscall.SysProcAttr to syscall.CREATE_NEW_PROCESS_GROUP.Remorseful
T
-2

You can ignore the syscall.SIGINT signal, then it won't be passed to the exec.Command.

func main() {
    var doneChannel = make(chan bool)

    signal.Ignore(syscall.SIGINT)

    go func() {
        log.Println("Sleep start")
        cmd := exec.Command("sleep", "10")
        cmd.Run()
        log.Println("Sleep stop")
        doneChannel <- true
    }()

    <-doneChannel
}
Tenement answered 25/5, 2017 at 21:7 Comment(2)
import("os/signal" "syscall")Spermatophyte
This only works because the child (in this case sleep) inherits parent's signal dispositions. The signal is still delivered and this might fail if the target application restores default signal behavior with e.g. signal(SIGINT, SIG_DFL) or sets it's own signal handler.Thrive

© 2022 - 2024 — McMap. All rights reserved.