github.com/jen20/docker@v1.13.1/daemon/kill.go (about) 1 package daemon 2 3 import ( 4 "fmt" 5 "runtime" 6 "strings" 7 "syscall" 8 "time" 9 10 "github.com/Sirupsen/logrus" 11 "github.com/docker/docker/container" 12 "github.com/docker/docker/pkg/signal" 13 ) 14 15 type errNoSuchProcess struct { 16 pid int 17 signal int 18 } 19 20 func (e errNoSuchProcess) Error() string { 21 return fmt.Sprintf("Cannot kill process (pid=%d) with signal %d: no such process.", e.pid, e.signal) 22 } 23 24 // isErrNoSuchProcess returns true if the error 25 // is an instance of errNoSuchProcess. 26 func isErrNoSuchProcess(err error) bool { 27 _, ok := err.(errNoSuchProcess) 28 return ok 29 } 30 31 // ContainerKill sends signal to the container 32 // If no signal is given (sig 0), then Kill with SIGKILL and wait 33 // for the container to exit. 34 // If a signal is given, then just send it to the container and return. 35 func (daemon *Daemon) ContainerKill(name string, sig uint64) error { 36 container, err := daemon.GetContainer(name) 37 if err != nil { 38 return err 39 } 40 41 if sig != 0 && !signal.ValidSignalForPlatform(syscall.Signal(sig)) { 42 return fmt.Errorf("The %s daemon does not support signal %d", runtime.GOOS, sig) 43 } 44 45 // If no signal is passed, or SIGKILL, perform regular Kill (SIGKILL + wait()) 46 if sig == 0 || syscall.Signal(sig) == syscall.SIGKILL { 47 return daemon.Kill(container) 48 } 49 return daemon.killWithSignal(container, int(sig)) 50 } 51 52 // killWithSignal sends the container the given signal. This wrapper for the 53 // host specific kill command prepares the container before attempting 54 // to send the signal. An error is returned if the container is paused 55 // or not running, or if there is a problem returned from the 56 // underlying kill command. 57 func (daemon *Daemon) killWithSignal(container *container.Container, sig int) error { 58 logrus.Debugf("Sending kill signal %d to container %s", sig, container.ID) 59 container.Lock() 60 defer container.Unlock() 61 62 // We could unpause the container for them rather than returning this error 63 if container.Paused { 64 return fmt.Errorf("Container %s is paused. Unpause the container before stopping", container.ID) 65 } 66 67 if !container.Running { 68 return errNotRunning{container.ID} 69 } 70 71 if container.Config.StopSignal != "" { 72 containerStopSignal, err := signal.ParseSignal(container.Config.StopSignal) 73 if err != nil { 74 return err 75 } 76 if containerStopSignal == syscall.Signal(sig) { 77 container.ExitOnNext() 78 } 79 } else { 80 container.ExitOnNext() 81 } 82 83 if !daemon.IsShuttingDown() { 84 container.HasBeenManuallyStopped = true 85 } 86 87 // if the container is currently restarting we do not need to send the signal 88 // to the process. Telling the monitor that it should exit on its next event 89 // loop is enough 90 if container.Restarting { 91 return nil 92 } 93 94 if err := daemon.kill(container, sig); err != nil { 95 err = fmt.Errorf("Cannot kill container %s: %s", container.ID, err) 96 // if container or process not exists, ignore the error 97 if strings.Contains(err.Error(), "container not found") || 98 strings.Contains(err.Error(), "no such process") { 99 logrus.Warnf("container kill failed because of 'container not found' or 'no such process': %s", err.Error()) 100 } else { 101 return err 102 } 103 } 104 105 attributes := map[string]string{ 106 "signal": fmt.Sprintf("%d", sig), 107 } 108 daemon.LogContainerEventWithAttributes(container, "kill", attributes) 109 return nil 110 } 111 112 // Kill forcefully terminates a container. 113 func (daemon *Daemon) Kill(container *container.Container) error { 114 if !container.IsRunning() { 115 return errNotRunning{container.ID} 116 } 117 118 // 1. Send SIGKILL 119 if err := daemon.killPossiblyDeadProcess(container, int(syscall.SIGKILL)); err != nil { 120 // While normally we might "return err" here we're not going to 121 // because if we can't stop the container by this point then 122 // its probably because its already stopped. Meaning, between 123 // the time of the IsRunning() call above and now it stopped. 124 // Also, since the err return will be environment specific we can't 125 // look for any particular (common) error that would indicate 126 // that the process is already dead vs something else going wrong. 127 // So, instead we'll give it up to 2 more seconds to complete and if 128 // by that time the container is still running, then the error 129 // we got is probably valid and so we return it to the caller. 130 if isErrNoSuchProcess(err) { 131 return nil 132 } 133 134 if _, err2 := container.WaitStop(2 * time.Second); err2 != nil { 135 return err 136 } 137 } 138 139 // 2. Wait for the process to die, in last resort, try to kill the process directly 140 if err := killProcessDirectly(container); err != nil { 141 if isErrNoSuchProcess(err) { 142 return nil 143 } 144 return err 145 } 146 147 container.WaitStop(-1 * time.Second) 148 return nil 149 } 150 151 // killPossibleDeadProcess is a wrapper around killSig() suppressing "no such process" error. 152 func (daemon *Daemon) killPossiblyDeadProcess(container *container.Container, sig int) error { 153 err := daemon.killWithSignal(container, sig) 154 if err == syscall.ESRCH { 155 e := errNoSuchProcess{container.GetPID(), sig} 156 logrus.Debug(e) 157 return e 158 } 159 return err 160 } 161 162 func (daemon *Daemon) kill(c *container.Container, sig int) error { 163 return daemon.containerd.Signal(c.ID, sig) 164 }