remove metrics client factory method

This commit is contained in:
David Eads 2018-02-21 11:01:09 -05:00
parent c153aff99f
commit f084311326
11 changed files with 145 additions and 136 deletions

View File

@ -40,7 +40,6 @@ go_library(
"//vendor/k8s.io/client-go/kubernetes:go_default_library", "//vendor/k8s.io/client-go/kubernetes:go_default_library",
"//vendor/k8s.io/client-go/rest:go_default_library", "//vendor/k8s.io/client-go/rest:go_default_library",
"//vendor/k8s.io/client-go/rest/fake:go_default_library", "//vendor/k8s.io/client-go/rest/fake:go_default_library",
"//vendor/k8s.io/metrics/pkg/client/clientset_generated/clientset:go_default_library",
], ],
) )

View File

@ -49,7 +49,6 @@ import (
"k8s.io/kubernetes/pkg/kubectl/scheme" "k8s.io/kubernetes/pkg/kubectl/scheme"
"k8s.io/kubernetes/pkg/kubectl/validation" "k8s.io/kubernetes/pkg/kubectl/validation"
"k8s.io/kubernetes/pkg/printers" "k8s.io/kubernetes/pkg/printers"
metricsclientset "k8s.io/metrics/pkg/client/clientset_generated/clientset"
) )
// +k8s:deepcopy-gen=true // +k8s:deepcopy-gen=true
@ -245,7 +244,6 @@ type TestFactory struct {
Command string Command string
TmpDir string TmpDir string
CategoryExpander categories.CategoryExpander CategoryExpander categories.CategoryExpander
MetricsClientSet metricsclientset.Interface
ClientForMappingFunc func(mapping *meta.RESTMapping) (resource.RESTClient, error) ClientForMappingFunc func(mapping *meta.RESTMapping) (resource.RESTClient, error)
UnstructuredClientForMappingFunc func(mapping *meta.RESTMapping) (resource.RESTClient, error) UnstructuredClientForMappingFunc func(mapping *meta.RESTMapping) (resource.RESTClient, error)
@ -299,10 +297,6 @@ func (f *FakeFactory) KubernetesClientSet() (*kubernetes.Clientset, error) {
return nil, nil return nil, nil
} }
func (f *FakeFactory) MetricsClientSet() (metricsclientset.Interface, error) {
return f.tf.MetricsClientSet, f.tf.Err
}
func (f *FakeFactory) ClientSet() (internalclientset.Interface, error) { func (f *FakeFactory) ClientSet() (internalclientset.Interface, error) {
return nil, nil return nil, nil
} }
@ -555,10 +549,6 @@ func (f *fakeAPIFactory) KubernetesClientSet() (*kubernetes.Clientset, error) {
return clientset, f.tf.Err return clientset, f.tf.Err
} }
func (f *fakeAPIFactory) MetricsClientSet() (metricsclientset.Interface, error) {
return f.tf.MetricsClientSet, f.tf.Err
}
func (f *fakeAPIFactory) ClientSet() (internalclientset.Interface, error) { func (f *fakeAPIFactory) ClientSet() (internalclientset.Interface, error) {
// Swap the HTTP client out of the REST client with the fake // Swap the HTTP client out of the REST client with the fake
// version. // version.

View File

@ -52,8 +52,10 @@ func NewCmdTop(f cmdutil.Factory, out, errOut io.Writer) *cobra.Command {
} }
// create subcommands // create subcommands
topPod, _ := NewCmdTopPod(f, nil, out)
cmd.AddCommand(NewCmdTopNode(f, nil, out)) cmd.AddCommand(NewCmdTopNode(f, nil, out))
cmd.AddCommand(NewCmdTopPod(f, nil, out)) cmd.AddCommand(topPod)
return cmd return cmd
} }

View File

@ -131,7 +131,12 @@ func (o *TopNodeOptions) Complete(f cmdutil.Factory, cmd *cobra.Command, args []
} }
o.DiscoveryClient = clientset.DiscoveryClient o.DiscoveryClient = clientset.DiscoveryClient
o.MetricsClient, err = f.MetricsClientSet()
config, err := f.ClientConfig()
if err != nil {
return err
}
o.MetricsClient, err = metricsclientset.NewForConfig(config)
if err != nil { if err != nil {
return err return err
} }

