97 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			97 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
/*
 | 
						|
   Copyright The containerd Authors.
 | 
						|
 | 
						|
   Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
   you may not use this file except in compliance with the License.
 | 
						|
   You may obtain a copy of the License at
 | 
						|
 | 
						|
       http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 | 
						|
   Unless required by applicable law or agreed to in writing, software
 | 
						|
   distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
   See the License for the specific language governing permissions and
 | 
						|
   limitations under the License.
 | 
						|
*/
 | 
						|
 | 
						|
package snapshotters
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
 | 
						|
	"github.com/containerd/containerd/v2/images"
 | 
						|
	"github.com/containerd/containerd/v2/labels"
 | 
						|
	"github.com/containerd/log"
 | 
						|
	ocispec "github.com/opencontainers/image-spec/specs-go/v1"
 | 
						|
)
 | 
						|
 | 
						|
// NOTE: The following labels contain "cri" prefix but they are not specific to CRI and
 | 
						|
// can be used by non-CRI clients as well for enabling remote snapshotters. We need to
 | 
						|
// retain that string for keeping compatibility with snapshotter implementations.
 | 
						|
const (
 | 
						|
	// TargetRefLabel is a label which contains image reference and will be passed
 | 
						|
	// to snapshotters.
 | 
						|
	TargetRefLabel = "containerd.io/snapshot/cri.image-ref"
 | 
						|
	// TargetManifestDigestLabel is a label which contains manifest digest and will be passed
 | 
						|
	// to snapshotters.
 | 
						|
	TargetManifestDigestLabel = "containerd.io/snapshot/cri.manifest-digest"
 | 
						|
	// TargetLayerDigestLabel is a label which contains layer digest and will be passed
 | 
						|
	// to snapshotters.
 | 
						|
	TargetLayerDigestLabel = "containerd.io/snapshot/cri.layer-digest"
 | 
						|
	// TargetImageLayersLabel is a label which contains layer digests contained in
 | 
						|
	// the target image and will be passed to snapshotters for preparing layers in
 | 
						|
	// parallel. Skipping some layers is allowed and only affects performance.
 | 
						|
	TargetImageLayersLabel = "containerd.io/snapshot/cri.image-layers"
 | 
						|
)
 | 
						|
 | 
						|
// AppendInfoHandlerWrapper makes a handler which appends some basic information
 | 
						|
// of images like digests for manifest and their child layers as annotations during unpack.
 | 
						|
// These annotations will be passed to snapshotters as labels. These labels will be
 | 
						|
// used mainly by remote snapshotters for querying image contents from the remote location.
 | 
						|
func AppendInfoHandlerWrapper(ref string) func(f images.Handler) images.Handler {
 | 
						|
	return func(f images.Handler) images.Handler {
 | 
						|
		return images.HandlerFunc(func(ctx context.Context, desc ocispec.Descriptor) ([]ocispec.Descriptor, error) {
 | 
						|
			children, err := f.Handle(ctx, desc)
 | 
						|
			if err != nil {
 | 
						|
				return nil, err
 | 
						|
			}
 | 
						|
			if images.IsManifestType(desc.MediaType) {
 | 
						|
				for i := range children {
 | 
						|
					c := &children[i]
 | 
						|
					if images.IsLayerType(c.MediaType) {
 | 
						|
						if c.Annotations == nil {
 | 
						|
							c.Annotations = make(map[string]string)
 | 
						|
						}
 | 
						|
						c.Annotations[TargetRefLabel] = ref
 | 
						|
						c.Annotations[TargetLayerDigestLabel] = c.Digest.String()
 | 
						|
						c.Annotations[TargetImageLayersLabel] = getLayers(ctx, TargetImageLayersLabel, children[i:], labels.Validate)
 | 
						|
						c.Annotations[TargetManifestDigestLabel] = desc.Digest.String()
 | 
						|
					}
 | 
						|
				}
 | 
						|
			}
 | 
						|
			return children, nil
 | 
						|
		})
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// getLayers returns comma-separated digests based on the passed list of
 | 
						|
// descriptors. The returned list contains as many digests as possible as well
 | 
						|
// as meets the label validation.
 | 
						|
func getLayers(ctx context.Context, key string, descs []ocispec.Descriptor, validate func(k, v string) error) (layers string) {
 | 
						|
	for _, l := range descs {
 | 
						|
		if images.IsLayerType(l.MediaType) {
 | 
						|
			item := l.Digest.String()
 | 
						|
			if layers != "" {
 | 
						|
				item = "," + item
 | 
						|
			}
 | 
						|
			// This avoids the label hits the size limitation.
 | 
						|
			if err := validate(key, layers+item); err != nil {
 | 
						|
				log.G(ctx).WithError(err).WithField("label", key).WithField("digest", l.Digest.String()).Debug("omitting digest in the layers list")
 | 
						|
				break
 | 
						|
			}
 | 
						|
			layers += item
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return
 | 
						|
}
 |