mirror of
https://github.com/kemko/nomad.git
synced 2026-01-09 11:55:42 +03:00
* command/csi: csi, csi_plugin, csi_volume * helper/funcs: move ExtraKeys from parse_config to UnusedKeys * command/agent/config_parse: use helper.UnusedKeys * api/csi: annotate CSIVolumes with hcl fields * command/csi_plugin: add Synopsis * command/csi_volume_register: use hcl.Decode style parsing * command/csi_volume_list * command/csi_volume_status: list format, cleanup * command/csi_plugin_list * command/csi_plugin_status * command/csi_volume_deregister * command/csi_volume: add Synopsis * api/contexts/contexts: add csi search contexts to the constants * command/commands: register csi commands * api/csi: fix struct tag for linter * command/csi_plugin_list: unused struct vars * command/csi_plugin_status: unused struct vars * command/csi_volume_list: unused struct vars * api/csi: add allocs to CSIPlugin * command/csi_plugin_status: format the allocs * api/allocations: copy Allocation.Stub in from structs * nomad/client_rpc: add some error context with Errorf * api/csi: collapse read & write alloc maps to a stub list * command/csi_volume_status: cleanup allocation display * command/csi_volume_list: use Schedulable instead of Healthy * command/csi_volume_status: use Schedulable instead of Healthy * command/csi_volume_list: sprintf string * command/csi: delete csi.go, csi_plugin.go * command/plugin: refactor csi components to sub-command plugin status * command/plugin: remove csi * command/plugin_status: remove csi * command/volume: remove csi * command/volume_status: split out csi specific * helper/funcs: add RemoveEqualFold * command/agent/config_parse: use helper.RemoveEqualFold * api/csi: do ,unusedKeys right * command/volume: refactor csi components to `nomad volume` * command/volume_register: split out csi specific * command/commands: use the new top level commands * command/volume_deregister: hardwired type csi for now * command/volume_status: csiFormatVolumes rescued from volume_list * command/plugin_status: avoid a panic on no args * command/volume_status: avoid a panic on no args * command/plugin_status: predictVolumeType * command/volume_status: predictVolumeType * nomad/csi_endpoint_test: move CreateTestPlugin to testing * command/plugin_status_test: use CreateTestCSIPlugin * nomad/structs/structs: add CSIPlugins and CSIVolumes search consts * nomad/state/state_store: add CSIPlugins and CSIVolumesByIDPrefix * nomad/search_endpoint: add CSIPlugins and CSIVolumes * command/plugin_status: move the header to the csi specific * command/volume_status: move the header to the csi specific * nomad/state/state_store: CSIPluginByID prefix * command/status: rename the search context to just Plugins/Volumes * command/plugin,volume_status: test return ids now * command/status: rename the search context to just Plugins/Volumes * command/plugin_status: support -json and -t * command/volume_status: support -json and -t * command/plugin_status_csi: comments * command/*_status: clean up text * api/csi: fix stale comments * command/volume: make deregister sound less fearsome * command/plugin_status: set the id length * command/plugin_status_csi: more compact plugin health * command/volume: better error message, comment
131 lines
2.9 KiB
Go
131 lines
2.9 KiB
Go
package command
|
|
|
|
import (
|
|
"fmt"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/hashicorp/hcl"
|
|
"github.com/hashicorp/hcl/hcl/ast"
|
|
"github.com/posener/complete"
|
|
)
|
|
|
|
type VolumeRegisterCommand struct {
|
|
Meta
|
|
}
|
|
|
|
func (c *VolumeRegisterCommand) Help() string {
|
|
helpText := `
|
|
Usage: nomad volume register [options] <input>
|
|
|
|
Creates or updates a volume in Nomad. The volume must exist on the remote
|
|
storage provider before it can be used by a task.
|
|
|
|
If the supplied path is "-" the volume file is read from stdin. Otherwise, it
|
|
is read from the file at the supplied path.
|
|
|
|
General Options:
|
|
|
|
` + generalOptionsUsage()
|
|
|
|
return strings.TrimSpace(helpText)
|
|
}
|
|
|
|
func (c *VolumeRegisterCommand) AutocompleteFlags() complete.Flags {
|
|
return c.Meta.AutocompleteFlags(FlagSetClient)
|
|
}
|
|
|
|
func (c *VolumeRegisterCommand) AutocompleteArgs() complete.Predictor {
|
|
return complete.PredictFiles("*")
|
|
}
|
|
|
|
func (c *VolumeRegisterCommand) Synopsis() string {
|
|
return "Create or update a volume"
|
|
}
|
|
|
|
func (c *VolumeRegisterCommand) Name() string { return "volume register" }
|
|
|
|
func (c *VolumeRegisterCommand) Run(args []string) int {
|
|
flags := c.Meta.FlagSet(c.Name(), FlagSetClient)
|
|
flags.Usage = func() { c.Ui.Output(c.Help()) }
|
|
|
|
if err := flags.Parse(args); err != nil {
|
|
c.Ui.Error(fmt.Sprintf("Error parsing arguments %s", err))
|
|
return 1
|
|
}
|
|
|
|
// Check that we get exactly one argument
|
|
args = flags.Args()
|
|
if l := len(args); l != 1 {
|
|
c.Ui.Error("This command takes one argument: <input>")
|
|
c.Ui.Error(commandErrorText(c))
|
|
return 1
|
|
}
|
|
|
|
// Read the file contents
|
|
file := args[0]
|
|
var rawVolume []byte
|
|
var err error
|
|
if file == "-" {
|
|
rawVolume, err = ioutil.ReadAll(os.Stdin)
|
|
if err != nil {
|
|
c.Ui.Error(fmt.Sprintf("Failed to read stdin: %v", err))
|
|
return 1
|
|
}
|
|
} else {
|
|
rawVolume, err = ioutil.ReadFile(file)
|
|
if err != nil {
|
|
c.Ui.Error(fmt.Sprintf("Failed to read file: %v", err))
|
|
return 1
|
|
}
|
|
}
|
|
|
|
ast, volType, err := parseVolumeType(string(rawVolume))
|
|
if err != nil {
|
|
c.Ui.Error(fmt.Sprintf("Error parsing the volume type: %s", err))
|
|
return 1
|
|
}
|
|
volType = strings.ToLower(volType)
|
|
|
|
// Get the HTTP client
|
|
client, err := c.Meta.Client()
|
|
if err != nil {
|
|
c.Ui.Error(fmt.Sprintf("Error initializing client: %s", err))
|
|
return 1
|
|
}
|
|
|
|
switch volType {
|
|
case "csi":
|
|
code := c.csiRegister(client, ast)
|
|
if code != 0 {
|
|
return code
|
|
}
|
|
default:
|
|
c.Ui.Error(fmt.Sprintf("Error unknown volume type: %s", volType))
|
|
return 1
|
|
}
|
|
|
|
return 0
|
|
}
|
|
|
|
// parseVolume is used to parse the quota specification from HCL
|
|
func parseVolumeType(input string) (*ast.File, string, error) {
|
|
// Parse the AST first
|
|
ast, err := hcl.Parse(input)
|
|
if err != nil {
|
|
return nil, "", fmt.Errorf("parse error: %v", err)
|
|
}
|
|
|
|
// Decode the type, so we can dispatch on it
|
|
dispatch := &struct {
|
|
T string `hcl:"type"`
|
|
}{}
|
|
err = hcl.DecodeObject(dispatch, ast)
|
|
if err != nil {
|
|
return nil, "", fmt.Errorf("dispatch error: %v", err)
|
|
}
|
|
|
|
return ast, dispatch.T, nil
|
|
}
|