package command import ( "fmt" "strings" "github.com/hashicorp/vault/api" "github.com/hashicorp/vault/meta" "github.com/posener/complete" ) // MountCommand is a Command that mounts a new mount. type MountCommand struct { meta.Meta } func (c *MountCommand) Run(args []string) int { var description, path, defaultLeaseTTL, maxLeaseTTL, pluginName string var local, forceNoCache bool flags := c.Meta.FlagSet("mount", meta.FlagSetDefault) flags.StringVar(&description, "description", "", "") flags.StringVar(&path, "path", "", "") flags.StringVar(&defaultLeaseTTL, "default-lease-ttl", "", "") flags.StringVar(&maxLeaseTTL, "max-lease-ttl", "", "") flags.StringVar(&pluginName, "plugin-name", "", "") flags.BoolVar(&forceNoCache, "force-no-cache", false, "") flags.BoolVar(&local, "local", false, "") flags.Usage = func() { c.Ui.Error(c.Help()) } if err := flags.Parse(args); err != nil { return 1 } args = flags.Args() if len(args) != 1 { flags.Usage() c.Ui.Error(fmt.Sprintf( "\nmount expects one argument: the type to mount.")) return 1 } mountType := args[0] // If no path is specified, we default the path to the backend type // or use the plugin name if it's a plugin backend if path == "" { if mountType == "plugin" { path = pluginName } else { path = mountType } } client, err := c.Client() if err != nil { c.Ui.Error(fmt.Sprintf( "Error initializing client: %s", err)) return 2 } mountInfo := &api.MountInput{ Type: mountType, Description: description, Config: api.MountConfigInput{ DefaultLeaseTTL: defaultLeaseTTL, MaxLeaseTTL: maxLeaseTTL, ForceNoCache: forceNoCache, PluginName: pluginName, }, Local: local, } if err := client.Sys().Mount(path, mountInfo); err != nil { c.Ui.Error(fmt.Sprintf( "Mount error: %s", err)) return 2 } mountTypeOutput := fmt.Sprintf("'%s'", mountType) if mountType == "plugin" { mountTypeOutput = fmt.Sprintf("plugin '%s'", pluginName) } c.Ui.Output(fmt.Sprintf( "Successfully mounted %s at '%s'!", mountTypeOutput, path)) return 0 } func (c *MountCommand) Synopsis() string { return "Mount a logical backend" } func (c *MountCommand) Help() string { helpText := ` Usage: vault mount [options] type Mount a logical backend. This command mounts a logical backend for storing and/or generating secrets. General Options: ` + meta.GeneralOptionsUsage() + ` Mount Options: -description= Human-friendly description of the purpose for the mount. This shows up in the mounts command. -path= Mount point for the logical backend. This defaults to the type of the mount. -default-lease-ttl= Default lease time-to-live for this backend. If not specified, uses the global default, or the previously set value. Set to '0' to explicitly set it to use the global default. -max-lease-ttl= Max lease time-to-live for this backend. If not specified, uses the global default, or the previously set value. Set to '0' to explicitly set it to use the global default. -force-no-cache Forces the backend to disable caching. If not specified, uses the global default. This does not affect caching of the underlying encrypted data storage. -plugin-name Name of the plugin to mount based from the name in the plugin catalog. -local Mark the mount as a local mount. Local mounts are not replicated nor (if a secondary) removed by replication. ` return strings.TrimSpace(helpText) } func (c *MountCommand) AutocompleteArgs() complete.Predictor { // This list does not contain deprecated backends return complete.PredictSet( "aws", "consul", "pki", "transit", "ssh", "rabbitmq", "database", "totp", "plugin", ) } func (c *MountCommand) AutocompleteFlags() complete.Flags { return complete.Flags{ "-description": complete.PredictNothing, "-path": complete.PredictNothing, "-default-lease-ttl": complete.PredictNothing, "-max-lease-ttl": complete.PredictNothing, "-force-no-cache": complete.PredictNothing, "-plugin-name": complete.PredictNothing, "-local": complete.PredictNothing, } }