• KubeCube 新增版本转换:K8s 尝鲜再也不用担心影响老版本了


    多租户可视化 K8s 管理平台 KubeCube近日迎来了新版本的发布,新版本增加了 K8s 版本转化、HNC GA 版本适配、审计信息国际化、warden 主动上报模式,为集群和项目设置 Ingress 域名后缀等特性,也修复了若干已知问题,详见 ChangeLog。

    该版本中最主要的特性是 Version-Conversion 能力的支持,使得接入 KubeCube 的用户无需感知被 KubeCube 接管的 K8s 集群版本,可以使用指定版本的 K8s API 来操作 K8s 资源,KubeCube 会做自适应转化;同时 KubeCube 也将这个能力包装成 SDK 供外部使用。

    为什么需要多 K8s 版本转化?

    在实际的生产场景中,用户的 K8s 集群往往固置于某一稳定版本,并随着时间的推移,在该 K8s 集群中沉淀了大量的业务、工具、方案等,同时 K8s 社区又会不断的推出更高的版本,此时升级 K8s 版本往往需要比较高的代价。

    K8s 的版本升级,并不总是保证 API 的完美兼容,绝大多数的 API 会经历从 Development level --> Alpha level --> Beta level --> Stable level 的发展阶段,理想情况下,用户应该使用 Stable level 的 API 用于生产环境,但是现实中,用户所使用的某一资源的 API 很可能处于 Stable level 以下的阶段,比如 extensions/v1beta1 的 Deployment 和 apps/v1 的 Deployment。详见 K8S API 变动规划

    当用户需要在控制面纳管多 K8s 集群时,用户暂时不希望升级老的稳定的 K8s 集群,又希望新增的 K8s 集群是比较高的版本,这时,管控面的 KubeCube,就能够提供访问多版本 K8s 的能力,对外暴露统一的 K8s 风格的 RESTfule API,用户既可以使用精确的 GVR 去访问不同版本的 K8s 资源,也可用使用统一版本的 GVR 去访问不同版本的 K8s 资源,KubeCube 会做自适应转化。

    K8s native convert

    1. K8s api workflow

    2. K8s version convert

     

    K8s 版本转换原则

    1. 同一个 group 的不同 version 都可以转换成该 group 的 internalVersion
    2. 某一 group 的 internalVersion 可以转换成该 group 下的任一 version

    K8s 版本转化的核心 ——scheme

    Scheme 中拥有 concerter 转化器,其内部存放了各个 API 注册的版本转化函数。

    1. type Scheme struct {
    2. ...
    3. // converter stores all registered conversion functions. It also has
    4. // default converting behavior.
    5. converter *conversion.Converter
    6. ...
    7. }
    8. // Converter knows how to convert one type to another.
    9. type Converter struct {
    10. // Map from the conversion pair to a function which can
    11. // do the conversion.
    12. conversionFuncs ConversionFuncs
    13. generatedConversionFuncs ConversionFuncs
    14. // Set of conversions that should be treated as a no-op
    15. ignoredUntypedConversions map[typePair]struct{}
    16. }

    我们已经知道 internalVersion 和指定 version 之间的转换规则,它们的转换函数位于 k8s apis 的定义文件夹下,如:pkg/apis/apps/v1/zz_generated.conversion.go

    1. // Code generated by conversion-gen. DO NOT EDIT.
    2. package v1
    3. import (...)
    4. func init() {
    5. localSchemeBuilder.Register(RegisterConversions)
    6. }
    7. // RegisterConversions adds conversion functions to the given scheme.
    8. // Public to allow building arbitrary schemes.
    9. func RegisterConversions(s *runtime.Scheme) error {...}

    这些转化函数,一般由 install 包下的 Install(scheme) 函数注册到 Scheme 中。

    1. // Package install installs the apps API group, making it available as
    2. // an option to all of the API encoding/decoding machinery.
    3. package install
    4. import (...)
    5. func init() {
    6. Install(legacyscheme.Scheme)
    7. }
    8. // Install registers the API group and adds types to a scheme
    9. func Install(scheme *runtime.Scheme) {
    10. utilruntime.Must(apps.AddToScheme(scheme))
    11. utilruntime.Must(v1beta1.AddToScheme(scheme))
    12. utilruntime.Must(v1beta2.AddToScheme(scheme))
    13. utilruntime.Must(v1.AddToScheme(scheme))
    14. utilruntime.Must(scheme.SetVersionPriority(v1.SchemeGroupVersion, v1beta2.SchemeGroupVersion, v1beta1.SchemeGroupVersion))
    15. }

    注册完的转会函数,将会在 Convert () 方法中使用。

    1. // Convert will translate src to dest if it knows how. Both must be pointers.
    2. // If no conversion func is registered and the default copying mechanism
    3. // doesn't work on this type pair, an error will be returned.
    4. // 'meta' is given to allow you to pass information to conversion functions,
    5. // it is not used by Convert() other than storing it in the scope.
    6. // Not safe for objects with cyclic references!
    7. func (c *Converter) Convert(src, dest interface{}, meta *Meta) error {
    8. // 转换函数 map 的 key
    9. pair := typePair{reflect.TypeOf(src), reflect.TypeOf(dest)}
    10. // 实际的 convert 句柄
    11. scope := &scope{
    12. converter: c,
    13. meta: meta,
    14. }
    15. // ignore conversions of this type
    16. if _, ok := c.ignoredUntypedConversions[pair]; ok {
    17. return nil
    18. }
    19. // 使用预先注册的转换函数进行转换
    20. if fn, ok := c.conversionFuncs.untyped[pair]; ok {
    21. return fn(src, dest, scope)
    22. }
    23. if fn, ok := c.generatedConversionFuncs.untyped[pair]; ok {
    24. return fn(src, dest, scope)
    25. }
    26. dv, err := EnforcePtr(dest)
    27. if err != nil {
    28. return err
    29. }
    30. sv, err := EnforcePtr(src)
    31. if err != nil {
    32. return err
    33. }
    34. return fmt.Errorf("converting (%s) to (%s): unknown conversion", sv.Type(), dv.Type())
    35. }

    KubeCube version conversion

    了解了 K8s 版本转换的大致思路后,KubeCube 如果需要做版本转换的能力,需要做到以下几点:

    1. 维护版本转换专用的 Scheme
    2. 注册所有的 K8s 的 API 转换函数,并提供拓展方法
    3. 使用 discovery client 提早做 src api 和 dest api 的转换检查

    1. Conversion func register

    KubeCube 会默认注册所有 K8s 原生资源的转换函数,同时也提供注册自定义资源转换函数的入口。

    2. Greeting target cluster

    3. Controller-runtime client support

    KubeCube 的版本转化 SDK 提供了 Wrap controller-runtime 的 client.Client 的能力,可以将 client.Client 升级为具有版本转化能力的句柄。

    写在最后

    未来我们会持续提供更多功能,帮助企业简化容器化落地。也欢迎大家参与贡献,提出宝贵的建议。

    【了解网易技术干货、活动】

  • 相关阅读:
    Python数据分析与挖掘————图像的处理
    网格(grid)布局
    JavaScript之DOM常用属性及方法详解
    手把手教你用AirtestIDE无线连接手机
    Maven进阶-分模块开发与设计
    LLVM系列第二十九章:写一个简单的常量加法“消除”工具(Pass)
    华为云云服务器评测|详解 Nacos 安装部署
    Oracle-Rman duplicate文件坏块问题处理ORA-19849 19612
    VisualSVN 是 Visual Studio 的专业级 Subversion 集成插件
    14:00面试,14:06就出来了,问的问题有点变态。。。
  • 原文地址:https://blog.csdn.net/wangyiyungw/article/details/127430874