diff options
author | Matthew Heon <matthew.heon@gmail.com> | 2018-03-31 20:53:05 -0400 |
---|---|---|
committer | Atomic Bot <atomic-devel@projectatomic.io> | 2018-04-03 14:57:16 +0000 |
commit | 489d977b22a01c310c3484a24238b7ae158c0d8e (patch) | |
tree | 6a1a8e6097d8d0daff7571f788424bbeed88922c /libpod/container_internal.go | |
parent | 0fb38659b7f87ee021d79dd9b2e2142a86c55deb (diff) | |
download | podman-489d977b22a01c310c3484a24238b7ae158c0d8e.tar.gz podman-489d977b22a01c310c3484a24238b7ae158c0d8e.tar.bz2 podman-489d977b22a01c310c3484a24238b7ae158c0d8e.zip |
Ensure dependencies are running before initializing containers
Signed-off-by: Matthew Heon <matthew.heon@gmail.com>
Closes: #577
Approved by: rhatdan
Diffstat (limited to 'libpod/container_internal.go')
-rw-r--r-- | libpod/container_internal.go | 52 |
1 files changed, 39 insertions, 13 deletions
diff --git a/libpod/container_internal.go b/libpod/container_internal.go index 81dc2c820..523d088d5 100644 --- a/libpod/container_internal.go +++ b/libpod/container_internal.go @@ -352,12 +352,38 @@ func (c *Container) save() error { // Initialize a container, creating it in the runtime func (c *Container) init() error { + deps := c.Dependencies() + depCtrs := make(map[string]*Container, len(deps)) + // Ensure that dependencies are up before we begin + for _, dep := range deps { + // Get the dependency container + depCtr, err := c.runtime.state.Container(dep) + if err != nil { + return errors.Wrapf(err, "error retrieving dependency %s of container %s from state", dep, c.ID()) + } + + // Check the status + state, err := depCtr.State() + if err != nil { + return errors.Wrapf(err, "error retrieving state of dependency %s of container %s", dep, c.ID()) + } + if state != ContainerStateRunning { + return errors.Wrapf(ErrCtrStateInvalid, "dependency %s of container %s is not running, cannot initialize container", dep, c.ID()) + } + depCtrs[dep] = depCtr + + // There is potential race here + // A dep container may stop before this container starts + // We can't really control this, but it probably can't cause + // anything insidious. + } + if err := c.makeBindMounts(); err != nil { return err } // Generate the OCI spec - spec, err := c.generateSpec() + spec, err := c.generateSpec(depCtrs) if err != nil { return err } @@ -848,7 +874,8 @@ func (c *Container) generateHosts() (string, error) { } // Generate spec for a container -func (c *Container) generateSpec() (*spec.Spec, error) { +// Accepts a map of the container's dependencies +func (c *Container) generateSpec(deps map[string]*Container) (*spec.Spec, error) { g := generate.NewFromSpec(c.config.Spec) // If network namespace was requested, add it now @@ -894,37 +921,37 @@ func (c *Container) generateSpec() (*spec.Spec, error) { // Add shared namespaces from other containers if c.config.IPCNsCtr != "" { - if err := c.addNamespaceContainer(&g, IPCNS, c.config.IPCNsCtr, spec.IPCNamespace); err != nil { + if err := c.addNamespaceContainer(&g, IPCNS, deps[c.config.IPCNsCtr], spec.IPCNamespace); err != nil { return nil, err } } if c.config.MountNsCtr != "" { - if err := c.addNamespaceContainer(&g, MountNS, c.config.MountNsCtr, spec.MountNamespace); err != nil { + if err := c.addNamespaceContainer(&g, MountNS, deps[c.config.MountNsCtr], spec.MountNamespace); err != nil { return nil, err } } if c.config.NetNsCtr != "" { - if err := c.addNamespaceContainer(&g, NetNS, c.config.NetNsCtr, spec.NetworkNamespace); err != nil { + if err := c.addNamespaceContainer(&g, NetNS, deps[c.config.NetNsCtr], spec.NetworkNamespace); err != nil { return nil, err } } if c.config.PIDNsCtr != "" { - if err := c.addNamespaceContainer(&g, PIDNS, c.config.PIDNsCtr, string(spec.PIDNamespace)); err != nil { + if err := c.addNamespaceContainer(&g, PIDNS, deps[c.config.PIDNsCtr], string(spec.PIDNamespace)); err != nil { return nil, err } } if c.config.UserNsCtr != "" { - if err := c.addNamespaceContainer(&g, UserNS, c.config.UserNsCtr, spec.UserNamespace); err != nil { + if err := c.addNamespaceContainer(&g, UserNS, deps[c.config.UserNsCtr], spec.UserNamespace); err != nil { return nil, err } } if c.config.UTSNsCtr != "" { - if err := c.addNamespaceContainer(&g, UTSNS, c.config.UTSNsCtr, spec.UTSNamespace); err != nil { + if err := c.addNamespaceContainer(&g, UTSNS, deps[c.config.UTSNsCtr], spec.UTSNamespace); err != nil { return nil, err } } if c.config.CgroupNsCtr != "" { - if err := c.addNamespaceContainer(&g, CgroupNS, c.config.CgroupNsCtr, spec.CgroupNamespace); err != nil { + if err := c.addNamespaceContainer(&g, CgroupNS, deps[c.config.CgroupNsCtr], spec.CgroupNamespace); err != nil { return nil, err } } @@ -952,10 +979,9 @@ func (c *Container) generateSpec() (*spec.Spec, error) { } // Add an existing container's namespace to the spec -func (c *Container) addNamespaceContainer(g *generate.Generator, ns LinuxNS, nsCtrID string, specNS string) error { - nsCtr, err := c.runtime.state.Container(nsCtrID) - if err != nil { - return err +func (c *Container) addNamespaceContainer(g *generate.Generator, ns LinuxNS, nsCtr *Container, specNS string) error { + if nsCtr == nil { + return errors.Wrapf(ErrNoSuchCtr, "nil container passed to addNamespaceContainer") } nsPath, err := nsCtr.NamespacePath(ns) |