k8s源码学习-资源转换器

网友投稿 519 2022-09-12

k8s源码学习-资源转换器

在Kubernetes系统中,同一资源拥有多个资源版本,Kubernetes系统允许同一资源的不同资源版本进行转换,例如Deployment资源对象,当前运行的是v1beta1资源版本,但v1beta1资源版本的某些功能或字段不如v1资源版本完善,则可以将Deployment资源对象的v1beta1资源版本转换为v1版本。可通过kubectl convert命令进行资源版本转换

kubectl convert 默认没有安装,安装方法参考:​​安装 kubectl convert 插件​​

比如:我定义了一个 v1beta1deploy.yaml

apiVersion: apps/v1beta1kind: Deploymentmetadata: name: nginx-deployment labels: app: nginxspec: replicas: 3 selector: matchLabels: app: nginx template: metadata: labels: app: nginx spec: containers: - name: nginx image: nginx:1.14.2 ports: - containerPort: 80

创建deployment报错了,我想把它转为v1版本再创建,命令:

kubectl convert -f v1beta1deploy.yaml --output-version=apps/v1 > /tmp/v1deploy.yaml

定义一个YAML Manifest File资源描述文件v1beta1deploy.yaml,该文件中定义Deployment资源版本为v1beta1。通过执行kubect convert命令,--output-version将资源版本转换为指定的资源版本v1。如果指定的资源版本不在Scheme资源注册表中,则会报错。如果不指定资源版本,则默认转换为资源的首选版本。

➜ ~ kubectl create -f v1beta1deploy.yamlerror: unable to recognize "v1beta1deploy.yaml": no matches for kind "Deployment" in version "apps/v1beta1"➜ ~ kubectl convert -f v1beta1deploy.yaml --output-version=apps/v1kubectl convert is DEPRECATED and will be removed in a future version.In order to convert, kubectl apply the object to the cluster, then kubectl get at the desired version.apiVersion: apps/v1kind: Deploymentmetadata: creationTimestamp: null labels: app: nginx name: nginx-deploymentspec: progressDeadlineSeconds: 600 replicas: 3 revisionHistoryLimit: 2 selector: matchLabels: app: nginx strategy: rollingUpdate: maxSurge: 25% maxUnavailable: 25% type: RollingUpdate template: metadata: creationTimestamp: null labels: app: nginx spec: containers: - image: nginx:1.14.2 imagePullPolicy: IfNotPresent name: nginx ports: - containerPort: 80 protocol: TCP resources: {} terminationMessagePath: /dev/termination-log terminationMessagePolicy: File dnsPolicy: ClusterFirst restartPolicy: Always schedulerName: default-scheduler securityContext: {} terminationGracePeriodSeconds: 30status: {}➜ ~ kubectl convert -f v1beta1deploy.yaml --output-version=apps/v1 > /tmp/v1deploy.yamlkubectl convert is DEPRECATED and will be removed in a future version.In order to convert, kubectl apply the object to the cluster, then kubectl get at the desired version.➜ ~ kubectl create -f /tmp/v1deploy.yamldeployment.apps/nginx-deployment created➜ ~ kubectl get pod --all-namespacesNAMESPACE NAME READY STATUS RESTARTS AGEdefault nginx-deployment-9456bbbf9-78z87 1/1 Running 0 27sdefault nginx-deployment-9456bbbf9-7tgnm 1/1 Running 0 27sdefault nginx-deployment-9456bbbf9-n4mch 1/1 Running 0 27s

Converter资源版本转换器主要用于解决多资源版本转换问题,Kubernetes系统中的一个资源支持多个资源版本,如果要在每个资源版本之间转换,最直接的方式是,每个资源版本都支持其他资源版本的转换,但这样处理起来非常麻烦。例如,某个资源对象支持3个资源版本,那么就需要提前定义一个资源版本转换到其他两个资源版本(v1→v1alpha1,v1→v1beta1)、(v1alpha1→v1,v1alpha1→v1beta1)及(v1beta1→v1,v1beta1→v1alpha1),随着资源版本的增加,资源版本转换的定义会越来越多。为了解决这个问题,Kubernetes通过内部版本(Internal Version)机制实现资源版本转换,Converter资源版本转换过程:

当需要在两个资源版本之间转换时,例如v1alpha1→v1beta1或v1alpha1→v1。Converter资源版本转换器先将第一个资源版本转换为__internal内部版本,再转换为相应的资源版本。每个资源只要能支持内部版本,就能与其他任何资源版本进行间接的资源版本转换。

Converter转换器数据结构

Converter转换器数据结构主要存放转换函数(即Conversion Funcs)。Converter转换器数据结构代码,代码路径:vendor/k8s.io/apimachinery/pkg/conversion/converter.go

