Files
velero/pkg/cmd/util/flag/map.go
Wenkai Yin(尹文开) 44199db79d Enhance the map flag to support parsing input value contains entry delimiters
Enhance the map flag to support parsing input value contains entry delimiters

Signed-off-by: Wenkai Yin(尹文开) <yinw@vmware.com>
2022-05-17 13:56:41 +08:00

106 lines
2.8 KiB
Go

/*
Copyright 2017 the Velero contributors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package flag
import (
"encoding/csv"
"fmt"
"strings"
"github.com/pkg/errors"
)
// Map is a Cobra-compatible wrapper for defining a flag containing
// map data (i.e. a collection of key-value pairs).
type Map struct {
data map[string]string
entryDelimiter rune
keyValueDelimiter rune
}
// NewMap returns a Map using the default delimiters ('=' between keys and
// values, and ',' between map entries, e.g. k1=v1,k2=v2)
func NewMap() Map {
m := Map{
data: make(map[string]string),
}
return m.WithEntryDelimiter(',').WithKeyValueDelimiter('=')
}
// WithEntryDelimiter sets the delimiter to be used between map
// entries.
//
// For example, in "k1=v1&k2=v2", the entry delimiter is '&'
func (m Map) WithEntryDelimiter(delimiter rune) Map {
m.entryDelimiter = delimiter
return m
}
// WithKeyValueDelimiter sets the delimiter to be used between
// keys and values.
//
// For example, in "k1=v1&k2=v2", the key-value delimiter is '='
func (m Map) WithKeyValueDelimiter(delimiter rune) Map {
m.keyValueDelimiter = delimiter
return m
}
// String returns a string representation of the Map flag.
func (m *Map) String() string {
var a []string
for k, v := range m.data {
a = append(a, fmt.Sprintf("%s%s%s", k, string(m.keyValueDelimiter), v))
}
return strings.Join(a, string(m.entryDelimiter))
}
// Set parses the provided string according to the delimiters and
// assigns the result to the Map receiver. It returns an error if
// the string is not parseable.
func (m *Map) Set(s string) error {
// use csv.Reader to support parsing input string contains entry delimiters.
// e.g. `"k1=a=b,c=d",k2=v2` will be parsed into two parts: `k1=a=b,c=d` and `k2=v2`
r := csv.NewReader(strings.NewReader(s))
r.Comma = m.entryDelimiter
parts, err := r.Read()
if err != nil {
return errors.Wrapf(err, "error parsing %q", s)
}
for _, part := range parts {
kvs := strings.SplitN(part, string(m.keyValueDelimiter), 2)
if len(kvs) != 2 {
return errors.Errorf("error parsing %q", part)
}
m.data[kvs[0]] = kvs[1]
}
return nil
}
// Type returns a string representation of the
// Map type.
func (m *Map) Type() string {
return "mapStringString"
}
// Data returns the underlying golang map storing
// the flag data.
func (m *Map) Data() map[string]string {
return m.data
}