mirror of https://github.com/containers/podman.git
libpod: Split out the common code from GetContainerStats
This moves the cgroups code to a new method getPlatformContainerStats. [NO NEW TESTS NEEDED] Signed-off-by: Doug Rabson <dfr@rabson.org>
This commit is contained in:
parent
47bd9e8110
commit
b3e978e43b
|
@ -0,0 +1,49 @@
|
||||||
|
//go:build linux || freebsd
|
||||||
|
// +build linux freebsd
|
||||||
|
|
||||||
|
package libpod
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
|
||||||
|
"github.com/containers/podman/v4/libpod/define"
|
||||||
|
)
|
||||||
|
|
||||||
|
// GetContainerStats gets the running stats for a given container.
|
||||||
|
// The previousStats is used to correctly calculate cpu percentages. You
|
||||||
|
// should pass nil if there is no previous stat for this container.
|
||||||
|
func (c *Container) GetContainerStats(previousStats *define.ContainerStats) (*define.ContainerStats, error) {
|
||||||
|
stats := new(define.ContainerStats)
|
||||||
|
stats.ContainerID = c.ID()
|
||||||
|
stats.Name = c.Name()
|
||||||
|
|
||||||
|
if c.config.NoCgroups {
|
||||||
|
return nil, fmt.Errorf("cannot run top on container %s as it did not create a cgroup: %w", c.ID(), define.ErrNoCgroups)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !c.batched {
|
||||||
|
c.lock.Lock()
|
||||||
|
defer c.lock.Unlock()
|
||||||
|
if err := c.syncContainer(); err != nil {
|
||||||
|
return stats, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// returns stats with the fields' default values respective of their type
|
||||||
|
if c.state.State != define.ContainerStateRunning && c.state.State != define.ContainerStatePaused {
|
||||||
|
return stats, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if previousStats == nil {
|
||||||
|
previousStats = &define.ContainerStats{
|
||||||
|
// if we have no prev stats use the container start time as prev time
|
||||||
|
// otherwise we cannot correctly calculate the CPU percentage
|
||||||
|
SystemNano: uint64(c.state.StartedTime.UnixNano()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := c.getPlatformContainerStats(stats, previousStats); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return stats, nil
|
||||||
|
}
|
|
@ -16,57 +16,33 @@ import (
|
||||||
"github.com/containers/podman/v4/libpod/define"
|
"github.com/containers/podman/v4/libpod/define"
|
||||||
)
|
)
|
||||||
|
|
||||||
// GetContainerStats gets the running stats for a given container.
|
// getPlatformContainerStats gets the platform-specific running stats
|
||||||
// The previousStats is used to correctly calculate cpu percentages. You
|
// for a given container. The previousStats is used to correctly
|
||||||
// should pass nil if there is no previous stat for this container.
|
// calculate cpu percentages. You should pass nil if there is no
|
||||||
func (c *Container) GetContainerStats(previousStats *define.ContainerStats) (*define.ContainerStats, error) {
|
// previous stat for this container.
|
||||||
stats := new(define.ContainerStats)
|
func (c *Container) getPlatformContainerStats(stats *define.ContainerStats, previousStats *define.ContainerStats) error {
|
||||||
stats.ContainerID = c.ID()
|
|
||||||
stats.Name = c.Name()
|
|
||||||
|
|
||||||
if c.config.NoCgroups {
|
if c.config.NoCgroups {
|
||||||
return nil, fmt.Errorf("cannot run top on container %s as it did not create a cgroup: %w", c.ID(), define.ErrNoCgroups)
|
return fmt.Errorf("cannot run top on container %s as it did not create a cgroup: %w", c.ID(), define.ErrNoCgroups)
|
||||||
}
|
|
||||||
|
|
||||||
if !c.batched {
|
|
||||||
c.lock.Lock()
|
|
||||||
defer c.lock.Unlock()
|
|
||||||
if err := c.syncContainer(); err != nil {
|
|
||||||
return stats, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// returns stats with the fields' default values respective of their type
|
|
||||||
if c.state.State != define.ContainerStateRunning && c.state.State != define.ContainerStatePaused {
|
|
||||||
return stats, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
if previousStats == nil {
|
|
||||||
previousStats = &define.ContainerStats{
|
|
||||||
// if we have no prev stats use the container start time as prev time
|
|
||||||
// otherwise we cannot correctly calculate the CPU percentage
|
|
||||||
SystemNano: uint64(c.state.StartedTime.UnixNano()),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
cgroupPath, err := c.cGroupPath()
|
cgroupPath, err := c.cGroupPath()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return err
|
||||||
}
|
}
|
||||||
cgroup, err := cgroups.Load(cgroupPath)
|
cgroup, err := cgroups.Load(cgroupPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return stats, fmt.Errorf("unable to load cgroup at %s: %w", cgroupPath, err)
|
return fmt.Errorf("unable to load cgroup at %s: %w", cgroupPath, err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ubuntu does not have swap memory in cgroups because swap is often not enabled.
|
// Ubuntu does not have swap memory in cgroups because swap is often not enabled.
|
||||||
cgroupStats, err := cgroup.Stat()
|
cgroupStats, err := cgroup.Stat()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return stats, fmt.Errorf("unable to obtain cgroup stats: %w", err)
|
return fmt.Errorf("unable to obtain cgroup stats: %w", err)
|
||||||
}
|
}
|
||||||
conState := c.state.State
|
conState := c.state.State
|
||||||
netStats, err := getContainerNetIO(c)
|
netStats, err := getContainerNetIO(c)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the current total usage in the cgroup is less than what was previously
|
// If the current total usage in the cgroup is less than what was previously
|
||||||
|
@ -103,7 +79,7 @@ func (c *Container) GetContainerStats(previousStats *define.ContainerStats) (*de
|
||||||
stats.NetOutput = 0
|
stats.NetOutput = 0
|
||||||
}
|
}
|
||||||
|
|
||||||
return stats, nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// getMemory limit returns the memory limit for a container
|
// getMemory limit returns the memory limit for a container
|
||||||
|
@ -133,7 +109,9 @@ func (c *Container) getMemLimit() uint64 {
|
||||||
|
|
||||||
// calculateCPUPercent calculates the cpu usage using the latest measurement in stats.
|
// calculateCPUPercent calculates the cpu usage using the latest measurement in stats.
|
||||||
// previousCPU is the last value of stats.CPU.Usage.Total measured at the time previousSystem.
|
// previousCPU is the last value of stats.CPU.Usage.Total measured at the time previousSystem.
|
||||||
// (now - previousSystem) is the time delta in nanoseconds, between the measurement in previousCPU
|
//
|
||||||
|
// (now - previousSystem) is the time delta in nanoseconds, between the measurement in previousCPU
|
||||||
|
//
|
||||||
// and the updated value in stats.
|
// and the updated value in stats.
|
||||||
func calculateCPUPercent(stats *runccgroup.Stats, previousCPU, now, previousSystem uint64) float64 {
|
func calculateCPUPercent(stats *runccgroup.Stats, previousCPU, now, previousSystem uint64) float64 {
|
||||||
var (
|
var (
|
||||||
|
|
Loading…
Reference in New Issue