mirror of https://github.com/helm/helm
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
422 lines
13 KiB
422 lines
13 KiB
9 years ago
|
/*
|
||
|
Copyright 2015 The Kubernetes Authors All rights reserved.
|
||
|
|
||
|
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 registry
|
||
|
|
||
|
import (
|
||
9 years ago
|
"github.com/google/go-github/github"
|
||
9 years ago
|
"github.com/kubernetes/deployment-manager/pkg/common"
|
||
|
"github.com/kubernetes/deployment-manager/pkg/util"
|
||
9 years ago
|
"golang.org/x/oauth2"
|
||
9 years ago
|
"golang.org/x/oauth2/google"
|
||
|
storage "google.golang.org/api/storage/v1"
|
||
9 years ago
|
|
||
|
"fmt"
|
||
9 years ago
|
"log"
|
||
9 years ago
|
"net/http"
|
||
9 years ago
|
"net/url"
|
||
|
"regexp"
|
||
|
"strings"
|
||
|
"sync"
|
||
|
)
|
||
|
|
||
9 years ago
|
// RegistryProvider is a factory for Registry instances.
|
||
9 years ago
|
type RegistryProvider interface {
|
||
|
GetRegistryByShortURL(URL string) (Registry, error)
|
||
|
GetRegistryByName(registryName string) (Registry, error)
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
type registryProvider struct {
|
||
|
sync.RWMutex
|
||
|
rs common.RegistryService
|
||
|
grp GithubRegistryProvider
|
||
9 years ago
|
gcsrp GCSRegistryProvider
|
||
9 years ago
|
cp common.CredentialProvider
|
||
9 years ago
|
registries map[string]Registry
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
// NewDefaultRegistryProvider creates a default registry provider with the supplied credential.
|
||
9 years ago
|
func NewDefaultRegistryProvider(cp common.CredentialProvider, rs common.RegistryService) RegistryProvider {
|
||
|
return NewRegistryProvider(rs, NewGithubRegistryProvider(cp), NewGCSRegistryProvider(cp), cp)
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
// NewRegistryProvider creates a new registry provider using the supplied arguments.
|
||
9 years ago
|
func NewRegistryProvider(rs common.RegistryService, grp GithubRegistryProvider, gcsrp GCSRegistryProvider, cp common.CredentialProvider) RegistryProvider {
|
||
9 years ago
|
if rs == nil {
|
||
|
rs = NewInmemRegistryService()
|
||
|
}
|
||
|
|
||
9 years ago
|
if cp == nil {
|
||
|
cp = NewInmemCredentialProvider()
|
||
|
}
|
||
|
|
||
9 years ago
|
if grp == nil {
|
||
9 years ago
|
grp = NewGithubRegistryProvider(cp)
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
if gcsrp == nil {
|
||
|
gcsrp = NewGCSRegistryProvider(cp)
|
||
|
}
|
||
|
|
||
9 years ago
|
registries := make(map[string]Registry)
|
||
9 years ago
|
rp := ®istryProvider{rs: rs, grp: grp, gcsrp: gcsrp, cp: cp, registries: registries}
|
||
9 years ago
|
return rp
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
func (rp *registryProvider) getRegistry(cr common.Registry) (Registry, error) {
|
||
9 years ago
|
switch cr.Type {
|
||
|
case common.GithubRegistryType:
|
||
|
return rp.grp.GetGithubRegistry(cr)
|
||
|
case common.GCSRegistryType:
|
||
|
log.Printf("Creating a bigstore client using %#v", rp.gcsrp)
|
||
|
return rp.gcsrp.GetGCSRegistry(cr)
|
||
|
default:
|
||
|
return nil, fmt.Errorf("unknown registry type: %s", cr.Type)
|
||
|
}
|
||
|
}
|
||
|
|
||
9 years ago
|
// GetRegistryByShortURL returns the registry identified by a short URL.
|
||
9 years ago
|
func (rp *registryProvider) GetRegistryByShortURL(URL string) (Registry, error) {
|
||
9 years ago
|
rp.RLock()
|
||
|
defer rp.RUnlock()
|
||
9 years ago
|
|
||
9 years ago
|
result := rp.findRegistryByShortURL(URL)
|
||
|
if result == nil {
|
||
9 years ago
|
cr, err := rp.rs.GetByURL(URL)
|
||
9 years ago
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
9 years ago
|
r, err := rp.getRegistry(*cr)
|
||
9 years ago
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
9 years ago
|
rp.registries[r.GetRegistryName()] = r
|
||
9 years ago
|
result = r
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
return result, nil
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
// findRegistryByShortURL trims the scheme from both the supplied URL
|
||
|
// and the short URL returned by GetRegistryShortURL.
|
||
9 years ago
|
func (rp *registryProvider) findRegistryByShortURL(URL string) Registry {
|
||
9 years ago
|
trimmed := util.TrimURLScheme(URL)
|
||
|
for _, r := range rp.registries {
|
||
|
if strings.HasPrefix(trimmed, util.TrimURLScheme(r.GetRegistryShortURL())) {
|
||
9 years ago
|
return r
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return nil
|
||
|
}
|
||
|
|
||
9 years ago
|
// GetRegistryByName returns a registry by name.
|
||
9 years ago
|
func (rp *registryProvider) GetRegistryByName(registryName string) (Registry, error) {
|
||
9 years ago
|
rp.RLock()
|
||
|
defer rp.RUnlock()
|
||
9 years ago
|
|
||
9 years ago
|
cr, err := rp.rs.Get(registryName)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
9 years ago
|
|
||
9 years ago
|
r, err := rp.getRegistry(*cr)
|
||
9 years ago
|
if err != nil {
|
||
|
return nil, err
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
rp.registries[r.GetRegistryName()] = r
|
||
|
|
||
|
return r, nil
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
// ParseRegistryFormat creates a map from a registry format string.
|
||
9 years ago
|
func ParseRegistryFormat(rf common.RegistryFormat) map[common.RegistryFormat]bool {
|
||
|
split := strings.Split(string(rf), ";")
|
||
9 years ago
|
var result = map[common.RegistryFormat]bool{}
|
||
9 years ago
|
for _, format := range split {
|
||
|
result[common.RegistryFormat(format)] = true
|
||
|
}
|
||
|
|
||
|
return result
|
||
|
}
|
||
|
|
||
9 years ago
|
// GithubRegistryProvider is a factory for GithubRegistry instances.
|
||
|
type GithubRegistryProvider interface {
|
||
|
GetGithubRegistry(cr common.Registry) (GithubRegistry, error)
|
||
|
}
|
||
|
|
||
|
type githubRegistryProvider struct {
|
||
9 years ago
|
cp common.CredentialProvider
|
||
9 years ago
|
}
|
||
|
|
||
|
// NewGithubRegistryProvider creates a GithubRegistryProvider.
|
||
9 years ago
|
func NewGithubRegistryProvider(cp common.CredentialProvider) GithubRegistryProvider {
|
||
|
if cp == nil {
|
||
9 years ago
|
// TODO: replace this panic with an error return.
|
||
9 years ago
|
panic(fmt.Errorf("no credential provider"))
|
||
9 years ago
|
}
|
||
|
return &githubRegistryProvider{cp: cp}
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
func (grp githubRegistryProvider) createGithubClient(credentialName string) (*http.Client, *github.Client, error) {
|
||
9 years ago
|
if credentialName == "" {
|
||
9 years ago
|
return http.DefaultClient, github.NewClient(nil), nil
|
||
9 years ago
|
}
|
||
|
c, err := grp.cp.GetCredential(credentialName)
|
||
|
|
||
|
if err != nil {
|
||
|
log.Printf("Failed to fetch credential %s: %v", credentialName, err)
|
||
|
log.Print("Trying to use unauthenticated client")
|
||
9 years ago
|
return http.DefaultClient, github.NewClient(nil), nil
|
||
9 years ago
|
}
|
||
|
|
||
|
if c != nil {
|
||
|
if c.APIToken != "" {
|
||
|
ts := oauth2.StaticTokenSource(
|
||
|
&oauth2.Token{AccessToken: string(c.APIToken)},
|
||
|
)
|
||
|
tc := oauth2.NewClient(oauth2.NoContext, ts)
|
||
9 years ago
|
return tc, github.NewClient(tc), nil
|
||
9 years ago
|
}
|
||
|
if c.BasicAuth.Username != "" && c.BasicAuth.Password != "" {
|
||
9 years ago
|
tp := github.BasicAuthTransport{
|
||
|
Username: c.BasicAuth.Username,
|
||
|
Password: c.BasicAuth.Password,
|
||
|
}
|
||
9 years ago
|
return tp.Client(), github.NewClient(tp.Client()), nil
|
||
9 years ago
|
}
|
||
|
|
||
|
}
|
||
9 years ago
|
return nil, nil, fmt.Errorf("No suitable credential found for %s", credentialName)
|
||
9 years ago
|
|
||
|
}
|
||
|
|
||
|
// GetGithubRegistry returns a new GithubRegistry. If there's a credential that is specified, will try
|
||
|
// to fetch it and use it, and if there's no credential found, will fall back to unauthenticated client.
|
||
9 years ago
|
func (grp githubRegistryProvider) GetGithubRegistry(cr common.Registry) (GithubRegistry, error) {
|
||
9 years ago
|
// If there's a credential that we need to use, fetch it and create a client for it.
|
||
|
httpClient, client, err := grp.createGithubClient(cr.CredentialName)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
9 years ago
|
|
||
9 years ago
|
fMap := ParseRegistryFormat(cr.Format)
|
||
|
if fMap[common.UnversionedRegistry] && fMap[common.OneLevelRegistry] {
|
||
|
return NewGithubPackageRegistry(cr.Name, cr.URL, nil, httpClient, client)
|
||
|
}
|
||
9 years ago
|
|
||
9 years ago
|
if fMap[common.VersionedRegistry] && fMap[common.CollectionRegistry] {
|
||
|
return NewGithubTemplateRegistry(cr.Name, cr.URL, nil, httpClient, client)
|
||
|
}
|
||
|
|
||
|
return nil, fmt.Errorf("unknown registry format: %s", cr.Format)
|
||
|
}
|
||
|
|
||
|
// GCSRegistryProvider is a factory for GCS Registry instances.
|
||
|
type GCSRegistryProvider interface {
|
||
|
GetGCSRegistry(cr common.Registry) (ObjectStorageRegistry, error)
|
||
|
}
|
||
|
|
||
|
type gcsRegistryProvider struct {
|
||
|
cp common.CredentialProvider
|
||
|
}
|
||
|
|
||
|
// NewGCSRegistryProvider creates a GCSRegistryProvider.
|
||
|
func NewGCSRegistryProvider(cp common.CredentialProvider) GCSRegistryProvider {
|
||
|
if cp == nil {
|
||
9 years ago
|
// TODO: replace this panic with an error return.
|
||
9 years ago
|
panic(fmt.Errorf("no credential provider"))
|
||
9 years ago
|
}
|
||
|
return &gcsRegistryProvider{cp: cp}
|
||
|
}
|
||
|
|
||
|
// GetGCSRegistry returns a new Google Cloud Storage . If there's a credential that is specified, will try
|
||
|
// to fetch it and use it, and if there's no credential found, will fall back to unauthenticated client.
|
||
|
func (gcsrp gcsRegistryProvider) GetGCSRegistry(cr common.Registry) (ObjectStorageRegistry, error) {
|
||
|
client, err := gcsrp.createGCSClient(cr.CredentialName)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
service, err := storage.New(client)
|
||
|
if err != nil {
|
||
|
log.Fatalf("Unable to create storage service: %v", err)
|
||
|
}
|
||
|
return NewGCSRegistry(cr.Name, cr.URL, client, service)
|
||
|
}
|
||
9 years ago
|
|
||
9 years ago
|
func (gcsrp gcsRegistryProvider) createGCSClient(credentialName string) (*http.Client, error) {
|
||
9 years ago
|
if credentialName == "" {
|
||
|
return http.DefaultClient, nil
|
||
|
}
|
||
|
|
||
9 years ago
|
c, err := gcsrp.cp.GetCredential(credentialName)
|
||
|
if err != nil {
|
||
|
log.Printf("Failed to fetch credential %s: %v", credentialName, err)
|
||
9 years ago
|
log.Print("Trying to use unauthenticated client")
|
||
|
return http.DefaultClient, nil
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
config, err := google.JWTConfigFromJSON([]byte(c.ServiceAccount), storage.DevstorageReadOnlyScope)
|
||
|
|
||
|
if err != nil {
|
||
|
log.Fatalf("Unable to parse client secret file to config: %v", err)
|
||
|
}
|
||
|
return config.Client(oauth2.NoContext), nil
|
||
9 years ago
|
}
|
||
|
|
||
|
// RE for a registry type that does support versions and has collections.
|
||
|
var TemplateRegistryMatcher = regexp.MustCompile("github.com/(.*)/(.*)/(.*)/(.*):(.*)")
|
||
|
|
||
|
// RE for a registry type that does not support versions and does not have collections.
|
||
|
var PackageRegistryMatcher = regexp.MustCompile("github.com/(.*)/(.*)/(.*)")
|
||
|
|
||
9 years ago
|
// RE for GCS storage
|
||
|
var GCSRegistryMatcher = regexp.MustCompile("gs://(.*)/(.*)")
|
||
|
|
||
9 years ago
|
// IsGithubShortType returns whether a given type is a type description in a short format to a github repository type.
|
||
|
// For now, this means using github types:
|
||
|
// github.com/owner/repo/qualifier/type:version
|
||
|
// for example:
|
||
|
// github.com/kubernetes/application-dm-templates/storage/redis:v1
|
||
|
func IsGithubShortType(t string) bool {
|
||
|
return TemplateRegistryMatcher.MatchString(t)
|
||
|
}
|
||
|
|
||
|
// IsGithubShortPackageType returns whether a given type is a type description in a short format to a github
|
||
|
// package repository type.
|
||
|
// For now, this means using github types:
|
||
|
// github.com/owner/repo/type
|
||
|
// for example:
|
||
|
// github.com/helm/charts/cassandra
|
||
|
func IsGithubShortPackageType(t string) bool {
|
||
|
return PackageRegistryMatcher.MatchString(t)
|
||
|
}
|
||
|
|
||
9 years ago
|
// IsGCSShortType returns whether a given type is a type description in a short format to GCS
|
||
|
func IsGCSShortType(t string) bool {
|
||
|
return strings.HasPrefix(t, "gs://")
|
||
|
}
|
||
|
|
||
9 years ago
|
// GetDownloadURLs checks a type to see if it is either a short git hub url or a fully specified URL
|
||
|
// and returns the URLs that should be used to fetch it. If the url is not fetchable (primitive type
|
||
|
// for example), it returns an empty slice.
|
||
9 years ago
|
func GetDownloadURLs(rp RegistryProvider, t string) ([]string, Registry, error) {
|
||
9 years ago
|
if IsGithubShortType(t) {
|
||
|
return ShortTypeToDownloadURLs(rp, t)
|
||
|
} else if IsGithubShortPackageType(t) {
|
||
|
return ShortTypeToPackageDownloadURLs(rp, t)
|
||
9 years ago
|
} else if IsGCSShortType(t) {
|
||
|
return ShortTypeToGCSDownloadUrls(rp, t)
|
||
9 years ago
|
} else if util.IsHTTPURL(t) {
|
||
9 years ago
|
result, err := url.Parse(t)
|
||
|
if err != nil {
|
||
9 years ago
|
return nil, nil, fmt.Errorf("cannot parse download URL %s: %s", t, err)
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
return []string{result.String()}, nil, nil
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
return []string{}, nil, nil
|
||
9 years ago
|
}
|
||
|
|
||
|
// ShortTypeToDownloadURLs converts a github URL into downloadable URL from github.
|
||
|
// Input must be of the type and is assumed to have been validated before this call:
|
||
|
// github.com/owner/repo/qualifier/type:version
|
||
|
// for example:
|
||
|
// github.com/kubernetes/application-dm-templates/storage/redis:v1
|
||
9 years ago
|
func ShortTypeToDownloadURLs(rp RegistryProvider, t string) ([]string, Registry, error) {
|
||
9 years ago
|
m := TemplateRegistryMatcher.FindStringSubmatch(t)
|
||
|
if len(m) != 6 {
|
||
9 years ago
|
return nil, nil, fmt.Errorf("cannot parse short github url: %s", t)
|
||
9 years ago
|
}
|
||
|
|
||
|
r, err := rp.GetRegistryByShortURL(t)
|
||
|
if err != nil {
|
||
9 years ago
|
return nil, nil, err
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
if r == nil {
|
||
9 years ago
|
return nil, nil, fmt.Errorf("cannot get github registry for %s", t)
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
tt, err := NewType(m[3], m[4], m[5])
|
||
|
if err != nil {
|
||
9 years ago
|
return nil, r, err
|
||
9 years ago
|
}
|
||
|
|
||
|
urls, err := r.GetDownloadURLs(tt)
|
||
|
if err != nil {
|
||
9 years ago
|
return nil, r, err
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
return util.ConvertURLsToStrings(urls), r, err
|
||
9 years ago
|
}
|
||
|
|
||
|
// ShortTypeToPackageDownloadURLs converts a github URL into downloadable URLs from github.
|
||
|
// Input must be of the type and is assumed to have been validated before this call:
|
||
|
// github.com/owner/repo/type
|
||
|
// for example:
|
||
|
// github.com/helm/charts/cassandra
|
||
9 years ago
|
func ShortTypeToPackageDownloadURLs(rp RegistryProvider, t string) ([]string, Registry, error) {
|
||
9 years ago
|
m := PackageRegistryMatcher.FindStringSubmatch(t)
|
||
|
if len(m) != 4 {
|
||
9 years ago
|
return nil, nil, fmt.Errorf("Failed to parse short github url: %s", t)
|
||
9 years ago
|
}
|
||
|
|
||
|
r, err := rp.GetRegistryByShortURL(t)
|
||
|
if err != nil {
|
||
9 years ago
|
return nil, nil, err
|
||
9 years ago
|
}
|
||
|
|
||
|
tt, err := NewType("", m[3], "")
|
||
|
if err != nil {
|
||
9 years ago
|
return nil, r, err
|
||
9 years ago
|
}
|
||
|
|
||
|
urls, err := r.GetDownloadURLs(tt)
|
||
|
if err != nil {
|
||
9 years ago
|
return nil, r, err
|
||
|
}
|
||
|
|
||
|
return util.ConvertURLsToStrings(urls), r, err
|
||
|
}
|
||
|
|
||
9 years ago
|
// ShortTypeToGCSDownloadUrls returns the download URLs for a short type name.
|
||
9 years ago
|
func ShortTypeToGCSDownloadUrls(rp RegistryProvider, t string) ([]string, Registry, error) {
|
||
|
m := GCSRegistryMatcher.FindStringSubmatch(t)
|
||
|
if len(m) != 3 {
|
||
|
return nil, nil, fmt.Errorf("Failed to parse short gs url: %s", t)
|
||
9 years ago
|
}
|
||
|
|
||
9 years ago
|
r, err := rp.GetRegistryByShortURL(t)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
tt, err := NewType(m[1], m[2], "")
|
||
|
if err != nil {
|
||
|
return nil, r, err
|
||
|
}
|
||
|
|
||
|
urls, err := r.GetDownloadURLs(tt)
|
||
|
if err != nil {
|
||
|
return nil, r, err
|
||
|
}
|
||
|
return util.ConvertURLsToStrings(urls), r, err
|
||
9 years ago
|
}
|