63422515ba
This allows the legacy builder to apply changes to the `FROM scratch`
layer.
Signed-off-by: Paweł Gronowski <pawel.gronowski@docker.com>
(cherry picked from commit dfaff9598c
)
Signed-off-by: Sebastiaan van Stijn <github@gone.nl>
120 lines
3.5 KiB
Go
120 lines
3.5 KiB
Go
package containerd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/containerd/containerd"
|
|
cerrdefs "github.com/containerd/containerd/errdefs"
|
|
containerdimages "github.com/containerd/containerd/images"
|
|
"github.com/containerd/containerd/leases"
|
|
"github.com/containerd/containerd/platforms"
|
|
"github.com/containerd/containerd/snapshots"
|
|
"github.com/docker/docker/errdefs"
|
|
"github.com/opencontainers/image-spec/identity"
|
|
ocispec "github.com/opencontainers/image-spec/specs-go/v1"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
// PrepareSnapshot prepares a snapshot from a parent image for a container
|
|
func (i *ImageService) PrepareSnapshot(ctx context.Context, id string, parentImage string, platform *ocispec.Platform) error {
|
|
var parentSnapshot string
|
|
if parentImage != "" {
|
|
img, err := i.resolveImage(ctx, parentImage)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
cs := i.client.ContentStore()
|
|
|
|
matcher := platforms.Default()
|
|
if platform != nil {
|
|
matcher = platforms.Only(*platform)
|
|
}
|
|
|
|
platformImg := containerd.NewImageWithPlatform(i.client, img, matcher)
|
|
unpacked, err := platformImg.IsUnpacked(ctx, i.snapshotter)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !unpacked {
|
|
if err := platformImg.Unpack(ctx, i.snapshotter); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
desc, err := containerdimages.Config(ctx, cs, img.Target, matcher)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
diffIDs, err := containerdimages.RootFS(ctx, cs, desc)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
parentSnapshot = identity.ChainID(diffIDs).String()
|
|
}
|
|
|
|
// Add a lease so that containerd doesn't garbage collect our snapshot
|
|
ls := i.client.LeasesService()
|
|
lease, err := ls.Create(ctx, leases.WithID(id))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := ls.AddResource(ctx, lease, leases.Resource{
|
|
ID: id,
|
|
Type: "snapshots/" + i.StorageDriver(),
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
s := i.client.SnapshotService(i.StorageDriver())
|
|
_, err = s.Prepare(ctx, id, parentSnapshot)
|
|
return err
|
|
}
|
|
|
|
// calculateSnapshotParentUsage returns the usage of all ancestors of the
|
|
// provided snapshot. It doesn't include the size of the snapshot itself.
|
|
func calculateSnapshotParentUsage(ctx context.Context, snapshotter snapshots.Snapshotter, snapshotID string) (snapshots.Usage, error) {
|
|
info, err := snapshotter.Stat(ctx, snapshotID)
|
|
if err != nil {
|
|
if cerrdefs.IsNotFound(err) {
|
|
return snapshots.Usage{}, errdefs.NotFound(err)
|
|
}
|
|
return snapshots.Usage{}, errdefs.System(errors.Wrapf(err, "snapshotter.Stat failed for %s", snapshotID))
|
|
}
|
|
if info.Parent == "" {
|
|
return snapshots.Usage{}, errdefs.NotFound(fmt.Errorf("snapshot %s has no parent", snapshotID))
|
|
}
|
|
|
|
return calculateSnapshotTotalUsage(ctx, snapshotter, info.Parent)
|
|
}
|
|
|
|
// calculateSnapshotTotalUsage returns the total usage of that snapshot
|
|
// including all of its ancestors.
|
|
func calculateSnapshotTotalUsage(ctx context.Context, snapshotter snapshots.Snapshotter, snapshotID string) (snapshots.Usage, error) {
|
|
var total snapshots.Usage
|
|
next := snapshotID
|
|
|
|
for next != "" {
|
|
usage, err := snapshotter.Usage(ctx, next)
|
|
if err != nil {
|
|
if cerrdefs.IsNotFound(err) {
|
|
return total, errdefs.NotFound(errors.Wrapf(err, "non-existing ancestor of %s", snapshotID))
|
|
}
|
|
return total, errdefs.System(errors.Wrapf(err, "snapshotter.Usage failed for %s", next))
|
|
}
|
|
total.Size += usage.Size
|
|
total.Inodes += usage.Inodes
|
|
|
|
info, err := snapshotter.Stat(ctx, next)
|
|
if err != nil {
|
|
return total, errdefs.System(errors.Wrapf(err, "snapshotter.Stat failed for %s", next))
|
|
}
|
|
next = info.Parent
|
|
}
|
|
return total, nil
|
|
}
|