mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-11-04 07:49:35 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			298 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			298 lines
		
	
	
		
			12 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 (
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"k8s.io/apimachinery/pkg/api/meta"
 | 
						|
	"k8s.io/apimachinery/pkg/runtime"
 | 
						|
	"k8s.io/kubernetes/pkg/kubectl"
 | 
						|
	"k8s.io/kubernetes/pkg/kubectl/cmd/templates"
 | 
						|
	kubectlscheme "k8s.io/kubernetes/pkg/kubectl/scheme"
 | 
						|
	"k8s.io/kubernetes/pkg/printers"
 | 
						|
	printersinternal "k8s.io/kubernetes/pkg/printers/internalversion"
 | 
						|
 | 
						|
	"github.com/spf13/cobra"
 | 
						|
	"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
 | 
						|
	"k8s.io/kubernetes/pkg/api/legacyscheme"
 | 
						|
)
 | 
						|
 | 
						|
// AddPrinterFlags adds printing related flags to a command (e.g. output format, no headers, template path)
 | 
						|
func AddPrinterFlags(cmd *cobra.Command) {
 | 
						|
	AddNonDeprecatedPrinterFlags(cmd)
 | 
						|
 | 
						|
	cmd.Flags().String("output-version", "", "DEPRECATED: To use a specific API version, fully-qualify the resource, version, and group (for example: 'jobs.v1.batch/myjob').")
 | 
						|
	cmd.Flags().MarkDeprecated("output-version", "The resource is used exactly as fetched from the API. To get a specific API version, fully-qualify the resource, version, and group (for example: 'jobs.v1.batch/myjob').")
 | 
						|
	cmd.Flags().MarkHidden("output-version")
 | 
						|
}
 | 
						|
 | 
						|
// AddNonDeprecatedPrinterFlags supports the conversion case which must logically have output-version.  Once output-version
 | 
						|
// is completely removed, this function can go away.
 | 
						|
func AddNonDeprecatedPrinterFlags(cmd *cobra.Command) {
 | 
						|
	AddOutputFlags(cmd)
 | 
						|
	AddNoHeadersFlags(cmd)
 | 
						|
	cmd.Flags().Bool("show-labels", false, "When printing, show all labels as the last column (default hide labels column)")
 | 
						|
	cmd.Flags().String("template", "", "Template string or path to template file to use when -o=go-template, -o=go-template-file. The template format is golang templates [http://golang.org/pkg/text/template/#pkg-overview].")
 | 
						|
	cmd.MarkFlagFilename("template")
 | 
						|
	cmd.Flags().String("sort-by", "", "If non-empty, sort list types using this field specification.  The field specification is expressed as a JSONPath expression (e.g. '{.metadata.name}'). The field in the API resource specified by this JSONPath expression must be an integer or a string.")
 | 
						|
	cmd.Flags().BoolP("show-all", "a", true, "When printing, show all resources (default show all pods including terminated one.)")
 | 
						|
	cmd.Flags().MarkDeprecated("show-all", "will be removed in an upcoming release")
 | 
						|
}
 | 
						|
 | 
						|
// AddOutputFlagsForMutation adds output related flags to a command. Used by mutations only.
 | 
						|
func AddOutputFlagsForMutation(cmd *cobra.Command) {
 | 
						|
	cmd.Flags().StringP("output", "o", "", "Output mode. Use \"-o name\" for shorter output (resource/name).")
 | 
						|
}
 | 
						|
 | 
						|
// AddOutputVarFlagsForMutation adds output related flags to a command. Used by mutations only.
 | 
						|
func AddOutputVarFlagsForMutation(cmd *cobra.Command, output *string) {
 | 
						|
	cmd.Flags().StringVarP(output, "output", "o", *output, "Output mode. Use \"-o name\" for shorter output (resource/name).")
 | 
						|
}
 | 
						|
 | 
						|
// AddOutputFlags adds output related flags to a command.
 | 
						|
