/* 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 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"` Username string `json:"username"` Password string `json:"password"` 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() r.setCredentials() 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) } // If HttpGetter is used, this method sets the configured repository credentials on the HttpGetter. func (r *ChartRepository) setCredentials() { if t, ok := r.Client.(*getter.HttpGetter); ok { t.SetCredentials(r.Config.Username, r.Config.Password) } } // 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 repositories func FindChartInRepoURL(repoURL, chartName, chartVersion, certFile, keyFile, caFile string, getters getter.Providers) (string, error) { return FindChartInAuthRepoURL(repoURL, "", "", chartName, chartVersion, certFile, keyFile, caFile, getters) } // FindChartInAuthRepoURL finds chart in chart repository pointed by repoURL // without adding repo to repositories, like FindChartInRepoURL, // but it also receives credentials for the chart repository. func FindChartInAuthRepoURL(repoURL, username, password, 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, Username: username, Password: password, 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) } chartURL := cv.URLs[0] absoluteChartURL, err := ResolveReferenceURL(repoURL, chartURL) if err != nil { return "", fmt.Errorf("failed to make chart URL absolute: %v", err) } return absoluteChartURL, nil } // ResolveReferenceURL resolves refURL relative to baseURL. // If refURL is absolute, it simply returns refURL. func ResolveReferenceURL(baseURL, refURL string) (string, error) { parsedBaseURL, err := url.Parse(baseURL) if err != nil { return "", fmt.Errorf("failed to parse %s as URL: %v", baseURL, err) } parsedRefURL, err := url.Parse(refURL) if err != nil { return "", fmt.Errorf("failed to parse %s as URL: %v", refURL, err) } return parsedBaseURL.ResolveReference(parsedRefURL).String(), nil }