client-go的四种客户端

client-go提供了四种与Kubernetes APIServer交互的客户端:RESTClient、DiscoveryClient、ClientSet、DynamicClient。 经过学习之后,我感觉这个东西还是蛮不复杂的,以下是这四种客户端获取所有Pods的demo,然后感受一下他们的区别。

运行在下面的四个代码时,需要准备好kubeconfig,就是~/.kube/config这个文件,可以从服务器上拷到本地上。或者直接放在服务器上跑这些代码。

RESTClient

RESTClient是一个最基础的client,他是其他三种client的底层,也就是说,其他三种client都是封装的它。所以操作RESTClient会感觉比较原始。我们

go 复制代码
import (
    "context"
    "flag"
    "fmt"
    corev1 "k8s.io/api/core/v1"
    "k8s.io/client-go/kubernetes/scheme"
    "k8s.io/client-go/rest"
    "k8s.io/client-go/tools/clientcmd"
    "k8s.io/client-go/util/homedir"
    "path/filepath"
)

func main() {
    var kubeconfig *string
    if home := homedir.HomeDir(); home != "" {
       kubeconfig = flag.String("kubeconfig", filepath.Join(home, ".kube", "config"), "(optional) absolute path to the kubeconfig file")
    } else {
       kubeconfig = flag.String("kubeconfig", "", "absolute path to the kubeconfig file")
    }
    flag.Parse()

    config, err := clientcmd.BuildConfigFromFlags("", *kubeconfig)
    if err != nil {
       panic(err.Error())
    }

    config.APIPath = "api"
    // 设置 corev1 groupVersion
    config.GroupVersion = &corev1.SchemeGroupVersion
    // 设置解析器,用于用于解析 scheme
    config.NegotiatedSerializer = scheme.Codecs.WithoutConversion()
    // 初始化 RESTClient
    restClient, err := rest.RESTClientFor(config)
    if err != nil {
       panic(err)
    }
    list := &corev1.PodList{}
    // namespace为空,就返回所有namespace中的资源
    err = restClient.Get().Namespace("").Resource("pods").Do(context.TODO()).Into(list)
    if err != nil {
       panic(err)
    }

    for _, item := range list.Items {
       fmt.Printf("ns:%s , pod:%s\n", item.Namespace, item.Name)
    }
}

ClientSet

ClientSet他是用来操作K8s内置资源的,底层给每一种资源封了RESTClient,如果想操作CRD,需要自己定义ClientSet。

go 复制代码
func main() {
    var kubeconfig *string
    if home := homedir.HomeDir(); home != "" {
       kubeconfig = flag.String("kubeconfig", filepath.Join(home, ".kube", "config"), "(optional) absolute path to the kubeconfig file")
    } else {
       kubeconfig = flag.String("kubeconfig", "", "absolute path to the kubeconfig file")
    }
    flag.Parse()

    // 从kubeconfig中获取配置
    config, err := clientcmd.BuildConfigFromFlags("", *kubeconfig)
    if err != nil {
       panic(err.Error())
    }

    // 从配置中获得clientset
    clientset, err := kubernetes.NewForConfig(config)
    if err != nil {
       panic(err.Error())
    }
    pods, err := clientset.CoreV1().Pods("").List(context.TODO(), metav1.ListOptions{})
    if err != nil {
       panic(err.Error())
    }
    fmt.Printf("There are %d pods in the cluster\n", len(pods.Items))

    for _, item := range pods.Items {
       fmt.Printf("ns:%s , pod:%s\n", item.Namespace, item.Name)
    }
}

其实最后来时落到了这里,所以他是帮我们把一些操作封了一下。

go 复制代码
func (c *pods) List(ctx context.Context, opts metav1.ListOptions) (result *v1.PodList, err error) {
    var timeout time.Duration
    if opts.TimeoutSeconds != nil {
       timeout = time.Duration(*opts.TimeoutSeconds) * time.Second
    }
    result = &v1.PodList{}
    err = c.client.Get().
       Namespace(c.ns).
       Resource("pods").
       VersionedParams(&opts, scheme.ParameterCodec).
       Timeout(timeout).
       Do(ctx).
       Into(result)
    return
}

DynamicClient

