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.
208 lines
5.9 KiB
208 lines
5.9 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 registry // import "helm.sh/helm/v4/pkg/registry"
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"slices"
|
|
"strings"
|
|
"time"
|
|
|
|
"helm.sh/helm/v4/internal/tlsutil"
|
|
chart "helm.sh/helm/v4/pkg/chart/v2"
|
|
"helm.sh/helm/v4/pkg/chart/v2/loader"
|
|
helmtime "helm.sh/helm/v4/pkg/time"
|
|
|
|
"github.com/Masterminds/semver/v3"
|
|
ocispec "github.com/opencontainers/image-spec/specs-go/v1"
|
|
)
|
|
|
|
var immutableOciAnnotations = []string{
|
|
ocispec.AnnotationVersion,
|
|
ocispec.AnnotationTitle,
|
|
}
|
|
|
|
// IsOCI determines whether a URL is to be treated as an OCI URL
|
|
func IsOCI(url string) bool {
|
|
return strings.HasPrefix(url, fmt.Sprintf("%s://", OCIScheme))
|
|
}
|
|
|
|
// ContainsTag determines whether a tag is found in a provided list of tags
|
|
func ContainsTag(tags []string, tag string) bool {
|
|
return slices.Contains(tags, tag)
|
|
}
|
|
|
|
func GetTagMatchingVersionOrConstraint(tags []string, versionString string) (string, error) {
|
|
var constraint *semver.Constraints
|
|
if versionString == "" {
|
|
// If string is empty, set wildcard constraint
|
|
constraint, _ = semver.NewConstraint("*")
|
|
} else {
|
|
// when customer inputs specific version, check whether there's an exact match first
|
|
for _, v := range tags {
|
|
if versionString == v {
|
|
return v, nil
|
|
}
|
|
}
|
|
|
|
// Otherwise set constraint to the string given
|
|
var err error
|
|
constraint, err = semver.NewConstraint(versionString)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
}
|
|
|
|
// Otherwise try to find the first available version matching the string,
|
|
// in case it is a constraint
|
|
for _, v := range tags {
|
|
test, err := semver.NewVersion(v)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
if constraint.Check(test) {
|
|
return v, nil
|
|
}
|
|
}
|
|
|
|
return "", fmt.Errorf("could not locate a version matching provided version string %s", versionString)
|
|
}
|
|
|
|
// extractChartMeta is used to extract a chart metadata from a byte array
|
|
func extractChartMeta(chartData []byte) (*chart.Metadata, error) {
|
|
ch, err := loader.LoadArchive(bytes.NewReader(chartData))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return ch.Metadata, nil
|
|
}
|
|
|
|
// NewRegistryClientWithTLS is a helper function to create a new registry client with TLS enabled.
|
|
func NewRegistryClientWithTLS(out io.Writer, certFile, keyFile, caFile string, insecureSkipTLSverify bool, registryConfig string, debug bool) (*Client, error) {
|
|
tlsConf, err := tlsutil.NewTLSConfig(
|
|
tlsutil.WithInsecureSkipVerify(insecureSkipTLSverify),
|
|
tlsutil.WithCertKeyPairFiles(certFile, keyFile),
|
|
tlsutil.WithCAFile(caFile),
|
|
)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("can't create TLS config for client: %s", err)
|
|
}
|
|
// Create a new registry client
|
|
registryClient, err := NewClient(
|
|
ClientOptDebug(debug),
|
|
ClientOptEnableCache(true),
|
|
ClientOptWriter(out),
|
|
ClientOptCredentialsFile(registryConfig),
|
|
ClientOptHTTPClient(&http.Client{
|
|
Transport: &http.Transport{
|
|
TLSClientConfig: tlsConf,
|
|
Proxy: http.ProxyFromEnvironment,
|
|
},
|
|
}),
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return registryClient, nil
|
|
}
|
|
|
|
// generateOCIAnnotations will generate OCI annotations to include within the OCI manifest
|
|
func generateOCIAnnotations(meta *chart.Metadata, creationTime string) map[string]string {
|
|
|
|
// Get annotations from Chart attributes
|
|
ociAnnotations := generateChartOCIAnnotations(meta, creationTime)
|
|
|
|
// Copy Chart annotations
|
|
annotations:
|
|
for chartAnnotationKey, chartAnnotationValue := range meta.Annotations {
|
|
|
|
// Avoid overriding key properties
|
|
for _, immutableOciKey := range immutableOciAnnotations {
|
|
if immutableOciKey == chartAnnotationKey {
|
|
continue annotations
|
|
}
|
|
}
|
|
|
|
// Add chart annotation
|
|
ociAnnotations[chartAnnotationKey] = chartAnnotationValue
|
|
}
|
|
|
|
return ociAnnotations
|
|
}
|
|
|
|
// generateChartOCIAnnotations will generate OCI annotations from the provided chart
|
|
func generateChartOCIAnnotations(meta *chart.Metadata, creationTime string) map[string]string {
|
|
chartOCIAnnotations := map[string]string{}
|
|
|
|
chartOCIAnnotations = addToMap(chartOCIAnnotations, ocispec.AnnotationDescription, meta.Description)
|
|
chartOCIAnnotations = addToMap(chartOCIAnnotations, ocispec.AnnotationTitle, meta.Name)
|
|
chartOCIAnnotations = addToMap(chartOCIAnnotations, ocispec.AnnotationVersion, meta.Version)
|
|
chartOCIAnnotations = addToMap(chartOCIAnnotations, ocispec.AnnotationURL, meta.Home)
|
|
|
|
if len(creationTime) == 0 {
|
|
creationTime = helmtime.Now().UTC().Format(time.RFC3339)
|
|
}
|
|
|
|
chartOCIAnnotations = addToMap(chartOCIAnnotations, ocispec.AnnotationCreated, creationTime)
|
|
|
|
if len(meta.Sources) > 0 {
|
|
chartOCIAnnotations = addToMap(chartOCIAnnotations, ocispec.AnnotationSource, meta.Sources[0])
|
|
}
|
|
|
|
if len(meta.Maintainers) > 0 {
|
|
var maintainerSb strings.Builder
|
|
|
|
for maintainerIdx, maintainer := range meta.Maintainers {
|
|
|
|
if len(maintainer.Name) > 0 {
|
|
maintainerSb.WriteString(maintainer.Name)
|
|
}
|
|
|
|
if len(maintainer.Email) > 0 {
|
|
maintainerSb.WriteString(" (")
|
|
maintainerSb.WriteString(maintainer.Email)
|
|
maintainerSb.WriteString(")")
|
|
}
|
|
|
|
if maintainerIdx < len(meta.Maintainers)-1 {
|
|
maintainerSb.WriteString(", ")
|
|
}
|
|
|
|
}
|
|
|
|
chartOCIAnnotations = addToMap(chartOCIAnnotations, ocispec.AnnotationAuthors, maintainerSb.String())
|
|
|
|
}
|
|
|
|
return chartOCIAnnotations
|
|
}
|
|
|
|
// addToMap takes an existing map and adds an item if the value is not empty
|
|
func addToMap(inputMap map[string]string, newKey string, newValue string) map[string]string {
|
|
|
|
// Add item to map if its
|
|
if len(strings.TrimSpace(newValue)) > 0 {
|
|
inputMap[newKey] = newValue
|
|
}
|
|
|
|
return inputMap
|
|
|
|
}
|