gpt4 book ai didi

amazon-web-services - k8s controller watch other controller CR

转载 作者:行者123 更新时间:2023-12-05 05:31:49 24 4
gpt4 key购买 nike

我有 k8s 运算符,它按预期工作,我需要向其他运算符 CRD(不是我的)添加一个“监视”,为了简单起见,我们将其称为 extCR,我们的运算符 cr 称为inCR,

我尝试了以下方法,但有一个问题是它如何触发协调。

func (r *Insiconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
For(&Inv1alpha1.Iget{}}).
Watches(&source.Kind{Type: &ext.Se{}}, handler.EnqueueRequestsFromMapFunc(r.FWatch)).
Complete(r)
}

func (r *Insiconciler) FWatch(c client.Object) []reconcile.Request {
val := c.(*ivi.Srv)
req := reconcile.Request{NamespacedName: types.NamespacedName{Name: val.Name, Namespace: val.Namespace}}
return []reconcile.Request{req}
}

这里的问题是我触发了与 extCR 的协调,我想在 FWatch 内部更新 inCR 并开始协调inCR 而不是 extCR,我该怎么做?

我的意思是,为了避免像下面的代码这样的事情,因为有时协调是针对 inCR 完成的,有时是针对 extCR 并且我可以得到一些丑陋的 if's

