如何用 Kubernetes API 对象的默认值惯用地填充空字段?

kis*_*ljr 3 go kubernetes client-go

我想比较两个 Kubernetes API 对象(例如v1.PodSpecs):其中一个是手动创建的(预期状态),另一个是从 Kubernetes API/客户端接收的(实际状态)。问题在于,即使两个对象在语义上相等,手动创建的结构体的未指定字段的值为零,而另一个结构体具有默认值,因此两者不匹配。这意味着简单的reflect.DeepEqual()调用不足以进行比较。

例如在此之后:

expected := &v1.Container{
    Name:  "busybox",
    Image: "busybox",
}

actual := getContainerSpecFromApi(...)
Run Code Online (Sandbox Code Playgroud)

expected.ImagePullPolicywill be "", while actual.ImagePullPolicywill be "IfNotPresent"(默认值),所以比较失败。

是否有一种惯用的方法可以将 Kubernetes API 结构中的零值替换为默认值?或者是一个构造函数,它使用某个地方可用的默认值来初始化结构?

编辑:目前我正在对每个 K8s API 对象类型使用手写相等测试,但这对我来说似乎无法维护。我正在寻找一个简单的(一组)函数,它“知道”所有内置 Kubernetes API 对象字段的默认值(可能在下面的某个地方k8s.io/api*?)。像这样的东西:

expected = api.ApplyContainerDefaults(expected)
if !reflect.DeepEqual(expected, actual) {
    reconcile(expected, actual)
}
Run Code Online (Sandbox Code Playgroud)

Nic*_*vin 5

有一些帮助程序可以填充默认值来代替空/零值。

例如,查看Deployment 的SetObjectDefaults_Deployment 。

看起来调用它的正确方法是 via (*runtime.Scheme).Default。下面是显示总体思路的片段:

import (
    "reflect"

    appsv1 "k8s.io/api/apps/v1"
    "k8s.io/client-go/kubernetes/scheme"
)

func compare() {
    scheme := scheme.Scheme

    // fetch the existing &appsv1.Deployment via API
    actual := ...
    expected := &appsv1.Deployment{}

    // fill in the fields to generate your expected state
    // ...

    scheme.Default(expected)
    // now you should have your empty values filled in
    if !reflect.DeepEqual(expected.Spec, actual.Spec) {
        reconcile(expected, actual)
    }
}
Run Code Online (Sandbox Code Playgroud)

如果您需要不太严格的比较,例如,如果您需要容忍一些注入的容器,那么应​​该使用更宽松的东西,如下所示