Report child error better (and later)

We use a unix domain socketpair instead of a pipe for the sync pipe,
which allows us to use two-way shutdown. After sending the
context we shut down the write side which lets the child know
it finished reading.

We then block on a read in the parent for the child closing the file
(ensuring we close our version of it too) to sync for when the child
is finished initializing. If the read is non-empty we assume this
is an error report and fail with an error. Otherwise we continue as
before.

This also means we're now calling back the start callback later,
meaning at that point its more likely to have succeeded, as well as
having consumed all the container resources (like volume mounts,
making it safe to e.g. unmount them when the start callback is
called).

Docker-DCO-1.1-Signed-off-by: Alexander Larsson <alexl@redhat.com> (github: alexlarsson)
This commit is contained in:
Alexander Larsson 2014-06-16 15:30:42 +02:00 committed by Michael Crosby
parent f975ff9159
commit ca9544522e
3 changed files with 44 additions and 5 deletions

View File

@ -32,6 +32,7 @@ func Exec(container *libcontainer.Config, term Terminal, rootfs, dataPath string
if err != nil { if err != nil {
return -1, err return -1, err
} }
defer syncPipe.Close()
if container.Tty { if container.Tty {
master, console, err = system.CreateMasterAndConsole() master, console, err = system.CreateMasterAndConsole()
@ -52,6 +53,9 @@ func Exec(container *libcontainer.Config, term Terminal, rootfs, dataPath string
return -1, err return -1, err
} }
// Now we passed the pipe to the child, close our side
syncPipe.CloseChild()
started, err := system.GetProcessStartTime(command.Process.Pid) started, err := system.GetProcessStartTime(command.Process.Pid)
if err != nil { if err != nil {
return -1, err return -1, err
@ -90,7 +94,9 @@ func Exec(container *libcontainer.Config, term Terminal, rootfs, dataPath string
defer libcontainer.DeleteState(dataPath) defer libcontainer.DeleteState(dataPath)
// Sync with child // Sync with child
syncPipe.Close() if err := syncPipe.BlockOnChild(); err != nil {
return -1, err
}
if startCallback != nil { if startCallback != nil {
startCallback() startCallback()

View File

@ -27,7 +27,13 @@ import (
// Move this to libcontainer package. // Move this to libcontainer package.
// Init is the init process that first runs inside a new namespace to setup mounts, users, networking, // Init is the init process that first runs inside a new namespace to setup mounts, users, networking,
// and other options required for the new container. // and other options required for the new container.
func Init(container *libcontainer.Config, uncleanRootfs, consolePath string, syncPipe *SyncPipe, args []string) error { func Init(container *libcontainer.Config, uncleanRootfs, consolePath string, syncPipe *SyncPipe, args []string) (err error) {
defer func() {
if err != nil {
syncPipe.ReportError(err)
}
}()
rootfs, err := utils.ResolveRootfs(uncleanRootfs) rootfs, err := utils.ResolveRootfs(uncleanRootfs)
if err != nil { if err != nil {
return err return err
@ -42,10 +48,8 @@ func Init(container *libcontainer.Config, uncleanRootfs, consolePath string, syn
// We always read this as it is a way to sync with the parent as well // We always read this as it is a way to sync with the parent as well
networkState, err := syncPipe.ReadFromParent() networkState, err := syncPipe.ReadFromParent()
if err != nil { if err != nil {
syncPipe.Close()
return err return err
} }
syncPipe.Close()
if consolePath != "" { if consolePath != "" {
if err := console.OpenAndDup(consolePath); err != nil { if err := console.OpenAndDup(consolePath); err != nil {

View File

@ -5,6 +5,7 @@ import (
"fmt" "fmt"
"io/ioutil" "io/ioutil"
"os" "os"
"syscall"
"github.com/docker/libcontainer/network" "github.com/docker/libcontainer/network"
) )
@ -18,10 +19,14 @@ type SyncPipe struct {
func NewSyncPipe() (s *SyncPipe, err error) { func NewSyncPipe() (s *SyncPipe, err error) {
s = &SyncPipe{} s = &SyncPipe{}
s.child, s.parent, err = os.Pipe()
fds, err := syscall.Socketpair(syscall.AF_LOCAL, syscall.SOCK_STREAM|syscall.SOCK_CLOEXEC, 0)
if err != nil { if err != nil {
return nil, err return nil, err
} }
s.child = os.NewFile(uintptr(fds[0]), "child syncpipe")
s.parent = os.NewFile(uintptr(fds[1]), "parent syncpipe")
return s, nil return s, nil
} }
@ -51,6 +56,18 @@ func (s *SyncPipe) SendToChild(networkState *network.NetworkState) error {
return err return err
} }
s.parent.Write(data) s.parent.Write(data)
syscall.Shutdown(int(s.parent.Fd()), syscall.SHUT_WR)
return nil
}
func (s *SyncPipe) BlockOnChild() error {
data, err := ioutil.ReadAll(s.parent)
if err != nil {
return nil
}
if len(data) > 0 {
return fmt.Errorf("Child error: %s", string(data))
}
return nil return nil
} }
@ -69,6 +86,11 @@ func (s *SyncPipe) ReadFromParent() (*network.NetworkState, error) {
} }
func (s *SyncPipe) ReportError(err error) {
s.child.Write([]byte(err.Error()))
s.CloseChild()
}
func (s *SyncPipe) Close() error { func (s *SyncPipe) Close() error {
if s.parent != nil { if s.parent != nil {
s.parent.Close() s.parent.Close()
@ -78,3 +100,10 @@ func (s *SyncPipe) Close() error {
} }
return nil return nil
} }
func (s *SyncPipe) CloseChild() {
if s.child != nil {
s.child.Close()
s.child = nil
}
}