mirror of https://github.com/k3s-io/k3s
619 lines
19 KiB
Go
619 lines
19 KiB
Go
/*
|
|
Copyright 2014 The Kubernetes Authors.
|
|
|
|
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 util
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/url"
|
|
"os"
|
|
"strings"
|
|
"time"
|
|
|
|
"k8s.io/kubernetes/pkg/api/errors"
|
|
"k8s.io/kubernetes/pkg/api/meta"
|
|
"k8s.io/kubernetes/pkg/api/unversioned"
|
|
"k8s.io/kubernetes/pkg/apimachinery/registered"
|
|
"k8s.io/kubernetes/pkg/apis/extensions"
|
|
"k8s.io/kubernetes/pkg/client/typed/discovery"
|
|
"k8s.io/kubernetes/pkg/client/unversioned/clientcmd"
|
|
"k8s.io/kubernetes/pkg/kubectl"
|
|
"k8s.io/kubernetes/pkg/kubectl/resource"
|
|
"k8s.io/kubernetes/pkg/runtime"
|
|
utilerrors "k8s.io/kubernetes/pkg/util/errors"
|
|
"k8s.io/kubernetes/pkg/util/strategicpatch"
|
|
|
|
"github.com/evanphx/json-patch"
|
|
"github.com/golang/glog"
|
|
"github.com/spf13/cobra"
|
|
"github.com/spf13/pflag"
|
|
)
|
|
|
|
const (
|
|
ApplyAnnotationsFlag = "save-config"
|
|
)
|
|
|
|
type debugError interface {
|
|
DebugError() (msg string, args []interface{})
|
|
}
|
|
|
|
// AddSourceToErr adds handleResourcePrefix and source string to error message.
|
|
// verb is the string like "creating", "deleting" etc.
|
|
// souce is the filename or URL to the template file(*.json or *.yaml), or stdin to use to handle the resource.
|
|
func AddSourceToErr(verb string, source string, err error) error {
|
|
if source != "" {
|
|
if statusError, ok := err.(errors.APIStatus); ok {
|
|
status := statusError.Status()
|
|
status.Message = fmt.Sprintf("error when %s %q: %v", verb, source, status.Message)
|
|
return &errors.StatusError{ErrStatus: status}
|
|
}
|
|
return fmt.Errorf("error when %s %q: %v", verb, source, err)
|
|
}
|
|
return err
|
|
}
|
|
|
|
var fatalErrHandler = fatal
|
|
|
|
// BehaviorOnFatal allows you to override the default behavior when a fatal
|
|
// error occurs, which is call os.Exit(1). You can pass 'panic' as a function
|
|
// here if you prefer the panic() over os.Exit(1).
|
|
func BehaviorOnFatal(f func(string)) {
|
|
fatalErrHandler = f
|
|
}
|
|
|
|
// DefaultBehaviorOnFatal allows you to undo any previous override. Useful in
|
|
// tests.
|
|
func DefaultBehaviorOnFatal() {
|
|
fatalErrHandler = fatal
|
|
}
|
|
|
|
// fatal prints the message and then exits. If V(2) or greater, glog.Fatal
|
|
// is invoked for extended information.
|
|
func fatal(msg string) {
|
|
// add newline if needed
|
|
if !strings.HasSuffix(msg, "\n") {
|
|
msg += "\n"
|
|
}
|
|
|
|
if glog.V(2) {
|
|
glog.FatalDepth(2, msg)
|
|
}
|
|
fmt.Fprint(os.Stderr, msg)
|
|
os.Exit(1)
|
|
}
|
|
|
|
// CheckErr prints a user friendly error to STDERR and exits with a non-zero
|
|
// exit code. Unrecognized errors will be printed with an "error: " prefix.
|
|
//
|
|
// This method is generic to the command in use and may be used by non-Kubectl
|
|
// commands.
|
|
func CheckErr(err error) {
|
|
checkErr("", err, fatalErrHandler)
|
|
}
|
|
|
|
// checkErrWithPrefix works like CheckErr, but adds a caller-defined prefix to non-nil errors
|
|
func checkErrWithPrefix(prefix string, err error) {
|
|
checkErr(prefix, err, fatalErrHandler)
|
|
}
|
|
|
|
func checkErr(pref string, err error, handleErr func(string)) {
|
|
if err == nil {
|
|
return
|
|
}
|
|
|
|
if errors.IsInvalid(err) {
|
|
details := err.(*errors.StatusError).Status().Details
|
|
prefix := fmt.Sprintf("%sThe %s %q is invalid.\n", pref, details.Kind, details.Name)
|
|
errs := statusCausesToAggrError(details.Causes)
|
|
handleErr(MultilineError(prefix, errs))
|
|
}
|
|
|
|
if meta.IsNoResourceMatchError(err) {
|
|
noMatch := err.(*meta.NoResourceMatchError)
|
|
|
|
switch {
|
|
case len(noMatch.PartialResource.Group) > 0 && len(noMatch.PartialResource.Version) > 0:
|
|
handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in group %q and version %q", pref, noMatch.PartialResource.Resource, noMatch.PartialResource.Group, noMatch.PartialResource.Version))
|
|
case len(noMatch.PartialResource.Group) > 0:
|
|
handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in group %q", pref, noMatch.PartialResource.Resource, noMatch.PartialResource.Group))
|
|
case len(noMatch.PartialResource.Version) > 0:
|
|
handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q in version %q", pref, noMatch.PartialResource.Resource, noMatch.PartialResource.Version))
|
|
default:
|
|
handleErr(fmt.Sprintf("%sthe server doesn't have a resource type %q", pref, noMatch.PartialResource.Resource))
|
|
}
|
|
return
|
|
}
|
|
|
|
// handle multiline errors
|
|
if clientcmd.IsConfigurationInvalid(err) {
|
|
handleErr(MultilineError(fmt.Sprintf("%sError in configuration: ", pref), err))
|
|
}
|
|
if agg, ok := err.(utilerrors.Aggregate); ok && len(agg.Errors()) > 0 {
|
|
handleErr(MultipleErrors(pref, agg.Errors()))
|
|
}
|
|
|
|
msg, ok := StandardErrorMessage(err)
|
|
if !ok {
|
|
msg = err.Error()
|
|
if !strings.HasPrefix(msg, "error: ") {
|
|
msg = fmt.Sprintf("error: %s", msg)
|
|
}
|
|
}
|
|
handleErr(fmt.Sprintf("%s%s", pref, msg))
|
|
}
|
|
|
|
func statusCausesToAggrError(scs []unversioned.StatusCause) utilerrors.Aggregate {
|
|
errs := make([]error, len(scs))
|
|
for i, sc := range scs {
|
|
errs[i] = fmt.Errorf("%s: %s", sc.Field, sc.Message)
|
|
}
|
|
return utilerrors.NewAggregate(errs)
|
|
}
|
|
|
|
// StandardErrorMessage translates common errors into a human readable message, or returns
|
|
// false if the error is not one of the recognized types. It may also log extended
|
|
// information to glog.
|
|
//
|
|
// This method is generic to the command in use and may be used by non-Kubectl
|
|
// commands.
|
|
func StandardErrorMessage(err error) (string, bool) {
|
|
if debugErr, ok := err.(debugError); ok {
|
|
glog.V(4).Infof(debugErr.DebugError())
|
|
}
|
|
status, isStatus := err.(errors.APIStatus)
|
|
switch {
|
|
case isStatus:
|
|
switch s := status.Status(); {
|
|
case s.Reason == "Unauthorized":
|
|
return fmt.Sprintf("error: You must be logged in to the server (%s)", s.Message), true
|
|
default:
|
|
return fmt.Sprintf("Error from server: %s", err.Error()), true
|
|
}
|
|
case errors.IsUnexpectedObjectError(err):
|
|
return fmt.Sprintf("Server returned an unexpected response: %s", err.Error()), true
|
|
}
|
|
switch t := err.(type) {
|
|
case *url.Error:
|
|
glog.V(4).Infof("Connection error: %s %s: %v", t.Op, t.URL, t.Err)
|
|
switch {
|
|
case strings.Contains(t.Err.Error(), "connection refused"):
|
|
host := t.URL
|
|
if server, err := url.Parse(t.URL); err == nil {
|
|
host = server.Host
|
|
}
|
|
return fmt.Sprintf("The connection to the server %s was refused - did you specify the right host or port?", host), true
|
|
}
|
|
return fmt.Sprintf("Unable to connect to the server: %v", t.Err), true
|
|
}
|
|
return "", false
|
|
}
|
|
|
|
// MultilineError returns a string representing an error that splits sub errors into their own
|
|
// lines. The returned string will end with a newline.
|
|
func MultilineError(prefix string, err error) string {
|
|
if agg, ok := err.(utilerrors.Aggregate); ok {
|
|
errs := utilerrors.Flatten(agg).Errors()
|
|
buf := &bytes.Buffer{}
|
|
switch len(errs) {
|
|
case 0:
|
|
return fmt.Sprintf("%s%v\n", prefix, err)
|
|
case 1:
|
|
return fmt.Sprintf("%s%v\n", prefix, messageForError(errs[0]))
|
|
default:
|
|
fmt.Fprintln(buf, prefix)
|
|
for _, err := range errs {
|
|
fmt.Fprintf(buf, "* %v\n", messageForError(err))
|
|
}
|
|
return buf.String()
|
|
}
|
|
}
|
|
return fmt.Sprintf("%s%s\n", prefix, err)
|
|
}
|
|
|
|
// MultipleErrors returns a newline delimited string containing
|
|
// the prefix and referenced errors in standard form.
|
|
func MultipleErrors(prefix string, errs []error) string {
|
|
buf := &bytes.Buffer{}
|
|
for _, err := range errs {
|
|
fmt.Fprintf(buf, "%s%v\n", prefix, messageForError(err))
|
|
}
|
|
return buf.String()
|
|
}
|
|
|
|
// messageForError returns the string representing the error.
|
|
func messageForError(err error) string {
|
|
msg, ok := StandardErrorMessage(err)
|
|
if !ok {
|
|
msg = err.Error()
|
|
}
|
|
return msg
|
|
}
|
|
|
|
func UsageError(cmd *cobra.Command, format string, args ...interface{}) error {
|
|
msg := fmt.Sprintf(format, args...)
|
|
return fmt.Errorf("%s\nSee '%s -h' for help and examples.", msg, cmd.CommandPath())
|
|
}
|
|
|
|
// Whether this cmd need watching objects.
|
|
func isWatch(cmd *cobra.Command) bool {
|
|
if w, err := cmd.Flags().GetBool("watch"); w && err == nil {
|
|
return true
|
|
}
|
|
|
|
if wo, err := cmd.Flags().GetBool("watch-only"); wo && err == nil {
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func getFlag(cmd *cobra.Command, flag string) *pflag.Flag {
|
|
f := cmd.Flags().Lookup(flag)
|
|
if f == nil {
|
|
glog.Fatalf("flag accessed but not defined for command %s: %s", cmd.Name(), flag)
|
|
}
|
|
return f
|
|
}
|
|
|
|
func GetFlagString(cmd *cobra.Command, flag string) string {
|
|
s, err := cmd.Flags().GetString(flag)
|
|
if err != nil {
|
|
glog.Fatalf("err accessing flag %s for command %s: %v", flag, cmd.Name(), err)
|
|
}
|
|
return s
|
|
}
|
|
|
|
// GetFlagStringList can be used to accept multiple argument with flag repetition (e.g. -f arg1 -f arg2 ...)
|
|
func GetFlagStringSlice(cmd *cobra.Command, flag string) []string {
|
|
s, err := cmd.Flags().GetStringSlice(flag)
|
|
if err != nil {
|
|
glog.Fatalf("err accessing flag %s for command %s: %v", flag, cmd.Name(), err)
|
|
}
|
|
return s
|
|
}
|
|
|
|
// GetWideFlag is used to determine if "-o wide" is used
|
|
func GetWideFlag(cmd *cobra.Command) bool {
|
|
f := cmd.Flags().Lookup("output")
|
|
if f.Value.String() == "wide" {
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func GetFlagBool(cmd *cobra.Command, flag string) bool {
|
|
b, err := cmd.Flags().GetBool(flag)
|
|
if err != nil {
|
|
glog.Fatalf("err accessing flag %s for command %s: %v", flag, cmd.Name(), err)
|
|
}
|
|
return b
|
|
}
|
|
|
|
// Assumes the flag has a default value.
|
|
func GetFlagInt(cmd *cobra.Command, flag string) int {
|
|
i, err := cmd.Flags().GetInt(flag)
|
|
if err != nil {
|
|
glog.Fatalf("err accessing flag %s for command %s: %v", flag, cmd.Name(), err)
|
|
}
|
|
return i
|
|
}
|
|
|
|
// Assumes the flag has a default value.
|
|
func GetFlagInt64(cmd *cobra.Command, flag string) int64 {
|
|
i, err := cmd.Flags().GetInt64(flag)
|
|
if err != nil {
|
|
glog.Fatalf("err accessing flag %s for command %s: %v", flag, cmd.Name(), err)
|
|
}
|
|
return i
|
|
}
|
|
|
|
func GetFlagDuration(cmd *cobra.Command, flag string) time.Duration {
|
|
d, err := cmd.Flags().GetDuration(flag)
|
|
if err != nil {
|
|
glog.Fatalf("err accessing flag %s for command %s: %v", flag, cmd.Name(), err)
|
|
}
|
|
return d
|
|
}
|
|
|
|
func AddValidateFlags(cmd *cobra.Command) {
|
|
cmd.Flags().Bool("validate", true, "If true, use a schema to validate the input before sending it")
|
|
cmd.Flags().String("schema-cache-dir", fmt.Sprintf("~/%s/%s", clientcmd.RecommendedHomeDir, clientcmd.RecommendedSchemaName), fmt.Sprintf("If non-empty, load/store cached API schemas in this directory, default is '$HOME/%s/%s'", clientcmd.RecommendedHomeDir, clientcmd.RecommendedSchemaName))
|
|
cmd.MarkFlagFilename("schema-cache-dir")
|
|
}
|
|
|
|
func AddRecursiveFlag(cmd *cobra.Command, value *bool) {
|
|
cmd.Flags().BoolVarP(value, "recursive", "R", *value, "Process the directory used in -f, --filename recursively. Useful when you want to manage related manifests organized within the same directory.")
|
|
}
|
|
|
|
// AddDryRunFlag adds dry-run flag to a command. Usually used by mutations.
|
|
func AddDryRunFlag(cmd *cobra.Command) {
|
|
cmd.Flags().Bool("dry-run", false, "If true, only print the object that would be sent, without sending it.")
|
|
}
|
|
|
|
func AddApplyAnnotationFlags(cmd *cobra.Command) {
|
|
cmd.Flags().Bool(ApplyAnnotationsFlag, false, "If true, the configuration of current object will be saved in its annotation. This is useful when you want to perform kubectl apply on this object in the future.")
|
|
}
|
|
|
|
// AddGeneratorFlags adds flags common to resource generation commands
|
|
// TODO: need to take a pass at other generator commands to use this set of flags
|
|
func AddGeneratorFlags(cmd *cobra.Command, defaultGenerator string) {
|
|
cmd.Flags().String("generator", defaultGenerator, "The name of the API generator to use.")
|
|
AddDryRunFlag(cmd)
|
|
}
|
|
|
|
func ReadConfigDataFromReader(reader io.Reader, source string) ([]byte, error) {
|
|
data, err := ioutil.ReadAll(reader)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(data) == 0 {
|
|
return nil, fmt.Errorf(`Read from %s but no data found`, source)
|
|
}
|
|
|
|
return data, nil
|
|
}
|
|
|
|
// Merge requires JSON serialization
|
|
// TODO: merge assumes JSON serialization, and does not properly abstract API retrieval
|
|
func Merge(codec runtime.Codec, dst runtime.Object, fragment, kind string) (runtime.Object, error) {
|
|
// encode dst into versioned json and apply fragment directly too it
|
|
target, err := runtime.Encode(codec, dst)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
patched, err := jsonpatch.MergePatch(target, []byte(fragment))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
out, err := runtime.Decode(codec, patched)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return out, nil
|
|
}
|
|
|
|
// DumpReaderToFile writes all data from the given io.Reader to the specified file
|
|
// (usually for temporary use).
|
|
func DumpReaderToFile(reader io.Reader, filename string) error {
|
|
f, err := os.OpenFile(filename, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0600)
|
|
defer f.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
buffer := make([]byte, 1024)
|
|
for {
|
|
count, err := reader.Read(buffer)
|
|
if err == io.EOF {
|
|
break
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, err = f.Write(buffer[:count])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// UpdateObject updates resource object with updateFn
|
|
func UpdateObject(info *resource.Info, codec runtime.Codec, updateFn func(runtime.Object) error) (runtime.Object, error) {
|
|
helper := resource.NewHelper(info.Client, info.Mapping)
|
|
|
|
if err := updateFn(info.Object); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Update the annotation used by kubectl apply
|
|
if err := kubectl.UpdateApplyAnnotation(info, codec); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if _, err := helper.Replace(info.Namespace, info.Name, true, info.Object); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return info.Object, nil
|
|
}
|
|
|
|
// AddCmdRecordFlag adds --record flag to command
|
|
func AddRecordFlag(cmd *cobra.Command) {
|
|
cmd.Flags().Bool("record", false, "Record current kubectl command in the resource annotation.")
|
|
}
|
|
|
|
func GetRecordFlag(cmd *cobra.Command) bool {
|
|
return GetFlagBool(cmd, "record")
|
|
}
|
|
|
|
func GetDryRunFlag(cmd *cobra.Command) bool {
|
|
return GetFlagBool(cmd, "dry-run")
|
|
}
|
|
|
|
// RecordChangeCause annotate change-cause to input runtime object.
|
|
func RecordChangeCause(obj runtime.Object, changeCause string) error {
|
|
accessor, err := meta.Accessor(obj)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
annotations := accessor.GetAnnotations()
|
|
if annotations == nil {
|
|
annotations = make(map[string]string)
|
|
}
|
|
annotations[kubectl.ChangeCauseAnnotation] = changeCause
|
|
accessor.SetAnnotations(annotations)
|
|
return nil
|
|
}
|
|
|
|
// ChangeResourcePatch creates a strategic merge patch between the origin input resource info
|
|
// and the annotated with change-cause input resource info.
|
|
func ChangeResourcePatch(info *resource.Info, changeCause string) ([]byte, error) {
|
|
oldData, err := json.Marshal(info.Object)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err := RecordChangeCause(info.Object, changeCause); err != nil {
|
|
return nil, err
|
|
}
|
|
newData, err := json.Marshal(info.Object)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return strategicpatch.CreateTwoWayMergePatch(oldData, newData, info.Object)
|
|
}
|
|
|
|
// containsChangeCause checks if input resource info contains change-cause annotation.
|
|
func ContainsChangeCause(info *resource.Info) bool {
|
|
annotations, err := info.Mapping.MetadataAccessor.Annotations(info.Object)
|
|
if err != nil {
|
|
return false
|
|
}
|
|
return len(annotations[kubectl.ChangeCauseAnnotation]) > 0
|
|
}
|
|
|
|
// ShouldRecord checks if we should record current change cause
|
|
func ShouldRecord(cmd *cobra.Command, info *resource.Info) bool {
|
|
return GetRecordFlag(cmd) || ContainsChangeCause(info)
|
|
}
|
|
|
|
// GetThirdPartyGroupVersions returns the thirdparty "group/versions"s and
|
|
// resources supported by the server. A user may delete a thirdparty resource
|
|
// when this function is running, so this function may return a "NotFound" error
|
|
// due to the race.
|
|
func GetThirdPartyGroupVersions(discovery discovery.DiscoveryInterface) ([]unversioned.GroupVersion, []unversioned.GroupVersionKind, error) {
|
|
result := []unversioned.GroupVersion{}
|
|
gvks := []unversioned.GroupVersionKind{}
|
|
|
|
groupList, err := discovery.ServerGroups()
|
|
if err != nil {
|
|
// On forbidden or not found, just return empty lists.
|
|
if errors.IsForbidden(err) || errors.IsNotFound(err) {
|
|
return result, gvks, nil
|
|
}
|
|
|
|
return nil, nil, err
|
|
}
|
|
|
|
for ix := range groupList.Groups {
|
|
group := &groupList.Groups[ix]
|
|
for jx := range group.Versions {
|
|
gv, err2 := unversioned.ParseGroupVersion(group.Versions[jx].GroupVersion)
|
|
if err2 != nil {
|
|
return nil, nil, err
|
|
}
|
|
// Skip GroupVersionKinds that have been statically registered.
|
|
if registered.IsRegisteredVersion(gv) {
|
|
continue
|
|
}
|
|
result = append(result, gv)
|
|
|
|
resourceList, err := discovery.ServerResourcesForGroupVersion(group.Versions[jx].GroupVersion)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
for kx := range resourceList.APIResources {
|
|
gvks = append(gvks, gv.WithKind(resourceList.APIResources[kx].Kind))
|
|
}
|
|
}
|
|
}
|
|
return result, gvks, nil
|
|
}
|
|
|
|
func GetIncludeThirdPartyAPIs(cmd *cobra.Command) bool {
|
|
if cmd.Flags().Lookup("include-extended-apis") == nil {
|
|
return false
|
|
}
|
|
return GetFlagBool(cmd, "include-extended-apis")
|
|
}
|
|
|
|
func AddInclude3rdPartyFlags(cmd *cobra.Command) {
|
|
cmd.Flags().Bool("include-extended-apis", true, "If true, include definitions of new APIs via calls to the API server. [default true]")
|
|
}
|
|
|
|
// GetResourcesAndPairs retrieves resources and "KEY=VALUE or KEY-" pair args from given args
|
|
func GetResourcesAndPairs(args []string, pairType string) (resources []string, pairArgs []string, err error) {
|
|
foundPair := false
|
|
for _, s := range args {
|
|
nonResource := strings.Contains(s, "=") || strings.HasSuffix(s, "-")
|
|
switch {
|
|
case !foundPair && nonResource:
|
|
foundPair = true
|
|
fallthrough
|
|
case foundPair && nonResource:
|
|
pairArgs = append(pairArgs, s)
|
|
case !foundPair && !nonResource:
|
|
resources = append(resources, s)
|
|
case foundPair && !nonResource:
|
|
err = fmt.Errorf("all resources must be specified before %s changes: %s", pairType, s)
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
// ParsePairs retrieves new and remove pairs (if supportRemove is true) from "KEY=VALUE or KEY-" pair args
|
|
func ParsePairs(pairArgs []string, pairType string, supportRemove bool) (newPairs map[string]string, removePairs []string, err error) {
|
|
newPairs = map[string]string{}
|
|
if supportRemove {
|
|
removePairs = []string{}
|
|
}
|
|
var invalidBuf bytes.Buffer
|
|
|
|
for _, pairArg := range pairArgs {
|
|
if strings.Index(pairArg, "=") != -1 {
|
|
parts := strings.SplitN(pairArg, "=", 2)
|
|
if len(parts) != 2 || len(parts[1]) == 0 {
|
|
if invalidBuf.Len() > 0 {
|
|
invalidBuf.WriteString(", ")
|
|
}
|
|
invalidBuf.WriteString(fmt.Sprintf(pairArg))
|
|
} else {
|
|
newPairs[parts[0]] = parts[1]
|
|
}
|
|
} else if supportRemove && strings.HasSuffix(pairArg, "-") {
|
|
removePairs = append(removePairs, pairArg[:len(pairArg)-1])
|
|
} else {
|
|
if invalidBuf.Len() > 0 {
|
|
invalidBuf.WriteString(", ")
|
|
}
|
|
invalidBuf.WriteString(fmt.Sprintf(pairArg))
|
|
}
|
|
}
|
|
if invalidBuf.Len() > 0 {
|
|
err = fmt.Errorf("invalid %s format: %s", pairType, invalidBuf.String())
|
|
return
|
|
}
|
|
|
|
return
|
|
}
|
|
|
|
// MaybeConvertObject attempts to convert an object to a specific group/version. If the object is
|
|
// a third party resource it is simply passed through.
|
|
func MaybeConvertObject(obj runtime.Object, gv unversioned.GroupVersion, converter runtime.ObjectConvertor) (runtime.Object, error) {
|
|
switch obj.(type) {
|
|
case *extensions.ThirdPartyResourceData:
|
|
// conversion is not supported for 3rd party objects
|
|
return obj, nil
|
|
default:
|
|
return converter.ConvertToVersion(obj, gv)
|
|
}
|
|
}
|