132 lines
2.7 KiB
Go
132 lines
2.7 KiB
Go
package kvbuilder
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/hashicorp/vault/sdk/helper/jsonutil"
|
|
"github.com/mitchellh/mapstructure"
|
|
)
|
|
|
|
// Builder is a struct to build a key/value mapping based on a list
|
|
// of "k=v" pairs, where the value might come from stdin, a file, etc.
|
|
type Builder struct {
|
|
Stdin io.Reader
|
|
|
|
result map[string]interface{}
|
|
stdin bool
|
|
}
|
|
|
|
// Map returns the built map.
|
|
func (b *Builder) Map() map[string]interface{} {
|
|
return b.result
|
|
}
|
|
|
|
// Add adds to the mapping with the given args.
|
|
func (b *Builder) Add(args ...string) error {
|
|
for _, a := range args {
|
|
if err := b.add(a); err != nil {
|
|
return fmt.Errorf("invalid key/value pair %q: %w", a, err)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (b *Builder) add(raw string) error {
|
|
// Regardless of validity, make sure we make our result
|
|
if b.result == nil {
|
|
b.result = make(map[string]interface{})
|
|
}
|
|
|
|
// Empty strings are fine, just ignored
|
|
if raw == "" {
|
|
return nil
|
|
}
|
|
|
|
// Split into key/value
|
|
parts := strings.SplitN(raw, "=", 2)
|
|
|
|
// If the arg is exactly "-", then we need to read from stdin
|
|
// and merge the results into the resulting structure.
|
|
if len(parts) == 1 {
|
|
if raw == "-" {
|
|
if b.Stdin == nil {
|
|
return fmt.Errorf("stdin is not supported")
|
|
}
|
|
if b.stdin {
|
|
return fmt.Errorf("stdin already consumed")
|
|
}
|
|
|
|
b.stdin = true
|
|
return b.addReader(b.Stdin)
|
|
}
|
|
|
|
// If the arg begins with "@" then we need to read a file directly
|
|
if raw[0] == '@' {
|
|
f, err := os.Open(raw[1:])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer f.Close()
|
|
|
|
return b.addReader(f)
|
|
}
|
|
}
|
|
|
|
if len(parts) != 2 {
|
|
return fmt.Errorf("format must be key=value")
|
|
}
|
|
key, value := parts[0], parts[1]
|
|
|
|
if len(value) > 0 {
|
|
if value[0] == '@' {
|
|
contents, err := ioutil.ReadFile(value[1:])
|
|
if err != nil {
|
|
return fmt.Errorf("error reading file: %w", err)
|
|
}
|
|
|
|
value = string(contents)
|
|
} else if value[0] == '\\' && value[1] == '@' {
|
|
value = value[1:]
|
|
} else if value == "-" {
|
|
if b.Stdin == nil {
|
|
return fmt.Errorf("stdin is not supported")
|
|
}
|
|
if b.stdin {
|
|
return fmt.Errorf("stdin already consumed")
|
|
}
|
|
b.stdin = true
|
|
|
|
var buf bytes.Buffer
|
|
if _, err := io.Copy(&buf, b.Stdin); err != nil {
|
|
return err
|
|
}
|
|
|
|
value = buf.String()
|
|
}
|
|
}
|
|
|
|
// Repeated keys will be converted into a slice
|
|
if existingValue, ok := b.result[key]; ok {
|
|
var sliceValue []interface{}
|
|
if err := mapstructure.WeakDecode(existingValue, &sliceValue); err != nil {
|
|
return err
|
|
}
|
|
sliceValue = append(sliceValue, value)
|
|
b.result[key] = sliceValue
|
|
return nil
|
|
}
|
|
|
|
b.result[key] = value
|
|
return nil
|
|
}
|
|
|
|
func (b *Builder) addReader(r io.Reader) error {
|
|
return jsonutil.DecodeJSONFromReader(r, &b.result)
|
|
}
|