从零开始设计一个CRD

前言

经过前面的铺垫,相信现在对kubebuilder的工作模式已初具了解,那么从本篇开始,正式设计一个CRD。本文对于Unit的设计,是基于我的个人场景下的需求提炼出来的,不一定适合你,但着重点在于思路,希望能有帮助。

面临的现状

在我的场景下,应该也是大多数人的场景下,通常一个运行服务(姑且这么称呼),使用一系列build-in 类型资源进行组合,来保障运行和提供服务,例如,最常用的组合有:StatefulSet/Deployment/Ingress/Service/Ingress这几种资源的按需组合,如下图:

这些资源类型每一种都是可选项,根据使用需求的不同,来灵活(弱绑定?)进行组合。

例如:

  • 非web服务不需要Ingress资源
  • 自发现和注册的应用不需要Service
  • 无状态的应用选用Deployment,有状态的应用选用StatefulSet
  • 有的应用无需持久存储,有的应用需求PVC来实现持久存储

按需组合,不一而同

这样的弱绑定关系在个人认为管理上不足够的友好,每种资源的增删改查等逻辑,需要分别操作,没有整体的连贯性和协同性,需要分而治之。

想要实现什么

与市面上大多数专属于某个应用或者某类应用的CRD不同,我这里想要实现的CRD目标是:每一个运行服务,所用到的的各类资源,以强申明将它们绑定在一起,进行统一的生命周期管理

因此,我给这个CRD的命名是Unit,这意味着,StatefulSet/Deployment/Ingress/Service/Ingress这些可选的build-in资源,由原本松散组合构成一个运行服务的模式,变为集合在一个Unit单元里,形成统一的管理生命周期。如下图:

设计的目标已经设定了,下面就开始实际设计和填充Unit结构体。

结构设计

如同所有的build-in GVR一样,结构体的构成除了主要分为两个部分,一个是Spec,一个是Status,声明式的目标即是Status满足Spec的声明要求。因此,Unit的结构体设计也围绕Spec和Status两部分来。下面直接贴代码,代码中会有注释。

按照上面的描述,Unit总共可管理这5种资源:StatefulSet/Deployment/Ingress/Service/Ingress,为了结构清晰,将它们各自的结构体分别放在不同的文件中:

原本的build-in资源结构体,以Deployment举例,Deployment结构体字段非常多,不够精简,因此我不想每种资源的每个字段都完整声明,而是在前端以最精简地方式声明,只提供必要的字段,而最后填充出完整的Deployment的结构体的方法放在后端来进行,尽可能地保证前端使用用户的友好度。

Spec

直接贴代码

UnitSpec

// UnitSpec defines the desired state of Unit
type UnitSpec struct {
    // INSERT ADDITIONAL SPEC FIELDS - desired state of cluster
    // Important: Run "make" to regenerate code after modifying this file

    // Category 支持两种: Deployment / StatefulSet ,二者只能选其一,在admission validating webhook里会做校验.

    Category string `json:"category"`

    // Replicas和Selector这两个字段在mutate webhook里默认会有填充
    Replicas *int32                `json:"replicas,omitempty"`
    Selector *metav1.LabelSelector `json:"selector,omitempty"`

    // Template describes the pods that will be created.
    Template         corev1.PodTemplateSpec    `json:"template"`
    RelationResource UnitRelationResourceSpec `json:"relationResource,omitempty"`
}

UnitRelationResourceSpec

type UnitRelationResourceSpec struct {
    Service *OwnService `json:"serviceInfo,omitempty"`
    PVC     *OwnPVC     `json:"pvcInfo,omitempty"`
    Ingress *OwnIngress `json:"ingressInfo,omitempty"`
}

这里指定OwnService/OwnIngress/OwnPVC,至于具体的ownDeployment / ownStatefulSet,由于限制二种部署方式只能取其一,因此会结合UnitSpec.Category和UnitSpec.Template来动态生成,怎么生成放在后面说,先来看看结构体。

OwnStatefulSet