type typePair struct { source reflect.Type dest reflect.Type}type NameFunc func(t reflect.Type) stringvar DefaultNameFunc = func(t reflect.Type) string { return t.Name() }// ConversionFunc converts the object a into the object b, reusing arrays or objects// or pointers if necessary. It should return an error if the object cannot be converted// or if some data is invalid. If you do not wish a and b to share fields or nested// objects, you must copy a before calling this function.type ConversionFunc func(a, b interface{}, scope Scope) error// Converter knows how to convert one type to another.type Converter struct { // Map from the conversion pair to a function which can // do the conversion. conversionFuncs ConversionFuncs generatedConversionFuncs ConversionFuncs // Set of conversions that should be treated as a no-op ignoredUntypedConversions map[typePair]struct{}}func NewConversionFuncs() ConversionFuncs { return ConversionFuncs{ untyped: make(map[typePair]ConversionFunc), }}type ConversionFuncs struct { untyped map[typePair]ConversionFunc}

conversionFuncs :默认转换函数。这些转换函数一般定义在资源目录下的conversion.go代码文件中。generatedConversionFuncs :自动生成的转换函数。这些转换函数一般定义在资源目录下的zz_generated.conversion.go代码文件中,是由代码生成器自动生成的转换函数。ignoredConversions :若资源对象注册到此字段,则忽略此资源对象的转换操作。nameFunc :在转换过程中其用于获取资源种类的名称

Converter转换器数据结构中存放的转换函数(即Conversion Funcs)可以分为两类,分别为默认的转换函数(即conversionFuncs字段)和自动生成的转换函数(即generatedConversionFuncs字段)。它们都通过ConversionFuncs来管理转换函数。

ConversionFunc类型函数(即Type Function)定义了转换函数实现的结构,将资源对象a转换为资源对象b。a参数定义了转换源(即source)的资源类型,b参数定义了转换目标(即dest)的资源类型。scope定义了多次转换机制(即递归调用转换函数)。

ConversionFunc类型函数的资源对象传参必须是指针,否则无法进行转换并抛出异常。

Converter注册转换函数

Converter转换函数需要通过注册才能在Kubernetes内部使用,目前Kubernetes支持5个注册转换函数,分别介绍如下。

scheme.AddIgnoredConversionType :注册忽略的资源类型,不会执行转换操作,忽略资源对象的转换操作。scheme.AddConversionFuncs :注册多个Conversion Func转换函数。scheme.AddConversionFunc :注册单个Conversion Func转换函数。scheme.AddGeneratedConversionFunc :注册自动生成的转换函数。scheme.AddFieldLabelConversionFunc :注册字段标签(FieldLabel)的转换函数。

以apps/v1资源组、资源版本为例,通过scheme.AddConversionFuncs函数注册所有资源的转换函数,代码路径:pkg/apis/apps/v1/conversion.go

