Call start in containerd

This fixes a sync issue when the containerd api returns after a
container has started.  It fixes it by calling the runtime start inside
containerd after the oom handler has been setup.

Signed-off-by: Michael Crosby <crosbymichael@gmail.com>
This commit is contained in:
Michael Crosby 2016-06-24 12:02:53 -07:00
parent 14e79494da
commit 654c537d38
6 changed files with 54 additions and 30 deletions

View file

@ -136,13 +136,6 @@ func start(log *os.File) error {
Height: uint16(msg.Height), Height: uint16(msg.Height),
} }
term.SetWinsize(p.console.Fd(), &ws) term.SetWinsize(p.console.Fd(), &ws)
case 2:
// tell runtime to execute the init process
if err := p.start(); err != nil {
p.delete()
p.Wait()
return err
}
} }
} }
} }

View file

@ -206,26 +206,6 @@ func (p *process) create() error {
return nil return nil
} }
func (p *process) start() error {
cwd, err := os.Getwd()
if err != nil {
return err
}
logPath := filepath.Join(cwd, "log.json")
args := append([]string{
"--log", logPath,
"--log-format", "json",
}, p.state.RuntimeArgs...)
args = append(args, "start", p.id)
cmd := exec.Command(p.runtime, args...)
cmd.Dir = p.bundle
cmd.Stdin = p.stdio.stdin
cmd.Stdout = p.stdio.stdout
cmd.Stderr = p.stdio.stderr
cmd.SysProcAttr = setPDeathSig()
return cmd.Run()
}
func (p *process) pid() int { func (p *process) pid() int {
return p.containerPid return p.containerPid
} }

View file

@ -2,6 +2,7 @@ package main
import ( import (
"fmt" "fmt"
"os"
"path/filepath" "path/filepath"
"syscall" "syscall"
"time" "time"
@ -326,6 +327,9 @@ func (cs *ContainerdSuite) TestOOM(t *check.C) {
spec.Linux.Resources.Memory = &ocs.Memory{ spec.Linux.Resources.Memory = &ocs.Memory{
Limit: &limit, Limit: &limit,
} }
if swapEnabled() {
spec.Linux.Resources.Memory.Swap = &limit
}
}); err != nil { }); err != nil {
t.Fatal(err) t.Fatal(err)
} }
@ -362,7 +366,7 @@ func (cs *ContainerdSuite) TestOOM(t *check.C) {
evt.Timestamp = e.Timestamp evt.Timestamp = e.Timestamp
t.Assert(*e, checker.Equals, evt) t.Assert(*e, checker.Equals, evt)
case <-time.After(60 * time.Second): case <-time.After(60 * time.Second):
t.Fatal("Container took more than 10 seconds to terminate") t.Fatalf("Container took more than 60 seconds to %s", evt.Type)
} }
} }
} }
@ -479,3 +483,8 @@ func (cs *ContainerdSuite) TestRestart(t *check.C) {
t.Assert(containers[i].Status, checker.Equals, "running") t.Assert(containers[i].Status, checker.Equals, "running")
} }
} }
func swapEnabled() bool {
_, err := os.Stat("/sys/fs/cgroup/memory/memory.memsw.limit_in_bytes")
return err == nil
}

View file

@ -260,8 +260,33 @@ func (p *process) Signal(s os.Signal) error {
// This should only be called on the process with ID "init" // This should only be called on the process with ID "init"
func (p *process) Start() error { func (p *process) Start() error {
if p.ID() == InitProcessID { if p.ID() == InitProcessID {
_, err := fmt.Fprintf(p.controlPipe, "%d %d %d\n", 2, 0, 0) var (
return err errC = make(chan error, 1)
shimExit = make(chan struct{}, 1)
args = append(p.container.runtimeArgs, "start", p.container.id)
cmd = exec.Command(p.container.runtime, args...)
)
go func() {
out, err := cmd.CombinedOutput()
if err != nil {
errC <- fmt.Errorf("%s: %q", err.Error(), out)
}
errC <- nil
}()
go func() {
p.Wait()
close(shimExit)
}()
select {
case err := <-errC:
if err != nil {
return err
}
case <-shimExit:
cmd.Process.Kill()
cmd.Wait()
return ErrShimExited
}
} }
return nil return nil
} }

View file

@ -20,6 +20,9 @@ var (
// ErrContainerStartTimeout is returned if a container takes too // ErrContainerStartTimeout is returned if a container takes too
// long to start // long to start
ErrContainerStartTimeout = errors.New("containerd: container did not start before the specified timeout") ErrContainerStartTimeout = errors.New("containerd: container did not start before the specified timeout")
// ErrShimExited is returned if the shim or the contianer's init process
// exits before completing
ErrShimExited = errors.New("containerd: shim exited before container process was started")
errNoPidFile = errors.New("containerd: no process pid file found") errNoPidFile = errors.New("containerd: no process pid file found")
errInvalidPidInt = errors.New("containerd: process pid is invalid") errInvalidPidInt = errors.New("containerd: process pid is invalid")

View file

@ -62,9 +62,23 @@ func (w *worker) Start() {
} }
if err := w.s.monitorProcess(process); err != nil { if err := w.s.monitorProcess(process); err != nil {
logrus.WithField("error", err).Error("containerd: add process to monitor") logrus.WithField("error", err).Error("containerd: add process to monitor")
t.Err <- err
evt := &DeleteTask{
ID: t.Container.ID(),
NoEvent: true,
}
w.s.SendTask(evt)
continue
} }
if err := process.Start(); err != nil { if err := process.Start(); err != nil {
logrus.WithField("error", err).Error("containerd: start init process") logrus.WithField("error", err).Error("containerd: start init process")
t.Err <- err
evt := &DeleteTask{
ID: t.Container.ID(),
NoEvent: true,
}
w.s.SendTask(evt)
continue
} }
ContainerStartTimer.UpdateSince(started) ContainerStartTimer.UpdateSince(started)
t.Err <- nil t.Err <- nil