View File

@ -303,13 +303,25 @@ func TestTopNodeAllMetricsFromMetricsServer(t *testing.T) {
fakemetricsClientset.AddReactor("list", "nodes", func(action core.Action) (handled bool, ret runtime.Object, err error) { fakemetricsClientset.AddReactor("list", "nodes", func(action core.Action) (handled bool, ret runtime.Object, err error) {
return true, expectedMetrics, nil return true, expectedMetrics, nil
}) })
tf.MetricsClientSet = fakemetricsClientset
tf.Namespace = "test" tf.Namespace = "test"
tf.ClientConfig = defaultClientConfig() tf.ClientConfig = defaultClientConfig()
buf := bytes.NewBuffer([]byte{}) buf := bytes.NewBuffer([]byte{})
cmd := NewCmdTopNode(f, nil, buf) cmd := NewCmdTopNode(f, nil, buf)
cmd.Run(cmd, []string{})
// TODO in the long run, we want to test most of our commands like this. Wire the options struct with specific mocks
// TODO then check the particular Run functionality and harvest results from fake clients
cmdOptions := &TopNodeOptions{}
if err := cmdOptions.Complete(f, cmd, []string{}, buf); err != nil {
t.Fatal(err)
}
cmdOptions.MetricsClient = fakemetricsClientset
if err := cmdOptions.Validate(); err != nil {
t.Fatal(err)
}
if err := cmdOptions.RunTopNode(); err != nil {
t.Fatal(err)
}
// Check the presence of node names in the output. // Check the presence of node names in the output.
result := buf.String() result := buf.String()
@ -355,13 +367,25 @@ func TestTopNodeWithNameMetricsFromMetricsServer(t *testing.T) {
fakemetricsClientset.AddReactor("get", "nodes", func(action core.Action) (handled bool, ret runtime.Object, err error) { fakemetricsClientset.AddReactor("get", "nodes", func(action core.Action) (handled bool, ret runtime.Object, err error) {
return true, &expectedMetrics, nil return true, &expectedMetrics, nil
}) })
tf.MetricsClientSet = fakemetricsClientset
tf.Namespace = "test" tf.Namespace = "test"
tf.ClientConfig = defaultClientConfig() tf.ClientConfig = defaultClientConfig()
buf := bytes.NewBuffer([]byte{}) buf := bytes.NewBuffer([]byte{})
cmd := NewCmdTopNode(f, nil, buf) cmd := NewCmdTopNode(f, nil, buf)
cmd.Run(cmd, []string{expectedMetrics.Name})
// TODO in the long run, we want to test most of our commands like this. Wire the options struct with specific mocks
// TODO then check the particular Run functionality and harvest results from fake clients
cmdOptions := &TopNodeOptions{}
if err := cmdOptions.Complete(f, cmd, []string{expectedMetrics.Name}, buf); err != nil {
t.Fatal(err)
}
cmdOptions.MetricsClient = fakemetricsClientset
if err := cmdOptions.Validate(); err != nil {
t.Fatal(err)
}
if err := cmdOptions.RunTopNode(); err != nil {
t.Fatal(err)
}
// Check the presence of node names in the output. // Check the presence of node names in the output.
result := buf.String() result := buf.String()
@ -418,14 +442,26 @@ func TestTopNodeWithLabelSelectorMetricsFromMetricsServer(t *testing.T) {
fakemetricsClientset.AddReactor("list", "nodes", func(action core.Action) (handled bool, ret runtime.Object, err error) { fakemetricsClientset.AddReactor("list", "nodes", func(action core.Action) (handled bool, ret runtime.Object, err error) {
return true, expectedMetrics, nil return true, expectedMetrics, nil
}) })
tf.MetricsClientSet = fakemetricsClientset
tf.Namespace = "test" tf.Namespace = "test"
tf.ClientConfig = defaultClientConfig() tf.ClientConfig = defaultClientConfig()
buf := bytes.NewBuffer([]byte{}) buf := bytes.NewBuffer([]byte{})
cmd := NewCmdTopNode(f, nil, buf) cmd := NewCmdTopNode(f, nil, buf)
cmd.Flags().Set("selector", label) cmd.Flags().Set("selector", label)
cmd.Run(cmd, []string{})
// TODO in the long run, we want to test most of our commands like this. Wire the options struct with specific mocks
// TODO then check the particular Run functionality and harvest results from fake clients
cmdOptions := &TopNodeOptions{}
if err := cmdOptions.Complete(f, cmd, []string{}, buf); err != nil {
t.Fatal(err)
}
cmdOptions.MetricsClient = fakemetricsClientset
if err := cmdOptions.Validate(); err != nil {
t.Fatal(err)
}
if err := cmdOptions.RunTopNode(); err != nil {
t.Fatal(err)
}
// Check the presence of node names in the output. // Check the presence of node names in the output.
result := buf.String() result := buf.String()

View File

@ -78,7 +78,7 @@ var (
kubectl top pod -l name=myLabel`)) kubectl top pod -l name=myLabel`))
) )
func NewCmdTopPod(f cmdutil.Factory, options *TopPodOptions, out io.Writer) *cobra.Command { func NewCmdTopPod(f cmdutil.Factory, options *TopPodOptions, out io.Writer) (*cobra.Command, *TopPodOptions) {
if options == nil { if options == nil {
options = &TopPodOptions{} options = &TopPodOptions{}
} }
@ -106,7 +106,7 @@ func NewCmdTopPod(f cmdutil.Factory, options *TopPodOptions, out io.Writer) *cob
cmd.Flags().BoolVar(&options.PrintContainers, "containers", false, "If present, print usage of containers within a pod.") cmd.Flags().BoolVar(&options.PrintContainers, "containers", false, "If present, print usage of containers within a pod.")
cmd.Flags().BoolVar(&options.AllNamespaces, "all-namespaces", false, "If present, list the requested object(s) across all namespaces. Namespace in current context is ignored even if specified with --namespace.") cmd.Flags().BoolVar(&options.AllNamespaces, "all-namespaces", false, "If present, list the requested object(s) across all namespaces. Namespace in current context is ignored even if specified with --namespace.")
options.HeapsterOptions.Bind(cmd.Flags()) options.HeapsterOptions.Bind(cmd.Flags())
return cmd return cmd, options
} }
func (o *TopPodOptions) Complete(f cmdutil.Factory, cmd *cobra.Command, args []string, out io.Writer) error { func (o *TopPodOptions) Complete(f cmdutil.Factory, cmd *cobra.Command, args []string, out io.Writer) error {
@ -127,7 +127,11 @@ func (o *TopPodOptions) Complete(f cmdutil.Factory, cmd *cobra.Command, args []s
} }
o.DiscoveryClient = clientset.DiscoveryClient o.DiscoveryClient = clientset.DiscoveryClient
o.MetricsClient, err = f.MetricsClientSet() config, err := f.ClientConfig()
if err != nil {
return err
}
o.MetricsClient, err = metricsclientset.NewForConfig(config)
if err != nil { if err != nil {
return err return err
} }

View File

@ -190,7 +190,7 @@ func TestTopPod(t *testing.T) {
tf.ClientConfig = defaultClientConfig() tf.ClientConfig = defaultClientConfig()
buf := bytes.NewBuffer([]byte{}) buf := bytes.NewBuffer([]byte{})
cmd := NewCmdTopPod(f, nil, buf) cmd, _ := NewCmdTopPod(f, nil, buf)
for name, value := range testCase.flags { for name, value := range testCase.flags {
cmd.Flags().Set(name, value) cmd.Flags().Set(name, value)
} }
@ -270,91 +270,103 @@ func TestTopPodWithMetricsServer(t *testing.T) {
} }
initTestErrorHandler(t) initTestErrorHandler(t)
for _, testCase := range testCases { for _, testCase := range testCases {
t.Logf("Running test case: %s", testCase.name) t.Run(testCase.name, func(t *testing.T) {
metricsList := testV1beta1PodMetricsData() metricsList := testV1beta1PodMetricsData()
var expectedMetrics []metricsv1beta1api.PodMetrics var expectedMetrics []metricsv1beta1api.PodMetrics
var expectedContainerNames, nonExpectedMetricsNames []string var expectedContainerNames, nonExpectedMetricsNames []string
for n, m := range metricsList { for n, m := range metricsList {
if n < len(testCase.namespaces) { if n < len(testCase.namespaces) {
m.Namespace = testCase.namespaces[n] m.Namespace = testCase.namespaces[n]
expectedMetrics = append(expectedMetrics, m) expectedMetrics = append(expectedMetrics, m)
for _, c := range m.Containers { for _, c := range m.Containers {
expectedContainerNames = append(expectedContainerNames, c.Name) expectedContainerNames = append(expectedContainerNames, c.Name)
}
} else {
nonExpectedMetricsNames = append(nonExpectedMetricsNames, m.Name)
} }
}
fakemetricsClientset := &metricsfake.Clientset{}
if len(expectedMetrics) == 1 {
fakemetricsClientset.AddReactor("get", "pods", func(action core.Action) (handled bool, ret runtime.Object, err error) {
return true, &expectedMetrics[0], nil
})
} else { } else {
nonExpectedMetricsNames = append(nonExpectedMetricsNames, m.Name) fakemetricsClientset.AddReactor("list", "pods", func(action core.Action) (handled bool, ret runtime.Object, err error) {
res := &metricsv1beta1api.PodMetricsList{
ListMeta: metav1.ListMeta{
ResourceVersion: "2",
},
Items: expectedMetrics,
}
return true, res, nil
})
} }
}
fakemetricsClientset := &metricsfake.Clientset{} f, tf := cmdtesting.NewAPIFactory()
ns := legacyscheme.Codecs
if len(expectedMetrics) == 1 { tf.Client = &fake.RESTClient{
fakemetricsClientset.AddReactor("get", "pods", func(action core.Action) (handled bool, ret runtime.Object, err error) { NegotiatedSerializer: ns,
return true, &expectedMetrics[0], nil Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) {
}) switch p := req.URL.Path; {
} else { case p == "/api":
fakemetricsClientset.AddReactor("list", "pods", func(action core.Action) (handled bool, ret runtime.Object, err error) { return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: ioutil.NopCloser(bytes.NewReader([]byte(apibody)))}, nil
res := &metricsv1beta1api.PodMetricsList{ case p == "/apis":
ListMeta: metav1.ListMeta{ return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: ioutil.NopCloser(bytes.NewReader([]byte(apisbodyWithMetrics)))}, nil
ResourceVersion: "2", default:
}, t.Fatalf("%s: unexpected request: %#v\nGot URL: %#v",
Items: expectedMetrics, testCase.name, req, req.URL)
} return nil, nil
return true, res, nil }
}) }),
} }
tf.Namespace = testNS
tf.ClientConfig = defaultClientConfig()
buf := bytes.NewBuffer([]byte{})
f, tf := cmdtesting.NewAPIFactory() cmd, cmdOptions := NewCmdTopPod(f, nil, buf)
ns := legacyscheme.Codecs for name, value := range testCase.flags {
cmd.Flags().Set(name, value)
}
tf.Client = &fake.RESTClient{ // TODO in the long run, we want to test most of our commands like this. Wire the options struct with specific mocks
NegotiatedSerializer: ns, // TODO then check the particular Run functionality and harvest results from fake clients. We probably end up skipping the factory altogether.
Client: fake.CreateHTTPClient(func(req *http.Request) (*http.Response, error) { if err := cmdOptions.Complete(f, cmd, testCase.args, buf); err != nil {
switch p := req.URL.Path; { t.Fatal(err)
case p == "/api": }
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: ioutil.NopCloser(bytes.NewReader([]byte(apibody)))}, nil cmdOptions.MetricsClient = fakemetricsClientset
case p == "/apis": if err := cmdOptions.Validate(); err != nil {
return &http.Response{StatusCode: 200, Header: defaultHeader(), Body: ioutil.NopCloser(bytes.NewReader([]byte(apisbodyWithMetrics)))}, nil t.Fatal(err)
default: }
t.Fatalf("%s: unexpected request: %#v\nGot URL: %#v", if err := cmdOptions.RunTopPod(); err != nil {
testCase.name, req, req.URL) t.Fatal(err)
return nil, nil }
}
}),
}
tf.MetricsClientSet = fakemetricsClientset
tf.Namespace = testNS
tf.ClientConfig = defaultClientConfig()
buf := bytes.NewBuffer([]byte{})
cmd := NewCmdTopPod(f, nil, buf) // Check the presence of pod names&namespaces/container names in the output.
for name, value := range testCase.flags { result := buf.String()
cmd.Flags().Set(name, value) if testCase.containers {
} for _, containerName := range expectedContainerNames {
cmd.Run(cmd, testCase.args) if !strings.Contains(result, containerName) {
t.Errorf("missing metrics for container %s: \n%s", containerName, result)
// Check the presence of pod names&namespaces/container names in the output. }
result := buf.String()
if testCase.containers {
for _, containerName := range expectedContainerNames {
if !strings.Contains(result, containerName) {
t.Errorf("%s: missing metrics for container %s: \n%s", testCase.name, containerName, result)
} }
} }
} for _, m := range expectedMetrics {
for _, m := range expectedMetrics { if !strings.Contains(result, m.Name) {
if !strings.Contains(result, m.Name) { t.Errorf("missing metrics for %s: \n%s", m.Name, result)
t.Errorf("%s: missing metrics for %s: \n%s", testCase.name, m.Name, result) }
if testCase.listsNamespaces && !strings.Contains(result, m.Namespace) {
t.Errorf("missing metrics for %s/%s: \n%s", m.Namespace, m.Name, result)
}
} }
if testCase.listsNamespaces && !strings.Contains(result, m.Namespace) { for _, name := range nonExpectedMetricsNames {
t.Errorf("%s: missing metrics for %s/%s: \n%s", testCase.name, m.Namespace, m.Name, result) if strings.Contains(result, name) {
t.Errorf("unexpected metrics for %s: \n%s", name, result)
}
} }
} })
for _, name := range nonExpectedMetricsNames {
if strings.Contains(result, name) {
t.Errorf("%s: unexpected metrics for %s: \n%s", testCase.name, name, result)
}
}
} }
} }
@ -520,7 +532,7 @@ func TestTopPodCustomDefaults(t *testing.T) {
}, },
DiscoveryClient: &fakeDiscovery{}, DiscoveryClient: &fakeDiscovery{},
} }
cmd := NewCmdTopPod(f, opts, buf) cmd, _ := NewCmdTopPod(f, opts, buf)
for name, value := range testCase.flags { for name, value := range testCase.flags {
cmd.Flags().Set(name, value) cmd.Flags().Set(name, value)
} }

