Ensure a reliable way to kill ghost containers on reboot
Docker-DCO-1.1-Signed-off-by: Michael Crosby <michael@crosbymichael.com> (github: crosbymichael)
This commit is contained in:
parent
345365e7b0
commit
b16b1a04ff
3 changed files with 44 additions and 5 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
|
||||||
}
|
}
|
||||||
|
|
|
@ -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
system/proc.go
Normal file
26
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
|
||||||
|
}
|
Loading…
Reference in a new issue