inspect.go 9.2 KB

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