View File

@ -81,7 +81,6 @@ go_library(
"//vendor/k8s.io/client-go/scale:go_default_library", "//vendor/k8s.io/client-go/scale:go_default_library",
"//vendor/k8s.io/client-go/tools/clientcmd:go_default_library", "//vendor/k8s.io/client-go/tools/clientcmd:go_default_library",
"//vendor/k8s.io/client-go/util/homedir:go_default_library", "//vendor/k8s.io/client-go/util/homedir:go_default_library",
"//vendor/k8s.io/metrics/pkg/client/clientset_generated/clientset:go_default_library",
"//vendor/k8s.io/utils/exec:go_default_library", "//vendor/k8s.io/utils/exec:go_default_library",
], ],
) )

View File

@ -27,7 +27,6 @@ import (
"k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset" "k8s.io/kubernetes/pkg/client/clientset_generated/internalclientset"
oldclient "k8s.io/kubernetes/pkg/client/unversioned" oldclient "k8s.io/kubernetes/pkg/client/unversioned"
"k8s.io/kubernetes/pkg/version" "k8s.io/kubernetes/pkg/version"
metricsclientset "k8s.io/metrics/pkg/client/clientset_generated/clientset"
) )
func NewClientCache(loader clientcmd.ClientConfig, discoveryClientFactory DiscoveryClientFactory) *ClientCache { func NewClientCache(loader clientcmd.ClientConfig, discoveryClientFactory DiscoveryClientFactory) *ClientCache {
@ -59,7 +58,6 @@ type ClientCache struct {
discoveryClient discovery.DiscoveryInterface discoveryClient discovery.DiscoveryInterface
kubernetesClientCache kubernetesClientCache kubernetesClientCache kubernetesClientCache
metricsClientCache metricsClientCache
} }
// kubernetesClientCache creates a new kubernetes.Clientset one time // kubernetesClientCache creates a new kubernetes.Clientset one time
@ -73,17 +71,6 @@ type kubernetesClientCache struct {
err error err error
} }
// metricsClientCache creates a new metricsclientset.Clientset one time
// and then returns the result for all future requests
type metricsClientCache struct {
// once makes sure the client is only initialized once
once sync.Once
// client is the cached client value
client *metricsclientset.Clientset
// err is the cached error value
err error
}
// KubernetesClientSetForVersion returns a new kubernetes.Clientset. It will cache the value // KubernetesClientSetForVersion returns a new kubernetes.Clientset. It will cache the value
// the first time it is called and return the cached value on subsequent calls. // the first time it is called and return the cached value on subsequent calls.
// If an error is encountered the first time KubernetesClientSetForVersion is called, // If an error is encountered the first time KubernetesClientSetForVersion is called,
@ -100,22 +87,6 @@ func (c *ClientCache) KubernetesClientSetForVersion(requiredVersion *schema.Grou
return c.kubernetesClientCache.client, c.kubernetesClientCache.err return c.kubernetesClientCache.client, c.kubernetesClientCache.err
} }
// MetricsClientSetForVersion returns a new kubernetes.Clientset. It will cache the value
// the first time it is called and return the cached value on subsequent calls.
// If an error is encountered the first time MetircsClientSetForVersion is called,
// the error will be cached.
func (c *ClientCache) MetricsClientSetForVersion(requiredVersion *schema.GroupVersion) (*metricsclientset.Clientset, error) {
c.metricsClientCache.once.Do(func() {
config, err := c.ClientConfigForVersion(requiredVersion)
if err != nil {
c.kubernetesClientCache.err = err
return
}
c.metricsClientCache.client, c.metricsClientCache.err = metricsclientset.NewForConfig(config)
})
return c.metricsClientCache.client, c.metricsClientCache.err
}
// also looks up the discovery client. We can't do this during init because the flags won't have been set // also looks up the discovery client. We can't do this during init because the flags won't have been set
// because this is constructed pre-command execution before the command tree is // because this is constructed pre-command execution before the command tree is
// even set up. Requires the lock to already be acquired // even set up. Requires the lock to already be acquired

