2023-09-08 16:20:14 +00:00
|
|
|
// Copyright (c) HashiCorp, Inc.
|
2023-09-14 22:19:04 +00:00
|
|
|
// SPDX-License-Identifier: BUSL-1.1
|
2023-09-08 16:20:14 +00:00
|
|
|
|
|
|
|
package apply
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
2023-10-13 13:24:16 +00:00
|
|
|
"io"
|
2023-09-08 16:20:14 +00:00
|
|
|
|
|
|
|
"github.com/mitchellh/cli"
|
|
|
|
|
2023-09-11 14:06:00 +00:00
|
|
|
"github.com/hashicorp/consul/command/resource"
|
2023-09-22 20:32:08 +00:00
|
|
|
"github.com/hashicorp/consul/command/resource/client"
|
2023-09-08 16:20:14 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func New(ui cli.Ui) *cmd {
|
|
|
|
c := &cmd{UI: ui}
|
|
|
|
c.init()
|
|
|
|
return c
|
|
|
|
}
|
|
|
|
|
|
|
|
type cmd struct {
|
2024-01-30 18:12:09 +00:00
|
|
|
UI cli.Ui
|
|
|
|
flags *flag.FlagSet
|
|
|
|
grpcFlags *client.GRPCFlags
|
|
|
|
help string
|
2023-09-08 16:20:14 +00:00
|
|
|
|
|
|
|
filePath string
|
2023-10-13 13:24:16 +00:00
|
|
|
|
|
|
|
testStdin io.Reader
|
2023-09-08 16:20:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *cmd) init() {
|
|
|
|
c.flags = flag.NewFlagSet("", flag.ContinueOnError)
|
|
|
|
c.flags.StringVar(&c.filePath, "f", "",
|
|
|
|
"File path with resource definition")
|
|
|
|
|
2024-01-30 18:12:09 +00:00
|
|
|
c.grpcFlags = &client.GRPCFlags{}
|
|
|
|
client.MergeFlags(c.flags, c.grpcFlags.ClientFlags())
|
|
|
|
c.help = client.Usage(help, c.flags)
|
2023-09-08 16:20:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *cmd) Run(args []string) int {
|
|
|
|
if err := c.flags.Parse(args); err != nil {
|
2023-09-11 14:06:00 +00:00
|
|
|
if !errors.Is(err, flag.ErrHelp) {
|
|
|
|
c.UI.Error(fmt.Sprintf("Failed to parse args: %v", err))
|
|
|
|
return 1
|
2023-09-08 16:20:14 +00:00
|
|
|
}
|
2024-01-30 18:12:09 +00:00
|
|
|
c.UI.Error(fmt.Sprintf("Failed to run apply command: %v", err))
|
|
|
|
return 1
|
2023-09-08 16:20:14 +00:00
|
|
|
}
|
|
|
|
|
2024-01-30 18:12:09 +00:00
|
|
|
// parse resource
|
2023-10-13 13:24:16 +00:00
|
|
|
input := c.filePath
|
2024-01-30 18:12:09 +00:00
|
|
|
if input == "" {
|
|
|
|
c.UI.Error("Required '-f' flag was not provided to specify where to load the resource content from")
|
|
|
|
return 1
|
2023-10-13 13:24:16 +00:00
|
|
|
}
|
2024-01-30 18:12:09 +00:00
|
|
|
parsedResource, err := resource.ParseResourceInput(input, c.testStdin)
|
|
|
|
if err != nil {
|
|
|
|
c.UI.Error(fmt.Sprintf("Failed to decode resource from input file: %v", err))
|
2023-09-08 16:20:14 +00:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
if parsedResource == nil {
|
|
|
|
c.UI.Error("Unable to parse the file argument")
|
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2024-01-30 18:12:09 +00:00
|
|
|
// initialize client
|
|
|
|
config, err := client.LoadGRPCConfig(nil)
|
2023-09-08 16:20:14 +00:00
|
|
|
if err != nil {
|
2024-01-30 18:12:09 +00:00
|
|
|
c.UI.Error(fmt.Sprintf("Error loading config: %s", err))
|
2023-09-08 16:20:14 +00:00
|
|
|
return 1
|
|
|
|
}
|
2024-01-30 18:12:09 +00:00
|
|
|
c.grpcFlags.MergeFlagsIntoGRPCConfig(config)
|
|
|
|
resourceClient, err := client.NewGRPCClient(config)
|
2023-09-08 16:20:14 +00:00
|
|
|
if err != nil {
|
2024-01-30 18:12:09 +00:00
|
|
|
c.UI.Error(fmt.Sprintf("Error connect to Consul agent: %s", err))
|
2023-09-08 16:20:14 +00:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2024-01-30 18:12:09 +00:00
|
|
|
// write resource
|
|
|
|
res := resource.ResourceGRPC{C: resourceClient}
|
|
|
|
entry, err := res.Apply(parsedResource)
|
2023-09-08 16:20:14 +00:00
|
|
|
if err != nil {
|
2024-01-30 18:12:09 +00:00
|
|
|
c.UI.Error(fmt.Sprintf("Error writing resource %s/%s: %v", parsedResource.Id.Type, parsedResource.Id.GetName(), err))
|
2023-09-08 16:20:14 +00:00
|
|
|
return 1
|
|
|
|
}
|
|
|
|
|
2024-01-30 18:12:09 +00:00
|
|
|
// display response
|
|
|
|
b, err := json.MarshalIndent(entry, "", resource.JSON_INDENT)
|
2023-09-08 16:20:14 +00:00
|
|
|
if err != nil {
|
|
|
|
c.UI.Error("Failed to encode output data")
|
|
|
|
return 1
|
|
|
|
}
|
2024-01-30 18:12:09 +00:00
|
|
|
c.UI.Info(fmt.Sprintf("%s.%s.%s '%s' created.", parsedResource.Id.Type.Group, parsedResource.Id.Type.GroupVersion, parsedResource.Id.Type.Kind, parsedResource.Id.GetName()))
|
2023-09-08 16:20:14 +00:00
|
|
|
c.UI.Info(string(b))
|
2024-01-30 18:12:09 +00:00
|
|
|
|
2023-09-08 16:20:14 +00:00
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *cmd) Synopsis() string {
|
|
|
|
return synopsis
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *cmd) Help() string {
|
2024-01-30 18:12:09 +00:00
|
|
|
return client.Usage(c.help, nil)
|
2023-09-08 16:20:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const synopsis = "Writes/updates resource information"
|
2023-09-14 22:19:04 +00:00
|
|
|
|
2023-09-08 16:20:14 +00:00
|
|
|
const help = `
|
2023-10-13 13:24:16 +00:00
|
|
|
Usage: consul resource apply [options] <resource>
|
|
|
|
|
|
|
|
Write and/or update a resource by providing the definition. The configuration
|
|
|
|
argument is either a file path or '-' to indicate that the resource
|
|
|
|
should be read from stdin. The data should be either in HCL or
|
|
|
|
JSON form.
|
|
|
|
|
|
|
|
Example (with flag):
|
2023-09-08 16:20:14 +00:00
|
|
|
|
2023-10-13 13:24:16 +00:00
|
|
|
$ consul resource apply -f=demo.hcl
|
2023-09-08 16:20:14 +00:00
|
|
|
|
2023-10-13 13:24:16 +00:00
|
|
|
Example (from stdin):
|
2023-09-08 16:20:14 +00:00
|
|
|
|
2024-01-30 18:12:09 +00:00
|
|
|
$ consul resource apply -f - < demo.hcl
|
2023-10-13 13:24:16 +00:00
|
|
|
|
|
|
|
Sample demo.hcl:
|
|
|
|
|
|
|
|
ID {
|
|
|
|
Type = gvk("group.version.kind")
|
|
|
|
Name = "resource-name"
|
|
|
|
Tenancy {
|
2024-01-30 18:12:09 +00:00
|
|
|
Namespace = "default"
|
|
|
|
Partition = "default"
|
2023-10-13 13:24:16 +00:00
|
|
|
}
|
2023-09-08 16:20:14 +00:00
|
|
|
}
|
|
|
|
|
2023-10-13 13:24:16 +00:00
|
|
|
Data {
|
|
|
|
Name = "demo"
|
|
|
|
}
|
2023-09-08 16:20:14 +00:00
|
|
|
|
2023-10-13 13:24:16 +00:00
|
|
|
Metadata = {
|
|
|
|
"foo" = "bar"
|
|
|
|
}
|
2023-09-08 16:20:14 +00:00
|
|
|
`
|