2015-01-07 21:02:08 -05:00
|
|
|
package daemon
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2015-06-12 11:27:39 -04:00
|
|
|
"io"
|
|
|
|
|
2015-05-30 13:25:51 -04:00
|
|
|
"github.com/docker/docker/api/types"
|
2015-09-24 16:56:57 -04:00
|
|
|
"github.com/docker/docker/api/types/versions/v1p20"
|
2015-01-19 18:29:42 -05:00
|
|
|
"github.com/docker/docker/daemon/execdriver"
|
2015-08-24 05:17:15 -04:00
|
|
|
"github.com/docker/docker/pkg/version"
|
2015-09-02 19:43:28 -04:00
|
|
|
"github.com/docker/libnetwork/osl"
|
2015-07-16 19:00:55 -04:00
|
|
|
"github.com/opencontainers/runc/libcontainer"
|
2015-01-07 21:02:08 -05:00
|
|
|
)
|
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
// ContainerStatsConfig holds information for configuring the runtime
|
|
|
|
// behavior of a daemon.ContainerStats() call.
|
2015-06-16 10:33:49 -04:00
|
|
|
type ContainerStatsConfig struct {
|
|
|
|
Stream bool
|
|
|
|
OutStream io.Writer
|
|
|
|
Stop <-chan bool
|
2015-08-24 05:17:15 -04:00
|
|
|
Version version.Version
|
2015-06-16 10:33:49 -04:00
|
|
|
}
|
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
// ContainerStats writes information about the container to the stream
|
|
|
|
// given in the config object.
|
2015-09-29 13:51:40 -04:00
|
|
|
func (daemon *Daemon) ContainerStats(prefixOrName string, config *ContainerStatsConfig) error {
|
2015-09-16 17:16:55 -04:00
|
|
|
|
2015-09-29 13:51:40 -04:00
|
|
|
container, err := daemon.Get(prefixOrName)
|
2015-09-16 17:16:55 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the container is not running and requires no stream, return an empty stats.
|
|
|
|
if !container.IsRunning() && !config.Stream {
|
|
|
|
return json.NewEncoder(config.OutStream).Encode(&types.Stats{})
|
|
|
|
}
|
|
|
|
|
2015-09-08 10:12:46 -04:00
|
|
|
updates, err := daemon.subscribeToContainerStats(container)
|
2015-01-07 21:02:08 -05:00
|
|
|
if err != nil {
|
2015-03-25 03:44:12 -04:00
|
|
|
return err
|
2015-01-07 21:02:08 -05:00
|
|
|
}
|
2015-06-12 11:27:39 -04:00
|
|
|
|
2015-06-16 10:33:49 -04:00
|
|
|
if config.Stream {
|
2015-09-19 05:58:43 -04:00
|
|
|
// Write an empty chunk of data.
|
|
|
|
// This is to ensure that the HTTP status code is sent immediately,
|
|
|
|
// even if the container has not yet produced any data.
|
2015-06-16 10:33:49 -04:00
|
|
|
config.OutStream.Write(nil)
|
|
|
|
}
|
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
var preCPUStats types.CPUStats
|
2015-08-24 05:17:15 -04:00
|
|
|
getStatJSON := func(v interface{}) *types.StatsJSON {
|
2015-01-19 18:29:42 -05:00
|
|
|
update := v.(*execdriver.ResourceStats)
|
2015-07-01 12:54:26 -04:00
|
|
|
// Retrieve the nw statistics from libnetwork and inject them in the Stats
|
2015-09-08 10:12:46 -04:00
|
|
|
if nwStats, err := daemon.getNetworkStats(container); err == nil {
|
2015-07-01 12:54:26 -04:00
|
|
|
update.Stats.Interfaces = nwStats
|
|
|
|
}
|
2015-06-12 11:27:39 -04:00
|
|
|
ss := convertStatsToAPITypes(update.Stats)
|
2015-07-30 17:01:53 -04:00
|
|
|
ss.PreCPUStats = preCPUStats
|
2015-01-07 21:02:08 -05:00
|
|
|
ss.MemoryStats.Limit = uint64(update.MemoryLimit)
|
|
|
|
ss.Read = update.Read
|
2015-07-23 05:40:54 -04:00
|
|
|
ss.CPUStats.SystemUsage = update.SystemUsage
|
2015-07-30 17:01:53 -04:00
|
|
|
preCPUStats = ss.CPUStats
|
2015-06-12 11:27:39 -04:00
|
|
|
return ss
|
|
|
|
}
|
|
|
|
|
2015-06-16 10:33:49 -04:00
|
|
|
enc := json.NewEncoder(config.OutStream)
|
2015-06-12 11:27:39 -04:00
|
|
|
|
2015-09-08 10:12:46 -04:00
|
|
|
defer daemon.unsubscribeToContainerStats(container, updates)
|
2015-06-12 11:27:39 -04:00
|
|
|
|
2015-06-16 10:33:49 -04:00
|
|
|
noStreamFirstFrame := true
|
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case v, ok := <-updates:
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2015-08-24 05:17:15 -04:00
|
|
|
statsJSON := getStatJSON(v)
|
|
|
|
if config.Version.LessThan("1.21") {
|
|
|
|
var (
|
|
|
|
rxBytes uint64
|
|
|
|
rxPackets uint64
|
|
|
|
rxErrors uint64
|
|
|
|
rxDropped uint64
|
|
|
|
txBytes uint64
|
|
|
|
txPackets uint64
|
|
|
|
txErrors uint64
|
|
|
|
txDropped uint64
|
|
|
|
)
|
|
|
|
for _, v := range statsJSON.Networks {
|
|
|
|
rxBytes += v.RxBytes
|
|
|
|
rxPackets += v.RxPackets
|
|
|
|
rxErrors += v.RxErrors
|
|
|
|
rxDropped += v.RxDropped
|
|
|
|
txBytes += v.TxBytes
|
|
|
|
txPackets += v.TxPackets
|
|
|
|
txErrors += v.TxErrors
|
|
|
|
txDropped += v.TxDropped
|
|
|
|
}
|
2015-09-24 16:56:57 -04:00
|
|
|
statsJSONPre121 := &v1p20.StatsJSON{
|
2015-08-24 05:17:15 -04:00
|
|
|
Stats: statsJSON.Stats,
|
|
|
|
Network: types.NetworkStats{
|
|
|
|
RxBytes: rxBytes,
|
|
|
|
RxPackets: rxPackets,
|
|
|
|
RxErrors: rxErrors,
|
|
|
|
RxDropped: rxDropped,
|
|
|
|
TxBytes: txBytes,
|
|
|
|
TxPackets: txPackets,
|
|
|
|
TxErrors: txErrors,
|
|
|
|
TxDropped: txDropped,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if !config.Stream && noStreamFirstFrame {
|
|
|
|
// prime the cpu stats so they aren't 0 in the final output
|
|
|
|
noStreamFirstFrame = false
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := enc.Encode(statsJSONPre121); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !config.Stream {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-16 10:33:49 -04:00
|
|
|
if !config.Stream && noStreamFirstFrame {
|
|
|
|
// prime the cpu stats so they aren't 0 in the final output
|
|
|
|
noStreamFirstFrame = false
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2015-08-24 05:17:15 -04:00
|
|
|
if err := enc.Encode(statsJSON); err != nil {
|
2015-06-16 10:33:49 -04:00
|
|
|
return err
|
|
|
|
}
|
2015-06-12 11:27:39 -04:00
|
|
|
|
2015-06-16 10:33:49 -04:00
|
|
|
if !config.Stream {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
case <-config.Stop:
|
|
|
|
return nil
|
2015-01-07 21:02:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-07-01 12:54:26 -04:00
|
|
|
|
2015-09-08 10:12:46 -04:00
|
|
|
func (daemon *Daemon) getNetworkStats(c *Container) ([]*libcontainer.NetworkInterface, error) {
|
2015-07-01 12:54:26 -04:00
|
|
|
var list []*libcontainer.NetworkInterface
|
|
|
|
|
2015-09-02 19:43:28 -04:00
|
|
|
sb, err := daemon.netController.SandboxByID(c.NetworkSettings.SandboxID)
|
2015-07-01 12:54:26 -04:00
|
|
|
if err != nil {
|
|
|
|
return list, err
|
|
|
|
}
|
|
|
|
|
2015-09-02 19:43:28 -04:00
|
|
|
stats, err := sb.Statistics()
|
2015-07-01 12:54:26 -04:00
|
|
|
if err != nil {
|
|
|
|
return list, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Convert libnetwork nw stats into libcontainer nw stats
|
|
|
|
for ifName, ifStats := range stats {
|
|
|
|
list = append(list, convertLnNetworkStats(ifName, ifStats))
|
|
|
|
}
|
|
|
|
|
|
|
|
return list, nil
|
|
|
|
}
|
|
|
|
|
2015-09-02 19:43:28 -04:00
|
|
|
func convertLnNetworkStats(name string, stats *osl.InterfaceStatistics) *libcontainer.NetworkInterface {
|
2015-07-01 12:54:26 -04:00
|
|
|
n := &libcontainer.NetworkInterface{Name: name}
|
|
|
|
n.RxBytes = stats.RxBytes
|
|
|
|
n.RxPackets = stats.RxPackets
|
|
|
|
n.RxErrors = stats.RxErrors
|
|
|
|
n.RxDropped = stats.RxDropped
|
|
|
|
n.TxBytes = stats.TxBytes
|
|
|
|
n.TxPackets = stats.TxPackets
|
|
|
|
n.TxErrors = stats.TxErrors
|
|
|
|
n.TxDropped = stats.TxDropped
|
|
|
|
return n
|
|
|
|
}
|