Introduce 2 libs workBuilder
and workApplier
to help user to build the manifest objects into manifestWorks
and apply/delete the manifestWorks easily.
The code is moved to https://github.com/open-cluster-management-io/sdk-go/tree/main/pkg/apis/work/v1, the code here will be removed in the future.
There is a limit size of manifest objects in a single manifestWork. And currently the limit is 50K.
if the size of the manifest objects exceed the limit, workBuilder
can help build the manifest objects into
multiple manifestWorks and update the existing manifestWorks.
-
Create a
WorkBuilder
instance.import ( "open-cluster-management.io/api/helpers/work/workbuilder" ) workBuilder := workbuilder.NewWorkBuilder()
WorkBuilder
will build a manifestWork with 80% size of the limit, the user can customize the limit size for the manifestWorks.workBuilder := workbuilder.NewWorkBuilder().WithManifestsLimit(limtSize)
-
Define a
GenerateManifestWorkObjectMeta
func to generate the meta info of the new manifestWorks like name, namespace, labels etc. The inputindex
is the count of the existing manifestWorks.func myGenerateWorkObjectMeta(index int) metav1.ObjectMeta { return metav1.ObjectMeta{ Name: <unique name for the manifestWork>, Namespace: clusterNamespace, } }
-
Build the manifest objects into manifestWorks if there is no existing manifestWorks.
applyWorks, _, err := workBuilder.Build(manifestObjects, myGenerateWorkObjectMeta)
The input
manifestObjects
is a slice ofruntime.Object
. And the outputapplyWorks
is a slice of built manifestWorks. -
Build the manifest objects into manifestWorks to update the existing manifestWorks on the Hub cluster.
applyWorks, deleteWorks, err := workBuilder.Build(manifestObjects, myGenerateWorkObjectMeta, workbuilder.ExistingManifestWorksOption(existingWorks))
The
existingWorks
is a slice of existing manifestWorks including the old manifests objects.The
applyWorks
is a slice of manifestWorks including the new created manifestWorks and the updated existing manifestWorks.The
deleteWorks
is a slice of manifestWorks which needs to be deleted since all its manifests are removed. -
Build manifestWorks with Options.
The user can configure the
ManifestConfigs
,Executor
orDeleteOption
asWorkBuilderOption
to the built manifestWorks.applyWorks, deleteWorks, err := workBuilder.Build(manifestObjects, myGenerateWorkObjectMeta, workbuilder.ExistingManifestWorksOption(existingWorks), workbuilder.ManifestConfigOption(manifestOptions), workbuilder.ManifestWorkExecutorOption(executorOption), workbuilder.DeletionOption(deletionOption))
-
Build and apply the manifestWorks.
The user also can use
BuildAndApply
method to apply the built manifestWorks with aworkApplier
instance.err := workBuilder.BuildAndApply(context, manifestObjects, myGenerateWorkObjectMeta, workApplier, workbuilder.ExistingManifestWorksOption(existingWorks), workbuilder.ManifestConfigOption(manifestOptions), workbuilder.ManifestWorkExecutorOption(executorOption), workbuilder.DeletionOption(deletionOption))
workApplier
can help the user to apply or delete a manifestWork easily, which has a cache to reduce redundant updates.
-
Create a
WorkApplier
instance.There is a default
WorkApplier
instance with ManifestWork typed client.One is
NewWorkApplierWithTypedClient(workClient workv1client.Interface, workLister worklister.ManifestWorkLister)
with manifestWork typed client.You can also define the instance with runtime client, for example:
func NewWorkApplierWithRuntimeClient(workClient client.Client) *WorkApplier { return &WorkApplier{ cache: newWorkCache(), getWork: func(ctx context.Context, namespace, name string) (*workapiv1.ManifestWork, error) { work := &workapiv1.ManifestWork{} err := workClient.Get(ctx, types.NamespacedName{Namespace: namespace, Name: name}, work) return work, err }, deleteWork: func(ctx context.Context, namespace, name string) error { work := &workapiv1.ManifestWork{ ObjectMeta: metav1.ObjectMeta{ Name: name, Namespace: namespace, }, } return workClient.Delete(ctx, work) }, patchWork: func(ctx context.Context, namespace, name string, pt types.PatchType, data []byte) (*workapiv1.ManifestWork, error) { work := &workapiv1.ManifestWork{ ObjectMeta: metav1.ObjectMeta{ Name: name, Namespace: namespace, }, } if err := workClient.Patch(ctx, work, client.RawPatch(pt, data)); err != nil { return nil, err } if err := workClient.Get(ctx, types.NamespacedName{Namespace: namespace, Name: name}, work); err != nil { return nil, err } return work, nil }, createWork: func(ctx context.Context, work *workapiv1.ManifestWork) (*workapiv1.ManifestWork, error) { err := workClient.Create(ctx, work) return work, err }, } }
-
Apply a manifestWork.
This method will create the manifestWork if the manifestWork does not exist, and will update the existing manifestWork.
appliedWork, err := workApplier.Apply(context, manifestWork)
-
Delete a manifestWork.
err := workApplier.Delete(context, namesapce, name)