mirror of
https://github.com/kubevela/kubevela.git
synced 2026-05-09 19:07:04 +00:00
* upgrade K8s dependency to v0.21 * update CRD for new version of controller-runtime * fix ci component revision create must fill Raw in runtime.RawExtension * try fix test * start control plane timeout set to 1min and fix tests * add timeout for env test start and stop * longer timeout time for BeforeSuit function * upgrade kubebuilder and k8s cluster version to match with v1.21.2 in github action * fix test * fix resource tracker ownerRef override * update developer guides
209 lines
6.9 KiB
Go
209 lines
6.9 KiB
Go
/*
|
|
Copyright 2021 The KubeVela Authors.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
package apply
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/oam-dev/kubevela/pkg/oam"
|
|
|
|
"github.com/pkg/errors"
|
|
kerrors "k8s.io/apimachinery/pkg/api/errors"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/apimachinery/pkg/types"
|
|
"k8s.io/klog/v2"
|
|
"sigs.k8s.io/controller-runtime/pkg/client"
|
|
)
|
|
|
|
// Applicator applies new state to an object or create it if not exist.
|
|
// It uses the same mechanism as `kubectl apply`, that is, for each resource being applied,
|
|
// computing a three-way diff merge in client side based on its current state, modified stated,
|
|
// and last-applied-state which is tracked through an specific annotation.
|
|
// If the resource doesn't exist before, Apply will create it.
|
|
type Applicator interface {
|
|
Apply(context.Context, client.Object, ...ApplyOption) error
|
|
}
|
|
|
|
// ApplyOption is called before applying state to the object.
|
|
// ApplyOption is still called even if the object does NOT exist.
|
|
// If the object does not exist, `existing` will be assigned as `nil`.
|
|
// nolint: golint
|
|
type ApplyOption func(ctx context.Context, existing, desired runtime.Object) error
|
|
|
|
// NewAPIApplicator creates an Applicator that applies state to an
|
|
// object or creates the object if not exist.
|
|
func NewAPIApplicator(c client.Client) *APIApplicator {
|
|
return &APIApplicator{
|
|
creator: creatorFn(createOrGetExisting),
|
|
patcher: patcherFn(threeWayMergePatch),
|
|
c: c,
|
|
}
|
|
}
|
|
|
|
type creator interface {
|
|
createOrGetExisting(context.Context, client.Client, client.Object, ...ApplyOption) (client.Object, error)
|
|
}
|
|
|
|
type creatorFn func(context.Context, client.Client, client.Object, ...ApplyOption) (client.Object, error)
|
|
|
|
func (fn creatorFn) createOrGetExisting(ctx context.Context, c client.Client, o client.Object, ao ...ApplyOption) (client.Object, error) {
|
|
return fn(ctx, c, o, ao...)
|
|
}
|
|
|
|
type patcher interface {
|
|
patch(c, m client.Object) (client.Patch, error)
|
|
}
|
|
|
|
type patcherFn func(c, m client.Object) (client.Patch, error)
|
|
|
|
func (fn patcherFn) patch(c, m client.Object) (client.Patch, error) {
|
|
return fn(c, m)
|
|
}
|
|
|
|
// APIApplicator implements Applicator
|
|
type APIApplicator struct {
|
|
creator
|
|
patcher
|
|
c client.Client
|
|
}
|
|
|
|
// loggingApply will record a log with desired object applied
|
|
func loggingApply(msg string, desired client.Object) {
|
|
d, ok := desired.(metav1.Object)
|
|
if !ok {
|
|
klog.InfoS(msg, "resource", desired.GetObjectKind().GroupVersionKind().String())
|
|
return
|
|
}
|
|
klog.InfoS(msg, "name", d.GetName(), "resource", desired.GetObjectKind().GroupVersionKind().String())
|
|
}
|
|
|
|
// Apply applies new state to an object or create it if not exist
|
|
func (a *APIApplicator) Apply(ctx context.Context, desired client.Object, ao ...ApplyOption) error {
|
|
existing, err := a.createOrGetExisting(ctx, a.c, desired, ao...)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if existing == nil {
|
|
return nil
|
|
}
|
|
|
|
// the object already exists, apply new state
|
|
if err := executeApplyOptions(ctx, existing, desired, ao); err != nil {
|
|
return err
|
|
}
|
|
loggingApply("patching object", desired)
|
|
patch, err := a.patcher.patch(existing, desired)
|
|
if err != nil {
|
|
return errors.Wrap(err, "cannot calculate patch by computing a three way diff")
|
|
}
|
|
return errors.Wrapf(a.c.Patch(ctx, desired, patch), "cannot patch object")
|
|
}
|
|
|
|
// createOrGetExisting will create the object if it does not exist
|
|
// or get and return the existing object
|
|
func createOrGetExisting(ctx context.Context, c client.Client, desired client.Object, ao ...ApplyOption) (client.Object, error) {
|
|
var create = func() (client.Object, error) {
|
|
// execute ApplyOptions even the object doesn't exist
|
|
if err := executeApplyOptions(ctx, nil, desired, ao); err != nil {
|
|
return nil, err
|
|
}
|
|
if err := addLastAppliedConfigAnnotation(desired); err != nil {
|
|
return nil, err
|
|
}
|
|
loggingApply("creating object", desired)
|
|
return nil, errors.Wrap(c.Create(ctx, desired), "cannot create object")
|
|
}
|
|
|
|
// allow to create object with only generateName
|
|
if desired.GetName() == "" && desired.GetGenerateName() != "" {
|
|
return create()
|
|
}
|
|
|
|
existing := &unstructured.Unstructured{}
|
|
existing.GetObjectKind().SetGroupVersionKind(desired.GetObjectKind().GroupVersionKind())
|
|
err := c.Get(ctx, types.NamespacedName{Name: desired.GetName(), Namespace: desired.GetNamespace()}, existing)
|
|
if kerrors.IsNotFound(err) {
|
|
return create()
|
|
}
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "cannot get object")
|
|
}
|
|
return existing, nil
|
|
}
|
|
|
|
func executeApplyOptions(ctx context.Context, existing, desired runtime.Object, aos []ApplyOption) error {
|
|
// if existing is nil, it means the object is going to be created.
|
|
// ApplyOption function should handle this situation carefully by itself.
|
|
for _, fn := range aos {
|
|
if err := fn(ctx, existing, desired); err != nil {
|
|
return errors.Wrap(err, "cannot apply ApplyOption")
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// MustBeControllableBy requires that the new object is controllable by an
|
|
// object with the supplied UID. An object is controllable if its controller
|
|
// reference includes the supplied UID.
|
|
func MustBeControllableBy(u types.UID) ApplyOption {
|
|
return func(_ context.Context, existing, _ runtime.Object) error {
|
|
if existing == nil {
|
|
return nil
|
|
}
|
|
c := metav1.GetControllerOf(existing.(metav1.Object))
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
if c.UID != u {
|
|
return errors.Errorf("existing object is not controlled by UID %q", u)
|
|
}
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// MustBeControllableByAny requires that the new object is controllable by any of the object with
|
|
// the supplied UID.
|
|
func MustBeControllableByAny(ctrlUIDs []types.UID) ApplyOption {
|
|
return func(_ context.Context, existing, _ runtime.Object) error {
|
|
if existing == nil || len(ctrlUIDs) == 0 {
|
|
return nil
|
|
}
|
|
existingObjMeta, _ := existing.(metav1.Object)
|
|
c := metav1.GetControllerOf(existingObjMeta)
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
// NOTE This is for backward compatibility after ApplicationContext is deprecated.
|
|
// In legacy clusters, existing resources are ctrl-owned by ApplicationContext or ResourceTracker (only for
|
|
// cx-namespace and cluster-scope resources). We use a particular annotation to identify legacy resources.
|
|
if len(existingObjMeta.GetAnnotations()[oam.AnnotationKubeVelaVersion]) == 0 {
|
|
// just skip checking UIDs, '3-way-merge' will remove the legacy ctrl-owner automatically
|
|
return nil
|
|
}
|
|
|
|
for _, u := range ctrlUIDs {
|
|
if c.UID == u {
|
|
return nil
|
|
}
|
|
}
|
|
return errors.Errorf("existing object is not controlled by any of UID %q", ctrlUIDs)
|
|
}
|
|
}
|