func Convert_apps_DeploymentSpec_To_v1_DeploymentSpec(in *apps.DeploymentSpec, out *appsv1.DeploymentSpec, s conversion.Scope) error { func Convert_v1_Deployment_To_apps_Deployment(in *appsv1.Deployment, out *apps.Deployment, s conversion.Scope) error {func Convert_apps_Deployment_To_v1_Deployment(in *apps.Deployment, out *appsv1.Deployment, s conversion.Scope) error {func Convert_apps_DaemonSet_To_v1_DaemonSet(in *apps.DaemonSet, out *appsv1.DaemonSet, s conversion.Scope) error {

Converter资源版本转换原理

Converter转换器在Kubernetes源码中实际应用非常广泛,例如Deployment资源对象,起初使用v1beta1资源版本,而v1资源版本更稳定,则会将v1beta1资源版本转换为v1资源版本。Converter资源版本转换过程

Converter转换器通过Converter函数转换资源版本,直接阅读代码会感觉比较晦涩,通过Converter Example代码示例来理解Converter转换器印象会更深刻,Converter Example代码示例如下:

package main import ("fmt" appsv1"k8s.io/api/apps/v1" appsv1beta1 "k8s.io/api/apps/v1beta1" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/runtime" "k8s.io/kubernetes/pkg/apis/apps")func main () { scheme := runtime.NewScheme() scheme.AddKnownTypes(appsv1beta1.SchemeGroupVersion &appsv1beta1.Deployment {}) scheme.AddKnownTypes (appsv1. SchemeGroupVersion, &appsv1.Deployment ()} scheme.AddKnownTypes (apps. SchemeGroupversion, &appsvl. Deployment (!) metavl.AddToGroupVersion(scheme, appsvlbetal.SchemeGroupVersion) metav1.AddToGroupVersion(scheme, appsv1. SchemeGroupVersion) v1betalDeployment:= &appsv1beta1.Deploymentf TypeMeta: metavl. TypeMeta1 Kind: "Deployment" APIVersion: "apps/v1beta1" ] // v1betal internal objInternal, err := scheme.ConvertToVersion( v1betalDeployment, apps.SchemeGroupVersion) if err != nil panic (err) } fmt. Println ("GVK: ", objInternal.GetObjectKind ().GroupVersionKind(). String () ) // output: // GVK: /, Kind= //internal - v` objv1, err := scheme.ConvertToVersion( objInternal, appsv1.SchemeGroupVersion) if err != nil { panic (err) } v1Deployment, ok : =objV1. (*appsv1.Deployment) if !ok { panic ("Got wrong type") fmt.Println ("GVK: ", v1Deployment.GetObjectKind().GroupVersionKind().String()) // output: // GVK: apps/v1, Kind-Deployment}

Convert Example代码示例分为3部分,执行过程如下。

第1部分 :实例化一个空的Scheme资源注册表,将v1beta1资源版本、v1资源版本及内部版本(__internal)的Deployment资源注册到Scheme资源注册表中。

第2部分 :实例化v1beta1Deployment资源对象,通过scheme.ConvertToVersion将其转换为目标资源版本(即__internale版本),得到objInternal资源对象,objInternal资源对象的GVK输出为“/,Kind=”。当资源对象的GVK输出为“/,Kind=”时,我们同样认为它是内部版本的资源对象,在后面章节中会说明原因。

第3部分 :将objInternal资源对象通过scheme.ConvertToVersion转换为目标资源版本(即v1资源版本),得到objV1资源对象,并通过断言的方式来验证是否转换成功,objV1资源对象的GVK输出为“apps/v1,Kind=Deployment”。

在Converter Example代码示例的第2部分中,将v1beta1资源版本转换为内部版本(即__internal版本),得到转换后资源对象的GVK为“/,Kind=”。在这里,肯定会产生疑问,为什么v1beta1资源版本转换为内部版本以后得到的GVK为“/,Kind=”而不是“apps/__internal,Kind=Deployment”。下面带着疑问来看看Kubernetes源码实现。

Scheme资源注册表可以通过两种方式进行版本转换,分别介绍如下。

● scheme.ConvertToVersion :将传入的(in)资源对象转换成目标(target)资源版本,在版本转换之前,会将资源对象深复制一份后再执行转换操作,相当于安全的内存对象转换操作。

●  scheme.UnsafeConvertToVersion :与scheme.ConvertToVersion功能相同,但在转换过程中不会深复制资源对象,而是直接对原资源对象进行转换操作,尽可能高效地实现转换。但该操作是非安全的内存对象转换操作。

scheme.ConvertToVersion与scheme.UnsafeConvertToVersion资源版本转换功能都依赖于s.convertToVersion函数来实现,Converter转换器流程图如图3-18所示。

下面详细介绍Converter转换器转换流程

1.获取传入的资源对象的反射类型

资源版本转换的类型可以是runtime.Object或runtime.Unstructured,它们都属于Go语言里的Struct数据结构,通过Go语言标准库reflect机制获取该资源类型的反射类型,因为在Scheme资源注册表中是以反射类型注册资源的。获取传入的资源对象的反射类型,代码示例如下:

代码路径:vendor/k8s.io/apimachinery/pkg/runtime/scheme.go

// unstructuredToTyped returns an Object, which must be a pointer to a struct. t = reflect.TypeOf(in).Elem()

2.从资源注册表中查找到传入的资源对象的GVK

从Scheme资源注册表中查找到传入的资源对象的所有GVK,验证传入的资源对象是否已经注册,如果未曾注册,则返回错误,代码示例如下:

kinds, ok := s.typeToGVK[t] if !ok || len(kinds) == 0 { return nil, NewNotRegisteredErrForType(s.schemeName, t) }

3.从多个GVK中选出与目标资源对象相匹配的GVK

target.KindForGroupVersionKinds函数从多个可转换的GVK中选出与目标资源对象相匹配的GVK。这里有一个优化点,转换过程是相对耗时的,大量的相同资源之间进行版本转换的耗时会比较长。在Kubernetes源码中判断,如果目标资源对象的GVK在可转换的GVK列表中,则直接将传入的资源对象的GVK设置为目标资源对象的GVK,而无须执行转换操作,缩短部分耗时。代码示例如下:

gvk, ok := target.KindForGroupVersionKinds(kinds) if !ok { // try to see if this type is listed as unversioned (for legacy support) // TODO: when we move to server API versions, we should completely remove the unversioned concept if unversionedKind, ok := s.unversionedTypes[t]; ok { if gvk, ok := target.KindForGroupVersionKinds([]schema.GroupVersionKind{unversionedKind}); ok { return copyAndSetTargetKind(copy, in, gvk) } return copyAndSetTargetKind(copy, in, unversionedKind) } return nil, NewNotRegisteredErrForTarget(s.schemeName, t, target) } // target wants to use the existing type, set kind and return (no conversion necessary) for _, kind := range kinds { if gvk == kind { return copyAndSetTargetKind(copy, in, gvk) } }

4.判断传入的资源对象是否属于Unversioned类型

对于Unversioned类型,前面曾介绍过,即无版本类型(UnversionedType)。属于该类型的资源对象并不需要进行转换操作,而是直接将传入的资源对象的GVK设置为目标资源对象的GVK。代码示例如下:

// type is unversioned, no conversion necessary if unversionedKind, ok := s.unversionedTypes[t]; ok { if gvk, ok := target.KindForGroupVersionKinds([]schema.GroupVersionKind{unversionedKind}); ok { return copyAndSetTargetKind(copy, in, gvk) } return copyAndSetTargetKind(copy, in, unversionedKind) }

5.执行转换操作

在执行转换操作之前,先判断是否需要对传入的资源对象执行深复制操作,然后通过s.converter.Convert转换函数执行转换操作,代码示例如下:

out, err := s.New(gvk) if err != nil { return nil, err } if copy { in = in.DeepCopyObject() } meta := s.generateConvertMeta(in) meta.Context = target if err := s.converter.Convert(in, out, meta); err != nil { return nil, err }

实际的转换函数是通过doConversion函数执行的,执行过程如下

● 从默认转换函数列表(即c.conversionFuncs)中查找出pair对应的转换函数,如果存在则执行该转换函数(即fn)并返回。

● 从自动生成的转换函数列表(即generatedConversionFuncs)中查找出pair对应的转换函数,如果存在则执行该转换函数(即fn)并返回。

● 如果默认转换函数列表和自动生成的转换函数列表中都不存在当前资源对象的转换函数,则使用doConversion函数传入的转换函数(即f)。调用f之前,需要将src与dest资源对象通过EnforcePtr函数取指针的值,因为doConversion函数传入的转换函数接收的是非指针资源对象。

doConversion转换函数代码示例如下,代码路径:vendor/k8s.io/apimachinery/pkg/conversion/converter.go

// Convert will translate src to dest if it knows how. Both must be pointers.// If no conversion func is registered and the default copying mechanism// doesn't work on this type pair, an error will be returned.// 'meta' is given to allow you to pass information to conversion functions,// it is not used by Convert() other than storing it in the scope.// Not safe for objects with cyclic references!func (c *Converter) Convert(src, dest interface{}, meta *Meta) error { pair := typePair{reflect.TypeOf(src), reflect.TypeOf(dest)} scope := &scope{ converter: c, meta: meta, } // ignore conversions of this type if _, ok := c.ignoredUntypedConversions[pair]; ok { return nil } if fn, ok := c.conversionFuncs.untyped[pair]; ok { return fn(src, dest, scope) } if fn, ok := c.generatedConversionFuncs.untyped[pair]; ok { return fn(src, dest, scope) } dv, err := EnforcePtr(dest) if err != nil { return err } sv, err := EnforcePtr(src) if err != nil { return err } return fmt.Errorf("converting (%s) to (%s): unknown conversion", sv.Type(), dv.Type())}

6.设置转换后资源对象的GVK

在Converter Example代码示例的第2部分中,将v1beta1资源版本转换为内部版本(即__internal版本),得到转换后资源对象的GVK为“/,Kind=”。原因在于setTargetKind函数,转换操作执行完成以后,通过setTargetKind函数设置转换后资源对象的GVK,判断当前资源对象是否为内部版本(即APIVersionInternal),是内部版本则设置GVK为schema.GroupVersionKind{}。代码示例如下:

代码路径:vendor/k8s.io/apimachinery/pkg/runtime/scheme.go

setTargetKind(out, gvk) // setTargetKind sets the kind on an object, taking into account whether the target kind is the internal version.func setTargetKind(obj Object, kind schema.GroupVersionKind) { if kind.Version == APIVersionInternal { // internal is a special case // TODO: look at removing the need to special case this obj.GetObjectKind().SetGroupVersionKind(schema.GroupVersionKind{}) return } obj.GetObjectKind().SetGroupVersionKind(kind)}

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:如何做好产品整体营销策划?
下一篇:实验:kubeadm方式来部署k8s
相关文章

 发表评论

暂时没有评论,来抢沙发吧~