func (r *Insiconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
var inCR FOO
var extCR BAR

if err := r.Get(ctx, req.NamespacedName, &inCR); err != nil {
return ctrl.Result{}, err
}

if err := r.Get(ctx, req.NamespacedName, &extCR); err != nil {
return ctrl.Result{}, err
}

我想知道处理这种情况的正确/干净的方法是什么

当您需要收听外部 CR(不是 Controller 的一部分)和内部 CR(来自您的 Controller )时的情况。

还有一件事——CR 是不同的 GVK,但是 exteranlCR 包含很多不需要的字段,只是其中的一些。但必填字段在两个 cr 上具有相同的名称

更新

type inCR struct {
metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"`

Spec inSpec `json:"spec,omitempty"` / / ————————here is the difference
Status InsightTargetStatus `json:"status,omitempty"`
}

//—————— 这是在不属于我们的其他程序上定义的,因此不能“重用”

type Bar struct {
metav1.TypeMeta `json:",inline"`
metav1.ObjectMeta `json:"metadata,omitempty"`

Spec extSpec `json:"spec,omitempty"` // ———————here is the difference
Status ServiceStatus `json:"status,omitempty"`
}

inSpec 具有以下字段(extSpec 的子集)

type inSpec struct {
name string
age int
}

extSpec 有那些字段以及更多不相关的字段

type extSpec struct {
name string
age int
foo string // not relevant
bar string // not relevant
bazz string // not relevant
}

最后,在 reconcile 中 我需要将 相关 字段移动到某些函数。 完全根据发生的事件(例如用户更新 extCR 或更新 inCR),完全y 有时需要 extCR 的字段,有时需要 inCR 的字段

更新2

func sharedLogic(r reconciler, ctx context.Context, c client.Object) (ctrl.Result, error) {


in := c.(*inCR)


vPass , e := vps.Get(ctx, r.Client, in.Spec.foo, in.Spec.bar)


return ctrl.Result{}, nil
}

But for extCR I should do the following


func sharedLogic(r reconciler, ctx context.Context, c client.Object) (ctrl.Result, error) {


ext := c.(*extCR)


vPass , e := vps.Get(ctx, r.Client, ext.Spec.val.foo, ext.Spec.val.bar)


return ctrl.Result{}, nil
}

最佳答案

需要注意的几点:

  • 每个 Controller 负责恰好一个资源。
  • 协调请求包含协调 Kubernetes 对象所需的信息。这包括唯一标识对象的信息 - 它的名称和命名空间。它不包含有关任何特定事件或对象内容本身的信息。

您可以在没有资源定义的情况下创建第二个 Controller 。在您的主文件中,两个 Controller 都将被注册。

如果 CRD 根本不相关,或者如果外部资源引用内部资源,这可能很有用,因此您可以在外部协调器中更改内部资源。

kubebuilder create api --group other --version v2 --kind External \
--resource=false --controller=true

这为您提供了一个带有 SetupWithManager 方法的 Controller ,如下所示。

func (r *ExternalReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
// Uncomment the following line adding a pointer to an instance of the controlled resource as an argument
// For().
Complete(r)
}

请注意 For 方法是如何被注释掉的,因为您需要从其他地方导入要观看的资源并引用它。

import (
...
otherv2 "other.io/external/api/v2"
)
...
func (r *ExternalReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
For(&otherv2.External{}).
Complete(r)
}

如果您无法导入外部资源,您可以回过头来自己模拟它,但这可能不是一种非常干净的方法。您真的应该尝试从其他 Controller 项目中导入它。

kubebuilder edit --multigroup=true
kubebuilder create api --group=other --version v2 --kind External \
--resource --controller

另一种方式是当资源相互关联时,内部资源在其规范中引用了外部资源,并且在协调时知道如何在其规范中获取外部资源。可以在此处找到这方面的示例 https://book.kubebuilder.io/reference/watching-resources/externally-managed.html

type InternalSpec struct {
// Name of an external resource
ExternalResource string `json:"externalResource,omitempty"`
}

这意味着在每个协调循环中, Controller 将查找外部资源并使用它来管理内部资源。

func (r *InternalReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
_ = log.FromContext(ctx)

internal := examplev1.Internal{}
if err := r.Get(context.TODO(), types.NamespacedName{
Name: req.Name,
Namespace: req.Namespace,
}, &internal); err != nil {
return ctrl.Result{}, err
}

external := otherv2.External{}
if err := r.Get(context.TODO(), types.NamespacedName{
// note how the name is taken from the internal spec
Name: internal.Spec.ExternalResource,
Namespace: req.Namespace,
}, &internal); err != nil {
return ctrl.Result{}, err
}

// do something with internal and external here

return ctrl.Result{}, nil
}

问题在于,当内部资源没有变化时,即使外部资源发生变化,也不会触发协调事件。为了解决这个问题,我们可以通过观察外部资源来触发协调。注意 Watches 方法:

func (r *InternalReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
For(&examplev1.Main{}).
Watches(
&source.Kind{Type: &otherv2.ExternalResource{}},
handler.EnqueueRequestsFromMapFunc(r.triggerReconcileBecauseExternalHasChanged),
builder.WithPredicates(predicate.ResourceVersionChangedPredicate{}),
).
Complete(r)
}

为了知道我们应该为哪个内部对象触发事件,我们使用映射函数来查找所有引用了外部资源的内部对象。

func (r *InternalReconciler) triggerReconcileBecauseExternalHasChanged(o client.Object) []reconcile.Request {
usedByInternals := &examplev1.InternalList{}
listOps := &client.ListOptions{
FieldSelector: fields.OneTermEqualSelector(".spec.ExternalResource", o.GetName()),
Namespace: o.GetNamespace(),
}
err := r.List(context.TODO(), usedByInternals, listOps)
if err != nil {
return []reconcile.Request{}
}
requests := make([]reconcile.Request, len(usedByInternals.Items))
for i, item := range usedByInternals.Items {
requests[i] = reconcile.Request{
NamespacedName: types.NamespacedName{
Name: item.GetName(),
Namespace: item.GetNamespace(),
},
}
}
return requests
}

由于您更新了问题,我建议您执行以下操作。

我正在创建一个新项目和 2 个 Controller 。注意第二个 Controller 命令没有资源与 Controller 一起创建。这是因为 Controller 将观看外部资源。

mkdir demo && cd demo
go mod init example.io/demo
kubebuilder init --domain example.io --repo example.io/demo --plugins=go/v4-alpha
kubebuilder create api --group=demo --version v1 --kind Internal --controller --resource
kubebuilder create api --group=other --version v2 --kind External --controller --resource=false
$ tree controllers
controllers
├── external_controller.go
├── internal_controller.go
└── suite_test.go

现在我们需要一些共享逻辑,例如将其添加到 Controller 包中。我们将从两个调解器中调用它。

// the interface may need tweaking
// depending on what you want to do with
// the reconiler
type reconciler interface {
client.Reader
client.Writer
client.StatusClient
}

func sharedLogic(r reconciler, kobj *demov1.Internal) (ctrl.Result, error) {
// do your shared logic here operating on the internal object struct
// this works out because the external controller will call this passing the
// internal object
return ctrl.Result{}, nil
}

这是内部协调器的示例。

func (r *InternalReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
_ = log.FromContext(ctx)
obj := demov1.Internal{}
if err := r.Get(ctx, req.NamespacedName, &obj); err != nil {
return ctrl.Result{}, err
}
return sharedLogic(r, &obj)
}

在外部调节器中,我们也这样做。

func (r *ExternalReconciler) Reconcile(ctx context.Context, req ctrl.Request) (ctrl.Result, error) {
_ = log.FromContext(ctx)
// note, we can use the internal object here as long as the external object
// does contain the same fields we want. That means when unmarshalling the extra
// fields are dropped. If this cannot be done, you could first unmarshal into the external
// resource and then assign the fields you need to the internal one, before passing it down
obj := demov1.Internal{}
if err := r.Get(ctx, req.NamespacedName, &obj); err != nil {
return ctrl.Result{}, err
}
return sharedLogic(r, &obj)
}

func (r *ExternalReconciler) SetupWithManager(mgr ctrl.Manager) error {
return ctrl.NewControllerManagedBy(mgr).
// note the external resource is imported from another project
// you may be able to watch this without import by creating a minimal
// type with the right GKV
For(otherv2.External{}).
Complete(r)
}

关于amazon-web-services - k8s controller watch other controller CR,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/74261141/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com