mirror of
https://github.com/containers/podman.git
synced 2025-05-22 01:27:07 +08:00
Merge pull request #7604 from vrothberg/fix-7406
system df: fix image-size calculations
This commit is contained in:
126
libpod/image/df.go
Normal file
126
libpod/image/df.go
Normal file
@ -0,0 +1,126 @@
|
|||||||
|
package image
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/containers/image/v5/docker/reference"
|
||||||
|
)
|
||||||
|
|
||||||
|
// DiskUsageStat gives disk-usage statistics for a specific image.
|
||||||
|
type DiskUsageStat struct {
|
||||||
|
// ID of the image.
|
||||||
|
ID string
|
||||||
|
// Repository of the first recorded name of the image.
|
||||||
|
Repository string
|
||||||
|
// Tag of the first recorded name of the image.
|
||||||
|
Tag string
|
||||||
|
// Created is the creation time of the image.
|
||||||
|
Created time.Time
|
||||||
|
// SharedSize is the amount of space shared with another image.
|
||||||
|
SharedSize uint64
|
||||||
|
// UniqueSize is the amount of space used only by this image.
|
||||||
|
UniqueSize uint64
|
||||||
|
// Size is the total size of the image (i.e., the sum of the shared and
|
||||||
|
// unique size).
|
||||||
|
Size uint64
|
||||||
|
// Number of containers using the image.
|
||||||
|
Containers int
|
||||||
|
}
|
||||||
|
|
||||||
|
// DiskUsage returns disk-usage statistics for the specified slice of images.
|
||||||
|
func (ir *Runtime) DiskUsage(ctx context.Context, images []*Image) ([]DiskUsageStat, error) {
|
||||||
|
stats := make([]DiskUsageStat, len(images))
|
||||||
|
|
||||||
|
// Build a layerTree to quickly compute (and cache!) parent/child
|
||||||
|
// relations.
|
||||||
|
tree, err := ir.layerTree()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculate the stats for each image.
|
||||||
|
for i, img := range images {
|
||||||
|
stat, err := diskUsageForImage(ctx, img, tree)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
stats[i] = *stat
|
||||||
|
}
|
||||||
|
|
||||||
|
return stats, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// diskUsageForImage returns the disk-usage statistics for the spcified image.
|
||||||
|
func diskUsageForImage(ctx context.Context, image *Image, tree *layerTree) (*DiskUsageStat, error) {
|
||||||
|
stat := DiskUsageStat{
|
||||||
|
ID: image.ID(),
|
||||||
|
Created: image.Created(),
|
||||||
|
}
|
||||||
|
|
||||||
|
// Repository and tag.
|
||||||
|
var name, repository, tag string
|
||||||
|
for _, n := range image.Names() {
|
||||||
|
if len(n) > 0 {
|
||||||
|
name = n
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if len(name) > 0 {
|
||||||
|
named, err := reference.ParseNormalizedNamed(name)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
repository = named.Name()
|
||||||
|
if tagged, isTagged := named.(reference.NamedTagged); isTagged {
|
||||||
|
tag = tagged.Tag()
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
repository = "<none>"
|
||||||
|
tag = "<none>"
|
||||||
|
}
|
||||||
|
stat.Repository = repository
|
||||||
|
stat.Tag = tag
|
||||||
|
|
||||||
|
// Shared, unique and total size.
|
||||||
|
parent, err := tree.parent(ctx, image)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
childIDs, err := tree.children(ctx, image, false)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
// Optimistically set unique size to the full size of the image.
|
||||||
|
size, err := image.Size(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
stat.UniqueSize = *size
|
||||||
|
|
||||||
|
if len(childIDs) > 0 {
|
||||||
|
// If we have children, we share everything.
|
||||||
|
stat.SharedSize = stat.UniqueSize
|
||||||
|
stat.UniqueSize = 0
|
||||||
|
} else if parent != nil {
|
||||||
|
// If we have no children but a parent, remove the parent
|
||||||
|
// (shared) size from the unique one.
|
||||||
|
size, err := parent.Size(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
stat.UniqueSize -= *size
|
||||||
|
stat.SharedSize = *size
|
||||||
|
}
|
||||||
|
|
||||||
|
stat.Size = stat.SharedSize + stat.UniqueSize
|
||||||
|
|
||||||
|
// Number of containers using the image.
|
||||||
|
containers, err := image.Containers()
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
stat.Containers = len(containers)
|
||||||
|
|
||||||
|
return &stat, nil
|
||||||
|
}
|
@ -17,7 +17,6 @@ import (
|
|||||||
"github.com/containers/podman/v2/pkg/rootless"
|
"github.com/containers/podman/v2/pkg/rootless"
|
||||||
"github.com/containers/podman/v2/pkg/util"
|
"github.com/containers/podman/v2/pkg/util"
|
||||||
"github.com/containers/podman/v2/utils"
|
"github.com/containers/podman/v2/utils"
|
||||||
"github.com/docker/distribution/reference"
|
|
||||||
"github.com/pkg/errors"
|
"github.com/pkg/errors"
|
||||||
"github.com/sirupsen/logrus"
|
"github.com/sirupsen/logrus"
|
||||||
"github.com/spf13/cobra"
|
"github.com/spf13/cobra"
|
||||||
@ -199,71 +198,32 @@ func (ic *ContainerEngine) SystemDf(ctx context.Context, options entities.System
|
|||||||
dfImages = []*entities.SystemDfImageReport{}
|
dfImages = []*entities.SystemDfImageReport{}
|
||||||
)
|
)
|
||||||
|
|
||||||
// Get Images and iterate them
|
// Compute disk-usage stats for all local images.
|
||||||
imgs, err := ic.Libpod.ImageRuntime().GetImages()
|
imgs, err := ic.Libpod.ImageRuntime().GetImages()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
for _, i := range imgs {
|
|
||||||
var sharedSize uint64
|
|
||||||
cons, err := i.Containers()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
imageSize, err := i.Size(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
uniqueSize := *imageSize
|
|
||||||
|
|
||||||
parent, err := i.GetParent(ctx)
|
imageStats, err := ic.Libpod.ImageRuntime().DiskUsage(ctx, imgs)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if parent != nil {
|
|
||||||
parentSize, err := parent.Size(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
uniqueSize = *parentSize - *imageSize
|
|
||||||
sharedSize = *imageSize - uniqueSize
|
|
||||||
}
|
|
||||||
var name, repository, tag string
|
|
||||||
for _, n := range i.Names() {
|
|
||||||
if len(n) > 0 {
|
|
||||||
name = n
|
|
||||||
break
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(name) > 0 {
|
|
||||||
named, err := reference.ParseNormalizedNamed(name)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
repository = named.Name()
|
|
||||||
if tagged, isTagged := named.(reference.NamedTagged); isTagged {
|
|
||||||
tag = tagged.Tag()
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
repository = "<none>"
|
|
||||||
tag = "<none>"
|
|
||||||
}
|
|
||||||
|
|
||||||
|
for _, stat := range imageStats {
|
||||||
report := entities.SystemDfImageReport{
|
report := entities.SystemDfImageReport{
|
||||||
Repository: repository,
|
Repository: stat.Repository,
|
||||||
Tag: tag,
|
Tag: stat.Tag,
|
||||||
ImageID: i.ID(),
|
ImageID: stat.ID,
|
||||||
Created: i.Created(),
|
Created: stat.Created,
|
||||||
Size: int64(*imageSize),
|
Size: int64(stat.Size),
|
||||||
SharedSize: int64(sharedSize),
|
SharedSize: int64(stat.SharedSize),
|
||||||
UniqueSize: int64(uniqueSize),
|
UniqueSize: int64(stat.UniqueSize),
|
||||||
Containers: len(cons),
|
Containers: stat.Containers,
|
||||||
}
|
}
|
||||||
dfImages = append(dfImages, &report)
|
dfImages = append(dfImages, &report)
|
||||||
}
|
}
|
||||||
|
|
||||||
// GetContainers and iterate them
|
// Get Containers and iterate them
|
||||||
cons, err := ic.Libpod.GetAllContainers()
|
cons, err := ic.Libpod.GetAllContainers()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
|
Reference in New Issue
Block a user