123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115 |
- package containerd
- import (
- "context"
- "sort"
- cplatforms "github.com/containerd/containerd/platforms"
- "github.com/docker/distribution/reference"
- imagetype "github.com/docker/docker/api/types/image"
- "github.com/docker/docker/errdefs"
- "github.com/docker/docker/pkg/platforms"
- "github.com/opencontainers/image-spec/identity"
- ocispec "github.com/opencontainers/image-spec/specs-go/v1"
- "github.com/pkg/errors"
- )
- // ImageHistory returns a slice of HistoryResponseItem structures for the
- // specified image name by walking the image lineage.
- func (i *ImageService) ImageHistory(ctx context.Context, name string) ([]*imagetype.HistoryResponseItem, error) {
- desc, err := i.resolveImage(ctx, name)
- if err != nil {
- return nil, err
- }
- cs := i.client.ContentStore()
- // TODO: pass platform in from the CLI
- platform := platforms.AllPlatformsWithPreference(cplatforms.Default())
- var presentImages []ocispec.Image
- err = i.walkImageManifests(ctx, desc, func(img *ImageManifest) error {
- conf, err := img.Config(ctx)
- if err != nil {
- return err
- }
- var ociimage ocispec.Image
- if err := readConfig(ctx, cs, conf, &ociimage); err != nil {
- return err
- }
- presentImages = append(presentImages, ociimage)
- return nil
- })
- if err != nil {
- return nil, err
- }
- if len(presentImages) == 0 {
- return nil, errdefs.NotFound(errors.New("failed to find image manifest"))
- }
- sort.SliceStable(presentImages, func(i, j int) bool {
- return platform.Less(presentImages[i].Platform, presentImages[j].Platform)
- })
- ociimage := presentImages[0]
- var (
- history []*imagetype.HistoryResponseItem
- sizes []int64
- )
- s := i.client.SnapshotService(i.snapshotter)
- diffIDs := ociimage.RootFS.DiffIDs
- for i := range diffIDs {
- chainID := identity.ChainID(diffIDs[0 : i+1]).String()
- use, err := s.Usage(ctx, chainID)
- if err != nil {
- return nil, err
- }
- sizes = append(sizes, use.Size)
- }
- for _, h := range ociimage.History {
- size := int64(0)
- if !h.EmptyLayer {
- if len(sizes) == 0 {
- return nil, errors.New("unable to find the size of the layer")
- }
- size = sizes[0]
- sizes = sizes[1:]
- }
- history = append([]*imagetype.HistoryResponseItem{{
- ID: "<missing>",
- Comment: h.Comment,
- CreatedBy: h.CreatedBy,
- Created: h.Created.Unix(),
- Size: size,
- Tags: nil,
- }}, history...)
- }
- if len(history) != 0 {
- history[0].ID = desc.Target.Digest.String()
- tagged, err := i.client.ImageService().List(ctx, "target.digest=="+desc.Target.Digest.String())
- if err != nil {
- return nil, err
- }
- var tags []string
- for _, t := range tagged {
- if isDanglingImage(t) {
- continue
- }
- name, err := reference.ParseNamed(t.Name)
- if err != nil {
- return nil, err
- }
- tags = append(tags, reference.FamiliarString(name))
- }
- history[0].Tags = tags
- }
- return history, nil
- }
|