2014-05-20 15:36:15 -04:00
|
|
|
package daemon
|
|
|
|
|
|
|
|
import (
|
2014-06-25 18:24:14 -04:00
|
|
|
"fmt"
|
2015-07-26 09:00:53 -04:00
|
|
|
"time"
|
2014-05-20 15:36:15 -04:00
|
|
|
|
2015-04-13 10:17:14 -04:00
|
|
|
"github.com/docker/docker/api/types"
|
2014-05-20 15:36:15 -04:00
|
|
|
)
|
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
// ContainerInspect returns low-level information about a
|
|
|
|
// container. Returns an error if the container cannot be found, or if
|
|
|
|
// there is an error getting the data.
|
2015-04-13 10:17:14 -04:00
|
|
|
func (daemon *Daemon) ContainerInspect(name string) (*types.ContainerJSON, error) {
|
|
|
|
container, err := daemon.Get(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2014-12-16 18:06:35 -05:00
|
|
|
}
|
2014-05-30 21:13:37 -04:00
|
|
|
|
2015-04-13 10:17:14 -04:00
|
|
|
container.Lock()
|
|
|
|
defer container.Unlock()
|
2014-06-25 18:24:14 -04:00
|
|
|
|
2015-06-02 17:37:59 -04:00
|
|
|
base, err := daemon.getInspectData(container)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2015-07-16 17:14:58 -04:00
|
|
|
mountPoints := addMountPoints(container)
|
2015-06-03 15:21:38 -04:00
|
|
|
|
|
|
|
return &types.ContainerJSON{base, mountPoints, container.Config}, nil
|
2015-06-02 17:37:59 -04:00
|
|
|
}
|
|
|
|
|
2015-08-24 13:57:39 -04:00
|
|
|
// ContainerInspect120 serializes the master version of a container into a json type.
|
|
|
|
func (daemon *Daemon) ContainerInspect120(name string) (*types.ContainerJSON120, error) {
|
|
|
|
container, err := daemon.Get(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
container.Lock()
|
|
|
|
defer container.Unlock()
|
|
|
|
|
|
|
|
base, err := daemon.getInspectData(container)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
mountPoints := addMountPoints(container)
|
|
|
|
config := &types.ContainerConfig120{
|
|
|
|
container.Config,
|
|
|
|
container.hostConfig.VolumeDriver,
|
|
|
|
}
|
|
|
|
|
|
|
|
return &types.ContainerJSON120{base, mountPoints, config}, nil
|
|
|
|
}
|
|
|
|
|
2015-06-02 17:37:59 -04:00
|
|
|
func (daemon *Daemon) getInspectData(container *Container) (*types.ContainerJSONBase, error) {
|
2015-04-13 10:17:14 -04:00
|
|
|
// make a copy to play with
|
|
|
|
hostConfig := *container.hostConfig
|
2014-12-23 17:03:20 -05:00
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
if children, err := daemon.children(container.Name); err == nil {
|
2014-12-16 18:06:35 -05:00
|
|
|
for linkAlias, child := range children {
|
2015-04-13 10:17:14 -04:00
|
|
|
hostConfig.Links = append(hostConfig.Links, fmt.Sprintf("%s:%s", child.Name, linkAlias))
|
2014-06-25 18:24:14 -04:00
|
|
|
}
|
2014-12-16 18:06:35 -05:00
|
|
|
}
|
2015-02-04 14:04:58 -05:00
|
|
|
// we need this trick to preserve empty log driver, so
|
|
|
|
// container will use daemon defaults even if daemon change them
|
2015-04-13 10:17:14 -04:00
|
|
|
if hostConfig.LogConfig.Type == "" {
|
2015-08-17 11:35:34 -04:00
|
|
|
hostConfig.LogConfig.Type = daemon.defaultLogConfig.Type
|
|
|
|
}
|
|
|
|
|
2015-08-17 18:27:44 -04:00
|
|
|
if len(hostConfig.LogConfig.Config) == 0 {
|
2015-08-17 11:35:34 -04:00
|
|
|
hostConfig.LogConfig.Config = daemon.defaultLogConfig.Config
|
2015-02-04 14:04:58 -05:00
|
|
|
}
|
2014-06-25 18:24:14 -04:00
|
|
|
|
2015-04-13 10:17:14 -04:00
|
|
|
containerState := &types.ContainerState{
|
2015-07-27 20:48:27 -04:00
|
|
|
Status: container.State.StateString(),
|
2015-04-13 10:17:14 -04:00
|
|
|
Running: container.State.Running,
|
|
|
|
Paused: container.State.Paused,
|
|
|
|
Restarting: container.State.Restarting,
|
|
|
|
OOMKilled: container.State.OOMKilled,
|
|
|
|
Dead: container.State.Dead,
|
|
|
|
Pid: container.State.Pid,
|
|
|
|
ExitCode: container.State.ExitCode,
|
|
|
|
Error: container.State.Error,
|
2015-07-26 09:00:53 -04:00
|
|
|
StartedAt: container.State.StartedAt.Format(time.RFC3339Nano),
|
|
|
|
FinishedAt: container.State.FinishedAt.Format(time.RFC3339Nano),
|
2015-04-13 10:17:14 -04:00
|
|
|
}
|
2014-06-25 18:24:14 -04:00
|
|
|
|
2015-06-02 17:37:59 -04:00
|
|
|
contJSONBase := &types.ContainerJSONBase{
|
2015-07-23 05:40:54 -04:00
|
|
|
ID: container.ID,
|
2015-07-26 09:00:53 -04:00
|
|
|
Created: container.Created.Format(time.RFC3339Nano),
|
2015-04-13 10:17:14 -04:00
|
|
|
Path: container.Path,
|
|
|
|
Args: container.Args,
|
|
|
|
State: containerState,
|
|
|
|
Image: container.ImageID,
|
|
|
|
NetworkSettings: container.NetworkSettings,
|
|
|
|
LogPath: container.LogPath,
|
|
|
|
Name: container.Name,
|
|
|
|
RestartCount: container.RestartCount,
|
|
|
|
Driver: container.Driver,
|
|
|
|
ExecDriver: container.ExecDriver,
|
|
|
|
MountLabel: container.MountLabel,
|
|
|
|
ProcessLabel: container.ProcessLabel,
|
2015-07-30 17:01:53 -04:00
|
|
|
ExecIDs: container.getExecIDs(),
|
2015-04-13 10:17:14 -04:00
|
|
|
HostConfig: &hostConfig,
|
2014-05-20 15:36:15 -04:00
|
|
|
}
|
2015-04-13 10:17:14 -04:00
|
|
|
|
2015-07-16 17:14:58 -04:00
|
|
|
// Now set any platform-specific fields
|
|
|
|
contJSONBase = setPlatformSpecificContainerFields(container, contJSONBase)
|
|
|
|
|
2015-06-15 14:05:10 -04:00
|
|
|
contJSONBase.GraphDriver.Name = container.Driver
|
|
|
|
graphDriverData, err := daemon.driver.GetMetadata(container.ID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
contJSONBase.GraphDriver.Data = graphDriverData
|
|
|
|
|
2015-06-02 17:37:59 -04:00
|
|
|
return contJSONBase, nil
|
2014-05-20 15:36:15 -04:00
|
|
|
}
|
2014-11-17 18:50:09 -05:00
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
// ContainerExecInspect returns low-level information about the exec
|
|
|
|
// command. An error is returned if the exec cannot be found.
|
|
|
|
func (daemon *Daemon) ContainerExecInspect(id string) (*ExecConfig, error) {
|
2014-11-17 18:50:09 -05:00
|
|
|
eConfig, err := daemon.getExecConfig(id)
|
|
|
|
if err != nil {
|
2015-04-11 18:15:34 -04:00
|
|
|
return nil, err
|
2014-11-17 18:50:09 -05:00
|
|
|
}
|
2015-04-11 18:15:34 -04:00
|
|
|
return eConfig, nil
|
2014-11-17 18:50:09 -05:00
|
|
|
}
|
2015-06-12 09:25:32 -04:00
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
// VolumeInspect looks up a volume by name. An error is returned if
|
|
|
|
// the volume cannot be found.
|
2015-06-12 09:25:32 -04:00
|
|
|
func (daemon *Daemon) VolumeInspect(name string) (*types.Volume, error) {
|
|
|
|
v, err := daemon.volumes.Get(name)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return volumeToAPIType(v), nil
|
|
|
|
}
|