inspect.go 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. package daemon
  2. import (
  3. "fmt"
  4. "time"
  5. "github.com/docker/docker/api/types/backend"
  6. "github.com/docker/docker/container"
  7. "github.com/docker/docker/daemon/network"
  8. "github.com/docker/engine-api/types"
  9. networktypes "github.com/docker/engine-api/types/network"
  10. "github.com/docker/engine-api/types/versions"
  11. "github.com/docker/engine-api/types/versions/v1p20"
  12. )
  13. // ContainerInspect returns low-level information about a
  14. // container. Returns an error if the container cannot be found, or if
  15. // there is an error getting the data.
  16. func (daemon *Daemon) ContainerInspect(name string, size bool, version string) (interface{}, error) {
  17. switch {
  18. case versions.LessThan(version, "1.20"):
  19. return daemon.containerInspectPre120(name)
  20. case versions.Equal(version, "1.20"):
  21. return daemon.containerInspect120(name)
  22. }
  23. return daemon.ContainerInspectCurrent(name, size)
  24. }
  25. // ContainerInspectCurrent returns low-level information about a
  26. // container in a most recent api version.
  27. func (daemon *Daemon) ContainerInspectCurrent(name string, size bool) (*types.ContainerJSON, error) {
  28. container, err := daemon.GetContainer(name)
  29. if err != nil {
  30. return nil, err
  31. }
  32. container.Lock()
  33. defer container.Unlock()
  34. base, err := daemon.getInspectData(container, size)
  35. if err != nil {
  36. return nil, err
  37. }
  38. mountPoints := addMountPoints(container)
  39. networkSettings := &types.NetworkSettings{
  40. NetworkSettingsBase: types.NetworkSettingsBase{
  41. Bridge: container.NetworkSettings.Bridge,
  42. SandboxID: container.NetworkSettings.SandboxID,
  43. HairpinMode: container.NetworkSettings.HairpinMode,
  44. LinkLocalIPv6Address: container.NetworkSettings.LinkLocalIPv6Address,
  45. LinkLocalIPv6PrefixLen: container.NetworkSettings.LinkLocalIPv6PrefixLen,
  46. Ports: container.NetworkSettings.Ports,
  47. SandboxKey: container.NetworkSettings.SandboxKey,
  48. SecondaryIPAddresses: container.NetworkSettings.SecondaryIPAddresses,
  49. SecondaryIPv6Addresses: container.NetworkSettings.SecondaryIPv6Addresses,
  50. },
  51. DefaultNetworkSettings: daemon.getDefaultNetworkSettings(container.NetworkSettings.Networks),
  52. Networks: container.NetworkSettings.Networks,
  53. }
  54. return &types.ContainerJSON{
  55. ContainerJSONBase: base,
  56. Mounts: mountPoints,
  57. Config: container.Config,
  58. NetworkSettings: networkSettings,
  59. }, nil
  60. }
  61. // containerInspect120 serializes the master version of a container into a json type.
  62. func (daemon *Daemon) containerInspect120(name string) (*v1p20.ContainerJSON, error) {
  63. container, err := daemon.GetContainer(name)
  64. if err != nil {
  65. return nil, err
  66. }
  67. container.Lock()
  68. defer container.Unlock()
  69. base, err := daemon.getInspectData(container, false)
  70. if err != nil {
  71. return nil, err
  72. }
  73. mountPoints := addMountPoints(container)
  74. config := &v1p20.ContainerConfig{
  75. Config: container.Config,
  76. MacAddress: container.Config.MacAddress,
  77. NetworkDisabled: container.Config.NetworkDisabled,
  78. ExposedPorts: container.Config.ExposedPorts,
  79. VolumeDriver: container.HostConfig.VolumeDriver,
  80. }
  81. networkSettings := daemon.getBackwardsCompatibleNetworkSettings(container.NetworkSettings)
  82. return &v1p20.ContainerJSON{
  83. ContainerJSONBase: base,
  84. Mounts: mountPoints,
  85. Config: config,
  86. NetworkSettings: networkSettings,
  87. }, nil
  88. }
  89. func (daemon *Daemon) getInspectData(container *container.Container, size bool) (*types.ContainerJSONBase, error) {
  90. // make a copy to play with
  91. hostConfig := *container.HostConfig
  92. children := daemon.children(container)
  93. hostConfig.Links = nil // do not expose the internal structure
  94. for linkAlias, child := range children {
  95. hostConfig.Links = append(hostConfig.Links, fmt.Sprintf("%s:%s", child.Name, linkAlias))
  96. }
  97. var containerHealth *types.Health
  98. if container.State.Health != nil {
  99. containerHealth = &types.Health{
  100. Status: container.State.Health.Status,
  101. FailingStreak: container.State.Health.FailingStreak,
  102. Log: append([]*types.HealthcheckResult{}, container.State.Health.Log...),
  103. }
  104. }
  105. containerState := &types.ContainerState{
  106. Status: container.State.StateString(),
  107. Running: container.State.Running,
  108. Paused: container.State.Paused,
  109. Restarting: container.State.Restarting,
  110. OOMKilled: container.State.OOMKilled,
  111. Dead: container.State.Dead,
  112. Pid: container.State.Pid,
  113. ExitCode: container.State.ExitCode(),
  114. Error: container.State.Error(),
  115. StartedAt: container.State.StartedAt.Format(time.RFC3339Nano),
  116. FinishedAt: container.State.FinishedAt.Format(time.RFC3339Nano),
  117. Health: containerHealth,
  118. }
  119. contJSONBase := &types.ContainerJSONBase{
  120. ID: container.ID,
  121. Created: container.Created.Format(time.RFC3339Nano),
  122. Path: container.Path,
  123. Args: container.Args,
  124. State: containerState,
  125. Image: container.ImageID.String(),
  126. LogPath: container.LogPath,
  127. Name: container.Name,
  128. RestartCount: container.RestartCount,
  129. Driver: container.Driver,
  130. MountLabel: container.MountLabel,
  131. ProcessLabel: container.ProcessLabel,
  132. ExecIDs: container.GetExecIDs(),
  133. HostConfig: &hostConfig,
  134. }
  135. var (
  136. sizeRw int64
  137. sizeRootFs int64
  138. )
  139. if size {
  140. sizeRw, sizeRootFs = daemon.getSize(container)
  141. contJSONBase.SizeRw = &sizeRw
  142. contJSONBase.SizeRootFs = &sizeRootFs
  143. }
  144. // Now set any platform-specific fields
  145. contJSONBase = setPlatformSpecificContainerFields(container, contJSONBase)
  146. contJSONBase.GraphDriver.Name = container.Driver
  147. graphDriverData, err := container.RWLayer.Metadata()
  148. if err != nil {
  149. return nil, err
  150. }
  151. contJSONBase.GraphDriver.Data = graphDriverData
  152. return contJSONBase, nil
  153. }
  154. // ContainerExecInspect returns low-level information about the exec
  155. // command. An error is returned if the exec cannot be found.
  156. func (daemon *Daemon) ContainerExecInspect(id string) (*backend.ExecInspect, error) {
  157. e, err := daemon.getExecConfig(id)
  158. if err != nil {
  159. return nil, err
  160. }
  161. pc := inspectExecProcessConfig(e)
  162. return &backend.ExecInspect{
  163. ID: e.ID,
  164. Running: e.Running,
  165. ExitCode: e.ExitCode,
  166. ProcessConfig: pc,
  167. OpenStdin: e.OpenStdin,
  168. OpenStdout: e.OpenStdout,
  169. OpenStderr: e.OpenStderr,
  170. CanRemove: e.CanRemove,
  171. ContainerID: e.ContainerID,
  172. DetachKeys: e.DetachKeys,
  173. }, nil
  174. }
  175. // VolumeInspect looks up a volume by name. An error is returned if
  176. // the volume cannot be found.
  177. func (daemon *Daemon) VolumeInspect(name string) (*types.Volume, error) {
  178. v, err := daemon.volumes.Get(name)
  179. if err != nil {
  180. return nil, err
  181. }
  182. apiV := volumeToAPIType(v)
  183. apiV.Mountpoint = v.Path()
  184. apiV.Status = v.Status()
  185. return apiV, nil
  186. }
  187. func (daemon *Daemon) getBackwardsCompatibleNetworkSettings(settings *network.Settings) *v1p20.NetworkSettings {
  188. result := &v1p20.NetworkSettings{
  189. NetworkSettingsBase: types.NetworkSettingsBase{
  190. Bridge: settings.Bridge,
  191. SandboxID: settings.SandboxID,
  192. HairpinMode: settings.HairpinMode,
  193. LinkLocalIPv6Address: settings.LinkLocalIPv6Address,
  194. LinkLocalIPv6PrefixLen: settings.LinkLocalIPv6PrefixLen,
  195. Ports: settings.Ports,
  196. SandboxKey: settings.SandboxKey,
  197. SecondaryIPAddresses: settings.SecondaryIPAddresses,
  198. SecondaryIPv6Addresses: settings.SecondaryIPv6Addresses,
  199. },
  200. DefaultNetworkSettings: daemon.getDefaultNetworkSettings(settings.Networks),
  201. }
  202. return result
  203. }
  204. // getDefaultNetworkSettings creates the deprecated structure that holds the information
  205. // about the bridge network for a container.
  206. func (daemon *Daemon) getDefaultNetworkSettings(networks map[string]*networktypes.EndpointSettings) types.DefaultNetworkSettings {
  207. var settings types.DefaultNetworkSettings
  208. if defaultNetwork, ok := networks["bridge"]; ok {
  209. settings.EndpointID = defaultNetwork.EndpointID
  210. settings.Gateway = defaultNetwork.Gateway
  211. settings.GlobalIPv6Address = defaultNetwork.GlobalIPv6Address
  212. settings.GlobalIPv6PrefixLen = defaultNetwork.GlobalIPv6PrefixLen
  213. settings.IPAddress = defaultNetwork.IPAddress
  214. settings.IPPrefixLen = defaultNetwork.IPPrefixLen
  215. settings.IPv6Gateway = defaultNetwork.IPv6Gateway
  216. settings.MacAddress = defaultNetwork.MacAddress
  217. }
  218. return settings
  219. }