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: "", 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 }