2016-11-22 19:32:10 +00:00
|
|
|
package layout
|
|
|
|
|
|
|
|
import (
|
2017-08-05 11:40:46 +00:00
|
|
|
"context"
|
2016-11-22 19:32:10 +00:00
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
2017-10-10 14:11:06 +00:00
|
|
|
"net/http"
|
2016-11-22 19:32:10 +00:00
|
|
|
"os"
|
2017-10-10 14:11:06 +00:00
|
|
|
"strconv"
|
2016-11-22 19:32:10 +00:00
|
|
|
|
2017-10-10 14:11:06 +00:00
|
|
|
"github.com/containers/image/pkg/tlsclientconfig"
|
2016-11-22 19:32:10 +00:00
|
|
|
"github.com/containers/image/types"
|
2017-10-10 14:11:06 +00:00
|
|
|
"github.com/docker/go-connections/tlsconfig"
|
2016-10-17 13:53:40 +00:00
|
|
|
"github.com/opencontainers/go-digest"
|
2016-11-22 19:32:10 +00:00
|
|
|
imgspecv1 "github.com/opencontainers/image-spec/specs-go/v1"
|
2017-10-10 14:11:06 +00:00
|
|
|
"github.com/pkg/errors"
|
2016-11-22 19:32:10 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type ociImageSource struct {
|
2017-05-17 17:18:35 +00:00
|
|
|
ref ociReference
|
2017-05-26 16:31:28 +00:00
|
|
|
descriptor imgspecv1.Descriptor
|
2017-10-10 14:11:06 +00:00
|
|
|
client *http.Client
|
2016-11-22 19:32:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// newImageSource returns an ImageSource for reading from an existing directory.
|
2017-10-10 14:11:06 +00:00
|
|
|
func newImageSource(ctx *types.SystemContext, ref ociReference) (types.ImageSource, error) {
|
|
|
|
tr := tlsclientconfig.NewTransport()
|
|
|
|
tr.TLSClientConfig = tlsconfig.ServerDefault()
|
|
|
|
|
|
|
|
if ctx != nil && ctx.OCICertPath != "" {
|
|
|
|
if err := tlsclientconfig.SetupCertificates(ctx.OCICertPath, tr.TLSClientConfig); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
tr.TLSClientConfig.InsecureSkipVerify = ctx.OCIInsecureSkipTLSVerify
|
|
|
|
}
|
|
|
|
|
|
|
|
client := &http.Client{}
|
|
|
|
client.Transport = tr
|
2017-05-17 17:18:35 +00:00
|
|
|
descriptor, err := ref.getManifestDescriptor()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2017-10-10 14:11:06 +00:00
|
|
|
return &ociImageSource{ref: ref, descriptor: descriptor, client: client}, nil
|
2016-11-22 19:32:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Reference returns the reference used to set up this source.
|
|
|
|
func (s *ociImageSource) Reference() types.ImageReference {
|
|
|
|
return s.ref
|
|
|
|
}
|
|
|
|
|
|
|
|
// Close removes resources associated with an initialized ImageSource, if any.
|
2017-03-13 16:33:17 +00:00
|
|
|
func (s *ociImageSource) Close() error {
|
|
|
|
return nil
|
2016-11-22 19:32:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetManifest returns the image's manifest along with its MIME type (which may be empty when it can't be determined but the manifest is available).
|
|
|
|
// It may use a remote (= slow) service.
|
|
|
|
func (s *ociImageSource) GetManifest() ([]byte, string, error) {
|
2017-05-17 17:18:35 +00:00
|
|
|
manifestPath, err := s.ref.blobPath(digest.Digest(s.descriptor.Digest))
|
2016-11-22 19:32:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, "", err
|
|
|
|
}
|
|
|
|
m, err := ioutil.ReadFile(manifestPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", err
|
|
|
|
}
|
|
|
|
|
2017-05-17 17:18:35 +00:00
|
|
|
return m, s.descriptor.MediaType, nil
|
2016-11-22 19:32:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *ociImageSource) GetTargetManifest(digest digest.Digest) ([]byte, string, error) {
|
|
|
|
manifestPath, err := s.ref.blobPath(digest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
m, err := ioutil.ReadFile(manifestPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, "", err
|
|
|
|
}
|
|
|
|
|
2017-03-13 16:33:17 +00:00
|
|
|
// XXX: GetTargetManifest means that we don't have the context of what
|
|
|
|
// mediaType the manifest has. In OCI this means that we don't know
|
|
|
|
// what reference it came from, so we just *assume* that its
|
|
|
|
// MediaTypeImageManifest.
|
|
|
|
return m, imgspecv1.MediaTypeImageManifest, nil
|
2016-11-22 19:32:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// GetBlob returns a stream for the specified blob, and the blob's size.
|
|
|
|
func (s *ociImageSource) GetBlob(info types.BlobInfo) (io.ReadCloser, int64, error) {
|
2017-10-10 14:11:06 +00:00
|
|
|
if len(info.URLs) != 0 {
|
|
|
|
return s.getExternalBlob(info.URLs)
|
|
|
|
}
|
|
|
|
|
2016-11-22 19:32:10 +00:00
|
|
|
path, err := s.ref.blobPath(info.Digest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, err
|
|
|
|
}
|
|
|
|
fi, err := r.Stat()
|
|
|
|
if err != nil {
|
|
|
|
return nil, 0, err
|
|
|
|
}
|
|
|
|
return r, fi.Size(), nil
|
|
|
|
}
|
|
|
|
|
2017-08-05 11:40:46 +00:00
|
|
|
func (s *ociImageSource) GetSignatures(context.Context) ([][]byte, error) {
|
2016-11-22 19:32:10 +00:00
|
|
|
return [][]byte{}, nil
|
|
|
|
}
|
2017-10-10 14:11:06 +00:00
|
|
|
|
|
|
|
func (s *ociImageSource) getExternalBlob(urls []string) (io.ReadCloser, int64, error) {
|
|
|
|
errWrap := errors.New("failed fetching external blob from all urls")
|
|
|
|
for _, url := range urls {
|
|
|
|
resp, err := s.client.Get(url)
|
|
|
|
if err != nil {
|
|
|
|
errWrap = errors.Wrapf(errWrap, "fetching %s failed %s", url, err.Error())
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
resp.Body.Close()
|
|
|
|
errWrap = errors.Wrapf(errWrap, "fetching %s failed, response code not 200", url)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp.Body, getBlobSize(resp), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, 0, errWrap
|
|
|
|
}
|
|
|
|
|
|
|
|
func getBlobSize(resp *http.Response) int64 {
|
|
|
|
size, err := strconv.ParseInt(resp.Header.Get("Content-Length"), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
size = -1
|
|
|
|
}
|
|
|
|
return size
|
|
|
|
}
|