2018-09-11 08:02:46 +00:00
|
|
|
package local
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/containerd/containerd/content"
|
|
|
|
"github.com/moby/buildkit/cache/remotecache"
|
|
|
|
"github.com/moby/buildkit/session"
|
|
|
|
sessioncontent "github.com/moby/buildkit/session/content"
|
|
|
|
digest "github.com/opencontainers/go-digest"
|
|
|
|
specs "github.com/opencontainers/image-spec/specs-go/v1"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
attrDigest = "digest"
|
2019-01-31 16:30:39 +00:00
|
|
|
attrSrc = "src"
|
|
|
|
attrDest = "dest"
|
2018-09-11 08:02:46 +00:00
|
|
|
contentStoreIDPrefix = "local:"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ResolveCacheExporterFunc for "local" cache exporter.
|
|
|
|
func ResolveCacheExporterFunc(sm *session.Manager) remotecache.ResolveCacheExporterFunc {
|
|
|
|
return func(ctx context.Context, attrs map[string]string) (remotecache.Exporter, error) {
|
2019-01-31 16:30:39 +00:00
|
|
|
store := attrs[attrDest]
|
2018-09-11 08:02:46 +00:00
|
|
|
if store == "" {
|
2019-01-31 16:30:39 +00:00
|
|
|
return nil, errors.New("local cache exporter requires dest")
|
2018-09-11 08:02:46 +00:00
|
|
|
}
|
|
|
|
csID := contentStoreIDPrefix + store
|
|
|
|
cs, err := getContentStore(ctx, sm, csID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return remotecache.NewExporter(cs), nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResolveCacheImporterFunc for "local" cache importer.
|
|
|
|
func ResolveCacheImporterFunc(sm *session.Manager) remotecache.ResolveCacheImporterFunc {
|
|
|
|
return func(ctx context.Context, attrs map[string]string) (remotecache.Importer, specs.Descriptor, error) {
|
|
|
|
dgstStr := attrs[attrDigest]
|
|
|
|
if dgstStr == "" {
|
|
|
|
return nil, specs.Descriptor{}, errors.New("local cache importer requires explicit digest")
|
|
|
|
}
|
|
|
|
dgst := digest.Digest(dgstStr)
|
2019-01-31 16:30:39 +00:00
|
|
|
store := attrs[attrSrc]
|
2018-09-11 08:02:46 +00:00
|
|
|
if store == "" {
|
2019-01-31 16:30:39 +00:00
|
|
|
return nil, specs.Descriptor{}, errors.New("local cache importer requires src")
|
2018-09-11 08:02:46 +00:00
|
|
|
}
|
|
|
|
csID := contentStoreIDPrefix + store
|
|
|
|
cs, err := getContentStore(ctx, sm, csID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, specs.Descriptor{}, err
|
|
|
|
}
|
|
|
|
info, err := cs.Info(ctx, dgst)
|
|
|
|
if err != nil {
|
|
|
|
return nil, specs.Descriptor{}, err
|
|
|
|
}
|
|
|
|
desc := specs.Descriptor{
|
|
|
|
// MediaType is typically MediaTypeDockerSchema2ManifestList,
|
|
|
|
// but we leave it empty until we get correct support for local index.json
|
|
|
|
Digest: dgst,
|
|
|
|
Size: info.Size,
|
|
|
|
}
|
|
|
|
return remotecache.NewImporter(cs), desc, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func getContentStore(ctx context.Context, sm *session.Manager, storeID string) (content.Store, error) {
|
|
|
|
sessionID := session.FromContext(ctx)
|
|
|
|
if sessionID == "" {
|
|
|
|
return nil, errors.New("local cache exporter/importer requires session")
|
|
|
|
}
|
|
|
|
timeoutCtx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
caller, err := sm.Get(timeoutCtx, sessionID)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return sessioncontent.NewCallerStore(caller, storeID), nil
|
|
|
|
}
|