mirror of https://github.com/pulumi/pulumi.git
134 lines
4.7 KiB
Go
134 lines
4.7 KiB
Go
|
// Copyright 2016-2023, Pulumi Corporation.
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
// you may not use this file except in compliance with the License.
|
||
|
// You may obtain a copy of the License at
|
||
|
//
|
||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software
|
||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
// See the License for the specific language governing permissions and
|
||
|
// limitations under the License.
|
||
|
|
||
|
package cmdutil
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"errors"
|
||
|
"os"
|
||
|
"os/exec"
|
||
|
"time"
|
||
|
)
|
||
|
|
||
|
// TerminateProcessGroup terminates the process group
|
||
|
// of the given process by sending a termination signal to it.
|
||
|
//
|
||
|
// - On Linux and macOS, it sends a SIGINT
|
||
|
// - On Windows, it sends a CTRL_BREAK_EVENT
|
||
|
//
|
||
|
// If the root process does not exit gracefully within the given duration,
|
||
|
// all processes in the group are forcibly terminated.
|
||
|
//
|
||
|
// Returns true if the process exited gracefully, false otherwise.
|
||
|
//
|
||
|
// Returns an error if the process could not be terminated,
|
||
|
// or if the process exited with a non-zero exit code.
|
||
|
func TerminateProcessGroup(proc *os.Process, cooldown time.Duration) (ok bool, err error) {
|
||
|
// The choice to use SIGINT and CTRL_BREAK_EVENT
|
||
|
// merits some explanation.
|
||
|
//
|
||
|
// On *nix, typically,
|
||
|
// SIGTERM is used for programmatic graceful shutdown,
|
||
|
// and SIGINT is used when the user presses Ctrl+C.
|
||
|
// e.g. Kubernetes sends SIGTERM to signal shutdown.
|
||
|
// So in short, SIGTERM is for computers, SIGINT is for humans.
|
||
|
//
|
||
|
// On Windows,
|
||
|
// there's CTRL_C_EVENT which is obviously analogous to SIGINT
|
||
|
// because they both handle Ctrl+C,
|
||
|
// and CTRL_BREAK_EVENT which is special to Windows,
|
||
|
// but we can decide it's analogous to SIGTERM.
|
||
|
//
|
||
|
// However, when writing a signal handler on Windows,
|
||
|
// different languages map these signals differently.
|
||
|
// Go maps both, CTRL_BREAK_EVENT and CTRL_C_EVENT to SIGINT,
|
||
|
// Node and Python map CTRL_BREAK_EVENT to SIGBREAK
|
||
|
// (which exists only on Windows), and CTRL_C_EVENT to SIGINT.
|
||
|
//
|
||
|
// In short:
|
||
|
//
|
||
|
// | OS | Signal sent | Language | Handled as |
|
||
|
// |------|------------------|----------|------------|
|
||
|
// | *nix | SIGTERM | Go | SIGTERM |
|
||
|
// | | | Node | SIGTERM |
|
||
|
// | | | Python | SIGTERM |
|
||
|
// | |------------------|----------|------------|
|
||
|
// | | SIGINT | Go | SIGINT |
|
||
|
// | | | Node | SIGINT |
|
||
|
// | | | Python | SIGINT |
|
||
|
// |------|------------------|----------|------------|
|
||
|
// | Win | CTRL_BREAK_EVENT | Go | SIGINT |
|
||
|
// | | | Node | SIGBREAK |
|
||
|
// | | | Python | SIGBREAK |
|
||
|
// | |------------------|----------|------------|
|
||
|
// | | CTRL_C_EVENT | Go | SIGINT |
|
||
|
// | | | Node | SIGINT |
|
||
|
// | | | Python | SIGINT |
|
||
|
//
|
||
|
// So the SIGINT+CTRL_C_EVENT combo would be the obvious choice here
|
||
|
// since it's consistent across languages and platforms;
|
||
|
// plugins would define a single SIGINT handler
|
||
|
// and it would work in all cases.
|
||
|
//
|
||
|
// Unfortunately, Winodws does not support sending CTRL_C_EVENT
|
||
|
// to a specific child process.
|
||
|
// It's "current process and all child processes" or nothing.
|
||
|
/// Per the docs [1], the CTRL_C_EVENT
|
||
|
// "cannot be limited to a specific process group."
|
||
|
//
|
||
|
// [1]: https://learn.microsoft.com/en-us/windows/console/generateconsolectrlevent
|
||
|
//
|
||
|
// So we have to use CTRL_BREAK_EVENT for Windows instead.
|
||
|
// At that point, using SIGINT for *nix makes sense because
|
||
|
// users will want to handle SIGINT anyway
|
||
|
// so that they can press Ctrl+C in the terminal.
|
||
|
|
||
|
if err := shutdownProcessGroup(proc.Pid); err != nil {
|
||
|
// Couldn't shut down the process gracefully.
|
||
|
// Let's just kill it.
|
||
|
return false, killProcessGroup(proc)
|
||
|
}
|
||
|
|
||
|
var waitErr error
|
||
|
ctx, cancel := context.WithTimeout(context.Background(), cooldown)
|
||
|
go func() {
|
||
|
defer cancel()
|
||
|
|
||
|
state, err := proc.Wait()
|
||
|
switch {
|
||
|
case err == nil && !state.Success():
|
||
|
// Non-zero exit code.
|
||
|
err = &exec.ExitError{ProcessState: state}
|
||
|
|
||
|
case isWaitAlreadyExited(err):
|
||
|
err = nil
|
||
|
}
|
||
|
|
||
|
waitErr = err
|
||
|
}()
|
||
|
|
||
|
// The context will be canceled when the timeout expires,
|
||
|
// or when the process exits, whichever happens first.
|
||
|
<-ctx.Done()
|
||
|
|
||
|
if err := ctx.Err(); errors.Is(err, context.DeadlineExceeded) {
|
||
|
// The process didn't exit within the given duration.
|
||
|
// Kill it.
|
||
|
return false, killProcessGroup(proc)
|
||
|
}
|
||
|
|
||
|
return true, waitErr
|
||
|
}
|