mirror of
https://github.com/k3s-io/kubernetes.git
synced 2025-10-15 11:39:24 +00:00
The unified RESTMapper and Typer follow the new rules, but on error will fallback to the legacy path (while still supporting Unstructured objects). This allows callers to handle the appropriate distinction themselves if necessary. Add a LocalParam() method to the resource.Builder that DRYs up a large chunk of complicated code in set commands.
208 lines
6.5 KiB
Go
208 lines
6.5 KiB
Go
/*
|
|
Copyright 2016 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.
|
|
*/
|
|
|
|
// this file contains factories with no other dependencies
|
|
|
|
package util
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
|
|
"github.com/spf13/cobra"
|
|
|
|
"k8s.io/apimachinery/pkg/api/meta"
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/kubernetes/pkg/kubectl/plugins"
|
|
"k8s.io/kubernetes/pkg/kubectl/resource"
|
|
"k8s.io/kubernetes/pkg/printers"
|
|
)
|
|
|
|
type ring2Factory struct {
|
|
clientAccessFactory ClientAccessFactory
|
|
objectMappingFactory ObjectMappingFactory
|
|
}
|
|
|
|
func NewBuilderFactory(clientAccessFactory ClientAccessFactory, objectMappingFactory ObjectMappingFactory) BuilderFactory {
|
|
f := &ring2Factory{
|
|
clientAccessFactory: clientAccessFactory,
|
|
objectMappingFactory: objectMappingFactory,
|
|
}
|
|
|
|
return f
|
|
}
|
|
|
|
func (f *ring2Factory) PrinterForCommand(cmd *cobra.Command, isLocal bool, outputOpts *printers.OutputOptions, options printers.PrintOptions) (printers.ResourcePrinter, error) {
|
|
var mapper meta.RESTMapper
|
|
var typer runtime.ObjectTyper
|
|
|
|
mapper, typer = f.objectMappingFactory.Object()
|
|
|
|
// TODO: used by the custom column implementation and the name implementation, break this dependency
|
|
decoders := []runtime.Decoder{f.clientAccessFactory.Decoder(true), unstructured.UnstructuredJSONScheme}
|
|
encoder := f.clientAccessFactory.JSONEncoder()
|
|
return PrinterForCommand(cmd, outputOpts, mapper, typer, encoder, decoders, options)
|
|
}
|
|
|
|
func (f *ring2Factory) PrinterForMapping(cmd *cobra.Command, isLocal bool, outputOpts *printers.OutputOptions, mapping *meta.RESTMapping, withNamespace bool) (printers.ResourcePrinter, error) {
|
|
// Some callers do not have "label-columns" so we can't use the GetFlagStringSlice() helper
|
|
columnLabel, err := cmd.Flags().GetStringSlice("label-columns")
|
|
if err != nil {
|
|
columnLabel = []string{}
|
|
}
|
|
|
|
options := printers.PrintOptions{
|
|
NoHeaders: GetFlagBool(cmd, "no-headers"),
|
|
WithNamespace: withNamespace,
|
|
Wide: GetWideFlag(cmd),
|
|
ShowAll: GetFlagBool(cmd, "show-all"),
|
|
ShowLabels: GetFlagBool(cmd, "show-labels"),
|
|
AbsoluteTimestamps: isWatch(cmd),
|
|
ColumnLabels: columnLabel,
|
|
}
|
|
|
|
printer, err := f.PrinterForCommand(cmd, isLocal, outputOpts, options)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Make sure we output versioned data for generic printers
|
|
if printer.IsGeneric() {
|
|
if mapping == nil {
|
|
return nil, fmt.Errorf("no serialization format found")
|
|
}
|
|
version := mapping.GroupVersionKind.GroupVersion()
|
|
if version.Empty() {
|
|
return nil, fmt.Errorf("no serialization format found")
|
|
}
|
|
|
|
printer = printers.NewVersionedPrinter(printer, mapping.ObjectConvertor, version, mapping.GroupVersionKind.GroupVersion())
|
|
|
|
}
|
|
|
|
return printer, nil
|
|
}
|
|
|
|
func (f *ring2Factory) PrintSuccess(mapper meta.RESTMapper, shortOutput bool, out io.Writer, resource, name string, dryRun bool, operation string) {
|
|
resource, _ = mapper.ResourceSingularizer(resource)
|
|
dryRunMsg := ""
|
|
if dryRun {
|
|
dryRunMsg = " (dry run)"
|
|
}
|
|
if shortOutput {
|
|
// -o name: prints resource/name
|
|
if len(resource) > 0 {
|
|
fmt.Fprintf(out, "%s/%s\n", resource, name)
|
|
} else {
|
|
fmt.Fprintf(out, "%s\n", name)
|
|
}
|
|
} else {
|
|
// understandable output by default
|
|
if len(resource) > 0 {
|
|
fmt.Fprintf(out, "%s \"%s\" %s%s\n", resource, name, operation, dryRunMsg)
|
|
} else {
|
|
fmt.Fprintf(out, "\"%s\" %s%s\n", name, operation, dryRunMsg)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (f *ring2Factory) PrintObject(cmd *cobra.Command, isLocal bool, mapper meta.RESTMapper, obj runtime.Object, out io.Writer) error {
|
|
// try to get a typed object
|
|
_, typer := f.objectMappingFactory.Object()
|
|
gvks, _, err := typer.ObjectKinds(obj)
|
|
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// Prefer the existing external version if specified
|
|
var preferredVersion []string
|
|
if gvks[0].Version != "" && gvks[0].Version != runtime.APIVersionInternal {
|
|
preferredVersion = []string{gvks[0].Version}
|
|
}
|
|
|
|
mapping, err := mapper.RESTMapping(gvks[0].GroupKind(), preferredVersion...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
printer, err := f.PrinterForMapping(cmd, isLocal, nil, mapping, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return printer.PrintObj(obj, out)
|
|
}
|
|
|
|
func (f *ring2Factory) PrintResourceInfoForCommand(cmd *cobra.Command, info *resource.Info, out io.Writer) error {
|
|
printer, err := f.PrinterForCommand(cmd, false, nil, printers.PrintOptions{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !printer.IsGeneric() {
|
|
printer, err = f.PrinterForMapping(cmd, false, nil, nil, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return printer.PrintObj(info.Object, out)
|
|
}
|
|
|
|
// NewBuilder returns a new resource builder for structured api objects.
|
|
func (f *ring2Factory) NewBuilder() *resource.Builder {
|
|
clientMapperFunc := resource.ClientMapperFunc(f.objectMappingFactory.ClientForMapping)
|
|
mapper, typer := f.objectMappingFactory.Object()
|
|
|
|
unstructuredClientMapperFunc := resource.ClientMapperFunc(f.objectMappingFactory.UnstructuredClientForMapping)
|
|
|
|
categoryExpander := f.objectMappingFactory.CategoryExpander()
|
|
|
|
return resource.NewBuilder(
|
|
&resource.Mapper{
|
|
RESTMapper: mapper,
|
|
ObjectTyper: typer,
|
|
ClientMapper: clientMapperFunc,
|
|
Decoder: f.clientAccessFactory.Decoder(true),
|
|
},
|
|
&resource.Mapper{
|
|
RESTMapper: mapper,
|
|
ObjectTyper: typer,
|
|
ClientMapper: unstructuredClientMapperFunc,
|
|
Decoder: unstructured.UnstructuredJSONScheme,
|
|
},
|
|
categoryExpander,
|
|
)
|
|
}
|
|
|
|
// PluginLoader loads plugins from a path set by the KUBECTL_PLUGINS_PATH env var.
|
|
// If this env var is not set, it defaults to
|
|
// "~/.kube/plugins", plus
|
|
// "./kubectl/plugins" directory under the "data dir" directory specified by the XDG
|
|
// system directory structure spec for the given platform.
|
|
func (f *ring2Factory) PluginLoader() plugins.PluginLoader {
|
|
if len(os.Getenv("KUBECTL_PLUGINS_PATH")) > 0 {
|
|
return plugins.KubectlPluginsPathPluginLoader()
|
|
}
|
|
return plugins.TolerantMultiPluginLoader{
|
|
plugins.XDGDataDirsPluginLoader(),
|
|
plugins.UserDirPluginLoader(),
|
|
}
|
|
}
|
|
|
|
func (f *ring2Factory) PluginRunner() plugins.PluginRunner {
|
|
return &plugins.ExecPluginRunner{}
|
|
}
|