2015-01-07 21:02:08 -05:00
|
|
|
package daemon
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2016-02-02 20:59:11 -05:00
|
|
|
"errors"
|
2016-06-07 03:45:21 -04:00
|
|
|
"fmt"
|
2016-02-02 20:59:11 -05:00
|
|
|
"runtime"
|
2016-09-07 19:08:51 -04:00
|
|
|
"time"
|
2015-06-12 11:27:39 -04:00
|
|
|
|
2016-03-25 14:33:54 -04:00
|
|
|
"golang.org/x/net/context"
|
|
|
|
|
2016-09-06 14:18:12 -04:00
|
|
|
"github.com/docker/docker/api/types"
|
2016-01-27 17:09:42 -05:00
|
|
|
"github.com/docker/docker/api/types/backend"
|
2016-09-06 14:18:12 -04:00
|
|
|
"github.com/docker/docker/api/types/versions"
|
|
|
|
"github.com/docker/docker/api/types/versions/v1p20"
|
2016-05-24 11:49:26 -04:00
|
|
|
"github.com/docker/docker/container"
|
2015-12-19 09:43:10 -05:00
|
|
|
"github.com/docker/docker/pkg/ioutils"
|
2015-01-07 21:02:08 -05:00
|
|
|
)
|
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
// ContainerStats writes information about the container to the stream
|
|
|
|
// given in the config object.
|
2016-03-25 14:33:54 -04:00
|
|
|
func (daemon *Daemon) ContainerStats(ctx context.Context, prefixOrName string, config *backend.ContainerStatsConfig) error {
|
2016-06-07 03:45:21 -04:00
|
|
|
if runtime.GOOS == "solaris" {
|
|
|
|
return fmt.Errorf("%+v does not support stats", runtime.GOOS)
|
|
|
|
}
|
2015-12-30 12:20:41 -05:00
|
|
|
// Remote API version (used for backwards compatibility)
|
2016-04-19 10:56:54 -04:00
|
|
|
apiVersion := config.Version
|
2016-02-02 20:59:11 -05:00
|
|
|
|
2015-12-11 12:39:28 -05:00
|
|
|
container, err := daemon.GetContainer(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-12-19 09:43:10 -05:00
|
|
|
outStream := config.OutStream
|
2015-06-16 10:33:49 -04:00
|
|
|
if config.Stream {
|
2015-12-19 09:43:10 -05:00
|
|
|
wf := ioutils.NewWriteFlusher(outStream)
|
|
|
|
defer wf.Close()
|
|
|
|
wf.Flush()
|
|
|
|
outStream = wf
|
2015-06-16 10:33:49 -04:00
|
|
|
}
|
|
|
|
|
2015-07-30 17:01:53 -04:00
|
|
|
var preCPUStats types.CPUStats
|
2016-09-07 19:08:51 -04:00
|
|
|
var preRead time.Time
|
2015-08-24 05:17:15 -04:00
|
|
|
getStatJSON := func(v interface{}) *types.StatsJSON {
|
2016-04-18 14:12:07 -04:00
|
|
|
ss := v.(types.StatsJSON)
|
2016-11-03 02:20:46 -04:00
|
|
|
ss.Name = container.Name
|
|
|
|
ss.ID = container.ID
|
2015-07-30 17:01:53 -04:00
|
|
|
ss.PreCPUStats = preCPUStats
|
2016-09-07 19:08:51 -04:00
|
|
|
ss.PreRead = preRead
|
2015-07-30 17:01:53 -04:00
|
|
|
preCPUStats = ss.CPUStats
|
2016-09-07 19:08:51 -04:00
|
|
|
preRead = ss.Read
|
2016-04-18 14:12:07 -04:00
|
|
|
return &ss
|
2015-06-12 11:27:39 -04:00
|
|
|
}
|
|
|
|
|
2015-12-19 09:43:10 -05:00
|
|
|
enc := json.NewEncoder(outStream)
|
2015-06-12 11:27:39 -04:00
|
|
|
|
2015-10-28 18:15:22 -04:00
|
|
|
updates := daemon.subscribeToContainerStats(container)
|
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-10-30 18:04:21 -04:00
|
|
|
var statsJSON interface{}
|
|
|
|
statsJSONPost120 := getStatJSON(v)
|
2016-04-19 10:56:54 -04:00
|
|
|
if versions.LessThan(apiVersion, "1.21") {
|
2016-09-07 19:08:51 -04:00
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
return errors.New("API versions pre v1.21 do not support stats on Windows")
|
|
|
|
}
|
2015-08-24 05:17:15 -04:00
|
|
|
var (
|
|
|
|
rxBytes uint64
|
|
|
|
rxPackets uint64
|
|
|
|
rxErrors uint64
|
|
|
|
rxDropped uint64
|
|
|
|
txBytes uint64
|
|
|
|
txPackets uint64
|
|
|
|
txErrors uint64
|
|
|
|
txDropped uint64
|
|
|
|
)
|
2015-10-30 18:04:21 -04:00
|
|
|
for _, v := range statsJSONPost120.Networks {
|
2015-08-24 05:17:15 -04:00
|
|
|
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-10-30 18:04:21 -04:00
|
|
|
statsJSON = &v1p20.StatsJSON{
|
|
|
|
Stats: statsJSONPost120.Stats,
|
2015-08-24 05:17:15 -04:00
|
|
|
Network: types.NetworkStats{
|
|
|
|
RxBytes: rxBytes,
|
|
|
|
RxPackets: rxPackets,
|
|
|
|
RxErrors: rxErrors,
|
|
|
|
RxDropped: rxDropped,
|
|
|
|
TxBytes: txBytes,
|
|
|
|
TxPackets: txPackets,
|
|
|
|
TxErrors: txErrors,
|
|
|
|
TxDropped: txDropped,
|
|
|
|
},
|
|
|
|
}
|
2015-10-30 18:04:21 -04:00
|
|
|
} else {
|
|
|
|
statsJSON = statsJSONPost120
|
2015-08-24 05:17:15 -04:00
|
|
|
}
|
|
|
|
|
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
|
|
|
|
}
|
2016-03-25 14:33:54 -04:00
|
|
|
case <-ctx.Done():
|
2015-06-16 10:33:49 -04:00
|
|
|
return nil
|
2015-01-07 21:02:08 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-05-24 11:49:26 -04:00
|
|
|
|
|
|
|
func (daemon *Daemon) subscribeToContainerStats(c *container.Container) chan interface{} {
|
|
|
|
return daemon.statsCollector.collect(c)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (daemon *Daemon) unsubscribeToContainerStats(c *container.Container, ch chan interface{}) {
|
|
|
|
daemon.statsCollector.unsubscribe(c, ch)
|
|
|
|
}
|
2016-05-27 05:32:26 -04:00
|
|
|
|
|
|
|
// GetContainerStats collects all the stats published by a container
|
|
|
|
func (daemon *Daemon) GetContainerStats(container *container.Container) (*types.StatsJSON, error) {
|
|
|
|
stats, err := daemon.stats(container)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-09-07 19:08:51 -04:00
|
|
|
// We already have the network stats on Windows directly from HCS.
|
|
|
|
if !container.Config.NetworkDisabled && runtime.GOOS != "windows" {
|
2016-08-20 18:43:15 -04:00
|
|
|
if stats.Networks, err = daemon.getNetworkStats(container); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2016-05-27 05:32:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
return stats, nil
|
|
|
|
}
|