// Copyright (c) HashiCorp, Inc. // SPDX-License-Identifier: MPL-2.0 package command import ( "context" "fmt" "net/http" "strings" "github.com/hashicorp/vault/api" "github.com/mitchellh/cli" "github.com/posener/complete" ) var ( _ cli.Command = (*NamespacePatchCommand)(nil) _ cli.CommandAutocomplete = (*NamespacePatchCommand)(nil) ) type NamespacePatchCommand struct { *BaseCommand flagCustomMetadata map[string]string flagRemoveCustomMetadata []string } func (c *NamespacePatchCommand) Synopsis() string { return "Patch an existing namespace" } func (c *NamespacePatchCommand) Help() string { helpText := ` Usage: vault namespace patch [options] PATH Patch an existing namespace. The namespace patched will be relative to the namespace provided in either the VAULT_NAMESPACE environment variable or -namespace CLI flag. Patch an existing child namespace by adding and removing custom-metadata (e.g. ns1/): $ vault namespace patch -custom-metadata=foo=abc -remove-custom-metadata=bar ns1 Patch an existing child namespace from a parent namespace (e.g. ns1/ns2/): $ vault namespace patch -namespace=ns1 -custom-metadata=foo=abc ns2 ` + c.Flags().Help() return strings.TrimSpace(helpText) } func (c *NamespacePatchCommand) Flags() *FlagSets { set := c.flagSet(FlagSetHTTP | FlagSetOutputField | FlagSetOutputFormat) f := set.NewFlagSet("Command Options") f.StringMapVar(&StringMapVar{ Name: "custom-metadata", Target: &c.flagCustomMetadata, Default: map[string]string{}, Usage: "Specifies arbitrary key=value metadata meant to describe a namespace." + "This can be specified multiple times to add multiple pieces of metadata.", }) f.StringSliceVar(&StringSliceVar{ Name: "remove-custom-metadata", Target: &c.flagRemoveCustomMetadata, Default: []string{}, Usage: "Key to remove from custom metadata. To specify multiple values, specify this flag multiple times.", }) return set } func (c *NamespacePatchCommand) AutocompleteArgs() complete.Predictor { return complete.PredictNothing } func (c *NamespacePatchCommand) AutocompleteFlags() complete.Flags { return c.Flags().Completions() } func (c *NamespacePatchCommand) Run(args []string) int { f := c.Flags() if err := f.Parse(args); err != nil { c.UI.Error(err.Error()) return 1 } args = f.Args() switch { case len(args) < 1: c.UI.Error(fmt.Sprintf("Not enough arguments (expected 1, got %d)", len(args))) return 1 case len(args) > 1: c.UI.Error(fmt.Sprintf("Too many arguments (expected 1, got %d)", len(args))) return 1 } namespacePath := strings.TrimSpace(args[0]) client, err := c.Client() if err != nil { c.UI.Error(err.Error()) return 2 } data := make(map[string]interface{}) customMetadata := make(map[string]interface{}) for key, value := range c.flagCustomMetadata { customMetadata[key] = value } for _, key := range c.flagRemoveCustomMetadata { // A null in a JSON merge patch payload will remove the associated key customMetadata[key] = nil } data["custom_metadata"] = customMetadata secret, err := client.Logical().JSONMergePatch(context.Background(), "sys/namespaces/"+namespacePath, data) if err != nil { if re, ok := err.(*api.ResponseError); ok && re.StatusCode == http.StatusNotFound { c.UI.Error("Namespace not found") return 2 } c.UI.Error(fmt.Sprintf("Error patching namespace: %s", err)) return 2 } // Handle single field output if c.flagField != "" { return PrintRawField(c.UI, secret, c.flagField) } return OutputSecret(c.UI, secret) }