2018-02-05 21:05:59 +00:00
|
|
|
package daemon // import "github.com/docker/docker/daemon"
|
2014-07-31 20:40:15 +00:00
|
|
|
|
2015-09-16 18:56:26 +00:00
|
|
|
import (
|
2017-03-30 20:52:40 +00:00
|
|
|
"context"
|
2015-11-02 23:25:26 +00:00
|
|
|
"time"
|
|
|
|
|
2017-03-31 03:01:41 +00:00
|
|
|
containerpkg "github.com/docker/docker/container"
|
2018-01-11 19:53:06 +00:00
|
|
|
"github.com/docker/docker/errdefs"
|
2017-07-19 14:20:13 +00:00
|
|
|
"github.com/pkg/errors"
|
2017-07-26 21:42:13 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2015-09-16 18:56:26 +00:00
|
|
|
)
|
2015-03-25 07:44:12 +00:00
|
|
|
|
2018-04-16 23:58:42 +00: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 17:39:28 +00:00
|
|
|
container, err := daemon.GetContainer(name)
|
2014-12-16 23:06:35 +00:00
|
|
|
if err != nil {
|
2015-03-25 07:44:12 +00:00
|
|
|
return err
|
2014-07-31 20:40:15 +00:00
|
|
|
}
|
2014-12-16 23:06:35 +00:00
|
|
|
if !container.IsRunning() {
|
2017-07-19 14:20:13 +00:00
|
|
|
return containerNotModifiedError{running: false}
|
2014-12-16 23:06:35 +00:00
|
|
|
}
|
2018-04-16 23:58:42 +00:00
|
|
|
if timeout == nil {
|
2016-06-07 03:29:05 +00:00
|
|
|
stopTimeout := container.StopTimeout()
|
2018-04-16 23:58:42 +00:00
|
|
|
timeout = &stopTimeout
|
2016-06-07 03:29:05 +00:00
|
|
|
}
|
2018-04-16 23:58:42 +00:00
|
|
|
if err := daemon.containerStop(container, *timeout); err != nil {
|
2017-11-29 04:09:37 +00:00
|
|
|
return errdefs.System(errors.Wrapf(err, "cannot stop container: %s", name))
|
2014-12-16 23:06:35 +00:00
|
|
|
}
|
2015-03-25 07:44:12 +00:00
|
|
|
return nil
|
2014-07-31 20:40:15 +00:00
|
|
|
}
|
2015-11-02 23:25:26 +00:00
|
|
|
|
2018-04-16 23:58:42 +00:00
|
|
|
// containerStop sends a stop signal, waits, sends a kill signal.
|
2017-03-31 03:01:41 +00:00
|
|
|
func (daemon *Daemon) containerStop(container *containerpkg.Container, seconds int) error {
|
2015-11-02 23:25:26 +00:00
|
|
|
if !container.IsRunning() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-01-27 08:06:45 +00:00
|
|
|
stopSignal := container.StopSignal()
|
|
|
|
// 1. Send a stop signal
|
|
|
|
if err := daemon.killPossiblyDeadProcess(container, stopSignal); err != nil {
|
2016-08-15 23:51:45 +00:00
|
|
|
// While normally we might "return err" here we're not going to
|
|
|
|
// because if we can't stop the container by this point then
|
2016-12-23 12:48:25 +00:00
|
|
|
// it's probably because it's already stopped. Meaning, between
|
2016-08-15 23:51:45 +00:00
|
|
|
// the time of the IsRunning() call above and now it stopped.
|
|
|
|
// Also, since the err return will be environment specific we can't
|
|
|
|
// look for any particular (common) error that would indicate
|
|
|
|
// that the process is already dead vs something else going wrong.
|
|
|
|
// So, instead we'll give it up to 2 more seconds to complete and if
|
|
|
|
// by that time the container is still running, then the error
|
|
|
|
// we got is probably valid and so we force kill it.
|
2017-03-30 20:52:40 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
|
2017-03-31 03:01:41 +00:00
|
|
|
if status := <-container.Wait(ctx, containerpkg.WaitConditionNotRunning); status.Err() != nil {
|
2016-08-15 23:51:45 +00:00
|
|
|
logrus.Infof("Container failed to stop after sending signal %d to the process, force killing", stopSignal)
|
|
|
|
if err := daemon.killPossiblyDeadProcess(container, 9); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2015-11-02 23:25:26 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// 2. Wait for the process to exit on its own
|
2018-04-16 23:58:42 +00:00
|
|
|
ctx := context.Background()
|
|
|
|
if seconds >= 0 {
|
|
|
|
var cancel context.CancelFunc
|
|
|
|
ctx, cancel = context.WithTimeout(ctx, time.Duration(seconds)*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
}
|
2017-03-30 20:52:40 +00:00
|
|
|
|
2017-03-31 03:01:41 +00:00
|
|
|
if status := <-container.Wait(ctx, containerpkg.WaitConditionNotRunning); status.Err() != nil {
|
2016-01-27 08:06:45 +00:00
|
|
|
logrus.Infof("Container %v failed to exit within %d seconds of signal %d - using the force", container.ID, seconds, stopSignal)
|
2015-11-02 23:25:26 +00:00
|
|
|
// 3. If it doesn't, then send SIGKILL
|
|
|
|
if err := daemon.Kill(container); err != nil {
|
2017-03-30 20:52:40 +00:00
|
|
|
// Wait without a timeout, ignore result.
|
2017-09-06 08:54:24 +00:00
|
|
|
<-container.Wait(context.Background(), containerpkg.WaitConditionNotRunning)
|
2015-11-02 23:25:26 +00:00
|
|
|
logrus.Warn(err) // Don't return error because we only care that container is stopped, not what function stopped it
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-03 17:33:13 +00:00
|
|
|
daemon.LogContainerEvent(container, "stop")
|
2015-11-02 23:25:26 +00:00
|
|
|
return nil
|
|
|
|
}
|