405 lines
14 KiB
Go
405 lines
14 KiB
Go
|
package openshift
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"encoding/json"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"io/ioutil"
|
||
|
"net/http"
|
||
|
"strings"
|
||
|
"time"
|
||
|
|
||
|
"github.com/Sirupsen/logrus"
|
||
|
"github.com/containers/image/docker"
|
||
|
"github.com/containers/image/manifest"
|
||
|
"github.com/containers/image/types"
|
||
|
"github.com/containers/image/version"
|
||
|
)
|
||
|
|
||
|
// openshiftClient is configuration for dealing with a single image stream, for reading or writing.
|
||
|
type openshiftClient struct {
|
||
|
ref openshiftReference
|
||
|
// Values from Kubernetes configuration
|
||
|
httpClient *http.Client
|
||
|
bearerToken string // "" if not used
|
||
|
username string // "" if not used
|
||
|
password string // if username != ""
|
||
|
}
|
||
|
|
||
|
// newOpenshiftClient creates a new openshiftClient for the specified reference.
|
||
|
func newOpenshiftClient(ref openshiftReference) (*openshiftClient, error) {
|
||
|
// We have already done this parsing in ParseReference, but thrown away
|
||
|
// httpClient. So, parse again.
|
||
|
// (We could also rework/split restClientFor to "get base URL" to be done
|
||
|
// in ParseReference, and "get httpClient" to be done here. But until/unless
|
||
|
// we support non-default clusters, this is good enough.)
|
||
|
|
||
|
// Overall, this is modelled on openshift/origin/pkg/cmd/util/clientcmd.New().ClientConfig() and openshift/origin/pkg/client.
|
||
|
cmdConfig := defaultClientConfig()
|
||
|
logrus.Debugf("cmdConfig: %#v", cmdConfig)
|
||
|
restConfig, err := cmdConfig.ClientConfig()
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
// REMOVED: SetOpenShiftDefaults (values are not overridable in config files, so hard-coded these defaults.)
|
||
|
logrus.Debugf("restConfig: %#v", restConfig)
|
||
|
baseURL, httpClient, err := restClientFor(restConfig)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
logrus.Debugf("URL: %#v", *baseURL)
|
||
|
if *baseURL != *ref.baseURL {
|
||
|
return nil, fmt.Errorf("Unexpected baseURL mismatch: default %#v, reference %#v", *baseURL, *ref.baseURL)
|
||
|
}
|
||
|
httpClient.Timeout = 1 * time.Minute
|
||
|
|
||
|
return &openshiftClient{
|
||
|
ref: ref,
|
||
|
httpClient: httpClient,
|
||
|
bearerToken: restConfig.BearerToken,
|
||
|
username: restConfig.Username,
|
||
|
password: restConfig.Password,
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
// doRequest performs a correctly authenticated request to a specified path, and returns response body or an error object.
|
||
|
func (c *openshiftClient) doRequest(method, path string, requestBody []byte) ([]byte, error) {
|
||
|
url := *c.ref.baseURL
|
||
|
url.Path = path
|
||
|
var requestBodyReader io.Reader
|
||
|
if requestBody != nil {
|
||
|
logrus.Debugf("Will send body: %s", requestBody)
|
||
|
requestBodyReader = bytes.NewReader(requestBody)
|
||
|
}
|
||
|
req, err := http.NewRequest(method, url.String(), requestBodyReader)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
if len(c.bearerToken) != 0 {
|
||
|
req.Header.Set("Authorization", "Bearer "+c.bearerToken)
|
||
|
} else if len(c.username) != 0 {
|
||
|
req.SetBasicAuth(c.username, c.password)
|
||
|
}
|
||
|
req.Header.Set("Accept", "application/json, */*")
|
||
|
req.Header.Set("User-Agent", fmt.Sprintf("skopeo/%s", version.Version))
|
||
|
if requestBody != nil {
|
||
|
req.Header.Set("Content-Type", "application/json")
|
||
|
}
|
||
|
|
||
|
logrus.Debugf("%s %s", method, url)
|
||
|
res, err := c.httpClient.Do(req)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
defer res.Body.Close()
|
||
|
body, err := ioutil.ReadAll(res.Body)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
logrus.Debugf("Got body: %s", body)
|
||
|
// FIXME: Just throwing this useful information away only to try to guess later...
|
||
|
logrus.Debugf("Got content-type: %s", res.Header.Get("Content-Type"))
|
||
|
|
||
|
var status status
|
||
|
statusValid := false
|
||
|
if err := json.Unmarshal(body, &status); err == nil && len(status.Status) > 0 {
|
||
|
statusValid = true
|
||
|
}
|
||
|
|
||
|
switch {
|
||
|
case res.StatusCode == http.StatusSwitchingProtocols: // FIXME?! No idea why this weird case exists in k8s.io/kubernetes/pkg/client/restclient.
|
||
|
if statusValid && status.Status != "Success" {
|
||
|
return nil, errors.New(status.Message)
|
||
|
}
|
||
|
case res.StatusCode >= http.StatusOK && res.StatusCode <= http.StatusPartialContent:
|
||
|
// OK.
|
||
|
default:
|
||
|
if statusValid {
|
||
|
return nil, errors.New(status.Message)
|
||
|
}
|
||
|
return nil, fmt.Errorf("HTTP error: status code: %d, body: %s", res.StatusCode, string(body))
|
||
|
}
|
||
|
|
||
|
return body, nil
|
||
|
}
|
||
|
|
||
|
// convertDockerImageReference takes an image API DockerImageReference value and returns a reference we can actually use;
|
||
|
// currently OpenShift stores the cluster-internal service IPs here, which are unusable from the outside.
|
||
|
func (c *openshiftClient) convertDockerImageReference(ref string) (string, error) {
|
||
|
parts := strings.SplitN(ref, "/", 2)
|
||
|
if len(parts) != 2 {
|
||
|
return "", fmt.Errorf("Invalid format of docker reference %s: missing '/'", ref)
|
||
|
}
|
||
|
// Sanity check that the reference is at least plausibly similar, i.e. uses the hard-coded port we expect.
|
||
|
if !strings.HasSuffix(parts[0], ":5000") {
|
||
|
return "", fmt.Errorf("Invalid format of docker reference %s: expecting port 5000", ref)
|
||
|
}
|
||
|
return c.dockerRegistryHostPart() + "/" + parts[1], nil
|
||
|
}
|
||
|
|
||
|
// dockerRegistryHostPart returns the host:port of the embedded Docker Registry API endpoint
|
||
|
// FIXME: There seems to be no way to discover the correct:host port using the API, so hard-code our knowledge
|
||
|
// about how the OpenShift Atomic Registry is configured, per examples/atomic-registry/run.sh:
|
||
|
// -p OPENSHIFT_OAUTH_PROVIDER_URL=https://${INSTALL_HOST}:8443,COCKPIT_KUBE_URL=https://${INSTALL_HOST},REGISTRY_HOST=${INSTALL_HOST}:5000
|
||
|
func (c *openshiftClient) dockerRegistryHostPart() string {
|
||
|
return strings.SplitN(c.ref.baseURL.Host, ":", 2)[0] + ":5000"
|
||
|
}
|
||
|
|
||
|
type openshiftImageSource struct {
|
||
|
client *openshiftClient
|
||
|
// Values specific to this image
|
||
|
certPath string // Only for parseDockerImageSource
|
||
|
tlsVerify bool // Only for parseDockerImageSource
|
||
|
// State
|
||
|
docker types.ImageSource // The Docker Registry endpoint, or nil if not resolved yet
|
||
|
imageStreamImageName string // Resolved image identifier, or "" if not known yet
|
||
|
}
|
||
|
|
||
|
// newImageSource creates a new ImageSource for the specified reference and connection specification.
|
||
|
func newImageSource(ref openshiftReference, certPath string, tlsVerify bool) (types.ImageSource, error) {
|
||
|
client, err := newOpenshiftClient(ref)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &openshiftImageSource{
|
||
|
client: client,
|
||
|
certPath: certPath,
|
||
|
tlsVerify: tlsVerify,
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
// Reference returns the reference used to set up this source, _as specified by the user_
|
||
|
// (not as the image itself, or its underlying storage, claims). This can be used e.g. to determine which public keys are trusted for this image.
|
||
|
func (s *openshiftImageSource) Reference() types.ImageReference {
|
||
|
return s.client.ref
|
||
|
}
|
||
|
|
||
|
func (s *openshiftImageSource) GetManifest(mimetypes []string) ([]byte, string, error) {
|
||
|
if err := s.ensureImageIsResolved(); err != nil {
|
||
|
return nil, "", err
|
||
|
}
|
||
|
return s.docker.GetManifest(mimetypes)
|
||
|
}
|
||
|
|
||
|
func (s *openshiftImageSource) GetBlob(digest string) (io.ReadCloser, int64, error) {
|
||
|
if err := s.ensureImageIsResolved(); err != nil {
|
||
|
return nil, 0, err
|
||
|
}
|
||
|
return s.docker.GetBlob(digest)
|
||
|
}
|
||
|
|
||
|
func (s *openshiftImageSource) GetSignatures() ([][]byte, error) {
|
||
|
return nil, nil
|
||
|
}
|
||
|
|
||
|
// ensureImageIsResolved sets up s.docker and s.imageStreamImageName
|
||
|
func (s *openshiftImageSource) ensureImageIsResolved() error {
|
||
|
if s.docker != nil {
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// FIXME: validate components per validation.IsValidPathSegmentName?
|
||
|
path := fmt.Sprintf("/oapi/v1/namespaces/%s/imagestreams/%s", s.client.ref.namespace, s.client.ref.stream)
|
||
|
body, err := s.client.doRequest("GET", path, nil)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
// Note: This does absolutely no kind/version checking or conversions.
|
||
|
var is imageStream
|
||
|
if err := json.Unmarshal(body, &is); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
var te *tagEvent
|
||
|
for _, tag := range is.Status.Tags {
|
||
|
if tag.Tag != s.client.ref.tag {
|
||
|
continue
|
||
|
}
|
||
|
if len(tag.Items) > 0 {
|
||
|
te = &tag.Items[0]
|
||
|
break
|
||
|
}
|
||
|
}
|
||
|
if te == nil {
|
||
|
return fmt.Errorf("No matching tag found")
|
||
|
}
|
||
|
logrus.Debugf("tag event %#v", te)
|
||
|
dockerRefString, err := s.client.convertDockerImageReference(te.DockerImageReference)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
logrus.Debugf("Resolved reference %#v", dockerRefString)
|
||
|
dockerRef, err := docker.ParseReference("//" + dockerRefString)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
d, err := dockerRef.NewImageSource(s.certPath, s.tlsVerify)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
s.docker = d
|
||
|
s.imageStreamImageName = te.Image
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
type openshiftImageDestination struct {
|
||
|
client *openshiftClient
|
||
|
docker types.ImageDestination // The Docker Registry endpoint
|
||
|
}
|
||
|
|
||
|
// newImageDestination creates a new ImageDestination for the specified reference and connection specification.
|
||
|
func newImageDestination(ref openshiftReference, certPath string, tlsVerify bool) (types.ImageDestination, error) {
|
||
|
client, err := newOpenshiftClient(ref)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
// FIXME: Should this always use a digest, not a tag? Uploading to Docker by tag requires the tag _inside_ the manifest to match,
|
||
|
// i.e. a single signed image cannot be available under multiple tags. But with types.ImageDestination, we don't know
|
||
|
// the manifest digest at this point.
|
||
|
dockerRefString := fmt.Sprintf("//%s/%s/%s:%s", client.dockerRegistryHostPart(), client.ref.namespace, client.ref.stream, client.ref.tag)
|
||
|
dockerRef, err := docker.ParseReference(dockerRefString)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
docker, err := dockerRef.NewImageDestination(certPath, tlsVerify)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
return &openshiftImageDestination{
|
||
|
client: client,
|
||
|
docker: docker,
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
// Reference returns the reference used to set up this destination. Note that this should directly correspond to user's intent,
|
||
|
// e.g. it should use the public hostname instead of the result of resolving CNAMEs or following redirects.
|
||
|
func (d *openshiftImageDestination) Reference() types.ImageReference {
|
||
|
return d.client.ref
|
||
|
}
|
||
|
|
||
|
func (d *openshiftImageDestination) SupportedManifestMIMETypes() []string {
|
||
|
return []string{
|
||
|
manifest.DockerV2Schema1SignedMIMEType,
|
||
|
manifest.DockerV2Schema1MIMEType,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (d *openshiftImageDestination) PutManifest(m []byte) error {
|
||
|
// Note: This does absolutely no kind/version checking or conversions.
|
||
|
manifestDigest, err := manifest.Digest(m)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
// FIXME: We can't do what respositorymiddleware.go does because we don't know the internal address. Does any of this matter?
|
||
|
dockerImageReference := fmt.Sprintf("%s/%s/%s@%s", d.client.dockerRegistryHostPart(), d.client.ref.namespace, d.client.ref.stream, manifestDigest)
|
||
|
ism := imageStreamMapping{
|
||
|
typeMeta: typeMeta{
|
||
|
Kind: "ImageStreamMapping",
|
||
|
APIVersion: "v1",
|
||
|
},
|
||
|
objectMeta: objectMeta{
|
||
|
Namespace: d.client.ref.namespace,
|
||
|
Name: d.client.ref.stream,
|
||
|
},
|
||
|
Image: image{
|
||
|
objectMeta: objectMeta{
|
||
|
Name: manifestDigest,
|
||
|
},
|
||
|
DockerImageReference: dockerImageReference,
|
||
|
DockerImageManifest: string(m),
|
||
|
},
|
||
|
Tag: d.client.ref.tag,
|
||
|
}
|
||
|
body, err := json.Marshal(ism)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
// FIXME: validate components per validation.IsValidPathSegmentName?
|
||
|
path := fmt.Sprintf("/oapi/v1/namespaces/%s/imagestreammappings", d.client.ref.namespace)
|
||
|
body, err = d.client.doRequest("POST", path, body)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
return d.docker.PutManifest(m)
|
||
|
}
|
||
|
|
||
|
func (d *openshiftImageDestination) PutBlob(digest string, stream io.Reader) error {
|
||
|
return d.docker.PutBlob(digest, stream)
|
||
|
}
|
||
|
|
||
|
func (d *openshiftImageDestination) PutSignatures(signatures [][]byte) error {
|
||
|
if len(signatures) != 0 {
|
||
|
return fmt.Errorf("Pushing signatures to an Atomic Registry is not supported")
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// These structs are subsets of github.com/openshift/origin/pkg/image/api/v1 and its dependencies.
|
||
|
type imageStream struct {
|
||
|
Status imageStreamStatus `json:"status,omitempty"`
|
||
|
}
|
||
|
type imageStreamStatus struct {
|
||
|
DockerImageRepository string `json:"dockerImageRepository"`
|
||
|
Tags []namedTagEventList `json:"tags,omitempty"`
|
||
|
}
|
||
|
type namedTagEventList struct {
|
||
|
Tag string `json:"tag"`
|
||
|
Items []tagEvent `json:"items"`
|
||
|
}
|
||
|
type tagEvent struct {
|
||
|
DockerImageReference string `json:"dockerImageReference"`
|
||
|
Image string `json:"image"`
|
||
|
}
|
||
|
type imageStreamImage struct {
|
||
|
Image image `json:"image"`
|
||
|
}
|
||
|
type image struct {
|
||
|
objectMeta `json:"metadata,omitempty"`
|
||
|
DockerImageReference string `json:"dockerImageReference,omitempty"`
|
||
|
// DockerImageMetadata runtime.RawExtension `json:"dockerImageMetadata,omitempty"`
|
||
|
DockerImageMetadataVersion string `json:"dockerImageMetadataVersion,omitempty"`
|
||
|
DockerImageManifest string `json:"dockerImageManifest,omitempty"`
|
||
|
// DockerImageLayers []ImageLayer `json:"dockerImageLayers"`
|
||
|
}
|
||
|
type imageStreamMapping struct {
|
||
|
typeMeta `json:",inline"`
|
||
|
objectMeta `json:"metadata,omitempty"`
|
||
|
Image image `json:"image"`
|
||
|
Tag string `json:"tag"`
|
||
|
}
|
||
|
type typeMeta struct {
|
||
|
Kind string `json:"kind,omitempty"`
|
||
|
APIVersion string `json:"apiVersion,omitempty"`
|
||
|
}
|
||
|
type objectMeta struct {
|
||
|
Name string `json:"name,omitempty"`
|
||
|
GenerateName string `json:"generateName,omitempty"`
|
||
|
Namespace string `json:"namespace,omitempty"`
|
||
|
SelfLink string `json:"selfLink,omitempty"`
|
||
|
ResourceVersion string `json:"resourceVersion,omitempty"`
|
||
|
Generation int64 `json:"generation,omitempty"`
|
||
|
DeletionGracePeriodSeconds *int64 `json:"deletionGracePeriodSeconds,omitempty"`
|
||
|
Labels map[string]string `json:"labels,omitempty"`
|
||
|
Annotations map[string]string `json:"annotations,omitempty"`
|
||
|
}
|
||
|
|
||
|
// A subset of k8s.io/kubernetes/pkg/api/unversioned/Status
|
||
|
type status struct {
|
||
|
Status string `json:"status,omitempty"`
|
||
|
Message string `json:"message,omitempty"`
|
||
|
// Reason StatusReason `json:"reason,omitempty"`
|
||
|
// Details *StatusDetails `json:"details,omitempty"`
|
||
|
Code int32 `json:"code,omitempty"`
|
||
|
}
|
||
|
|
||
|
func (s *openshiftImageSource) Delete() error {
|
||
|
return fmt.Errorf("openshift#openshiftImageSource.Delete() not implmented")
|
||
|
}
|