|
@@ -2,9 +2,10 @@ package daemon
|
|
|
|
|
|
import (
|
|
import (
|
|
"encoding/json"
|
|
"encoding/json"
|
|
|
|
+ "io"
|
|
|
|
+
|
|
"github.com/docker/docker/api/types"
|
|
"github.com/docker/docker/api/types"
|
|
"github.com/docker/docker/daemon/execdriver"
|
|
"github.com/docker/docker/daemon/execdriver"
|
|
- "io"
|
|
|
|
)
|
|
)
|
|
|
|
|
|
func (daemon *Daemon) ContainerStats(name string, stream bool, out io.Writer) error {
|
|
func (daemon *Daemon) ContainerStats(name string, stream bool, out io.Writer) error {
|
|
@@ -12,31 +13,39 @@ func (daemon *Daemon) ContainerStats(name string, stream bool, out io.Writer) er
|
|
if err != nil {
|
|
if err != nil {
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
- var pre_cpu_stats types.CpuStats
|
|
|
|
- for first_v := range updates {
|
|
|
|
- first_update := first_v.(*execdriver.ResourceStats)
|
|
|
|
- first_stats := convertToAPITypes(first_update.Stats)
|
|
|
|
- pre_cpu_stats = first_stats.CpuStats
|
|
|
|
- pre_cpu_stats.SystemUsage = first_update.SystemUsage
|
|
|
|
- break
|
|
|
|
- }
|
|
|
|
- enc := json.NewEncoder(out)
|
|
|
|
- for v := range updates {
|
|
|
|
|
|
+
|
|
|
|
+ var preCpuStats types.CpuStats
|
|
|
|
+ getStat := func(v interface{}) *types.Stats {
|
|
update := v.(*execdriver.ResourceStats)
|
|
update := v.(*execdriver.ResourceStats)
|
|
- ss := convertToAPITypes(update.Stats)
|
|
|
|
- ss.PreCpuStats = pre_cpu_stats
|
|
|
|
|
|
+ ss := convertStatsToAPITypes(update.Stats)
|
|
|
|
+ ss.PreCpuStats = preCpuStats
|
|
ss.MemoryStats.Limit = uint64(update.MemoryLimit)
|
|
ss.MemoryStats.Limit = uint64(update.MemoryLimit)
|
|
ss.Read = update.Read
|
|
ss.Read = update.Read
|
|
ss.CpuStats.SystemUsage = update.SystemUsage
|
|
ss.CpuStats.SystemUsage = update.SystemUsage
|
|
- pre_cpu_stats = ss.CpuStats
|
|
|
|
- if err := enc.Encode(ss); err != nil {
|
|
|
|
|
|
+ preCpuStats = ss.CpuStats
|
|
|
|
+ return ss
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ enc := json.NewEncoder(out)
|
|
|
|
+
|
|
|
|
+ if !stream {
|
|
|
|
+ // prime the cpu stats so they aren't 0 in the final output
|
|
|
|
+ s := getStat(<-updates)
|
|
|
|
+
|
|
|
|
+ // now pull stats again with the cpu stats primed
|
|
|
|
+ s = getStat(<-updates)
|
|
|
|
+ err := enc.Encode(s)
|
|
|
|
+ daemon.UnsubscribeToContainerStats(name, updates)
|
|
|
|
+ return err
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ for v := range updates {
|
|
|
|
+ s := getStat(v)
|
|
|
|
+ if err := enc.Encode(s); err != nil {
|
|
// TODO: handle the specific broken pipe
|
|
// TODO: handle the specific broken pipe
|
|
daemon.UnsubscribeToContainerStats(name, updates)
|
|
daemon.UnsubscribeToContainerStats(name, updates)
|
|
return err
|
|
return err
|
|
}
|
|
}
|
|
- if !stream {
|
|
|
|
- break
|
|
|
|
- }
|
|
|
|
}
|
|
}
|
|
return nil
|
|
return nil
|
|
}
|
|
}
|