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
} |