2020-02-24 19:14:46 +03:00
|
|
|
package plugins
|
|
|
|
|
|
|
|
/*
|
|
|
|
some of the code here is borrowed from the krew codebse (kubernetes)
|
|
|
|
and the copyright belongs to the respective authors.
|
|
|
|
|
|
|
|
source: https://github.com/kubernetes-sigs/krew/tree/master/internal
|
|
|
|
*/
|
|
|
|
|
|
|
|
import (
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
|
2021-05-14 22:09:01 +03:00
|
|
|
"github.com/goccy/go-yaml"
|
2021-06-16 14:44:15 +03:00
|
|
|
"github.com/hasura/graphql-engine/cli/v2/plugins/paths"
|
2020-02-24 19:14:46 +03:00
|
|
|
"github.com/pkg/errors"
|
2020-04-07 17:12:14 +03:00
|
|
|
"github.com/spf13/afero"
|
2020-02-24 19:14:46 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
func (c *Config) findPluginManifestFiles(indexDir string) ([]string, error) {
|
2020-11-12 12:25:48 +03:00
|
|
|
c.Logger.Debugf("finding plugin manifest files in directory %v", indexDir)
|
2020-02-24 19:14:46 +03:00
|
|
|
var out []string
|
2020-04-07 17:12:14 +03:00
|
|
|
fs := afero.Afero{
|
|
|
|
Fs: afero.NewOsFs(),
|
2020-02-24 19:14:46 +03:00
|
|
|
}
|
2020-04-07 17:12:14 +03:00
|
|
|
fs.Walk(indexDir, func(path string, info os.FileInfo, err error) error {
|
2020-11-12 12:25:48 +03:00
|
|
|
if info == nil {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2020-04-07 17:12:14 +03:00
|
|
|
if info.Mode().IsRegular() && filepath.Ext(info.Name()) == paths.ManifestExtension {
|
|
|
|
out = append(out, path)
|
2020-02-24 19:14:46 +03:00
|
|
|
}
|
2020-04-07 17:12:14 +03:00
|
|
|
return nil
|
|
|
|
})
|
2020-11-12 12:25:48 +03:00
|
|
|
|
2020-02-24 19:14:46 +03:00
|
|
|
return out, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// LoadPluginListFromFS will parse and retrieve all plugin files.
|
2020-04-07 17:12:14 +03:00
|
|
|
func (c *Config) LoadPluginListFromFS(indexDir string) (Plugins, error) {
|
2020-02-24 19:14:46 +03:00
|
|
|
indexDir, err := filepath.EvalSymlinks(indexDir)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
files, err := c.findPluginManifestFiles(indexDir)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to scan plugins in index directory")
|
|
|
|
}
|
|
|
|
|
2020-04-07 17:12:14 +03:00
|
|
|
return c.LoadPlugins(files), nil
|
2020-02-24 19:14:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// LoadPluginByName loads a plugins index file by its name. When plugin
|
|
|
|
// file not found, it returns an error that can be checked with os.IsNotExist.
|
2020-04-07 17:12:14 +03:00
|
|
|
func (c *Config) LoadPluginByName(pluginName string) (*PluginVersions, error) {
|
2020-11-12 12:25:48 +03:00
|
|
|
c.Logger.Debugf("loading plugin %s", pluginName)
|
2020-02-24 19:14:46 +03:00
|
|
|
if !IsSafePluginName(pluginName) {
|
2020-04-07 17:12:14 +03:00
|
|
|
return nil, errors.Errorf("plugin name %q not allowed", pluginName)
|
|
|
|
}
|
|
|
|
|
|
|
|
files, err := c.findPluginManifestFiles(c.Paths.IndexPluginsPath())
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.Wrap(err, "failed to scan plugins in index directory")
|
|
|
|
}
|
|
|
|
|
|
|
|
ps := c.LoadPlugins(files, pluginName)
|
|
|
|
if _, ok := ps[pluginName]; !ok {
|
|
|
|
return nil, os.ErrNotExist
|
2020-02-24 19:14:46 +03:00
|
|
|
}
|
|
|
|
|
2020-04-07 17:12:14 +03:00
|
|
|
return ps[pluginName], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Config) LoadPlugins(files []string, pluginName ...string) Plugins {
|
2020-11-12 12:25:48 +03:00
|
|
|
c.Logger.Debugf("loading plugins")
|
2020-04-07 17:12:14 +03:00
|
|
|
ps := Plugins{}
|
|
|
|
for _, file := range files {
|
|
|
|
p, err := c.ReadPluginFromFile(file)
|
|
|
|
if err != nil {
|
|
|
|
c.Logger.Debugf("failed to read or parse plugin manifest: %v", err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if p.ParsedVersion == nil {
|
|
|
|
c.Logger.Debugf("version of plugin %s cannot be nil", p.Name)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if len(pluginName) != 0 {
|
|
|
|
var isFound bool
|
|
|
|
for _, name := range pluginName {
|
|
|
|
if name == p.Name {
|
|
|
|
isFound = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if !isFound {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if _, ok := ps[p.Name]; !ok {
|
|
|
|
ps[p.Name] = NewPluginVersions()
|
|
|
|
}
|
|
|
|
err = ps[p.Name].Append(p)
|
|
|
|
if err != nil {
|
|
|
|
c.Logger.Debugf("failed to append version %s for plugin %s: %v", p.Version, p.Name, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ps
|
2020-02-24 19:14:46 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// ReadPluginFromFile loads a file from the FS. When plugin file not found, it
|
|
|
|
// returns an error that can be checked with os.IsNotExist.
|
|
|
|
func (c *Config) ReadPluginFromFile(path string) (Plugin, error) {
|
|
|
|
f, err := os.Open(path)
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return Plugin{}, err
|
|
|
|
} else if err != nil {
|
|
|
|
return Plugin{}, errors.Wrap(err, "failed to open index file")
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
var plugin Plugin
|
|
|
|
b, err := ioutil.ReadAll(f)
|
|
|
|
if err != nil {
|
|
|
|
return plugin, err
|
|
|
|
}
|
|
|
|
err = yaml.Unmarshal(b, &plugin)
|
|
|
|
if err != nil {
|
|
|
|
return plugin, errors.Wrap(err, "failed to decode plugin manifest")
|
|
|
|
}
|
2020-04-07 17:12:14 +03:00
|
|
|
plugin.ParseVersion()
|
2020-02-24 19:14:46 +03:00
|
|
|
return plugin, errors.Wrap(plugin.ValidatePlugin(plugin.Name), "plugin manifest validation error")
|
|
|
|
}
|