mirror of
https://gitea.com/Lydanne/buildx.git
synced 2025-05-18 09:17:49 +08:00

full diff: https://github.com/moby/buildkit/compare/v0.13.0-rc2...v0.13.0-rc3 Signed-off-by: CrazyMax <1951866+crazy-max@users.noreply.github.com>
128 lines
3.1 KiB
Go
128 lines
3.1 KiB
Go
package contentutil
|
|
|
|
import (
|
|
"context"
|
|
"sync"
|
|
|
|
"github.com/containerd/containerd/content"
|
|
"github.com/containerd/containerd/errdefs"
|
|
"github.com/moby/buildkit/session"
|
|
digest "github.com/opencontainers/go-digest"
|
|
ocispecs "github.com/opencontainers/image-spec/specs-go/v1"
|
|
"github.com/pkg/errors"
|
|
)
|
|
|
|
// NewMultiProvider creates a new mutable provider with a base provider
|
|
func NewMultiProvider(base content.InfoReaderProvider) *MultiProvider {
|
|
return &MultiProvider{
|
|
base: base,
|
|
sub: map[digest.Digest]content.InfoReaderProvider{},
|
|
}
|
|
}
|
|
|
|
// MultiProvider is a provider backed by a mutable map of providers
|
|
type MultiProvider struct {
|
|
mu sync.RWMutex
|
|
base content.InfoReaderProvider
|
|
sub map[digest.Digest]content.InfoReaderProvider
|
|
}
|
|
|
|
func (mp *MultiProvider) SnapshotLabels(descs []ocispecs.Descriptor, index int) map[string]string {
|
|
if len(descs) < index {
|
|
return nil
|
|
}
|
|
desc := descs[index]
|
|
type snapshotLabels interface {
|
|
SnapshotLabels([]ocispecs.Descriptor, int) map[string]string
|
|
}
|
|
|
|
mp.mu.RLock()
|
|
if p, ok := mp.sub[desc.Digest]; ok {
|
|
mp.mu.RUnlock()
|
|
if cd, ok := p.(snapshotLabels); ok {
|
|
return cd.SnapshotLabels(descs, index)
|
|
}
|
|
} else {
|
|
mp.mu.RUnlock()
|
|
}
|
|
if cd, ok := mp.base.(snapshotLabels); ok {
|
|
return cd.SnapshotLabels(descs, index)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (mp *MultiProvider) CheckDescriptor(ctx context.Context, desc ocispecs.Descriptor) error {
|
|
type checkDescriptor interface {
|
|
CheckDescriptor(context.Context, ocispecs.Descriptor) error
|
|
}
|
|
|
|
mp.mu.RLock()
|
|
if p, ok := mp.sub[desc.Digest]; ok {
|
|
mp.mu.RUnlock()
|
|
if cd, ok := p.(checkDescriptor); ok {
|
|
return cd.CheckDescriptor(ctx, desc)
|
|
}
|
|
} else {
|
|
mp.mu.RUnlock()
|
|
}
|
|
if cd, ok := mp.base.(checkDescriptor); ok {
|
|
return cd.CheckDescriptor(ctx, desc)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ReaderAt returns a content.ReaderAt
|
|
func (mp *MultiProvider) ReaderAt(ctx context.Context, desc ocispecs.Descriptor) (content.ReaderAt, error) {
|
|
mp.mu.RLock()
|
|
if p, ok := mp.sub[desc.Digest]; ok {
|
|
mp.mu.RUnlock()
|
|
return p.ReaderAt(ctx, desc)
|
|
}
|
|
mp.mu.RUnlock()
|
|
if mp.base == nil {
|
|
return nil, errors.Wrapf(errdefs.ErrNotFound, "content %v", desc.Digest)
|
|
}
|
|
return mp.base.ReaderAt(ctx, desc)
|
|
}
|
|
|
|
// Info returns a content.Info
|
|
func (mp *MultiProvider) Info(ctx context.Context, dgst digest.Digest) (content.Info, error) {
|
|
mp.mu.RLock()
|
|
if p, ok := mp.sub[dgst]; ok {
|
|
mp.mu.RUnlock()
|
|
return p.Info(ctx, dgst)
|
|
}
|
|
mp.mu.RUnlock()
|
|
if mp.base == nil {
|
|
return content.Info{}, errors.Wrapf(errdefs.ErrNotFound, "content %v", dgst)
|
|
}
|
|
return mp.base.Info(ctx, dgst)
|
|
}
|
|
|
|
// Add adds a new child provider for a specific digest
|
|
func (mp *MultiProvider) Add(dgst digest.Digest, p content.InfoReaderProvider) {
|
|
mp.mu.Lock()
|
|
defer mp.mu.Unlock()
|
|
mp.sub[dgst] = p
|
|
}
|
|
|
|
func (mp *MultiProvider) UnlazySession(desc ocispecs.Descriptor) session.Group {
|
|
type unlazySession interface {
|
|
UnlazySession(ocispecs.Descriptor) session.Group
|
|
}
|
|
|
|
mp.mu.RLock()
|
|
if p, ok := mp.sub[desc.Digest]; ok {
|
|
mp.mu.RUnlock()
|
|
if cd, ok := p.(unlazySession); ok {
|
|
return cd.UnlazySession(desc)
|
|
}
|
|
} else {
|
|
mp.mu.RUnlock()
|
|
}
|
|
if cd, ok := mp.base.(unlazySession); ok {
|
|
return cd.UnlazySession(desc)
|
|
}
|
|
return nil
|
|
}
|