2018-02-05 16:05:59 -05:00
|
|
|
package daemon // import "github.com/docker/docker/daemon"
|
2014-07-31 16:40:15 -04:00
|
|
|
|
2015-09-16 14:56:26 -04:00
|
|
|
import (
|
2017-03-30 16:52:40 -04:00
|
|
|
"context"
|
2015-11-02 18:25:26 -05:00
|
|
|
"time"
|
|
|
|
|
2017-03-30 23:01:41 -04:00
|
|
|
containerpkg "github.com/docker/docker/container"
|
2018-01-11 14:53:06 -05:00
|
|
|
"github.com/docker/docker/errdefs"
|
2017-07-19 10:20:13 -04:00
|
|
|
"github.com/pkg/errors"
|
2017-07-26 17:42:13 -04:00
|
|
|
"github.com/sirupsen/logrus"
|
2015-09-16 14:56:26 -04:00
|
|
|
)
|
2015-03-25 03:44:12 -04:00
|
|
|
|
2018-04-16 19:58:42 -04:00
|
|
|
// ContainerStop looks for the given container and stops it.
|
|
|
|
// In case the container fails to stop gracefully within a time duration
|
|
|
|
// specified by the timeout argument, in seconds, it is forcefully
|
|
|
|
// terminated (killed).
|
|
|
|
//
|
|
|
|
// If the timeout is nil, the container's StopTimeout value is used, if set,
|
|
|
|
// otherwise the engine default. A negative timeout value can be specified,
|
|
|
|
// meaning no timeout, i.e. no forceful termination is performed.
|
|
|
|
func (daemon *Daemon) ContainerStop(name string, timeout *int) error {
|
2015-12-11 12:39:28 -05:00
|
|
|
container, err := daemon.GetContainer(name)
|
2014-12-16 18:06:35 -05:00
|
|
|
if err != nil {
|
2015-03-25 03:44:12 -04:00
|
|
|
return err
|
2014-07-31 16:40:15 -04:00
|
|
|
}
|
2014-12-16 18:06:35 -05:00
|
|
|
if !container.IsRunning() {
|
2017-07-19 10:20:13 -04:00
|
|
|
return containerNotModifiedError{running: false}
|
2014-12-16 18:06:35 -05:00
|
|
|
}
|
2018-04-16 19:58:42 -04:00
|
|
|
if timeout == nil {
|
2016-06-06 23:29:05 -04:00
|
|
|
stopTimeout := container.StopTimeout()
|
2018-04-16 19:58:42 -04:00
|
|
|
timeout = &stopTimeout
|
2016-06-06 23:29:05 -04:00
|
|
|
}
|
2018-04-16 19:58:42 -04:00
|
|
|
if err := daemon.containerStop(container, *timeout); err != nil {
|
2017-11-28 23:09:37 -05:00
|
|
|
return errdefs.System(errors.Wrapf(err, "cannot stop container: %s", name))
|
2014-12-16 18:06:35 -05:00
|
|
|
}
|
2015-03-25 03:44:12 -04:00
|
|
|
return nil
|
2014-07-31 16:40:15 -04:00
|
|
|
}
|
2015-11-02 18:25:26 -05:00
|
|
|
|
2018-04-16 19:58:42 -04:00
|
|
|
// containerStop sends a stop signal, waits, sends a kill signal.
|
2017-03-30 23:01:41 -04:00
|
|
|
func (daemon *Daemon) containerStop(container *containerpkg.Container, seconds int) error {
|
2020-10-24 16:15:58 -04:00
|
|
|
// TODO propagate a context down to this function
|
|
|
|
ctx := context.TODO()
|
2015-11-02 18:25:26 -05:00
|
|
|
if !container.IsRunning() {
|
|
|
|
return nil
|
|
|
|
}
|
2020-10-24 16:15:58 -04:00
|
|
|
var wait time.Duration
|
|
|
|
if seconds >= 0 {
|
|
|
|
wait = time.Duration(seconds) * time.Second
|
|
|
|
}
|
|
|
|
success := func() error {
|
|
|
|
daemon.LogContainerEvent(container, "stop")
|
|
|
|
return nil
|
|
|
|
}
|
2016-01-27 03:06:45 -05:00
|
|
|
stopSignal := container.StopSignal()
|
2017-03-30 16:52:40 -04:00
|
|
|
|
2020-10-24 16:15:58 -04:00
|
|
|
// 1. Send a stop signal
|
|
|
|
err := daemon.killPossiblyDeadProcess(container, stopSignal)
|
|
|
|
if err != nil {
|
|
|
|
wait = 2 * time.Second
|
2015-11-02 18:25:26 -05:00
|
|
|
}
|
|
|
|
|
2020-10-24 16:15:58 -04:00
|
|
|
var subCtx context.Context
|
|
|
|
var cancel context.CancelFunc
|
2018-04-16 19:58:42 -04:00
|
|
|
if seconds >= 0 {
|
2020-10-24 16:15:58 -04:00
|
|
|
subCtx, cancel = context.WithTimeout(ctx, wait)
|
|
|
|
} else {
|
|
|
|
subCtx, cancel = context.WithCancel(ctx)
|
2018-04-16 19:58:42 -04:00
|
|
|
}
|
2020-10-24 16:15:58 -04:00
|
|
|
defer cancel()
|
2017-03-30 16:52:40 -04:00
|
|
|
|
2020-10-24 16:15:58 -04:00
|
|
|
if status := <-container.Wait(subCtx, containerpkg.WaitConditionNotRunning); status.Err() == nil {
|
|
|
|
// container did exit, so ignore any previous errors and return
|
|
|
|
return success()
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
// the container has still not exited, and the kill function errored, so log the error here:
|
|
|
|
logrus.WithError(err).WithField("container", container.ID).Errorf("Error sending stop (signal %d) to container", stopSignal)
|
|
|
|
}
|
|
|
|
if seconds < 0 {
|
|
|
|
// if the client requested that we never kill / wait forever, but container.Wait was still
|
|
|
|
// interrupted (parent context cancelled, for example), we should propagate the signal failure
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.WithField("container", container.ID).Infof("Container failed to exit within %d seconds of signal %d - using the force", seconds, stopSignal)
|
|
|
|
// Stop either failed or container didnt exit, so fallback to kill.
|
|
|
|
if err := daemon.Kill(container); err != nil {
|
|
|
|
// got a kill error, but give container 2 more seconds to exit just in case
|
|
|
|
subCtx, cancel := context.WithTimeout(ctx, 2*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
if status := <-container.Wait(subCtx, containerpkg.WaitConditionNotRunning); status.Err() == nil {
|
|
|
|
// container did exit, so ignore error and return
|
|
|
|
return success()
|
2015-11-02 18:25:26 -05:00
|
|
|
}
|
2020-10-24 16:15:58 -04:00
|
|
|
logrus.WithError(err).WithField("container", container.ID).Error("Error killing the container")
|
|
|
|
return err
|
2015-11-02 18:25:26 -05:00
|
|
|
}
|
|
|
|
|
2020-10-24 16:15:58 -04:00
|
|
|
return success()
|
2015-11-02 18:25:26 -05:00
|
|
|
}
|