View File

@ -47,7 +47,6 @@ import (
"k8s.io/kubernetes/pkg/kubectl/resource" "k8s.io/kubernetes/pkg/kubectl/resource"
"k8s.io/kubernetes/pkg/kubectl/validation" "k8s.io/kubernetes/pkg/kubectl/validation"
"k8s.io/kubernetes/pkg/printers" "k8s.io/kubernetes/pkg/printers"
metricsclientset "k8s.io/metrics/pkg/client/clientset_generated/clientset"
) )
const ( const (
@ -95,9 +94,6 @@ type ClientAccessFactory interface {
// KubernetesClientSet gives you back an external clientset // KubernetesClientSet gives you back an external clientset
KubernetesClientSet() (*kubernetes.Clientset, error) KubernetesClientSet() (*kubernetes.Clientset, error)
// MetricsClientSet gives you back an external clientset for the metrics API
MetricsClientSet() (metricsclientset.Interface, error)
// Returns a RESTClient for accessing Kubernetes resources or an error. // Returns a RESTClient for accessing Kubernetes resources or an error.
RESTClient() (*restclient.RESTClient, error) RESTClient() (*restclient.RESTClient, error)
// Returns a client.Config for accessing the Kubernetes server. // Returns a client.Config for accessing the Kubernetes server.

View File

@ -31,7 +31,6 @@ import (
"time" "time"
"k8s.io/api/core/v1" "k8s.io/api/core/v1"
metricsclientset "k8s.io/metrics/pkg/client/clientset_generated/clientset"
"github.com/spf13/cobra" "github.com/spf13/cobra"
"github.com/spf13/pflag" "github.com/spf13/pflag"
@ -200,10 +199,6 @@ func (f *ring0Factory) KubernetesClientSet() (*kubernetes.Clientset, error) {
return f.clientCache.KubernetesClientSetForVersion(nil) return f.clientCache.KubernetesClientSetForVersion(nil)
} }
func (f *ring0Factory) MetricsClientSet() (metricsclientset.Interface, error) {
return f.clientCache.MetricsClientSetForVersion(nil)
}
func (f *ring0Factory) ClientSet() (internalclientset.Interface, error) { func (f *ring0Factory) ClientSet() (internalclientset.Interface, error) {
return f.clientCache.ClientSetForVersion(nil) return f.clientCache.ClientSetForVersion(nil)
} }