mirror of
https://github.com/moby/moby.git
synced 2022-11-09 12:21:53 -05:00
Merge pull request #4867 from crosbymichael/clean-shutdown
Cleanly shutdown docker
This commit is contained in:
commit
30ff3fa954
10 changed files with 128 additions and 17 deletions
|
@ -50,8 +50,13 @@ func (ns *linuxNs) Exec(container *libcontainer.Container, term Terminal, args [
|
||||||
if err := command.Start(); err != nil {
|
if err := command.Start(); err != nil {
|
||||||
return -1, err
|
return -1, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
started, err := system.GetProcessStartTime(command.Process.Pid)
|
||||||
|
if err != nil {
|
||||||
|
return -1, err
|
||||||
|
}
|
||||||
ns.logger.Printf("writting pid %d to file\n", command.Process.Pid)
|
ns.logger.Printf("writting pid %d to file\n", command.Process.Pid)
|
||||||
if err := ns.stateWriter.WritePid(command.Process.Pid); err != nil {
|
if err := ns.stateWriter.WritePid(command.Process.Pid, started); err != nil {
|
||||||
command.Process.Kill()
|
command.Process.Kill()
|
||||||
return -1, err
|
return -1, err
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,11 +54,6 @@ func (ns *linuxNs) Init(container *libcontainer.Container, uncleanRootfs, consol
|
||||||
return fmt.Errorf("setctty %s", err)
|
return fmt.Errorf("setctty %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// this is our best effort to let the process know that the parent has died and that it
|
|
||||||
// should it should act on it how it sees fit
|
|
||||||
if err := system.ParentDeathSignal(uintptr(syscall.SIGTERM)); err != nil {
|
|
||||||
return fmt.Errorf("parent death signal %s", err)
|
|
||||||
}
|
|
||||||
if err := setupNetwork(container, context); err != nil {
|
if err := setupNetwork(container, context); err != nil {
|
||||||
return fmt.Errorf("setup networking %s", err)
|
return fmt.Errorf("setup networking %s", err)
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,7 @@ import (
|
||||||
// StateWriter handles writing and deleting the pid file
|
// StateWriter handles writing and deleting the pid file
|
||||||
// on disk
|
// on disk
|
||||||
type StateWriter interface {
|
type StateWriter interface {
|
||||||
WritePid(pid int) error
|
WritePid(pid int, startTime string) error
|
||||||
DeletePid() error
|
DeletePid() error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,10 +19,18 @@ type DefaultStateWriter struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// writePidFile writes the namespaced processes pid to pid in the rootfs for the container
|
// writePidFile writes the namespaced processes pid to pid in the rootfs for the container
|
||||||
func (d *DefaultStateWriter) WritePid(pid int) error {
|
func (d *DefaultStateWriter) WritePid(pid int, startTime string) error {
|
||||||
return ioutil.WriteFile(filepath.Join(d.Root, "pid"), []byte(fmt.Sprint(pid)), 0655)
|
err := ioutil.WriteFile(filepath.Join(d.Root, "pid"), []byte(fmt.Sprint(pid)), 0655)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return ioutil.WriteFile(filepath.Join(d.Root, "start"), []byte(startTime), 0655)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *DefaultStateWriter) DeletePid() error {
|
func (d *DefaultStateWriter) DeletePid() error {
|
||||||
return os.Remove(filepath.Join(d.Root, "pid"))
|
err := os.Remove(filepath.Join(d.Root, "pid"))
|
||||||
|
if serr := os.Remove(filepath.Join(d.Root, "start")); err == nil {
|
||||||
|
err = serr
|
||||||
|
}
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
26
pkg/system/proc.go
Normal file
26
pkg/system/proc.go
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
package system
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io/ioutil"
|
||||||
|
"path/filepath"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
// look in /proc to find the process start time so that we can verify
|
||||||
|
// that this pid has started after ourself
|
||||||
|
func GetProcessStartTime(pid int) (string, error) {
|
||||||
|
data, err := ioutil.ReadFile(filepath.Join("/proc", strconv.Itoa(pid), "stat"))
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
parts := strings.Split(string(data), " ")
|
||||||
|
// the starttime is located at pos 22
|
||||||
|
// from the man page
|
||||||
|
//
|
||||||
|
// starttime %llu (was %lu before Linux 2.6)
|
||||||
|
// (22) The time the process started after system boot. In kernels before Linux 2.6, this
|
||||||
|
// value was expressed in jiffies. Since Linux 2.6, the value is expressed in clock ticks
|
||||||
|
// (divide by sysconf(_SC_CLK_TCK)).
|
||||||
|
return parts[22-1], nil // starts at 1
|
||||||
|
}
|
|
@ -915,7 +915,6 @@ func (container *Container) Stop(seconds int) error {
|
||||||
|
|
||||||
// 1. Send a SIGTERM
|
// 1. Send a SIGTERM
|
||||||
if err := container.KillSig(15); err != nil {
|
if err := container.KillSig(15); err != nil {
|
||||||
utils.Debugf("Error sending kill SIGTERM: %s", err)
|
|
||||||
log.Print("Failed to send SIGTERM to the process, force killing")
|
log.Print("Failed to send SIGTERM to the process, force killing")
|
||||||
if err := container.KillSig(9); err != nil {
|
if err := container.KillSig(9); err != nil {
|
||||||
return err
|
return err
|
||||||
|
|
|
@ -84,6 +84,7 @@ type Driver interface {
|
||||||
Name() string // Driver name
|
Name() string // Driver name
|
||||||
Info(id string) Info // "temporary" hack (until we move state from core to plugins)
|
Info(id string) Info // "temporary" hack (until we move state from core to plugins)
|
||||||
GetPidsForContainer(id string) ([]int, error) // Returns a list of pids for the given container.
|
GetPidsForContainer(id string) ([]int, error) // Returns a list of pids for the given container.
|
||||||
|
Terminate(c *Command) error // kill it with fire
|
||||||
}
|
}
|
||||||
|
|
||||||
// Network settings of the container
|
// Network settings of the container
|
||||||
|
|
|
@ -204,6 +204,10 @@ func (d *driver) Kill(c *execdriver.Command, sig int) error {
|
||||||
return KillLxc(c.ID, sig)
|
return KillLxc(c.ID, sig)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (d *driver) Terminate(c *execdriver.Command) error {
|
||||||
|
return KillLxc(c.ID, 9)
|
||||||
|
}
|
||||||
|
|
||||||
func (d *driver) version() string {
|
func (d *driver) version() string {
|
||||||
var (
|
var (
|
||||||
version string
|
version string
|
||||||
|
|
|
@ -117,9 +117,39 @@ func (d *driver) Run(c *execdriver.Command, pipes *execdriver.Pipes, startCallba
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *driver) Kill(p *execdriver.Command, sig int) error {
|
func (d *driver) Kill(p *execdriver.Command, sig int) error {
|
||||||
err := syscall.Kill(p.Process.Pid, syscall.Signal(sig))
|
return syscall.Kill(p.Process.Pid, syscall.Signal(sig))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *driver) Terminate(p *execdriver.Command) error {
|
||||||
|
// lets check the start time for the process
|
||||||
|
started, err := d.readStartTime(p)
|
||||||
|
if err != nil {
|
||||||
|
// if we don't have the data on disk then we can assume the process is gone
|
||||||
|
// because this is only removed after we know the process has stopped
|
||||||
|
if os.IsNotExist(err) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
currentStartTime, err := system.GetProcessStartTime(p.Process.Pid)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if started == currentStartTime {
|
||||||
|
err = syscall.Kill(p.Process.Pid, 9)
|
||||||
|
}
|
||||||
d.removeContainerRoot(p.ID)
|
d.removeContainerRoot(p.ID)
|
||||||
return err
|
return err
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *driver) readStartTime(p *execdriver.Command) (string, error) {
|
||||||
|
data, err := ioutil.ReadFile(filepath.Join(d.root, p.ID, "start"))
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return string(data), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *driver) Info(id string) execdriver.Info {
|
func (d *driver) Info(id string) execdriver.Info {
|
||||||
|
@ -235,9 +265,9 @@ type dockerStateWriter struct {
|
||||||
callback execdriver.StartCallback
|
callback execdriver.StartCallback
|
||||||
}
|
}
|
||||||
|
|
||||||
func (d *dockerStateWriter) WritePid(pid int) error {
|
func (d *dockerStateWriter) WritePid(pid int, started string) error {
|
||||||
d.c.ContainerPid = pid
|
d.c.ContainerPid = pid
|
||||||
err := d.dsw.WritePid(pid)
|
err := d.dsw.WritePid(pid, started)
|
||||||
if d.callback != nil {
|
if d.callback != nil {
|
||||||
d.callback(d.c)
|
d.callback(d.c)
|
||||||
}
|
}
|
||||||
|
|
|
@ -174,6 +174,7 @@ func (runtime *Runtime) Register(container *Container) error {
|
||||||
if container.State.IsGhost() {
|
if container.State.IsGhost() {
|
||||||
utils.Debugf("killing ghost %s", container.ID)
|
utils.Debugf("killing ghost %s", container.ID)
|
||||||
|
|
||||||
|
existingPid := container.State.Pid
|
||||||
container.State.SetGhost(false)
|
container.State.SetGhost(false)
|
||||||
container.State.SetStopped(0)
|
container.State.SetStopped(0)
|
||||||
|
|
||||||
|
@ -181,9 +182,23 @@ func (runtime *Runtime) Register(container *Container) error {
|
||||||
// no ghost processes are left when docker dies
|
// no ghost processes are left when docker dies
|
||||||
if container.ExecDriver == "" || strings.Contains(container.ExecDriver, "lxc") {
|
if container.ExecDriver == "" || strings.Contains(container.ExecDriver, "lxc") {
|
||||||
lxc.KillLxc(container.ID, 9)
|
lxc.KillLxc(container.ID, 9)
|
||||||
if err := container.Unmount(); err != nil {
|
} else {
|
||||||
utils.Debugf("ghost unmount error %s", err)
|
// use the current driver and ensure that the container is dead x.x
|
||||||
|
cmd := &execdriver.Command{
|
||||||
|
ID: container.ID,
|
||||||
}
|
}
|
||||||
|
var err error
|
||||||
|
cmd.Process, err = os.FindProcess(existingPid)
|
||||||
|
if err != nil {
|
||||||
|
utils.Debugf("cannot find existing process for %d", existingPid)
|
||||||
|
}
|
||||||
|
runtime.execDriver.Terminate(cmd)
|
||||||
|
}
|
||||||
|
if err := container.Unmount(); err != nil {
|
||||||
|
utils.Debugf("ghost unmount error %s", err)
|
||||||
|
}
|
||||||
|
if err := container.ToDisk(); err != nil {
|
||||||
|
utils.Debugf("saving ghost state to disk %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -778,8 +793,36 @@ func NewRuntimeFromDirectory(config *daemonconfig.Config, eng *engine.Engine) (*
|
||||||
return runtime, nil
|
return runtime, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (runtime *Runtime) shutdown() error {
|
||||||
|
group := sync.WaitGroup{}
|
||||||
|
utils.Debugf("starting clean shutdown of all containers...")
|
||||||
|
for _, container := range runtime.List() {
|
||||||
|
c := container
|
||||||
|
if c.State.IsRunning() {
|
||||||
|
utils.Debugf("stopping %s", c.ID)
|
||||||
|
group.Add(1)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
defer group.Done()
|
||||||
|
if err := c.KillSig(15); err != nil {
|
||||||
|
utils.Debugf("kill 15 error for %s - %s", c.ID, err)
|
||||||
|
}
|
||||||
|
c.Wait()
|
||||||
|
utils.Debugf("container stopped %s", c.ID)
|
||||||
|
}()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
group.Wait()
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (runtime *Runtime) Close() error {
|
func (runtime *Runtime) Close() error {
|
||||||
errorsStrings := []string{}
|
errorsStrings := []string{}
|
||||||
|
if err := runtime.shutdown(); err != nil {
|
||||||
|
utils.Errorf("runtime.shutdown(): %s", err)
|
||||||
|
errorsStrings = append(errorsStrings, err.Error())
|
||||||
|
}
|
||||||
if err := portallocator.ReleaseAll(); err != nil {
|
if err := portallocator.ReleaseAll(); err != nil {
|
||||||
utils.Errorf("portallocator.ReleaseAll(): %s", err)
|
utils.Errorf("portallocator.ReleaseAll(): %s", err)
|
||||||
errorsStrings = append(errorsStrings, err.Error())
|
errorsStrings = append(errorsStrings, err.Error())
|
||||||
|
|
|
@ -54,7 +54,7 @@ func InitServer(job *engine.Job) engine.Status {
|
||||||
gosignal.Notify(c, os.Interrupt, syscall.SIGTERM, syscall.SIGQUIT)
|
gosignal.Notify(c, os.Interrupt, syscall.SIGTERM, syscall.SIGQUIT)
|
||||||
go func() {
|
go func() {
|
||||||
sig := <-c
|
sig := <-c
|
||||||
log.Printf("Received signal '%v', exiting\n", sig)
|
log.Printf("Received signal '%v', starting shutdown of docker...\n", sig)
|
||||||
utils.RemovePidFile(srv.runtime.Config().Pidfile)
|
utils.RemovePidFile(srv.runtime.Config().Pidfile)
|
||||||
srv.Close()
|
srv.Close()
|
||||||
os.Exit(0)
|
os.Exit(0)
|
||||||
|
|
Loading…
Reference in a new issue