登录
首页 >  Golang >  Go问答

父进程传递给子进程无法处理的 SIGINT 信号,子进程突然终止

来源:stackoverflow

时间:2024-03-06 11:30:32 418浏览 收藏

今天golang学习网给大家带来了《父进程传递给子进程无法处理的 SIGINT 信号,子进程突然终止》,其中涉及到的知识点包括等等,无论你是小白还是老手,都适合看一看哦~有好的建议也欢迎大家在评论留言,若是看完有所收获,也希望大家能多多点赞支持呀!一起加油学习~

问题内容

我正在尝试使用 stdpipes 在 golang 中管理一个应用程序(需要通过特定过程关闭,在本例中是拯救世界)。

这是我想要实现的目标的一个简单示例,但我有一个对我来说非常具体的问题,但对其他人来说也可能很有趣(也许您可以建议如何概括它)。

我还添加了一个名为 interruptlistener 的函数,该函数创建一个 goroutine 并在发送终止信号时管理程序的停止

脚本正常运行

  • 启动我的世界服务器
  • 等待 40 秒,然后通过 stdin 发出“停止”命令

    (在这种情况下,它按预期打印有关保存过程的所有日志)

测试用例(演示问题出在哪里):

  • 启动我的世界服务器
  • 在脚本发出停止命令之前,用户发送 ctrl+c

    (在这种情况下,它应该完成打印有关保存过程的日志,然后退出,但它没有...似乎在收到终止信号后 scanner.scan() 返回 false,因此它只是退出)

您知道为什么会发生这种情况吗?我应该研究什么才能找到解决方案?

我真的迷路了,我已经花了 8 个多小时研究所有可能的代码组合......

package main

import (
    "bufio"
    "fmt"
    "io"
    "os"
    "os/exec"
    "os/signal"
    "strings"
    "sync"
    "syscall"
    "time"
)

var cmd *exec.Cmd

var wg sync.WaitGroup

var stdOut io.ReadCloser
var stdErr io.ReadCloser
var stdIn io.WriteCloser

func main() {
    interruptListener()

    cSplit := strings.Split("java -Xmx1024M -Xms1024M -jar server.jar nogui", " ")
    cmd = exec.Command(cSplit[0], cSplit[1:]...)
    cmd.Dir = "path/to/server/folder"
    stdOut, _ = cmd.StdoutPipe()
    stdErr, _ = cmd.StderrPipe()
    stdIn, _ = cmd.StdinPipe()
    wg.Add(2)
    go printer(stdOut)
    go printer(stdErr)
    err := cmd.Start()
    if err != nil {
        fmt.Println(err)
    }

    // test 1: wait 40 seconds for it to send the stop command and observe the output
    // test 2: in the 40 seconds (after the server has loaded press ctrl+c), there is no output
    time.Sleep(40 * time.Second)
    execute("stop")

    err = cmd.Wait()
    if err != nil {
        fmt.Println(err.Error())
    }
}

func printer(stdP io.ReadCloser) {
    defer func() {
        wg.Done()
        fmt.Println("printer is out")
    }()

    var line string

    scanner := bufio.NewScanner(stdP)

    for scanner.Scan() {
        line = scanner.Text()

        fmt.Println(line)
    }

    fmt.Printf("scanner error: %v\n", scanner.Err())
}

func execute(com string) {
    fmt.Println("sending", com, "to terminal")

    // needs to be added otherwise the virtual "enter" button is not pressed
    com += "\n"

    // write to cmd
    _, err := stdIn.Write([]byte(com))
    if err != nil {
        fmt.Println(err.Error())
    }
}

func interruptListener() {
    c := make(chan os.Signal, 1)
    signal.Notify(c, syscall.SIGHUP, syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT)
    go func() {
        select {
        case <-c:
            execute("stop")
            wg.Wait()
            os.Exit(0)
        }
    }()
}

解决方案


感谢在评论中回答的人,这是一个可以接受的解决方案(仍然不完美,如评论中所述)

只需在执行 cmd.start() 之前添加这几行:

// launch as new process group so that signals (ex: SIGINT) are not sent also the the child process
    cmd.SysProcAttr = &syscall.SysProcAttr{
        CreationFlags: syscall.CREATE_NEW_PROCESS_GROUP, // windows
        // Setpgid: true, // linux
    }

这是您问题的更完整答案 - Killing a child process and all of its children in Go

今天关于《父进程传递给子进程无法处理的 SIGINT 信号,子进程突然终止》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>