动态client的就是比较动态,然后他可以操作CRD,但是他返回的内容是map

go 复制代码
func main() {
    var kubeconfig *string
    if home := homedir.HomeDir(); home != "" {
       kubeconfig = flag.String("kubeconfig", filepath.Join(home, ".kube", "config"), "(optional) absolute path to the kubeconfig file")
    } else {
       kubeconfig = flag.String("kubeconfig", "", "absolute path to the kubeconfig file")
    }
    flag.Parse()

    // 从kubeconfig中获取配置
    config, err := clientcmd.BuildConfigFromFlags("", *kubeconfig)
    if err != nil {
       panic(err.Error())
    }

    dynamicClient, err := dynamic.NewForConfig(config)
    if err != nil {
       panic(err.Error())
    }

    // 因为是动态调用,不需要操作哪个资源,所以需要自己提供
    gvr := schema.GroupVersionResource{
       Group:    "",
       Version:  "v1",
       Resource: "pods",
    }

    result, err := dynamicClient.Resource(gvr).Namespace("").List(context.TODO(), meta_v1.ListOptions{})
    podList := &corev1.PodList{}
    // 将结果解析到 podList scheme 中
    err = runtime.DefaultUnstructuredConverter.FromUnstructured(
       result.UnstructuredContent(), podList)

    for _, item := range podList.Items {
       fmt.Printf("ns: %s,pod: %s\n", item.Namespace, item.Name)
    }
}

我们这里把的结构化数据转为了结构化数据。

DiscoveryClient

发现客户端是用来用于发现 Kube-apiserver 支持的资源组、资源版本、资源类型等。 通过DiscoveryClient发现所有gvr

go 复制代码
func main() {
    var kubeconfig *string
    if home := homedir.HomeDir(); home != "" {
       kubeconfig = flag.String("kubeconfig", filepath.Join(home, ".kube", "config"), "(optional) absolute path to the kubeconfig file")
    } else {
       kubeconfig = flag.String("kubeconfig", "", "absolute path to the kubeconfig file")
    }
    flag.Parse()

    // 从kubeconfig中获取配置
    config, err := clientcmd.BuildConfigFromFlags("", *kubeconfig)
    if err != nil {
       panic(err.Error())
    }

    // 初始化 DiscoveryClient
    discoveryClient, err := discovery.NewDiscoveryClientForConfig(config)
    if err != nil {
       panic(err.Error())
    }
    // 获取集群所有资源
    _, apiResourceList, err := discoveryClient.ServerGroupsAndResources()
    if err != nil {
       panic(err.Error())
    }

    for _, resources := range apiResourceList {
       gv, err := schema.ParseGroupVersion(resources.GroupVersion)
       if err != nil {
          panic(err.Error())
       }
       for _, resource := range resources.APIResources {
          fmt.Printf("group: %s, version: %s, resource: %s\n", gv.Group, gv.Version, resource.Name)
       }

    }
}
相关推荐
To_再飞行8 小时前
K8s访问控制(二)
linux·网络·云原生·容器·kubernetes
xy_recording11 小时前
学习番外:Docker和K8S理解
学习·docker·kubernetes
虚伪的空想家12 小时前
K8S的dashboard部署与访问
云原生·容器·kubernetes·k8s·web·dashboard
BBluster12 小时前
Kubernetes(K8S)入门以及命令指南
云原生·容器·kubernetes
走上未曾设想的道路12 小时前
gitlab流水线与k8s集群的联通
kubernetes·gitlab
橙*^O^*安20 小时前
Go 语言基础:变量与常量
运维·开发语言·后端·golang·kubernetes
_Walli_1 天前
k8s集群搭建(三)-------- Dashboard UI
云原生·容器·kubernetes
橙*^O^*安1 天前
Kubernetes集群部署Jenkins指南
云原生·容器·kubernetes·jenkins·devops
pwj去战斗吧1 天前
k8s+jenkins+harbor构建Devops平台
kubernetes·jenkins·devops
ChaITSimpleLove1 天前
零代码入侵:Kubernetes 部署时自动注入 kube-system UID 到 .NET 9 环境变量
kubernetes·.net·环境变量·uid·kube-system·集群环境唯一id