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.
376 lines
12 KiB
376 lines
12 KiB
/*
|
|
Copyright The Helm Authors.
|
|
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 downloader
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"net/url"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"helm.sh/helm/v3/internal/experimental/registry"
|
|
"helm.sh/helm/v3/internal/fileutil"
|
|
"helm.sh/helm/v3/internal/urlutil"
|
|
"helm.sh/helm/v3/pkg/getter"
|
|
"helm.sh/helm/v3/pkg/helmpath"
|
|
"helm.sh/helm/v3/pkg/provenance"
|
|
"helm.sh/helm/v3/pkg/repo"
|
|
)
|
|
|
|
// VerificationStrategy describes a strategy for determining whether to verify a chart.
|
|
type VerificationStrategy int
|
|
|
|
const (
|
|
// VerifyNever will skip all verification of a chart.
|
|
VerifyNever VerificationStrategy = iota
|
|
// VerifyIfPossible will attempt a verification, it will not error if verification
|
|
// data is missing. But it will not stop processing if verification fails.
|
|
VerifyIfPossible
|
|
// VerifyAlways will always attempt a verification, and will fail if the
|
|
// verification fails.
|
|
VerifyAlways
|
|
// VerifyLater will fetch verification data, but not do any verification.
|
|
// This is to accommodate the case where another step of the process will
|
|
// perform verification.
|
|
VerifyLater
|
|
)
|
|
|
|
// ErrNoOwnerRepo indicates that a given chart URL can't be found in any repos.
|
|
var ErrNoOwnerRepo = errors.New("could not find a repo containing the given URL")
|
|
|
|
// ChartDownloader handles downloading a chart.
|
|
//
|
|
// It is capable of performing verifications on charts as well.
|
|
type ChartDownloader struct {
|
|
// Out is the location to write warning and info messages.
|
|
Out io.Writer
|
|
// Verify indicates what verification strategy to use.
|
|
Verify VerificationStrategy
|
|
// Keyring is the keyring file used for verification.
|
|
Keyring string
|
|
// Getter collection for the operation
|
|
Getters getter.Providers
|
|
// Options provide parameters to be passed along to the Getter being initialized.
|
|
Options []getter.Option
|
|
RegistryClient *registry.Client
|
|
RepositoryConfig string
|
|
RepositoryCache string
|
|
}
|
|
|
|
// DownloadTo retrieves a chart. Depending on the settings, it may also download a provenance file.
|
|
//
|
|
// If Verify is set to VerifyNever, the verification will be nil.
|
|
// If Verify is set to VerifyIfPossible, this will return a verification (or nil on failure), and print a warning on failure.
|
|
// If Verify is set to VerifyAlways, this will return a verification or an error if the verification fails.
|
|
// If Verify is set to VerifyLater, this will download the prov file (if it exists), but not verify it.
|
|
//
|
|
// For VerifyNever and VerifyIfPossible, the Verification may be empty.
|
|
//
|
|
// Returns a string path to the location where the file was downloaded and a verification
|
|
// (if provenance was verified), or an error if something bad happened.
|
|
func (c *ChartDownloader) DownloadTo(ref, version, dest string) (string, *provenance.Verification, error) {
|
|
u, err := c.ResolveChartVersion(ref, version)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
|
|
g, err := c.Getters.ByScheme(u.Scheme)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
|
|
data, err := g.Get(u.String(), c.Options...)
|
|
if err != nil {
|
|
return "", nil, err
|
|
}
|
|
|
|
name := filepath.Base(u.Path)
|
|
if u.Scheme == "oci" {
|
|
name = fmt.Sprintf("%s-%s.tgz", name, version)
|
|
}
|
|
|
|
destfile := filepath.Join(dest, name)
|
|
if err := fileutil.AtomicWriteFile(destfile, data, 0644); err != nil {
|
|
return destfile, nil, err
|
|
}
|
|
|
|
// If provenance is requested, verify it.
|
|
ver := &provenance.Verification{}
|
|
if c.Verify > VerifyNever {
|
|
body, err := g.Get(u.String() + ".prov")
|
|
if err != nil {
|
|
if c.Verify == VerifyAlways {
|
|
return destfile, ver, errors.Errorf("failed to fetch provenance %q", u.String()+".prov")
|
|
}
|
|
fmt.Fprintf(c.Out, "WARNING: Verification not found for %s: %s\n", ref, err)
|
|
return destfile, ver, nil
|
|
}
|
|
provfile := destfile + ".prov"
|
|
if err := fileutil.AtomicWriteFile(provfile, body, 0644); err != nil {
|
|
return destfile, nil, err
|
|
}
|
|
|
|
if c.Verify != VerifyLater {
|
|
ver, err = VerifyChart(destfile, c.Keyring)
|
|
if err != nil {
|
|
// Fail always in this case, since it means the verification step
|
|
// failed.
|
|
return destfile, ver, err
|
|
}
|
|
}
|
|
}
|
|
return destfile, ver, nil
|
|
}
|
|
|
|
// ResolveChartVersion resolves a chart reference to a URL.
|
|
//
|
|
// It returns the URL and sets the ChartDownloader's Options that can fetch
|
|
// the URL using the appropriate Getter.
|
|
//
|
|
// A reference may be an HTTP URL, a 'reponame/chartname' reference, or a local path.
|
|
//
|
|
// A version is a SemVer string (1.2.3-beta.1+f334a6789).
|
|
//
|
|
// - For fully qualified URLs, the version will be ignored (since URLs aren't versioned)
|
|
// - For a chart reference
|
|
// * If version is non-empty, this will return the URL for that version
|
|
// * If version is empty, this will return the URL for the latest version
|
|
// * If no version can be found, an error is returned
|
|
func (c *ChartDownloader) ResolveChartVersion(ref, version string) (*url.URL, error) {
|
|
u, err := url.Parse(ref)
|
|
if err != nil {
|
|
return nil, errors.Errorf("invalid chart URL format: %s", ref)
|
|
}
|
|
c.Options = append(c.Options, getter.WithURL(ref))
|
|
|
|
rf, err := loadRepoConfig(c.RepositoryConfig)
|
|
if err != nil {
|
|
return u, err
|
|
}
|
|
|
|
if u.IsAbs() && len(u.Host) > 0 && len(u.Path) > 0 {
|
|
// In this case, we have to find the parent repo that contains this chart
|
|
// URL. And this is an unfortunate problem, as it requires actually going
|
|
// through each repo cache file and finding a matching URL. But basically
|
|
// we want to find the repo in case we have special SSL cert config
|
|
// for that repo.
|
|
|
|
rc, err := c.scanReposForURL(ref, rf)
|
|
if err != nil {
|
|
// If there is no special config, return the default HTTP client and
|
|
// swallow the error.
|
|
if err == ErrNoOwnerRepo {
|
|
return u, nil
|
|
}
|
|
return u, err
|
|
}
|
|
|
|
// If we get here, we don't need to go through the next phase of looking
|
|
// up the URL. We have it already. So we just set the parameters and return.
|
|
c.Options = append(
|
|
c.Options,
|
|
getter.WithURL(rc.URL),
|
|
)
|
|
if rc.CertFile != "" || rc.KeyFile != "" || rc.CAFile != "" {
|
|
c.Options = append(c.Options, getter.WithTLSClientConfig(rc.CertFile, rc.KeyFile, rc.CAFile))
|
|
}
|
|
if rc.Username != "" && rc.Password != "" {
|
|
c.Options = append(
|
|
c.Options,
|
|
getter.WithBasicAuth(rc.Username, rc.Password),
|
|
)
|
|
}
|
|
return u, nil
|
|
}
|
|
|
|
// See if it's of the form: repo/path_to_chart
|
|
p := strings.SplitN(u.Path, "/", 2)
|
|
if len(p) < 2 {
|
|
return u, errors.Errorf("non-absolute URLs should be in form of repo_name/path_to_chart, got: %s", u)
|
|
}
|
|
|
|
repoName := p[0]
|
|
chartName := p[1]
|
|
rc, err := pickChartRepositoryConfigByName(repoName, rf.Repositories)
|
|
|
|
if err != nil {
|
|
return u, err
|
|
}
|
|
|
|
r, err := repo.NewChartRepository(rc, c.Getters)
|
|
if err != nil {
|
|
return u, err
|
|
}
|
|
|
|
if r != nil && r.Config != nil {
|
|
if r.Config.CertFile != "" || r.Config.KeyFile != "" || r.Config.CAFile != "" {
|
|
c.Options = append(c.Options, getter.WithTLSClientConfig(r.Config.CertFile, r.Config.KeyFile, r.Config.CAFile))
|
|
}
|
|
if r.Config.Username != "" && r.Config.Password != "" {
|
|
c.Options = append(c.Options, getter.WithBasicAuth(r.Config.Username, r.Config.Password))
|
|
}
|
|
}
|
|
|
|
// Next, we need to load the index, and actually look up the chart.
|
|
idxFile := filepath.Join(c.RepositoryCache, helmpath.CacheIndexFile(r.Config.Name))
|
|
i, err := repo.LoadIndexFile(idxFile)
|
|
if err != nil {
|
|
return u, errors.Wrap(err, "no cached repo found. (try 'helm repo update')")
|
|
}
|
|
|
|
cv, err := i.Get(chartName, version)
|
|
if err != nil {
|
|
return u, errors.Wrapf(err, "chart %q matching %s not found in %s index. (try 'helm repo update')", chartName, version, r.Config.Name)
|
|
}
|
|
|
|
if len(cv.URLs) == 0 {
|
|
return u, errors.Errorf("chart %q has no downloadable URLs", ref)
|
|
}
|
|
|
|
// TODO: Seems that picking first URL is not fully correct
|
|
u, err = url.Parse(cv.URLs[0])
|
|
if err != nil {
|
|
return u, errors.Errorf("invalid chart URL format: %s", ref)
|
|
}
|
|
|
|
// If the URL is relative (no scheme), prepend the chart repo's base URL
|
|
if !u.IsAbs() {
|
|
repoURL, err := url.Parse(rc.URL)
|
|
if err != nil {
|
|
return repoURL, err
|
|
}
|
|
q := repoURL.Query()
|
|
// We need a trailing slash for ResolveReference to work, but make sure there isn't already one
|
|
repoURL.Path = strings.TrimSuffix(repoURL.Path, "/") + "/"
|
|
u = repoURL.ResolveReference(u)
|
|
u.RawQuery = q.Encode()
|
|
// TODO add user-agent
|
|
if _, err := getter.NewHTTPGetter(getter.WithURL(rc.URL)); err != nil {
|
|
return repoURL, err
|
|
}
|
|
return u, err
|
|
}
|
|
|
|
// TODO add user-agent
|
|
return u, nil
|
|
}
|
|
|
|
// VerifyChart takes a path to a chart archive and a keyring, and verifies the chart.
|
|
//
|
|
// It assumes that a chart archive file is accompanied by a provenance file whose
|
|
// name is the archive file name plus the ".prov" extension.
|
|
func VerifyChart(path, keyring string) (*provenance.Verification, error) {
|
|
// For now, error out if it's not a tar file.
|
|
switch fi, err := os.Stat(path); {
|
|
case err != nil:
|
|
return nil, err
|
|
case fi.IsDir():
|
|
return nil, errors.New("unpacked charts cannot be verified")
|
|
case !isTar(path):
|
|
return nil, errors.New("chart must be a tgz file")
|
|
}
|
|
|
|
provfile := path + ".prov"
|
|
if _, err := os.Stat(provfile); err != nil {
|
|
return nil, errors.Wrapf(err, "could not load provenance file %s", provfile)
|
|
}
|
|
|
|
sig, err := provenance.NewFromKeyring(keyring, "")
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to load keyring")
|
|
}
|
|
return sig.Verify(path, provfile)
|
|
}
|
|
|
|
// isTar tests whether the given file is a tar file.
|
|
//
|
|
// Currently, this simply checks extension, since a subsequent function will
|
|
// untar the file and validate its binary format.
|
|
func isTar(filename string) bool {
|
|
return strings.EqualFold(filepath.Ext(filename), ".tgz")
|
|
}
|
|
|
|
func pickChartRepositoryConfigByName(name string, cfgs []*repo.Entry) (*repo.Entry, error) {
|
|
for _, rc := range cfgs {
|
|
if rc.Name == name {
|
|
if rc.URL == "" {
|
|
return nil, errors.Errorf("no URL found for repository %s", name)
|
|
}
|
|
return rc, nil
|
|
}
|
|
}
|
|
return nil, errors.Errorf("repo %s not found", name)
|
|
}
|
|
|
|
// scanReposForURL scans all repos to find which repo contains the given URL.
|
|
//
|
|
// This will attempt to find the given URL in all of the known repositories files.
|
|
//
|
|
// If the URL is found, this will return the repo entry that contained that URL.
|
|
//
|
|
// If all of the repos are checked, but the URL is not found, an ErrNoOwnerRepo
|
|
// error is returned.
|
|
//
|
|
// Other errors may be returned when repositories cannot be loaded or searched.
|
|
//
|
|
// Technically, the fact that a URL is not found in a repo is not a failure indication.
|
|
// Charts are not required to be included in an index before they are valid. So
|
|
// be mindful of this case.
|
|
//
|
|
// The same URL can technically exist in two or more repositories. This algorithm
|
|
// will return the first one it finds. Order is determined by the order of repositories
|
|
// in the repositories.yaml file.
|
|
func (c *ChartDownloader) scanReposForURL(u string, rf *repo.File) (*repo.Entry, error) {
|
|
// FIXME: This is far from optimal. Larger installations and index files will
|
|
// incur a performance hit for this type of scanning.
|
|
for _, rc := range rf.Repositories {
|
|
r, err := repo.NewChartRepository(rc, c.Getters)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
idxFile := filepath.Join(c.RepositoryCache, helmpath.CacheIndexFile(r.Config.Name))
|
|
i, err := repo.LoadIndexFile(idxFile)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "no cached repo found. (try 'helm repo update')")
|
|
}
|
|
|
|
for _, entry := range i.Entries {
|
|
for _, ver := range entry {
|
|
for _, dl := range ver.URLs {
|
|
if urlutil.Equal(u, dl) {
|
|
return rc, nil
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// This means that there is no repo file for the given URL.
|
|
return nil, ErrNoOwnerRepo
|
|
}
|
|
|
|
func loadRepoConfig(file string) (*repo.File, error) {
|
|
r, err := repo.LoadFile(file)
|
|
if err != nil && !os.IsNotExist(errors.Cause(err)) {
|
|
return nil, err
|
|
}
|
|
return r, nil
|
|
}
|