2018-07-13 17:35:08 +00:00
|
|
|
package command
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/hashicorp/vault/api"
|
|
|
|
"github.com/mitchellh/cli"
|
|
|
|
"github.com/posener/complete"
|
|
|
|
)
|
|
|
|
|
2021-04-08 16:43:39 +00:00
|
|
|
var (
|
|
|
|
_ cli.Command = (*PluginListCommand)(nil)
|
|
|
|
_ cli.CommandAutocomplete = (*PluginListCommand)(nil)
|
|
|
|
)
|
2018-07-13 17:35:08 +00:00
|
|
|
|
|
|
|
type PluginListCommand struct {
|
|
|
|
*BaseCommand
|
2022-09-09 20:03:07 +00:00
|
|
|
|
|
|
|
flagDetailed bool
|
2018-07-13 17:35:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *PluginListCommand) Synopsis() string {
|
|
|
|
return "Lists available plugins"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *PluginListCommand) Help() string {
|
|
|
|
helpText := `
|
2018-11-07 01:21:24 +00:00
|
|
|
Usage: vault plugin list [options] [TYPE]
|
2018-07-13 17:35:08 +00:00
|
|
|
|
|
|
|
Lists available plugins registered in the catalog. This does not list whether
|
2018-11-07 01:21:24 +00:00
|
|
|
plugins are in use, but rather just their availability. The last argument of
|
|
|
|
type takes "auth", "database", or "secret".
|
2018-07-13 17:35:08 +00:00
|
|
|
|
|
|
|
List all available plugins in the catalog:
|
|
|
|
|
|
|
|
$ vault plugin list
|
|
|
|
|
2018-11-07 01:21:24 +00:00
|
|
|
List all available database plugins in the catalog:
|
|
|
|
|
|
|
|
$ vault plugin list database
|
|
|
|
|
2022-09-09 20:03:07 +00:00
|
|
|
List all available plugins with detailed output:
|
|
|
|
|
|
|
|
$ vault plugin list -detailed
|
|
|
|
|
2018-07-13 17:35:08 +00:00
|
|
|
` + c.Flags().Help()
|
|
|
|
|
|
|
|
return strings.TrimSpace(helpText)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *PluginListCommand) Flags() *FlagSets {
|
2022-09-09 20:03:07 +00:00
|
|
|
set := c.flagSet(FlagSetHTTP | FlagSetOutputFormat)
|
|
|
|
|
|
|
|
f := set.NewFlagSet("Command Options")
|
|
|
|
|
|
|
|
f.BoolVar(&BoolVar{
|
|
|
|
Name: "detailed",
|
|
|
|
Target: &c.flagDetailed,
|
|
|
|
Default: false,
|
|
|
|
Usage: "Print detailed plugin information such as plugin type, " +
|
|
|
|
"version, and deprecation status for each plugin. This option " +
|
|
|
|
"is only applicable to table-formatted output.",
|
|
|
|
})
|
|
|
|
|
|
|
|
return set
|
2018-07-13 17:35:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *PluginListCommand) AutocompleteArgs() complete.Predictor {
|
|
|
|
return complete.PredictNothing
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *PluginListCommand) AutocompleteFlags() complete.Flags {
|
|
|
|
return c.Flags().Completions()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *PluginListCommand) Run(args []string) int {
|
|
|
|
f := c.Flags()
|
|
|
|
|
|
|
|
if err := f.Parse(args); err != nil {
|
|
|
|
c.UI.Error(err.Error())
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
|
|
|
args = f.Args()
|
2018-11-07 01:21:24 +00:00
|
|
|
switch {
|
|
|
|
case len(args) > 1:
|
|
|
|
c.UI.Error(fmt.Sprintf("Too many arguments (expected 0 or 1, got %d)", len(args)))
|
2018-07-13 17:35:08 +00:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2023-02-06 14:41:56 +00:00
|
|
|
pluginType := api.PluginTypeUnknown
|
2018-11-07 01:21:24 +00:00
|
|
|
if len(args) > 0 {
|
|
|
|
pluginTypeStr := strings.TrimSpace(args[0])
|
|
|
|
if pluginTypeStr != "" {
|
|
|
|
var err error
|
2023-02-06 14:41:56 +00:00
|
|
|
pluginType, err = api.ParsePluginType(pluginTypeStr)
|
2018-11-07 01:21:24 +00:00
|
|
|
if err != nil {
|
|
|
|
c.UI.Error(fmt.Sprintf("Error parsing type: %s", err))
|
|
|
|
return 2
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-13 17:35:08 +00:00
|
|
|
client, err := c.Client()
|
|
|
|
if err != nil {
|
|
|
|
c.UI.Error(err.Error())
|
|
|
|
return 2
|
|
|
|
}
|
|
|
|
|
2018-11-07 01:21:24 +00:00
|
|
|
resp, err := client.Sys().ListPlugins(&api.ListPluginsInput{
|
|
|
|
Type: pluginType,
|
|
|
|
})
|
2018-07-13 17:35:08 +00:00
|
|
|
if err != nil {
|
|
|
|
c.UI.Error(fmt.Sprintf("Error listing available plugins: %s", err))
|
|
|
|
return 2
|
|
|
|
}
|
2018-12-12 20:36:28 +00:00
|
|
|
if resp == nil {
|
|
|
|
c.UI.Error("No response from server when listing plugins")
|
|
|
|
return 2
|
|
|
|
}
|
2018-07-13 17:35:08 +00:00
|
|
|
|
|
|
|
switch Format(c.UI) {
|
|
|
|
case "table":
|
2022-09-09 20:03:07 +00:00
|
|
|
if c.flagDetailed {
|
|
|
|
c.UI.Output(tableOutput(c.detailedResponse(resp), nil))
|
|
|
|
return 0
|
2018-11-07 01:21:24 +00:00
|
|
|
}
|
2022-09-29 17:22:33 +00:00
|
|
|
c.UI.Output(tableOutput(c.simpleResponse(resp, pluginType), nil))
|
2018-07-13 17:35:08 +00:00
|
|
|
return 0
|
|
|
|
default:
|
2018-11-07 01:21:24 +00:00
|
|
|
res := make(map[string]interface{})
|
|
|
|
for k, v := range resp.PluginsByType {
|
|
|
|
res[k.String()] = v
|
|
|
|
}
|
2022-09-29 17:22:33 +00:00
|
|
|
res["details"] = resp.Details
|
2018-11-07 01:21:24 +00:00
|
|
|
return OutputData(c.UI, res)
|
2018-07-13 17:35:08 +00:00
|
|
|
}
|
|
|
|
}
|
2022-09-09 20:03:07 +00:00
|
|
|
|
2023-02-06 14:41:56 +00:00
|
|
|
func (c *PluginListCommand) simpleResponse(plugins *api.ListPluginsResponse, pluginType api.PluginType) []string {
|
2022-09-29 17:22:33 +00:00
|
|
|
var out []string
|
|
|
|
switch pluginType {
|
2023-02-06 14:41:56 +00:00
|
|
|
case api.PluginTypeUnknown:
|
2022-09-29 17:22:33 +00:00
|
|
|
out = []string{"Name | Type | Version"}
|
|
|
|
for _, plugin := range plugins.Details {
|
|
|
|
out = append(out, fmt.Sprintf("%s | %s | %s", plugin.Name, plugin.Type, plugin.Version))
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
out = []string{"Name | Version"}
|
|
|
|
for _, plugin := range plugins.Details {
|
|
|
|
out = append(out, fmt.Sprintf("%s | %s", plugin.Name, plugin.Version))
|
2022-09-09 20:03:07 +00:00
|
|
|
}
|
|
|
|
}
|
2022-09-29 17:22:33 +00:00
|
|
|
|
|
|
|
return out
|
2022-09-09 20:03:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *PluginListCommand) detailedResponse(plugins *api.ListPluginsResponse) []string {
|
|
|
|
out := []string{"Name | Type | Version | Deprecation Status"}
|
|
|
|
for _, plugin := range plugins.Details {
|
|
|
|
out = append(out, fmt.Sprintf("%s | %s | %s | %s", plugin.Name, plugin.Type, plugin.Version, plugin.DeprecationStatus))
|
|
|
|
}
|
|
|
|
|
|
|
|
return out
|
|
|
|
}
|