/* 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 ( "errors" "flag" "fmt" "io" "net/http" "os" "path/filepath" "regexp" "strings" "time" "k8s.io/api/core/v1" "github.com/spf13/cobra" "github.com/spf13/pflag" appsv1 "k8s.io/api/apps/v1" appsv1beta1 "k8s.io/api/apps/v1beta1" appsv1beta2 "k8s.io/api/apps/v1beta2" batchv1 "k8s.io/api/batch/v1" batchv1beta1 "k8s.io/api/batch/v1beta1" batchv2alpha1 "k8s.io/api/batch/v2alpha1" extensionsv1beta1 "k8s.io/api/extensions/v1beta1" apierrors "k8s.io/apimachinery/pkg/api/errors" "k8s.io/apimachinery/pkg/api/meta" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/runtime/schema" utilflag "k8s.io/apiserver/pkg/util/flag" "k8s.io/client-go/discovery" "k8s.io/client-go/kubernetes" restclient "k8s.io/client-go/rest" "k8s.io/client-go/tools/clientcmd" "k8s.io/client-go/util/homedir" "k8s.io/kubernetes/pkg/api/legacyscheme" "k8s.io/kubernetes/pkg/apis/apps" api "k8s.io/kubernetes/pkg/apis/core" "k8s.io/kubernetes/pkg/apis/extensions" "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset" "k8s.io/kubernetes/pkg/kubectl" "k8s.io/kubernetes/pkg/kubectl/resource" "k8s.io/kubernetes/pkg/kubectl/util/transport" ) type ring0Factory struct { flags *pflag.FlagSet clientConfig clientcmd.ClientConfig discoveryFactory DiscoveryClientFactory clientCache *ClientCache } func NewClientAccessFactory(optionalClientConfig clientcmd.ClientConfig) ClientAccessFactory { flags := pflag.NewFlagSet("", pflag.ContinueOnError) clientConfig := optionalClientConfig if optionalClientConfig == nil { clientConfig = DefaultClientConfig(flags) } return NewClientAccessFactoryFromDiscovery(flags, clientConfig, &discoveryFactory{clientConfig: clientConfig}) } // NewClientAccessFactoryFromDiscovery allows an external caller to substitute a different discoveryFactory // Which allows for the client cache to be built in ring0, but still rely on a custom discovery client func NewClientAccessFactoryFromDiscovery(flags *pflag.FlagSet, clientConfig clientcmd.ClientConfig, discoveryFactory DiscoveryClientFactory) ClientAccessFactory { flags.SetNormalizeFunc(utilflag.WarnWordSepNormalizeFunc) // Warn for "_" flags clientCache := NewClientCache(clientConfig, discoveryFactory) f := &ring0Factory{ flags: flags, clientConfig: clientConfig, discoveryFactory: discoveryFactory, clientCache: clientCache, } return f } type discoveryFactory struct { clientConfig clientcmd.ClientConfig cacheDir string } func (f *discoveryFactory) DiscoveryClient() (discovery.CachedDiscoveryInterface, error) { cfg, err := f.clientConfig.ClientConfig() if err != nil { return nil, err } if f.cacheDir != "" { wt := cfg.WrapTransport cfg.WrapTransport = func(rt http.RoundTripper) http.RoundTripper { if wt != nil { rt = wt(rt) } return transport.NewCacheRoundTripper(f.cacheDir, rt) } } discoveryClient, err := discovery.NewDiscoveryClientForConfig(cfg) if err != nil { return nil, err } cacheDir := computeDiscoverCacheDir(filepath.Join(homedir.HomeDir(), ".kube", "cache", "discovery"), cfg.Host) return NewCachedDiscoveryClient(discoveryClient, cacheDir, time.Duration(10*time.Minute)), nil } func (f *discoveryFactory) BindFlags(flags *pflag.FlagSet) { defaultCacheDir := filepath.Join(homedir.HomeDir(), ".kube", "http-cache") flags.StringVar(&f.cacheDir, FlagHTTPCacheDir, defaultCacheDir, "Default HTTP cache directory") } // DefaultClientConfig creates a clientcmd.ClientConfig with the following hierarchy: // 1. Use the kubeconfig builder. The number of merges and overrides here gets a little crazy. Stay with me. // 1. Merge the kubeconfig itself. This is done with the following hierarchy rules: // 1. CommandLineLocation - this parsed from the command line, so it must be late bound. If you specify this, // then no other kubeconfig files are merged. This file must exist. // 2. If $KUBECONFIG is set, then it is treated as a list of files that should be merged. // 3. HomeDirectoryLocation // Empty filenames are ignored. Files with non-deserializable content produced errors. // The first file to set a particular value or map key wins and the value or map key is never changed. // This means that the first file to set CurrentContext will have its context preserved. It also means // that if two files specify a "red-user", only values from the first file's red-user are used. Even // non-conflicting entries from the second file's "red-user" are discarded. // 2. Determine the context to use based on the first hit in this chain // 1. command line argument - again, parsed from the command line, so it must be late bound // 2. CurrentContext from the merged kubeconfig file // 3. Empty is allowed at this stage // 3. Determine the cluster info and auth info to use. At this point, we may or may not have a context. They // are built based on the first hit in this chain. (run it twice, once for auth, once for cluster) // 1. command line argument // 2. If context is present, then use the context value // 3. Empty is allowed // 4. Determine the actual cluster info to use. At this point, we may or may not have a cluster info. Build // each piece of the cluster info based on the chain: // 1. command line argument // 2. If cluster info is present and a value for the attribute is present, use it. // 3. If you don't have a server location, bail. // 5. Auth info is build using the same rules as cluster info, EXCEPT that you can only have one authentication // technique per auth info. The following conditions result in an error: // 1. If there are two conflicting techniques specified from the command line, fail. // 2. If the command line does not specify one, and the auth info has conflicting techniques, fail. // 3. If the command line specifies one and the auth info specifies another, honor the command line technique. // 2. Use default values and potentially prompt for auth information // // However, if it appears that we're running in a kubernetes cluster // container environment, then run with the auth info kubernetes mounted for // us. Specifically: // The env vars KUBERNETES_SERVICE_HOST and KUBERNETES_SERVICE_PORT are // set, and the file /var/run/secrets/kubernetes.io/serviceaccount/token // exists and is not a directory. func DefaultClientConfig(flags *pflag.FlagSet) clientcmd.ClientConfig { loadingRules := clientcmd.NewDefaultClientConfigLoadingRules() // use the standard defaults for this client command // DEPRECATED: remove and replace with something more accurate loadingRules.DefaultClientConfig = &clientcmd.DefaultClientConfig flags.StringVar(&loadingRules.ExplicitPath, "kubeconfig", "", "Path to the kubeconfig file to use for CLI requests.") overrides := &clientcmd.ConfigOverrides{ClusterDefaults: clientcmd.ClusterDefaults} flagNames := clientcmd.RecommendedConfigOverrideFlags("") // short flagnames are disabled by default. These are here for compatibility with existing scripts flagNames.ClusterOverrideFlags.APIServer.ShortName = "s" clientcmd.BindOverrideFlags(overrides, flags, flagNames) clientConfig := clientcmd.NewInteractiveDeferredLoadingClientConfig(loadingRules, overrides, os.Stdin) return clientConfig } func (f *ring0Factory) DiscoveryClient() (discovery.CachedDiscoveryInterface, error) { return f.discoveryFactory.DiscoveryClient() } func (f *ring0Factory) KubernetesClientSet() (*kubernetes.Clientset, error) { return f.clientCache.KubernetesClientSetForVersion(nil) } func (f *ring0Factory) ClientSet() (internalclientset.Interface, error) { return f.clientCache.ClientSetForVersion(nil) } func (f *ring0Factory) ClientConfig() (*restclient.Config, error) { return f.clientCache.ClientConfigForVersion(nil) } func (f *ring0Factory) BareClientConfig() (*restclient.Config, error) { return f.clientConfig.ClientConfig() } func (f *ring0Factory) RESTClient() (*restclient.RESTClient, error) { clientConfig, err := f.clientCache.ClientConfigForVersion(nil) if err != nil { return nil, err } return restclient.RESTClientFor(clientConfig) } func (f *ring0Factory) UpdatePodSpecForObject(obj runtime.Object, fn func(*v1.PodSpec) error) (bool, error) { // TODO: replace with a swagger schema based approach (identify pod template via schema introspection) switch t := obj.(type) { case *v1.Pod: return true, fn(&t.Spec) // ReplicationController case *v1.ReplicationController: if t.Spec.Template == nil { t.Spec.Template = &v1.PodTemplateSpec{} } return true, fn(&t.Spec.Template.Spec) // Deployment case *extensionsv1beta1.Deployment: return true, fn(&t.Spec.Template.Spec) case *appsv1beta1.Deployment: return true, fn(&t.Spec.Template.Spec) case *appsv1beta2.Deployment: return true, fn(&t.Spec.Template.Spec) case *appsv1.Deployment: return true, fn(&t.Spec.Template.Spec) // DaemonSet case *extensionsv1beta1.DaemonSet: return true, fn(&t.Spec.Template.Spec) case *appsv1beta2.DaemonSet: return true, fn(&t.Spec.Template.Spec) case *appsv1.DaemonSet: return true, fn(&t.Spec.Template.Spec) // ReplicaSet case *extensionsv1beta1.ReplicaSet: return true, fn(&t.Spec.Template.Spec) case *appsv1beta2.ReplicaSet: return true, fn(&t.Spec.Template.Spec) case *appsv1.ReplicaSet: return true, fn(&t.Spec.Template.Spec) // StatefulSet case *appsv1beta1.StatefulSet: return true, fn(&t.Spec.Template.Spec) case *appsv1beta2.StatefulSet: return true, fn(&t.Spec.Template.Spec) case *appsv1.StatefulSet: return true, fn(&t.Spec.Template.Spec) // Job case *batchv1.Job: return true, fn(&t.Spec.Template.Spec) // CronJob case *batchv1beta1.CronJob: return true, fn(&t.Spec.JobTemplate.Spec.Template.Spec) case *batchv2alpha1.CronJob: return true, fn(&t.Spec.JobTemplate.Spec.Template.Spec) default: return false, fmt.Errorf("the object is not a pod or does not have a pod template") } } func (f *ring0Factory) MapBasedSelectorForObject(object runtime.Object) (string, error) { // TODO: replace with a swagger schema based approach (identify pod selector via schema introspection) switch t := object.(type) { case *api.ReplicationController: return kubectl.MakeLabels(t.Spec.Selector), nil case *api.Pod: if len(t.Labels) == 0 { return "", fmt.Errorf("the pod has no labels and cannot be exposed") } return kubectl.MakeLabels(t.Labels), nil case *api.Service: if t.Spec.Selector == nil { return "", fmt.Errorf("the service has no pod selector set") } return kubectl.MakeLabels(t.Spec.Selector), nil case *extensions.Deployment: // TODO(madhusudancs): Make this smarter by admitting MatchExpressions with Equals // operator, DoubleEquals operator and In operator with only one element in the set. if len(t.Spec.Selector.MatchExpressions) > 0 { return "", fmt.Errorf("couldn't convert expressions - \"%+v\" to map-based selector format", t.Spec.Selector.MatchExpressions) } return kubectl.MakeLabels(t.Spec.Selector.MatchLabels), nil case *extensions.ReplicaSet: // TODO(madhusudancs): Make this smarter by admitting MatchExpressions with Equals // operator, DoubleEquals operator and In operator with only one element in the set. if len(t.Spec.Selector.MatchExpressions) > 0 { return "", fmt.Errorf("couldn't convert expressions - \"%+v\" to map-based selector format", t.Spec.Selector.MatchExpressions) } return kubectl.MakeLabels(t.Spec.Selector.MatchLabels), nil default: return "", fmt.Errorf("cannot extract pod selector from %T", object) } } func (f *ring0Factory) PortsForObject(object runtime.Object) ([]string, error) { // TODO: replace with a swagger schema based approach (identify pod selector via schema introspection) switch t := object.(type) { case *api.ReplicationController: return getPorts(t.Spec.Template.Spec), nil case *api.Pod: return getPorts(t.Spec), nil case *api.Service: return getServicePorts(t.Spec), nil case *extensions.Deployment: return getPorts(t.Spec.Template.Spec), nil case *extensions.ReplicaSet: return getPorts(t.Spec.Template.Spec), nil default: return nil, fmt.Errorf("cannot extract ports from %T", object) } } func (f *ring0Factory) ProtocolsForObject(object runtime.Object) (map[string]string, error) { // TODO: replace with a swagger schema based approach (identify pod selector via schema introspection) switch t := object.(type) { case *api.ReplicationController: return getProtocols(t.Spec.Template.Spec), nil case *api.Pod: return getProtocols(t.Spec), nil case *api.Service: return getServiceProtocols(t.Spec), nil case *extensions.Deployment: return getProtocols(t.Spec.Template.Spec), nil case *extensions.ReplicaSet: return getProtocols(t.Spec.Template.Spec), nil default: return nil, fmt.Errorf("cannot extract protocols from %T", object) } } func (f *ring0Factory) LabelsForObject(object runtime.Object) (map[string]string, error) { return meta.NewAccessor().Labels(object) } func (f *ring0Factory) FlagSet() *pflag.FlagSet { return f.flags } // Set showSecrets false to filter out stuff like secrets. func (f *ring0Factory) Command(cmd *cobra.Command, showSecrets bool) string { if len(os.Args) == 0 { return "" } flags := "" parseFunc := func(flag *pflag.Flag, value string) error { flags = flags + " --" + flag.Name if set, ok := flag.Annotations["classified"]; showSecrets || !ok || len(set) == 0 { flags = flags + "=" + value } else { flags = flags + "=CLASSIFIED" } return nil } var err error err = cmd.Flags().ParseAll(os.Args[1:], parseFunc) if err != nil || !cmd.Flags().Parsed() { return "" } args := "" if arguments := cmd.Flags().Args(); len(arguments) > 0 { args = " " + strings.Join(arguments, " ") } base := filepath.Base(os.Args[0]) return base + args + flags } func (f *ring0Factory) BindFlags(flags *pflag.FlagSet) { // Merge factory's flags flags.AddFlagSet(f.flags) // Globally persistent flags across all subcommands. // TODO Change flag names to consts to allow safer lookup from subcommands. // TODO Add a verbose flag that turns on glog logging. Probably need a way // to do that automatically for every subcommand. flags.BoolVar(&f.clientCache.matchVersion, FlagMatchBinaryVersion, false, "Require server version to match client version") f.discoveryFactory.BindFlags(flags) // Normalize all flags that are coming from other packages or pre-configurations // a.k.a. change all "_" to "-". e.g. glog package flags.SetNormalizeFunc(utilflag.WordSepNormalizeFunc) } func (f *ring0Factory) BindExternalFlags(flags *pflag.FlagSet) { // any flags defined by external projects (not part of pflags) flags.AddGoFlagSet(flag.CommandLine) } func (f *ring0Factory) DefaultResourceFilterFunc() kubectl.Filters { return kubectl.NewResourceFilter() } func (f *ring0Factory) SuggestedPodTemplateResources() []schema.GroupResource { return []schema.GroupResource{ {Resource: "replicationcontroller"}, {Resource: "deployment"}, {Resource: "daemonset"}, {Resource: "job"}, {Resource: "replicaset"}, } } func (f *ring0Factory) Pauser(info *resource.Info) ([]byte, error) { switch obj := info.Object.(type) { case *extensions.Deployment: if obj.Spec.Paused { return nil, errors.New("is already paused") } obj.Spec.Paused = true return runtime.Encode(InternalVersionJSONEncoder(), info.Object) default: return nil, fmt.Errorf("pausing is not supported") } } func (f *ring0Factory) ResolveImage(name string) (string, error) { return name, nil } func (f *ring0Factory) Resumer(info *resource.Info) ([]byte, error) { switch obj := info.Object.(type) { case *extensions.Deployment: if !obj.Spec.Paused { return nil, errors.New("is not paused") } obj.Spec.Paused = false return runtime.Encode(InternalVersionJSONEncoder(), info.Object) default: return nil, fmt.Errorf("resuming is not supported") } } func (f *ring0Factory) DefaultNamespace() (string, bool, error) { return f.clientConfig.Namespace() } const ( // TODO(sig-cli): Enforce consistent naming for generators here. // See discussion in https://github.com/kubernetes/kubernetes/issues/46237 // before you add any more. RunV1GeneratorName = "run/v1" RunPodV1GeneratorName = "run-pod/v1" ServiceV1GeneratorName = "service/v1" ServiceV2GeneratorName = "service/v2" ServiceNodePortGeneratorV1Name = "service-nodeport/v1" ServiceClusterIPGeneratorV1Name = "service-clusterip/v1" ServiceLoadBalancerGeneratorV1Name = "service-loadbalancer/v1" ServiceExternalNameGeneratorV1Name = "service-externalname/v1" ServiceAccountV1GeneratorName = "serviceaccount/v1" HorizontalPodAutoscalerV1GeneratorName = "horizontalpodautoscaler/v1" DeploymentV1Beta1GeneratorName = "deployment/v1beta1" DeploymentAppsV1Beta1GeneratorName = "deployment/apps.v1beta1" DeploymentBasicV1Beta1GeneratorName = "deployment-basic/v1beta1" DeploymentBasicAppsV1Beta1GeneratorName = "deployment-basic/apps.v1beta1" JobV1GeneratorName = "job/v1" CronJobV2Alpha1GeneratorName = "cronjob/v2alpha1" CronJobV1Beta1GeneratorName = "cronjob/v1beta1" NamespaceV1GeneratorName = "namespace/v1" ResourceQuotaV1GeneratorName = "resourcequotas/v1" SecretV1GeneratorName = "secret/v1" SecretForDockerRegistryV1GeneratorName = "secret-for-docker-registry/v1" SecretForTLSV1GeneratorName = "secret-for-tls/v1" ConfigMapV1GeneratorName = "configmap/v1" ClusterRoleBindingV1GeneratorName = "clusterrolebinding.rbac.authorization.k8s.io/v1alpha1" RoleBindingV1GeneratorName = "rolebinding.rbac.authorization.k8s.io/v1alpha1" ClusterV1Beta1GeneratorName = "cluster/v1beta1" PodDisruptionBudgetV1GeneratorName = "poddisruptionbudget/v1beta1" PodDisruptionBudgetV2GeneratorName = "poddisruptionbudget/v1beta1/v2" PriorityClassV1Alpha1GeneratorName = "priorityclass/v1alpha1" ) // DefaultGenerators returns the set of default generators for use in Factory instances func DefaultGenerators(cmdName string) map[string]kubectl.Generator { var generator map[string]kubectl.Generator switch cmdName { case "expose": generator = map[string]kubectl.Generator{ ServiceV1GeneratorName: kubectl.ServiceGeneratorV1{}, ServiceV2GeneratorName: kubectl.ServiceGeneratorV2{}, } case "service-clusterip": generator = map[string]kubectl.Generator{ ServiceClusterIPGeneratorV1Name: kubectl.ServiceClusterIPGeneratorV1{}, } case "service-nodeport": generator = map[string]kubectl.Generator{ ServiceNodePortGeneratorV1Name: kubectl.ServiceNodePortGeneratorV1{}, } case "service-loadbalancer": generator = map[string]kubectl.Generator{ ServiceLoadBalancerGeneratorV1Name: kubectl.ServiceLoadBalancerGeneratorV1{}, } case "deployment": // Create Deployment has only StructuredGenerators and no // param-based Generators. // The StructuredGenerators are as follows (as of 2017-07-17): // DeploymentBasicV1Beta1GeneratorName -> kubectl.DeploymentBasicGeneratorV1 // DeploymentBasicAppsV1Beta1GeneratorName -> kubectl.DeploymentBasicAppsGeneratorV1 generator = map[string]kubectl.Generator{} case "run": generator = map[string]kubectl.Generator{ RunV1GeneratorName: kubectl.BasicReplicationController{}, RunPodV1GeneratorName: kubectl.BasicPod{}, DeploymentV1Beta1GeneratorName: kubectl.DeploymentV1Beta1{}, DeploymentAppsV1Beta1GeneratorName: kubectl.DeploymentAppsV1Beta1{}, JobV1GeneratorName: kubectl.JobV1{}, CronJobV2Alpha1GeneratorName: kubectl.CronJobV2Alpha1{}, CronJobV1Beta1GeneratorName: kubectl.CronJobV1Beta1{}, } case "namespace": generator = map[string]kubectl.Generator{ NamespaceV1GeneratorName: kubectl.NamespaceGeneratorV1{}, } case "quota": generator = map[string]kubectl.Generator{ ResourceQuotaV1GeneratorName: kubectl.ResourceQuotaGeneratorV1{}, } case "secret": generator = map[string]kubectl.Generator{ SecretV1GeneratorName: kubectl.SecretGeneratorV1{}, } case "secret-for-docker-registry": generator = map[string]kubectl.Generator{ SecretForDockerRegistryV1GeneratorName: kubectl.SecretForDockerRegistryGeneratorV1{}, } case "secret-for-tls": generator = map[string]kubectl.Generator{ SecretForTLSV1GeneratorName: kubectl.SecretForTLSGeneratorV1{}, } } return generator } // fallbackGeneratorNameIfNecessary returns the name of the old generator // if server does not support new generator. Otherwise, the // generator string is returned unchanged. // // If the generator name is changed, print a warning message to let the user // know. func FallbackGeneratorNameIfNecessary( generatorName string, discoveryClient discovery.DiscoveryInterface, cmdErr io.Writer, ) (string, error) { switch generatorName { case DeploymentBasicAppsV1Beta1GeneratorName: hasResource, err := HasResource(discoveryClient, appsv1beta1.SchemeGroupVersion.WithResource("deployments")) if err != nil { return "", err } if !hasResource { warning(cmdErr, DeploymentBasicAppsV1Beta1GeneratorName, DeploymentBasicV1Beta1GeneratorName) return DeploymentBasicV1Beta1GeneratorName, nil } case CronJobV2Alpha1GeneratorName: hasResource, err := HasResource(discoveryClient, batchv2alpha1.SchemeGroupVersion.WithResource("cronjobs")) if err != nil { return "", err } if !hasResource { warning(cmdErr, CronJobV2Alpha1GeneratorName, JobV1GeneratorName) return JobV1GeneratorName, nil } } return generatorName, nil } func warning(cmdErr io.Writer, newGeneratorName, oldGeneratorName string) { fmt.Fprintf(cmdErr, "WARNING: New deployments generator %q specified, "+ "but it isn't available. "+ "Falling back to %q.\n", newGeneratorName, oldGeneratorName, ) } func HasResource(client discovery.DiscoveryInterface, resource schema.GroupVersionResource) (bool, error) { resources, err := client.ServerResourcesForGroupVersion(resource.GroupVersion().String()) if apierrors.IsNotFound(err) { // entire group is missing return false, nil } if err != nil { // other errors error return false, fmt.Errorf("failed to discover supported resources: %v", err) } for _, serverResource := range resources.APIResources { if serverResource.Name == resource.Resource { return true, nil } } return false, nil } func Contains(resourcesList []*metav1.APIResourceList, resource schema.GroupVersionResource) bool { resources := discovery.FilteredBy(discovery.ResourcePredicateFunc(func(gv string, r *metav1.APIResource) bool { return resource.GroupVersion().String() == gv && resource.Resource == r.Name }), resourcesList) return len(resources) != 0 } func (f *ring0Factory) Generators(cmdName string) map[string]kubectl.Generator { return DefaultGenerators(cmdName) } func (f *ring0Factory) CanBeExposed(kind schema.GroupKind) error { switch kind { case api.Kind("ReplicationController"), api.Kind("Service"), api.Kind("Pod"), extensions.Kind("Deployment"), apps.Kind("Deployment"), extensions.Kind("ReplicaSet"), apps.Kind("ReplicaSet"): // nothing to do here default: return fmt.Errorf("cannot expose a %s", kind) } return nil } func (f *ring0Factory) CanBeAutoscaled(kind schema.GroupKind) error { switch kind { case api.Kind("ReplicationController"), extensions.Kind("ReplicaSet"), extensions.Kind("Deployment"), apps.Kind("Deployment"), apps.Kind("ReplicaSet"): // nothing to do here default: return fmt.Errorf("cannot autoscale a %v", kind) } return nil } func (f *ring0Factory) EditorEnvs() []string { return []string{"KUBE_EDITOR", "EDITOR"} } // overlyCautiousIllegalFileCharacters matches characters that *might* not be supported. Windows is really restrictive, so this is really restrictive var overlyCautiousIllegalFileCharacters = regexp.MustCompile(`[^(\w/\.)]`) // computeDiscoverCacheDir takes the parentDir and the host and comes up with a "usually non-colliding" name. func computeDiscoverCacheDir(parentDir, host string) string { // strip the optional scheme from host if its there: schemelessHost := strings.Replace(strings.Replace(host, "https://", "", 1), "http://", "", 1) // now do a simple collapse of non-AZ09 characters. Collisions are possible but unlikely. Even if we do collide the problem is short lived safeHost := overlyCautiousIllegalFileCharacters.ReplaceAllString(schemelessHost, "_") return filepath.Join(parentDir, safeHost) } // this method exists to help us find the points still relying on internal types. func InternalVersionDecoder() runtime.Decoder { return legacyscheme.Codecs.UniversalDecoder() } func InternalVersionJSONEncoder() runtime.Encoder { return legacyscheme.Codecs.LegacyCodec(legacyscheme.Registry.EnabledVersions()...) }