Maison >développement back-end >Golang >Comment puis-je exécuter un processus de longue durée dans Golang et le détacher de mon programme tout en contrôlant l'utilisateur, en redirigeant la sortie et en l'empêchant de devenir un zombie ?
Problème :
Vous souhaitez exécuter un long- exécuter un processus dans Golang tout en répondant à plusieurs exigences :
Tentative de solution :
Vous avez tenté une solution en utilisant exec.Command mais elle ne répond pas à toutes les exigences, en particulier lorsque votre programme reçoit des signaux SIGTERM/SIGKILL.
Point clé à comprendre :
Il est important de noter que vous ne pouvez pas changer le parent d'un processus une fois qu'il a été démarré - la relation parent-enfant d'un processus est fixe.
Bibliothèques alternatives :
Au lieu de réinventer la roue, il est recommandé d'utiliser une bibliothèque existante qui résout ce problème. Les bibliothèques suivantes sont suggérées :
Exemple avec go-reap :
<code class="go">import ( "fmt" "os" "os/exec" "strings" "sync" "time" "github.com/fatih/color" "github.com/hashicorp/go-reap" ) func main() { if reap.IsSupported() { done := make(chan struct{}) var reapLock sync.RWMutex pids := make(reap.PidCh, 1) errors := make(reap.ErrorCh, 1) go reap.ReapChildren(pids, errors, done, &reapLock) go report(pids, errors, done) // Here is where you would start your long-running process Sh() close(done) } else { fmt.Println("Sorry, go-reap isn't supported on your platform.") } } func report(pids reap.PidCh, errors reap.ErrorCh, done chan struct{}) { sprintf := color.New(color.FgWhite, color.Bold).SprintfFunc() for ;; { select { case pid := <-pids: println(sprintf("raeper pid %d", pid)) case err := <-errors: println(sprintf("raeper er %s", err)) case <-done: return } } } func Sh() { args := os.Args[1:] script := args[0:0] if len(args) >= 1 { if args[0] == "-c" { script = args[1:] } } if len(script) == 0 { fn.CyanBold("cmd: expecting sh -c 'foobar'") os.Exit(111) } var cmd *exec.Cmd parts, _ := shlex.Split(strings.Join(script, " ")) if len(parts) >= 2 { cmd = fn.Merge(exec.Command(parts[0], parts[1:]...), nil) } if len(parts) == 1 { cmd = fn.Merge(exec.Command(parts[0]), nil) } // ... Here you can customize how the process is started and controlled if fn.IfEnv("HANG") { fn.CyanBold("cmd: %v\n start", parts) ex := cmd.Start() if ex != nil { fn.CyanBold("cmd %v err: %s", parts, ex) } go func() { time.Sleep(time.Millisecond * 100) errw := cmd.Wait() if errw != nil { fn.CyanBold("cmd %v err: %s", parts, errw) } else { fn.CyanBold("cmd %v all done.", parts) } }() fn.CyanBold("cmd: %v\n dispatched, hanging forever (i.e. to keep docker running)", parts) for { time.Sleep(time.Millisecond * time.Duration(fn.EnvInt("HANG", 2888))) fn.SystemCyan("/bin/ps", "-e", "-o", "stat,comm,user,etime,pid,ppid") } } else { if fn.IfEnv("NOWAIT") { ex := cmd.Start() if ex != nil { fn.CyanBold("cmd %v start err: %s", parts, ex) } } else { ex := cmd.Run() if ex != nil { fn.CyanBold("cmd %v run err: %s", parts, ex) } } fn.CyanBold("cmd %v\n dispatched, exit docker.", parts) } }</code>
Cet exemple utilise go-reap pour démarrer un shell (fonction Sh()) et exécuter une commande dans ce shell. Il configure le Reaper pour gérer le nettoyage des processus enfants.
En utilisant une bibliothèque réputée, vous pouvez éviter les pièges courants et vous assurer que votre application se comporte comme prévu.
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!