type OwnStatefulSet struct {
    Spec appsv1.StatefulSetSpec
}

OwnDeployment

type OwnDeployment struct {
   Spec appsv1.DeploymentSpec `json:"spec"`
}

OwnService

// svc的端口映射关系
type ServicePort struct {
   Name       string             `json:"name,omitempty" protobuf:"bytes,1,opt,name=name"`
   Protocol   string             `json:"protocol,omitempty" protobuf:"bytes,2,opt,name=protocol,casttype=Protocol"`
   Port       int32              `json:"port" protobuf:"varint,3,opt,name=port"`
   TargetPort intstr.IntOrString `json:"targetPort,omitempty" protobuf:"bytes,4,opt,name=targetPort"`
   NodePort   int32              `json:"nodePort,omitempty" protobuf:"varint,5,opt,name=nodePort"`
}

type OwnService struct {
   Ports     []v1.ServicePort `json:"ports,omitempty" patchStrategy:"merge" patchMergeKey:"port" protobuf:"bytes,1,rep,name=ports"`
   ClusterIP string           `json:"clusterIP,omitempty" protobuf:"bytes,3,opt,name=clusterIP"`
}

OwnIngress

// ingress信息
type OwnIngress struct {
    Domains []string `json:"domain"`
}

OwnPVC

// pvc声明信息
type OwnPVC struct {
    Spec v1.PersistentVolumeClaimSpec` json:"spec"`
}

大部分都是利用原resource.Spec字段,这样在精简的同时,生成resource object的时候也可以比较方便。

Status

Unit.Status设计就比较天马行空了,例如我希望在状态里面看到与Service和Pod自动关联的Endpoints逻辑资源,因此把部分Endpoint的信息也加了进来。

UnitStatus

// UnitStatus defines the observed state of Unit
type UnitStatus struct {
   // INSERT ADDITIONAL STATUS FIELD - define observed state of cluster
   // Important: Run "make" to regenerate code after modifying this file
   Replicas       *int32      `json:"replicas,omitempty"`
   Selector string `json:"selector"`
   LastUpdateTime metav1.Time `json:"lastUpdateTime,omitempty"`

   BaseDeployment         appsv1.DeploymentStatus     `json:"deployment,omitempty"`
   BaseStatefulSet        appsv1.StatefulSetStatus    `json:"statefulSet,omitempty"`
   RelationResourceStatus UnitRelationResourceStatus `json:"relationResourceStatus,omitempty"`
}

type UnitRelationResourceStatus struct {
    Service  UnitRelationServiceStatus         `json:"service,omitempty"`
    Ingress  []v1beta1.IngressRule              `json:"ingress,omitempty"`
    Endpoint []UnitRelationEndpointStatus      `json:"endpoint,omitempty"`
    PVC      corev1.PersistentVolumeClaimStatus `json:"pvc,omitempty"`
}

UnitRelationServiceStatus

type UnitRelationServiceStatus struct {
   Type            v1.ServiceType      `json:"type,omitempty"`
   ClusterIP       string              `json:"clusterIP,omitempty"`
   Ports           []ServicePortStatus `json:"ports,omitempty"`
   SessionAffinity v1.ServiceAffinity  `json:"sessionAffinity,omitempty"`
}

UnitRelationEndpointStatus

type UnitRelationEndpointStatus struct {
   PodName  string `json:"podName"`
   PodIP    string `json:"podIP"`
   NodeName string `json:"nodeName"`
}

原ServiceStatus信息量太少,就用了一个自定义的status填充了一下,其余的资源类型的Status,如DeploymentStatus/StatefulSetStatus/PersistentVolumeClaimStatus 信息量已经足够了,偷下懒直接利用。

总结

Unit的实现目标确定后,设计和填充了它的结构体,同时为了保持结构体的精简,简化出了一层ownResource,同样也设计和填充ownResources的Spec和Status结构。基础已经铺垫好,下一篇开始设计和实现Unit Reconciler的控制逻辑,以管理Unit和ownResources之间的生命周期的联动。

results matching ""

    No results matching ""