daemon/stats.go
4f174aa7
 package daemon
 
 import (
 	"encoding/json"
855a056a
 	"io"
 
96123a1f
 	"github.com/docker/docker/api/types"
2f46b760
 	"github.com/docker/docker/daemon/execdriver"
8b40e44c
 	"github.com/docker/libnetwork/sandbox"
c86189d5
 	"github.com/opencontainers/runc/libcontainer"
4f174aa7
 )
 
1cbf5a54
 type ContainerStatsConfig struct {
 	Stream    bool
 	OutStream io.Writer
 	Stop      <-chan bool
 }
 
 func (daemon *Daemon) ContainerStats(name string, config *ContainerStatsConfig) error {
65a05634
 	updates, err := daemon.SubscribeToContainerStats(name)
4f174aa7
 	if err != nil {
c79b9bab
 		return err
4f174aa7
 	}
855a056a
 
1cbf5a54
 	if config.Stream {
 		config.OutStream.Write(nil)
 	}
 
855a056a
 	var preCpuStats types.CpuStats
 	getStat := func(v interface{}) *types.Stats {
2f46b760
 		update := v.(*execdriver.ResourceStats)
8b40e44c
 		// Retrieve the nw statistics from libnetwork and inject them in the Stats
 		if nwStats, err := daemon.getNetworkStats(name); err == nil {
 			update.Stats.Interfaces = nwStats
 		}
855a056a
 		ss := convertStatsToAPITypes(update.Stats)
 		ss.PreCpuStats = preCpuStats
4f174aa7
 		ss.MemoryStats.Limit = uint64(update.MemoryLimit)
 		ss.Read = update.Read
 		ss.CpuStats.SystemUsage = update.SystemUsage
855a056a
 		preCpuStats = ss.CpuStats
 		return ss
 	}
 
1cbf5a54
 	enc := json.NewEncoder(config.OutStream)
855a056a
 
1cbf5a54
 	defer daemon.UnsubscribeToContainerStats(name, updates)
855a056a
 
1cbf5a54
 	noStreamFirstFrame := true
 	for {
 		select {
 		case v, ok := <-updates:
 			if !ok {
 				return nil
 			}
 
 			s := getStat(v)
 			if !config.Stream && noStreamFirstFrame {
 				// prime the cpu stats so they aren't 0 in the final output
 				noStreamFirstFrame = false
 				continue
 			}
 
 			if err := enc.Encode(s); err != nil {
 				return err
 			}
855a056a
 
1cbf5a54
 			if !config.Stream {
 				return nil
 			}
 		case <-config.Stop:
 			return nil
4f174aa7
 		}
 	}
 }
8b40e44c
 
 func (daemon *Daemon) getNetworkStats(name string) ([]*libcontainer.NetworkInterface, error) {
 	var list []*libcontainer.NetworkInterface
 
 	c, err := daemon.Get(name)
 	if err != nil {
 		return list, err
 	}
 
 	nw, err := daemon.netController.NetworkByID(c.NetworkSettings.NetworkID)
 	if err != nil {
 		return list, err
 	}
 	ep, err := nw.EndpointByID(c.NetworkSettings.EndpointID)
 	if err != nil {
 		return list, err
 	}
 
 	stats, err := ep.Statistics()
 	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
 }
 
 func convertLnNetworkStats(name string, stats *sandbox.InterfaceStatistics) *libcontainer.NetworkInterface {
 	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
 }