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.
403 lines
12 KiB
403 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 repo
|
|
|
|
import (
|
|
"bytes"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"os"
|
|
"path/filepath"
|
|
"reflect"
|
|
"runtime"
|
|
"strings"
|
|
"testing"
|
|
"time"
|
|
|
|
"sigs.k8s.io/yaml"
|
|
|
|
"helm.sh/helm/v3/pkg/chart"
|
|
"helm.sh/helm/v3/pkg/cli"
|
|
"helm.sh/helm/v3/pkg/getter"
|
|
)
|
|
|
|
const (
|
|
testRepository = "testdata/repository"
|
|
testURL = "http://example-charts.com"
|
|
)
|
|
|
|
func TestLoadChartRepository(t *testing.T) {
|
|
r, err := NewChartRepository(&Entry{
|
|
Name: testRepository,
|
|
URL: testURL,
|
|
}, getter.All(&cli.EnvSettings{}))
|
|
if err != nil {
|
|
t.Errorf("Problem creating chart repository from %s: %v", testRepository, err)
|
|
}
|
|
|
|
if err := r.Load(); err != nil {
|
|
t.Errorf("Problem loading chart repository from %s: %v", testRepository, err)
|
|
}
|
|
|
|
paths := []string{
|
|
filepath.Join(testRepository, "frobnitz-1.2.3.tgz"),
|
|
filepath.Join(testRepository, "sprocket-1.1.0.tgz"),
|
|
filepath.Join(testRepository, "sprocket-1.2.0.tgz"),
|
|
filepath.Join(testRepository, "universe/zarthal-1.0.0.tgz"),
|
|
}
|
|
|
|
if r.Config.Name != testRepository {
|
|
t.Errorf("Expected %s as Name but got %s", testRepository, r.Config.Name)
|
|
}
|
|
|
|
if !reflect.DeepEqual(r.ChartPaths, paths) {
|
|
t.Errorf("Expected %#v but got %#v\n", paths, r.ChartPaths)
|
|
}
|
|
|
|
if r.Config.URL != testURL {
|
|
t.Errorf("Expected url for chart repository to be %s but got %s", testURL, r.Config.URL)
|
|
}
|
|
}
|
|
|
|
func TestIndex(t *testing.T) {
|
|
r, err := NewChartRepository(&Entry{
|
|
Name: testRepository,
|
|
URL: testURL,
|
|
}, getter.All(&cli.EnvSettings{}))
|
|
if err != nil {
|
|
t.Errorf("Problem creating chart repository from %s: %v", testRepository, err)
|
|
}
|
|
|
|
if err := r.Load(); err != nil {
|
|
t.Errorf("Problem loading chart repository from %s: %v", testRepository, err)
|
|
}
|
|
|
|
err = r.Index()
|
|
if err != nil {
|
|
t.Errorf("Error performing index: %v\n", err)
|
|
}
|
|
|
|
tempIndexPath := filepath.Join(testRepository, indexPath)
|
|
actual, err := LoadIndexFile(tempIndexPath)
|
|
defer os.Remove(tempIndexPath) // clean up
|
|
if err != nil {
|
|
t.Errorf("Error loading index file %v", err)
|
|
}
|
|
verifyIndex(t, actual)
|
|
|
|
// Re-index and test again.
|
|
err = r.Index()
|
|
if err != nil {
|
|
t.Errorf("Error performing re-index: %s\n", err)
|
|
}
|
|
second, err := LoadIndexFile(tempIndexPath)
|
|
if err != nil {
|
|
t.Errorf("Error re-loading index file %v", err)
|
|
}
|
|
verifyIndex(t, second)
|
|
}
|
|
|
|
type CustomGetter struct {
|
|
repoUrls []string
|
|
}
|
|
|
|
func (g *CustomGetter) Get(href string, _ ...getter.Option) (*bytes.Buffer, error) {
|
|
index := &IndexFile{
|
|
APIVersion: "v1",
|
|
Generated: time.Now(),
|
|
}
|
|
indexBytes, err := yaml.Marshal(index)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
g.repoUrls = append(g.repoUrls, href)
|
|
return bytes.NewBuffer(indexBytes), nil
|
|
}
|
|
|
|
func TestIndexCustomSchemeDownload(t *testing.T) {
|
|
repoName := "gcs-repo"
|
|
repoURL := "gs://some-gcs-bucket"
|
|
myCustomGetter := &CustomGetter{}
|
|
customGetterConstructor := func(_ ...getter.Option) (getter.Getter, error) {
|
|
return myCustomGetter, nil
|
|
}
|
|
providers := getter.Providers{{
|
|
Schemes: []string{"gs"},
|
|
New: customGetterConstructor,
|
|
}}
|
|
repo, err := NewChartRepository(&Entry{
|
|
Name: repoName,
|
|
URL: repoURL,
|
|
}, providers)
|
|
if err != nil {
|
|
t.Fatalf("Problem loading chart repository from %s: %v", repoURL, err)
|
|
}
|
|
repo.CachePath = t.TempDir()
|
|
|
|
tempIndexFile, err := os.CreateTemp("", "test-repo")
|
|
if err != nil {
|
|
t.Fatalf("Failed to create temp index file: %v", err)
|
|
}
|
|
defer os.Remove(tempIndexFile.Name())
|
|
|
|
idx, err := repo.DownloadIndexFile()
|
|
if err != nil {
|
|
t.Fatalf("Failed to download index file to %s: %v", idx, err)
|
|
}
|
|
|
|
if len(myCustomGetter.repoUrls) != 1 {
|
|
t.Fatalf("Custom Getter.Get should be called once")
|
|
}
|
|
|
|
expectedRepoIndexURL := repoURL + "/index.yaml"
|
|
if myCustomGetter.repoUrls[0] != expectedRepoIndexURL {
|
|
t.Fatalf("Custom Getter.Get should be called with %s", expectedRepoIndexURL)
|
|
}
|
|
}
|
|
|
|
func verifyIndex(t *testing.T, actual *IndexFile) {
|
|
var empty time.Time
|
|
if actual.Generated.Equal(empty) {
|
|
t.Errorf("Generated should be greater than 0: %s", actual.Generated)
|
|
}
|
|
|
|
if actual.APIVersion != APIVersionV1 {
|
|
t.Error("Expected v1 API")
|
|
}
|
|
|
|
entries := actual.Entries
|
|
if numEntries := len(entries); numEntries != 3 {
|
|
t.Errorf("Expected 3 charts to be listed in index file but got %v", numEntries)
|
|
}
|
|
|
|
expects := map[string]ChartVersions{
|
|
"frobnitz": {
|
|
{
|
|
Metadata: &chart.Metadata{
|
|
Name: "frobnitz",
|
|
Version: "1.2.3",
|
|
},
|
|
},
|
|
},
|
|
"sprocket": {
|
|
{
|
|
Metadata: &chart.Metadata{
|
|
Name: "sprocket",
|
|
Version: "1.2.0",
|
|
},
|
|
},
|
|
{
|
|
Metadata: &chart.Metadata{
|
|
Name: "sprocket",
|
|
Version: "1.1.0",
|
|
},
|
|
},
|
|
},
|
|
"zarthal": {
|
|
{
|
|
Metadata: &chart.Metadata{
|
|
Name: "zarthal",
|
|
Version: "1.0.0",
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
for name, versions := range expects {
|
|
got, ok := entries[name]
|
|
if !ok {
|
|
t.Errorf("Could not find %q entry", name)
|
|
continue
|
|
}
|
|
if len(versions) != len(got) {
|
|
t.Errorf("Expected %d versions, got %d", len(versions), len(got))
|
|
continue
|
|
}
|
|
for i, e := range versions {
|
|
g := got[i]
|
|
if e.Name != g.Name {
|
|
t.Errorf("Expected %q, got %q", e.Name, g.Name)
|
|
}
|
|
if e.Version != g.Version {
|
|
t.Errorf("Expected %q, got %q", e.Version, g.Version)
|
|
}
|
|
if len(g.Keywords) != 3 {
|
|
t.Error("Expected 3 keywords.")
|
|
}
|
|
if len(g.Maintainers) != 2 {
|
|
t.Error("Expected 2 maintainers.")
|
|
}
|
|
if g.Created.Equal(empty) {
|
|
t.Error("Expected created to be non-empty")
|
|
}
|
|
if g.Description == "" {
|
|
t.Error("Expected description to be non-empty")
|
|
}
|
|
if g.Home == "" {
|
|
t.Error("Expected home to be non-empty")
|
|
}
|
|
if g.Digest == "" {
|
|
t.Error("Expected digest to be non-empty")
|
|
}
|
|
if len(g.URLs) != 1 {
|
|
t.Error("Expected exactly 1 URL")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// startLocalServerForTests Start the local helm server
|
|
func startLocalServerForTests(handler http.Handler) (*httptest.Server, error) {
|
|
if handler == nil {
|
|
fileBytes, err := os.ReadFile("testdata/local-index.yaml")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
handler = http.HandlerFunc(func(w http.ResponseWriter, _ *http.Request) {
|
|
w.Write(fileBytes)
|
|
})
|
|
}
|
|
|
|
return httptest.NewServer(handler), nil
|
|
}
|
|
|
|
// startLocalTLSServerForTests Start the local helm server with TLS
|
|
func startLocalTLSServerForTests(handler http.Handler) (*httptest.Server, error) {
|
|
if handler == nil {
|
|
fileBytes, err := os.ReadFile("testdata/local-index.yaml")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
handler = http.HandlerFunc(func(w http.ResponseWriter, _ *http.Request) {
|
|
w.Write(fileBytes)
|
|
})
|
|
}
|
|
|
|
return httptest.NewTLSServer(handler), nil
|
|
}
|
|
|
|
func TestFindChartInAuthAndTLSAndPassRepoURL(t *testing.T) {
|
|
srv, err := startLocalTLSServerForTests(nil)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer srv.Close()
|
|
|
|
chartURL, err := FindChartInAuthAndTLSAndPassRepoURL(srv.URL, "", "", "nginx", "", "", "", "", true, false, getter.All(&cli.EnvSettings{}))
|
|
if err != nil {
|
|
t.Fatalf("%v", err)
|
|
}
|
|
if chartURL != "https://charts.helm.sh/stable/nginx-0.2.0.tgz" {
|
|
t.Errorf("%s is not the valid URL", chartURL)
|
|
}
|
|
|
|
// If the insecureSkipTLSVerify is false, it will return an error that contains "x509: certificate signed by unknown authority".
|
|
_, err = FindChartInAuthAndTLSAndPassRepoURL(srv.URL, "", "", "nginx", "0.1.0", "", "", "", false, false, getter.All(&cli.EnvSettings{}))
|
|
// Go communicates with the platform and different platforms return different messages. Go itself tests darwin
|
|
// differently for its message. On newer versions of Darwin the message includes the "Acme Co" portion while older
|
|
// versions of Darwin do not. As there are people developing Helm using both old and new versions of Darwin we test
|
|
// for both messages.
|
|
if runtime.GOOS == "darwin" {
|
|
if !strings.Contains(err.Error(), "x509: “Acme Co” certificate is not trusted") && !strings.Contains(err.Error(), "x509: certificate signed by unknown authority") {
|
|
t.Errorf("Expected TLS error for function FindChartInAuthAndTLSAndPassRepoURL not found, but got a different error (%v)", err)
|
|
}
|
|
} else if !strings.Contains(err.Error(), "x509: certificate signed by unknown authority") {
|
|
t.Errorf("Expected TLS error for function FindChartInAuthAndTLSAndPassRepoURL not found, but got a different error (%v)", err)
|
|
}
|
|
}
|
|
|
|
func TestFindChartInRepoURL(t *testing.T) {
|
|
srv, err := startLocalServerForTests(nil)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer srv.Close()
|
|
|
|
chartURL, err := FindChartInRepoURL(srv.URL, "nginx", "", "", "", "", getter.All(&cli.EnvSettings{}))
|
|
if err != nil {
|
|
t.Fatalf("%v", err)
|
|
}
|
|
if chartURL != "https://charts.helm.sh/stable/nginx-0.2.0.tgz" {
|
|
t.Errorf("%s is not the valid URL", chartURL)
|
|
}
|
|
|
|
chartURL, err = FindChartInRepoURL(srv.URL, "nginx", "0.1.0", "", "", "", getter.All(&cli.EnvSettings{}))
|
|
if err != nil {
|
|
t.Errorf("%s", err)
|
|
}
|
|
if chartURL != "https://charts.helm.sh/stable/nginx-0.1.0.tgz" {
|
|
t.Errorf("%s is not the valid URL", chartURL)
|
|
}
|
|
}
|
|
|
|
func TestErrorFindChartInRepoURL(t *testing.T) {
|
|
|
|
g := getter.All(&cli.EnvSettings{
|
|
RepositoryCache: t.TempDir(),
|
|
})
|
|
|
|
if _, err := FindChartInRepoURL("http://someserver/something", "nginx", "", "", "", "", g); err == nil {
|
|
t.Errorf("Expected error for bad chart URL, but did not get any errors")
|
|
} else if !strings.Contains(err.Error(), `looks like "http://someserver/something" is not a valid chart repository or cannot be reached`) {
|
|
t.Errorf("Expected error for bad chart URL, but got a different error (%v)", err)
|
|
}
|
|
|
|
srv, err := startLocalServerForTests(nil)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
defer srv.Close()
|
|
|
|
if _, err = FindChartInRepoURL(srv.URL, "nginx1", "", "", "", "", g); err == nil {
|
|
t.Errorf("Expected error for chart not found, but did not get any errors")
|
|
} else if err.Error() != `chart "nginx1" not found in `+srv.URL+` repository` {
|
|
t.Errorf("Expected error for chart not found, but got a different error (%v)", err)
|
|
}
|
|
|
|
if _, err = FindChartInRepoURL(srv.URL, "nginx1", "0.1.0", "", "", "", g); err == nil {
|
|
t.Errorf("Expected error for chart not found, but did not get any errors")
|
|
} else if err.Error() != `chart "nginx1" version "0.1.0" not found in `+srv.URL+` repository` {
|
|
t.Errorf("Expected error for chart not found, but got a different error (%v)", err)
|
|
}
|
|
|
|
if _, err = FindChartInRepoURL(srv.URL, "chartWithNoURL", "", "", "", "", g); err == nil {
|
|
t.Errorf("Expected error for no chart URLs available, but did not get any errors")
|
|
} else if err.Error() != `chart "chartWithNoURL" has no downloadable URLs` {
|
|
t.Errorf("Expected error for chart not found, but got a different error (%v)", err)
|
|
}
|
|
}
|
|
|
|
func TestResolveReferenceURL(t *testing.T) {
|
|
for _, tt := range []struct {
|
|
baseURL, refURL, chartURL string
|
|
}{
|
|
{"http://localhost:8123/charts/", "nginx-0.2.0.tgz", "http://localhost:8123/charts/nginx-0.2.0.tgz"},
|
|
{"http://localhost:8123/charts-with-no-trailing-slash", "nginx-0.2.0.tgz", "http://localhost:8123/charts-with-no-trailing-slash/nginx-0.2.0.tgz"},
|
|
{"http://localhost:8123", "https://charts.helm.sh/stable/nginx-0.2.0.tgz", "https://charts.helm.sh/stable/nginx-0.2.0.tgz"},
|
|
{"http://localhost:8123/charts%2fwith%2fescaped%2fslash", "nginx-0.2.0.tgz", "http://localhost:8123/charts%2fwith%2fescaped%2fslash/nginx-0.2.0.tgz"},
|
|
{"http://localhost:8123/charts?with=queryparameter", "nginx-0.2.0.tgz", "http://localhost:8123/charts/nginx-0.2.0.tgz?with=queryparameter"},
|
|
} {
|
|
chartURL, err := ResolveReferenceURL(tt.baseURL, tt.refURL)
|
|
if err != nil {
|
|
t.Errorf("unexpected error in ResolveReferenceURL(%q, %q): %s", tt.baseURL, tt.refURL, err)
|
|
}
|
|
if chartURL != tt.chartURL {
|
|
t.Errorf("expected ResolveReferenceURL(%q, %q) to equal %q, got %q", tt.baseURL, tt.refURL, tt.chartURL, chartURL)
|
|
}
|
|
}
|
|
}
|