func AddOutputFlags(cmd *cobra.Command) {
 | 
						|
	cmd.Flags().StringP("output", "o", "", "Output format. One of: json|yaml|wide|name|custom-columns=...|custom-columns-file=...|go-template=...|go-template-file=...|jsonpath=...|jsonpath-file=... See custom columns [http://kubernetes.io/docs/user-guide/kubectl-overview/#custom-columns], golang template [http://golang.org/pkg/text/template/#pkg-overview] and jsonpath template [http://kubernetes.io/docs/user-guide/jsonpath].")
 | 
						|
	cmd.Flags().Bool("allow-missing-template-keys", true, "If true, ignore any errors in templates when a field or map key is missing in the template. Only applies to golang and jsonpath output formats.")
 | 
						|
}
 | 
						|
 | 
						|
// AddNoHeadersFlags adds no-headers flags to a command.
 | 
						|
func AddNoHeadersFlags(cmd *cobra.Command) {
 | 
						|
	cmd.Flags().Bool("no-headers", false, "When using the default or custom-column output format, don't print headers (default print headers).")
 | 
						|
}
 | 
						|
 | 
						|
// ValidateOutputArgs validates -o flag args for mutations
 | 
						|
func ValidateOutputArgs(cmd *cobra.Command) error {
 | 
						|
	outputMode := GetFlagString(cmd, "output")
 | 
						|
	if outputMode != "" && outputMode != "name" {
 | 
						|
		return UsageErrorf(cmd, "Unexpected -o output mode: %v. We only support '-o name'.", outputMode)
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// PrintSuccess prints a success message and can do a "-o name" as "shortOutput"
 | 
						|
// TODO this should really just be a printer.  It's got just about the exact same signature.
 | 
						|
func PrintSuccess(shortOutput bool, out io.Writer, obj runtime.Object, dryRun bool, operation string) {
 | 
						|
	dryRunMsg := ""
 | 
						|
	if dryRun {
 | 
						|
		dryRunMsg = " (dry run)"
 | 
						|
	}
 | 
						|
 | 
						|
	// match name printer format
 | 
						|
	name := "<unknown>"
 | 
						|
	if acc, err := meta.Accessor(obj); err == nil {
 | 
						|
		if n := acc.GetName(); len(n) > 0 {
 | 
						|
			name = n
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	// legacy scheme to be sure we work ok with internal types.
 | 
						|
	// TODO internal types aren't supposed to exist here
 | 
						|
	groupKind := printers.GetObjectGroupKind(obj, legacyscheme.Scheme)
 | 
						|
	kindString := fmt.Sprintf("%s.%s", strings.ToLower(groupKind.Kind), groupKind.Group)
 | 
						|
	if len(groupKind.Group) == 0 {
 | 
						|
		kindString = strings.ToLower(groupKind.Kind)
 | 
						|
	}
 | 
						|
 | 
						|
	if shortOutput {
 | 
						|
		// -o name: prints resource/name
 | 
						|
		fmt.Fprintf(out, "%s/%s\n", kindString, name)
 | 
						|
		return
 | 
						|
	}
 | 
						|
 | 
						|
	// understandable output by default
 | 
						|
	fmt.Fprintf(out, "%s \"%s\" %s%s\n", kindString, name, operation, dryRunMsg)
 | 
						|
}
 | 
						|
 | 
						|
// PrintObject prints a single object based on the default command options
 | 
						|
// TODO this should go away once commands can embed the PrintOptions instead
 | 
						|
func PrintObject(cmd *cobra.Command, obj runtime.Object, out io.Writer) error {
 | 
						|
	printer, err := PrinterForOptions(ExtractCmdPrintOptions(cmd, false))
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	return printer.PrintObj(obj, out)
 | 
						|
}
 | 
						|
 | 
						|
// PrinterForOptions returns the printer for the outputOptions (if given) or
 | 
						|
// returns the default printer for the command.
 | 
						|
// TODO this should become a function on the PrintOptions struct
 | 
						|
func PrinterForOptions(options *printers.PrintOptions) (printers.ResourcePrinter, error) {
 | 
						|
	// TODO: used by the custom column implementation and the name implementation, break this dependency
 | 
						|
	decoders := []runtime.Decoder{kubectlscheme.Codecs.UniversalDecoder(), unstructured.UnstructuredJSONScheme}
 | 
						|
	encoder := kubectlscheme.Codecs.LegacyCodec(kubectlscheme.Registry.EnabledVersions()...)
 | 
						|
 | 
						|
	printer, err := printers.GetStandardPrinter(kubectlscheme.Scheme, encoder, decoders, *options)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	// we try to convert to HumanReadablePrinter, if return ok, it must be no generic
 | 
						|
	// we execute AddHandlers() here before maybeWrapSortingPrinter so that we don't
 | 
						|
	// need to convert to delegatePrinter again then invoke AddHandlers()
 | 
						|
	// TODO this looks highly questionable.  human readable printers are baked into code.  This can just live in the definition of the handler itself
 | 
						|
	// TODO or be registered there
 | 
						|
	if humanReadablePrinter, ok := printer.(printers.PrintHandler); ok {
 | 
						|
		printersinternal.AddHandlers(humanReadablePrinter)
 | 
						|
	}
 | 
						|
 | 
						|
	printer = maybeWrapSortingPrinter(printer, *options)
 | 
						|
 | 
						|
	// wrap the printer in a versioning printer that understands when to convert and when not to convert
 | 
						|
	printer = printers.NewVersionedPrinter(printer, legacyscheme.Scheme, legacyscheme.Scheme, kubectlscheme.Versions...)
 | 
						|
 | 
						|
	return printer, nil
 | 
						|
}
 | 
						|
 | 
						|
// ExtractCmdPrintOptions parses printer specific commandline args and
 | 
						|
// returns a PrintOptions object.
 | 
						|
// Requires that printer flags have been added to cmd (see AddPrinterFlags)
 | 
						|
func ExtractCmdPrintOptions(cmd *cobra.Command, withNamespace bool) *printers.PrintOptions {
 | 
						|
	flags := cmd.Flags()
 | 
						|
 | 
						|
	columnLabel, err := flags.GetStringSlice("label-columns")
 | 
						|
	if err != nil {
 | 
						|
		columnLabel = []string{}
 | 
						|
	}
 | 
						|
 | 
						|
	options := &printers.PrintOptions{
 | 
						|
		NoHeaders:          GetFlagBool(cmd, "no-headers"),
 | 
						|
		Wide:               GetWideFlag(cmd),
 | 
						|
		ShowAll:            GetFlagBool(cmd, "show-all"),
 | 
						|
		ShowLabels:         GetFlagBool(cmd, "show-labels"),
 | 
						|
		AbsoluteTimestamps: isWatch(cmd),
 | 
						|
		ColumnLabels:       columnLabel,
 | 
						|
		WithNamespace:      withNamespace,
 | 
						|
	}
 | 
						|
 | 
						|
	var outputFormat string
 | 
						|
	if flags.Lookup("output") != nil {
 | 
						|
		outputFormat = GetFlagString(cmd, "output")
 | 
						|
	}
 | 
						|
 | 
						|
	if flags.Lookup("sort-by") != nil {
 | 
						|
		options.SortBy = GetFlagString(cmd, "sort-by")
 | 
						|
	}
 | 
						|
 | 
						|
	// templates are logically optional for specifying a format.
 | 
						|
	// TODO once https://github.com/kubernetes/kubernetes/issues/12668 is fixed, this should fall back to GetFlagString
 | 
						|
	var templateFile string
 | 
						|
	if flag := flags.Lookup("template"); flag != nil {
 | 
						|
		if flag.Value.Type() == "string" {
 | 
						|
			templateFile = GetFlagString(cmd, "template")
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if len(outputFormat) == 0 && len(templateFile) != 0 {
 | 
						|
		outputFormat = "template"
 | 
						|
	}
 | 
						|
 | 
						|
	templateFormats := []string{
 | 
						|
		"go-template=", "go-template-file=", "jsonpath=", "jsonpath-file=", "custom-columns=", "custom-columns-file=",
 | 
						|
	}
 | 
						|
	for _, format := range templateFormats {
 | 
						|
		if strings.HasPrefix(outputFormat, format) {
 | 
						|
			templateFile = outputFormat[len(format):]
 | 
						|
			outputFormat = format[:len(format)-1]
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	// this function may be invoked by a command that did not call AddPrinterFlags first, so we need
 | 
						|
	// to be safe about how we access the allow-missing-template-keys flag
 | 
						|
	if flags.Lookup("allow-missing-template-keys") != nil {
 | 
						|
		options.AllowMissingKeys = GetFlagBool(cmd, "allow-missing-template-keys")
 | 
						|
	}
 | 
						|
 | 
						|
	options.OutputFormatType = outputFormat
 | 
						|
	options.OutputFormatArgument = templateFile
 | 
						|
 | 
						|
	return options
 | 
						|
}
 | 
						|
 | 
						|
func maybeWrapSortingPrinter(printer printers.ResourcePrinter, printOpts printers.PrintOptions) printers.ResourcePrinter {
 | 
						|
	if len(printOpts.SortBy) != 0 {
 | 
						|
		return &kubectl.SortingPrinter{
 | 
						|
			Delegate:  printer,
 | 
						|
			SortField: fmt.Sprintf("{%s}", printOpts.SortBy),
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return printer
 | 
						|
}
 | 
						|
 | 
						|
// ValidResourceTypeList returns a multi-line string containing the valid resources. May
 | 
						|
// be called before the factory is initialized.
 | 
						|
// TODO: This function implementation should be replaced with a real implementation from the
 | 
						|
//   discovery service.
 | 
						|
func ValidResourceTypeList(f ClientAccessFactory) string {
 | 
						|
	// TODO: Should attempt to use the cached discovery list or fallback to a static list
 | 
						|
	// that is calculated from code compiled into the factory.
 | 
						|
	return templates.LongDesc(`Valid resource types include:
 | 
						|
	
 | 
						|
			* all
 | 
						|
			* certificatesigningrequests (aka 'csr')
 | 
						|
			* clusterrolebindings
 | 
						|
			* clusterroles
 | 
						|
			* componentstatuses (aka 'cs')
 | 
						|
			* configmaps (aka 'cm')
 | 
						|
			* controllerrevisions
 | 
						|
			* cronjobs
 | 
						|
			* customresourcedefinition (aka 'crd')
 | 
						|
			* daemonsets (aka 'ds')
 | 
						|
			* deployments (aka 'deploy')
 | 
						|
			* endpoints (aka 'ep')
 | 
						|
			* events (aka 'ev')
 | 
						|
			* horizontalpodautoscalers (aka 'hpa')
 | 
						|
			* ingresses (aka 'ing')
 | 
						|
			* jobs
 | 
						|
			* limitranges (aka 'limits')
 | 
						|
			* namespaces (aka 'ns')
 | 
						|
			* networkpolicies (aka 'netpol')
 | 
						|
			* nodes (aka 'no')
 | 
						|
			* persistentvolumeclaims (aka 'pvc')
 | 
						|
			* persistentvolumes (aka 'pv')
 | 
						|
			* poddisruptionbudgets (aka 'pdb')
 | 
						|
			* podpreset
 | 
						|
			* pods (aka 'po')
 | 
						|
			* podsecuritypolicies (aka 'psp')
 | 
						|
			* podtemplates
 | 
						|
			* replicasets (aka 'rs')
 | 
						|
			* replicationcontrollers (aka 'rc')
 | 
						|
			* resourcequotas (aka 'quota')
 | 
						|
			* rolebindings
 | 
						|
			* roles
 | 
						|
			* secrets
 | 
						|
			* serviceaccounts (aka 'sa')
 | 
						|
			* services (aka 'svc')
 | 
						|
			* statefulsets (aka 'sts')
 | 
						|
			* storageclasses (aka 'sc')
 | 
						|
	
 | 
						|
	`)
 | 
						|
}
 | 
						|
 | 
						|
// Retrieve a list of handled resources from printer as valid args
 | 
						|
// TODO: This function implementation should be replaced with a real implementation from the discovery service.
 | 
						|
func ValidArgList(f ClientAccessFactory) []string {
 | 
						|
	validArgs := []string{}
 | 
						|
 | 
						|
	humanReadablePrinter := printers.NewHumanReadablePrinter(nil, nil, printers.PrintOptions{})
 | 
						|
	printersinternal.AddHandlers(humanReadablePrinter)
 | 
						|
	validArgs = humanReadablePrinter.HandledResources()
 | 
						|
 | 
						|
	return validArgs
 | 
						|
}
 |