mirror of
https://github.com/helm/helm.git
synced 2026-04-08 02:25:03 -04:00
Linting is specific to the chart versions. A v2 and v3 chart will lint differently. To accomplish this, packages like engine need to be able to handle different chart versions. This was accomplished by some changes: 1. The introduction of a Charter interface for charts 2. The ChartAccessor which is able to accept a chart and then provide access to its data via an interface. There is an interface, factory, and implementation for each version of chart. 3. Common packages were moved to a common and util packages. Due to some package loops, there are 2 packages which may get some consolidation in the future. The new interfaces provide the foundation to move the actions and cmd packages to be able to handle multiple apiVersions of charts. Signed-off-by: Matt Farina <matt.farina@suse.com>
140 lines
4.9 KiB
Go
140 lines
4.9 KiB
Go
/*
|
|
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 engine
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"log/slog"
|
|
"strings"
|
|
|
|
apierrors "k8s.io/apimachinery/pkg/api/errors"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
"k8s.io/client-go/discovery"
|
|
"k8s.io/client-go/dynamic"
|
|
"k8s.io/client-go/rest"
|
|
)
|
|
|
|
type lookupFunc = func(apiversion string, resource string, namespace string, name string) (map[string]interface{}, error)
|
|
|
|
// NewLookupFunction returns a function for looking up objects in the cluster.
|
|
//
|
|
// If the resource does not exist, no error is raised.
|
|
func NewLookupFunction(config *rest.Config) lookupFunc { //nolint:revive
|
|
return newLookupFunction(clientProviderFromConfig{config: config})
|
|
}
|
|
|
|
type ClientProvider interface {
|
|
// GetClientFor returns a dynamic.NamespaceableResourceInterface suitable for interacting with resources
|
|
// corresponding to the provided apiVersion and kind, as well as a boolean indicating whether the resources
|
|
// are namespaced.
|
|
GetClientFor(apiVersion, kind string) (dynamic.NamespaceableResourceInterface, bool, error)
|
|
}
|
|
|
|
type clientProviderFromConfig struct {
|
|
config *rest.Config
|
|
}
|
|
|
|
func (c clientProviderFromConfig) GetClientFor(apiVersion, kind string) (dynamic.NamespaceableResourceInterface, bool, error) {
|
|
return getDynamicClientOnKind(apiVersion, kind, c.config)
|
|
}
|
|
|
|
func newLookupFunction(clientProvider ClientProvider) lookupFunc {
|
|
return func(apiversion string, kind string, namespace string, name string) (map[string]interface{}, error) {
|
|
var client dynamic.ResourceInterface
|
|
c, namespaced, err := clientProvider.GetClientFor(apiversion, kind)
|
|
if err != nil {
|
|
return map[string]interface{}{}, err
|
|
}
|
|
if namespaced && namespace != "" {
|
|
client = c.Namespace(namespace)
|
|
} else {
|
|
client = c
|
|
}
|
|
if name != "" {
|
|
// this will return a single object
|
|
obj, err := client.Get(context.Background(), name, metav1.GetOptions{})
|
|
if err != nil {
|
|
if apierrors.IsNotFound(err) {
|
|
// Just return an empty interface when the object was not found.
|
|
// That way, users can use `if not (lookup ...)` in their templates.
|
|
return map[string]interface{}{}, nil
|
|
}
|
|
return map[string]interface{}{}, err
|
|
}
|
|
return obj.UnstructuredContent(), nil
|
|
}
|
|
// this will return a list
|
|
obj, err := client.List(context.Background(), metav1.ListOptions{})
|
|
if err != nil {
|
|
if apierrors.IsNotFound(err) {
|
|
// Just return an empty interface when the object was not found.
|
|
// That way, users can use `if not (lookup ...)` in their templates.
|
|
return map[string]interface{}{}, nil
|
|
}
|
|
return map[string]interface{}{}, err
|
|
}
|
|
return obj.UnstructuredContent(), nil
|
|
}
|
|
}
|
|
|
|
// getDynamicClientOnKind returns a dynamic client on an Unstructured type. This client can be further namespaced.
|
|
func getDynamicClientOnKind(apiversion string, kind string, config *rest.Config) (dynamic.NamespaceableResourceInterface, bool, error) {
|
|
gvk := schema.FromAPIVersionAndKind(apiversion, kind)
|
|
apiRes, err := getAPIResourceForGVK(gvk, config)
|
|
if err != nil {
|
|
slog.Error("unable to get apiresource", "groupVersionKind", gvk.String(), slog.Any("error", err))
|
|
return nil, false, fmt.Errorf("unable to get apiresource from unstructured: %s: %w", gvk.String(), err)
|
|
}
|
|
gvr := schema.GroupVersionResource{
|
|
Group: apiRes.Group,
|
|
Version: apiRes.Version,
|
|
Resource: apiRes.Name,
|
|
}
|
|
intf, err := dynamic.NewForConfig(config)
|
|
if err != nil {
|
|
slog.Error("unable to get dynamic client", slog.Any("error", err))
|
|
return nil, false, err
|
|
}
|
|
res := intf.Resource(gvr)
|
|
return res, apiRes.Namespaced, nil
|
|
}
|
|
|
|
func getAPIResourceForGVK(gvk schema.GroupVersionKind, config *rest.Config) (metav1.APIResource, error) {
|
|
res := metav1.APIResource{}
|
|
discoveryClient, err := discovery.NewDiscoveryClientForConfig(config)
|
|
if err != nil {
|
|
slog.Error("unable to create discovery client", slog.Any("error", err))
|
|
return res, err
|
|
}
|
|
resList, err := discoveryClient.ServerResourcesForGroupVersion(gvk.GroupVersion().String())
|
|
if err != nil {
|
|
slog.Error("unable to retrieve resource list", "GroupVersion", gvk.GroupVersion().String(), slog.Any("error", err))
|
|
return res, err
|
|
}
|
|
for _, resource := range resList.APIResources {
|
|
// if a resource contains a "/" it's referencing a subresource. we don't support subresource for now.
|
|
if resource.Kind == gvk.Kind && !strings.Contains(resource.Name, "/") {
|
|
res = resource
|
|
res.Group = gvk.Group
|
|
res.Version = gvk.Version
|
|
break
|
|
}
|
|
}
|
|
return res, nil
|
|
}
|