/* Copyright 2016 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 repo // import "k8s.io/helm/pkg/repo" import ( "fmt" "io/ioutil" "net/url" "os" "path/filepath" "strings" "github.com/ghodss/yaml" "k8s.io/helm/pkg/chartutil" "k8s.io/helm/pkg/getter" "k8s.io/helm/pkg/provenance" ) // Entry represents a collection of parameters for chart repository type Entry struct { Name string `json:"name"` Cache string `json:"cache"` URL string `json:"url"` CertFile string `json:"certFile"` KeyFile string `json:"keyFile"` CAFile string `json:"caFile"` } // ChartRepository represents a chart repository type ChartRepository struct { Config *Entry ChartPaths []string IndexFile *IndexFile Client getter.Getter } // NewChartRepository constructs ChartRepository func NewChartRepository(cfg *Entry, getters getter.Providers) (*ChartRepository, error) { u, err := url.Parse(cfg.URL) if err != nil { return nil, fmt.Errorf("invalid chart URL format: %s", cfg.URL) } getterConstructor, err := getters.ByScheme(u.Scheme) if err != nil { return nil, fmt.Errorf("Could not find protocol handler for: %s", u.Scheme) } client, err := getterConstructor(cfg.URL, cfg.CertFile, cfg.KeyFile, cfg.CAFile) if err != nil { return nil, fmt.Errorf("Could not construct protocol handler for: %s error: %v", u.Scheme, err) } return &ChartRepository{ Config: cfg, IndexFile: NewIndexFile(), Client: client, }, nil } // Load loads a directory of charts as if it were a repository. // // It requires the presence of an index.yaml file in the directory. func (r *ChartRepository) Load() error { dirInfo, err := os.Stat(r.Config.Name) if err != nil { return err } if !dirInfo.IsDir() { return fmt.Errorf("%q is not a directory", r.Config.Name) } // FIXME: Why are we recursively walking directories? // FIXME: Why are we not reading the repositories.yaml to figure out // what repos to use? filepath.Walk(r.Config.Name, func(path string, f os.FileInfo, err error) error { if !f.IsDir() { if strings.Contains(f.Name(), "-index.yaml") { i, err := LoadIndexFile(path) if err != nil { return nil } r.IndexFile = i } else if strings.HasSuffix(f.Name(), ".tgz") { r.ChartPaths = append(r.ChartPaths, path) } } return nil }) return nil } // DownloadIndexFile fetches the index from a repository. // // cachePath is prepended to any index that does not have an absolute path. This // is for pre-2.2.0 repo files. func (r *ChartRepository) DownloadIndexFile(cachePath string) error { var indexURL string parsedURL, err := url.Parse(r.Config.URL) if err != nil { return err } parsedURL.Path = strings.TrimSuffix(parsedURL.Path, "/") + "/index.yaml" indexURL = parsedURL.String() resp, err := r.Client.Get(indexURL) if err != nil { return err } index, err := ioutil.ReadAll(resp) if err != nil { return err } if _, err := loadIndex(index); err != nil { return err } // In Helm 2.2.0 the config.cache was accidentally switched to an absolute // path, which broke backward compatibility. This fixes it by prepending a // global cache path to relative paths. // // It is changed on DownloadIndexFile because that was the method that // originally carried the cache path. cp := r.Config.Cache if !filepath.IsAbs(cp) { cp = filepath.Join(cachePath, cp) } return ioutil.WriteFile(cp, index, 0644) } // Index generates an index for the chart repository and writes an index.yaml file. func (r *ChartRepository) Index() error { err := r.generateIndex() if err != nil { return err } return r.saveIndexFile() } func (r *ChartRepository) saveIndexFile() error { index, err := yaml.Marshal(r.IndexFile) if err != nil { return err } return ioutil.WriteFile(filepath.Join(r.Config.Name, indexPath), index, 0644) } func (r *ChartRepository) generateIndex() error { for _, path := range r.ChartPaths { ch, err := chartutil.Load(path) if err != nil { return err } digest, err := provenance.DigestFile(path) if err != nil { return err } if !r.IndexFile.Has(ch.Metadata.Name, ch.Metadata.Version) { r.IndexFile.Add(ch.Metadata, path, r.Config.URL, digest) } // TODO: If a chart exists, but has a different Digest, should we error? } r.IndexFile.SortEntries() return nil } // FindChartInRepoURL finds chart in chart repository pointed by repoURL // without adding repo to repostiories func FindChartInRepoURL(repoURL, chartName, chartVersion, certFile, keyFile, caFile string, getters getter.Providers) (string, error) { // Download and write the index file to a temporary location tempIndexFile, err := ioutil.TempFile("", "tmp-repo-file") if err != nil { return "", fmt.Errorf("cannot write index file for repository requested") } defer os.Remove(tempIndexFile.Name()) c := Entry{ URL: repoURL, CertFile: certFile, KeyFile: keyFile, CAFile: caFile, } r, err := NewChartRepository(&c, getters) if err != nil { return "", err } if err := r.DownloadIndexFile(tempIndexFile.Name()); err != nil { return "", fmt.Errorf("Looks like %q is not a valid chart repository or cannot be reached: %s", repoURL, err) } // Read the index file for the repository to get chart information and return chart URL repoIndex, err := LoadIndexFile(tempIndexFile.Name()) if err != nil { return "", err } errMsg := fmt.Sprintf("chart %q", chartName) if chartVersion != "" { errMsg = fmt.Sprintf("%s version %q", errMsg, chartVersion) } cv, err := repoIndex.Get(chartName, chartVersion) if err != nil { return "", fmt.Errorf("%s not found in %s repository", errMsg, repoURL) } if len(cv.URLs) == 0 { return "", fmt.Errorf("%s has no downloadable URLs", errMsg) } return cv.URLs[0], nil }