2019-01-12 04:58:27 +00:00
|
|
|
/*
|
|
|
|
Copyright 2018 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 genericclioptions
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
|
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
2019-04-07 17:07:55 +00:00
|
|
|
"k8s.io/cli-runtime/pkg/printers"
|
2019-01-12 04:58:27 +00:00
|
|
|
)
|
|
|
|
|
2021-03-18 22:40:29 +00:00
|
|
|
// NoCompatiblePrinterError is a struct that contains error information.
|
|
|
|
// It will be constructed when a invalid printing format is provided
|
2019-01-12 04:58:27 +00:00
|
|
|
type NoCompatiblePrinterError struct {
|
|
|
|
OutputFormat *string
|
|
|
|
AllowedFormats []string
|
|
|
|
Options interface{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e NoCompatiblePrinterError) Error() string {
|
|
|
|
output := ""
|
|
|
|
if e.OutputFormat != nil {
|
|
|
|
output = *e.OutputFormat
|
|
|
|
}
|
|
|
|
|
|
|
|
sort.Strings(e.AllowedFormats)
|
|
|
|
return fmt.Sprintf("unable to match a printer suitable for the output format %q, allowed formats are: %s", output, strings.Join(e.AllowedFormats, ","))
|
|
|
|
}
|
|
|
|
|
2021-03-18 22:40:29 +00:00
|
|
|
// IsNoCompatiblePrinterError returns true if it is a not a compatible printer
|
|
|
|
// otherwise it will return false
|
2019-01-12 04:58:27 +00:00
|
|
|
func IsNoCompatiblePrinterError(err error) bool {
|
|
|
|
if err == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
_, ok := err.(NoCompatiblePrinterError)
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
// PrintFlags composes common printer flag structs
|
|
|
|
// used across all commands, and provides a method
|
|
|
|
// of retrieving a known printer based on flag values provided.
|
|
|
|
type PrintFlags struct {
|
|
|
|
JSONYamlPrintFlags *JSONYamlPrintFlags
|
|
|
|
NamePrintFlags *NamePrintFlags
|
|
|
|
TemplatePrinterFlags *KubeTemplatePrintFlags
|
|
|
|
|
|
|
|
TypeSetterPrinter *printers.TypeSetterPrinter
|
|
|
|
|
|
|
|
OutputFormat *string
|
|
|
|
|
|
|
|
// OutputFlagSpecified indicates whether the user specifically requested a certain kind of output.
|
|
|
|
// Using this function allows a sophisticated caller to change the flag binding logic if they so desire.
|
|
|
|
OutputFlagSpecified func() bool
|
|
|
|
}
|
|
|
|
|
2021-03-18 22:40:29 +00:00
|
|
|
// Complete sets NamePrintFlags operation flag from sucessTemplate
|
2019-01-12 04:58:27 +00:00
|
|
|
func (f *PrintFlags) Complete(successTemplate string) error {
|
|
|
|
return f.NamePrintFlags.Complete(successTemplate)
|
|
|
|
}
|
|
|
|
|
2021-03-18 22:40:29 +00:00
|
|
|
// AllowedFormats returns slice of string of allowed JSONYaml/Name/Template printing format
|
2019-01-12 04:58:27 +00:00
|
|
|
func (f *PrintFlags) AllowedFormats() []string {
|
|
|
|
ret := []string{}
|
|
|
|
ret = append(ret, f.JSONYamlPrintFlags.AllowedFormats()...)
|
|
|
|
ret = append(ret, f.NamePrintFlags.AllowedFormats()...)
|
|
|
|
ret = append(ret, f.TemplatePrinterFlags.AllowedFormats()...)
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
2021-03-18 22:40:29 +00:00
|
|
|
// ToPrinter returns a printer capable of
|
|
|
|
// handling --output or --template printing.
|
|
|
|
// Returns false if the specified outputFormat does not match a supported format.
|
|
|
|
// Supported format types can be found in pkg/printers/printers.go
|
2019-01-12 04:58:27 +00:00
|
|
|
func (f *PrintFlags) ToPrinter() (printers.ResourcePrinter, error) {
|
|
|
|
outputFormat := ""
|
|
|
|
if f.OutputFormat != nil {
|
|
|
|
outputFormat = *f.OutputFormat
|
|
|
|
}
|
|
|
|
// For backwards compatibility we want to support a --template argument given, even when no --output format is provided.
|
|
|
|
// If no explicit output format has been provided via the --output flag, fallback
|
|
|
|
// to honoring the --template argument.
|
|
|
|
templateFlagSpecified := f.TemplatePrinterFlags != nil &&
|
|
|
|
f.TemplatePrinterFlags.TemplateArgument != nil &&
|
|
|
|
len(*f.TemplatePrinterFlags.TemplateArgument) > 0
|
|
|
|
outputFlagSpecified := f.OutputFlagSpecified != nil && f.OutputFlagSpecified()
|
|
|
|
if templateFlagSpecified && !outputFlagSpecified {
|
|
|
|
outputFormat = "go-template"
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.JSONYamlPrintFlags != nil {
|
|
|
|
if p, err := f.JSONYamlPrintFlags.ToPrinter(outputFormat); !IsNoCompatiblePrinterError(err) {
|
|
|
|
return f.TypeSetterPrinter.WrapToPrinter(p, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.NamePrintFlags != nil {
|
|
|
|
if p, err := f.NamePrintFlags.ToPrinter(outputFormat); !IsNoCompatiblePrinterError(err) {
|
|
|
|
return f.TypeSetterPrinter.WrapToPrinter(p, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.TemplatePrinterFlags != nil {
|
|
|
|
if p, err := f.TemplatePrinterFlags.ToPrinter(outputFormat); !IsNoCompatiblePrinterError(err) {
|
|
|
|
return f.TypeSetterPrinter.WrapToPrinter(p, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil, NoCompatiblePrinterError{OutputFormat: f.OutputFormat, AllowedFormats: f.AllowedFormats()}
|
|
|
|
}
|
|
|
|
|
2021-03-18 22:40:29 +00:00
|
|
|
// AddFlags receives a *cobra.Command reference and binds
|
|
|
|
// flags related to JSON/Yaml/Name/Template printing to it
|
2019-01-12 04:58:27 +00:00
|
|
|
func (f *PrintFlags) AddFlags(cmd *cobra.Command) {
|
|
|
|
f.JSONYamlPrintFlags.AddFlags(cmd)
|
|
|
|
f.NamePrintFlags.AddFlags(cmd)
|
|
|
|
f.TemplatePrinterFlags.AddFlags(cmd)
|
|
|
|
|
|
|
|
if f.OutputFormat != nil {
|
|
|
|
cmd.Flags().StringVarP(f.OutputFormat, "output", "o", *f.OutputFormat, fmt.Sprintf("Output format. One of: %s.", strings.Join(f.AllowedFormats(), "|")))
|
|
|
|
if f.OutputFlagSpecified == nil {
|
|
|
|
f.OutputFlagSpecified = func() bool {
|
|
|
|
return cmd.Flag("output").Changed
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithDefaultOutput sets a default output format if one is not provided through a flag value
|
|
|
|
func (f *PrintFlags) WithDefaultOutput(output string) *PrintFlags {
|
|
|
|
f.OutputFormat = &output
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
|
|
|
// WithTypeSetter sets a wrapper than will surround the returned printer with a printer to type resources
|
|
|
|
func (f *PrintFlags) WithTypeSetter(scheme *runtime.Scheme) *PrintFlags {
|
|
|
|
f.TypeSetterPrinter = printers.NewTypeSetter(scheme)
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
2021-03-18 22:40:29 +00:00
|
|
|
// NewPrintFlags returns a default *PrintFlags
|
2019-01-12 04:58:27 +00:00
|
|
|
func NewPrintFlags(operation string) *PrintFlags {
|
|
|
|
outputFormat := ""
|
|
|
|
|
|
|
|
return &PrintFlags{
|
|
|
|
OutputFormat: &outputFormat,
|
|
|
|
|
|
|
|
JSONYamlPrintFlags: NewJSONYamlPrintFlags(),
|
|
|
|
NamePrintFlags: NewNamePrintFlags(operation),
|
|
|
|
TemplatePrinterFlags: NewKubeTemplatePrintFlags(),
|
|
|
|
}
|
|
|
|
}
|