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.
250 lines
6.0 KiB
250 lines
6.0 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 plugin
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"path/filepath"
|
|
|
|
"go.yaml.in/yaml/v3"
|
|
)
|
|
|
|
func peekAPIVersion(r io.Reader) (string, error) {
|
|
type apiVersion struct {
|
|
APIVersion string `yaml:"apiVersion"`
|
|
}
|
|
|
|
var v apiVersion
|
|
d := yaml.NewDecoder(r)
|
|
if err := d.Decode(&v); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return v.APIVersion, nil
|
|
}
|
|
|
|
func loadMetadataLegacy(metadataData []byte) (*Metadata, error) {
|
|
|
|
var ml MetadataLegacy
|
|
d := yaml.NewDecoder(bytes.NewReader(metadataData))
|
|
if err := d.Decode(&ml); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := ml.Validate(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
m := fromMetadataLegacy(ml)
|
|
if err := m.Validate(); err != nil {
|
|
return nil, err
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
func loadMetadataV1(metadataData []byte) (*Metadata, error) {
|
|
|
|
var mv1 MetadataV1
|
|
d := yaml.NewDecoder(bytes.NewReader(metadataData))
|
|
if err := d.Decode(&mv1); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if err := mv1.Validate(); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
m, err := fromMetadataV1(mv1)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to convert MetadataV1 to Metadata: %w", err)
|
|
}
|
|
|
|
if err := m.Validate(); err != nil {
|
|
return nil, err
|
|
}
|
|
return m, nil
|
|
}
|
|
|
|
func loadMetadata(metadataData []byte) (*Metadata, error) {
|
|
apiVersion, err := peekAPIVersion(bytes.NewReader(metadataData))
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to peek %s API version: %w", PluginFileName, err)
|
|
}
|
|
|
|
switch apiVersion {
|
|
case "": // legacy
|
|
return loadMetadataLegacy(metadataData)
|
|
case "v1":
|
|
return loadMetadataV1(metadataData)
|
|
}
|
|
|
|
return nil, fmt.Errorf("invalid plugin apiVersion: %q", apiVersion)
|
|
}
|
|
|
|
type prototypePluginManager struct {
|
|
runtimes map[string]Runtime
|
|
}
|
|
|
|
func newPrototypePluginManager() *prototypePluginManager {
|
|
return &prototypePluginManager{
|
|
runtimes: map[string]Runtime{
|
|
"subprocess": &RuntimeSubprocess{},
|
|
},
|
|
}
|
|
}
|
|
|
|
func (pm *prototypePluginManager) RegisterRuntime(runtimeName string, runtime Runtime) {
|
|
pm.runtimes[runtimeName] = runtime
|
|
}
|
|
|
|
func (pm *prototypePluginManager) CreatePlugin(pluginPath string, metadata *Metadata) (Plugin, error) {
|
|
rt, ok := pm.runtimes[metadata.Runtime]
|
|
if !ok {
|
|
return nil, fmt.Errorf("unsupported plugin runtime type: %q", metadata.Runtime)
|
|
}
|
|
|
|
return rt.CreatePlugin(pluginPath, metadata)
|
|
}
|
|
|
|
// LoadDir loads a plugin from the given directory.
|
|
func LoadDir(dirname string) (Plugin, error) {
|
|
pluginfile := filepath.Join(dirname, PluginFileName)
|
|
metadataData, err := os.ReadFile(pluginfile)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to read plugin at %q: %w", pluginfile, err)
|
|
}
|
|
|
|
m, err := loadMetadata(metadataData)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to load plugin %q: %w", dirname, err)
|
|
}
|
|
|
|
pm := newPrototypePluginManager()
|
|
return pm.CreatePlugin(dirname, m)
|
|
}
|
|
|
|
// LoadAll loads all plugins found beneath the base directory.
|
|
//
|
|
// This scans only one directory level.
|
|
func LoadAll(basedir string) ([]Plugin, error) {
|
|
var plugins []Plugin
|
|
// We want basedir/*/plugin.yaml
|
|
scanpath := filepath.Join(basedir, "*", PluginFileName)
|
|
matches, err := filepath.Glob(scanpath)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to search for plugins in %q: %w", scanpath, err)
|
|
}
|
|
|
|
// empty dir should load
|
|
if len(matches) == 0 {
|
|
return plugins, nil
|
|
}
|
|
|
|
for _, yamlFile := range matches {
|
|
dir := filepath.Dir(yamlFile)
|
|
p, err := LoadDir(dir)
|
|
if err != nil {
|
|
return plugins, err
|
|
}
|
|
plugins = append(plugins, p)
|
|
}
|
|
return plugins, detectDuplicates(plugins)
|
|
}
|
|
|
|
// findFunc is a function that finds plugins in a directory
|
|
type findFunc func(pluginsDir string) ([]Plugin, error)
|
|
|
|
// filterFunc is a function that filters plugins
|
|
type filterFunc func(Plugin) bool
|
|
|
|
// FindPlugins returns a list of plugins that match the descriptor
|
|
func FindPlugins(pluginsDirs []string, descriptor Descriptor) ([]Plugin, error) {
|
|
return findPlugins(pluginsDirs, LoadAll, makeDescriptorFilter(descriptor))
|
|
}
|
|
|
|
// findPlugins is the internal implementation that uses the find and filter functions
|
|
func findPlugins(pluginsDirs []string, findFn findFunc, filterFn filterFunc) ([]Plugin, error) {
|
|
var found []Plugin
|
|
for _, pluginsDir := range pluginsDirs {
|
|
ps, err := findFn(pluginsDir)
|
|
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for _, p := range ps {
|
|
if filterFn(p) {
|
|
found = append(found, p)
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return found, nil
|
|
}
|
|
|
|
// makeDescriptorFilter creates a filter function from a descriptor
|
|
// Additional plugin filter criteria we wish to support can be added here
|
|
func makeDescriptorFilter(descriptor Descriptor) filterFunc {
|
|
return func(p Plugin) bool {
|
|
// If name is specified, it must match
|
|
if descriptor.Name != "" && p.Metadata().Name != descriptor.Name {
|
|
return false
|
|
|
|
}
|
|
// If type is specified, it must match
|
|
if descriptor.Type != "" && p.Metadata().Type != descriptor.Type {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
}
|
|
|
|
// FindPlugin returns a single plugin that matches the descriptor
|
|
func FindPlugin(dirs []string, descriptor Descriptor) (Plugin, error) {
|
|
plugins, err := FindPlugins(dirs, descriptor)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(plugins) > 0 {
|
|
return plugins[0], nil
|
|
}
|
|
|
|
return nil, fmt.Errorf("plugin: %+v not found", descriptor)
|
|
}
|
|
|
|
func detectDuplicates(plugs []Plugin) error {
|
|
names := map[string]string{}
|
|
|
|
for _, plug := range plugs {
|
|
if oldpath, ok := names[plug.Metadata().Name]; ok {
|
|
return fmt.Errorf(
|
|
"two plugins claim the name %q at %q and %q",
|
|
plug.Metadata().Name,
|
|
oldpath,
|
|
plug.Dir(),
|
|
)
|
|
}
|
|
names[plug.Metadata().Name] = plug.Dir()
|
|
}
|
|
|
|
return nil
|
|
}
|