diff --git a/go.mod b/go.mod index f89b0bb..870f225 100644 --- a/go.mod +++ b/go.mod @@ -86,6 +86,7 @@ require ( github.com/monochromegane/go-gitignore v0.0.0-20200626010858-205db1a8cc00 // indirect github.com/munnerz/goautoneg v0.0.0-20191010083416-a7dc8b61c822 // indirect github.com/mxk/go-flowrate v0.0.0-20140419014527-cca7078d478f // indirect + github.com/onsi/gomega v1.31.1 // indirect github.com/opencontainers/go-digest v1.0.1-0.20231025023718-d50d2fec9c98 // indirect github.com/opencontainers/image-spec v1.1.0 // indirect github.com/peterbourgon/diskv v2.0.1+incompatible // indirect @@ -136,6 +137,7 @@ require ( github.com/fluxcd/image-reflector-controller/api v0.31.2 github.com/fluxcd/kustomize-controller/api v1.2.2 github.com/fluxcd/notification-controller/api v1.2.4 + github.com/fluxcd/pkg/runtime v0.44.1 github.com/fluxcd/source-controller/api v1.2.4 github.com/go-chi/chi v1.5.5 github.com/go-logr/logr v1.4.1 // indirect diff --git a/go.sum b/go.sum index ebfadc3..e16c55a 100644 --- a/go.sum +++ b/go.sum @@ -111,6 +111,8 @@ github.com/fluxcd/pkg/apis/kustomize v1.3.0 h1:qvB46CfaOWcL1SyR2RiVWN/j7/035D0Ot github.com/fluxcd/pkg/apis/kustomize v1.3.0/go.mod h1:PCXf5kktTzNav0aH2Ns3jsowqwmA9xTcsrEOoPzx/K8= github.com/fluxcd/pkg/apis/meta v1.3.0 h1:KxeEc6olmSZvQ5pBONPE4IKxyoWQbqTJF1X6K5nIXpU= github.com/fluxcd/pkg/apis/meta v1.3.0/go.mod h1:3Ui8xFkoU4sYehqmscjpq7NjqH2YN1A2iX2okbO3/yA= +github.com/fluxcd/pkg/runtime v0.44.1 h1:XuPTcNIgn/NsoIo/A6qfPZaD9E7cbnJTDbeNw8O1SZQ= +github.com/fluxcd/pkg/runtime v0.44.1/go.mod h1:s1AhSOTCEBPaTfz/GdBD/Ws66uOByIuNP4Znrq+is9M= github.com/fluxcd/source-controller/api v1.2.4 h1:XjKTWhSSeLGsogWnTcLl5sUnyMlC5TKDbbBgP9SyJ5c= github.com/fluxcd/source-controller/api v1.2.4/go.mod h1:j3QSHpIPBP5sjaGIkVtsgWCx8JcOmcsutRmdJmRMOZg= github.com/foxcpp/go-mockdns v1.0.0 h1:7jBqxd3WDWwi/6WhDvacvH1XsN3rOLXyHM1uhvIx6FI= @@ -282,6 +284,8 @@ github.com/onsi/ginkgo/v2 v2.14.0 h1:vSmGj2Z5YPb9JwCWT6z6ihcUvDhuXLc3sJiqd3jMKAY github.com/onsi/ginkgo/v2 v2.14.0/go.mod h1:JkUdW7JkN0V6rFvsHcJ478egV3XH9NxpD27Hal/PhZw= github.com/onsi/gomega v1.30.0 h1:hvMK7xYz4D3HapigLTeGdId/NcfQx1VHMJc60ew99+8= github.com/onsi/gomega v1.30.0/go.mod h1:9sxs+SwGrKI0+PWe4Fxa9tFQQBG5xSsSbMXOI8PPpoQ= +github.com/onsi/gomega v1.31.1 h1:KYppCUK+bUgAZwHOu7EXVBKyQA6ILvOESHkn/tgoqvo= +github.com/onsi/gomega v1.31.1/go.mod h1:y40C95dwAD1Nz36SsEnxvfFe8FFfNxzI5eJ0EYGyAy0= github.com/opencontainers/go-digest v1.0.1-0.20231025023718-d50d2fec9c98 h1:H55sU3giNgBkIvmAo0vI/AAFwVTwfWsf6MN3+9H6U8o= github.com/opencontainers/go-digest v1.0.1-0.20231025023718-d50d2fec9c98/go.mod h1:RqnyioA3pIEZMkSbOIcrw32YSgETfn/VrLuEikEdPNU= github.com/opencontainers/image-spec v1.1.0 h1:8SG7/vwALn54lVB/0yZ/MMwhFrPYtpEHQb2IpWsCzug= diff --git a/pkg/api/api.go b/pkg/api/api.go index aafe1a1..c8ee2e7 100644 --- a/pkg/api/api.go +++ b/pkg/api/api.go @@ -216,6 +216,19 @@ func suspend(w http.ResponseWriter, r *http.Request) { w.Write([]byte("{}")) } +func resume(w http.ResponseWriter, r *http.Request) { + resource := r.URL.Query().Get("resource") + namespace := r.URL.Query().Get("namespace") + name := r.URL.Query().Get("name") + config, _ := r.Context().Value("config").(*rest.Config) + + reconcileCommand := flux.NewResumeCommand(resource) + go reconcileCommand.Run(config, namespace, name) + + w.WriteHeader(http.StatusOK) + w.Write([]byte("{}")) +} + func reconcile(w http.ResponseWriter, r *http.Request) { resource := r.URL.Query().Get("resource") namespace := r.URL.Query().Get("namespace") diff --git a/pkg/api/router.go b/pkg/api/router.go index ed549b2..e0da429 100644 --- a/pkg/api/router.go +++ b/pkg/api/router.go @@ -40,6 +40,7 @@ func SetupRouter( r.Get("/api/logs", streamLogs) r.Get("/api/stopLogs", stopLogs) r.Post("/api/suspend", suspend) + r.Post("/api/resume", resume) r.Post("/api/reconcile", reconcile) r.Get("/ws/", func(w http.ResponseWriter, r *http.Request) { streaming.ServeWs(clientHub, w, r) diff --git a/pkg/flux/helmrelease.go b/pkg/flux/helmrelease.go index 610d06c..24031f7 100644 --- a/pkg/flux/helmrelease.go +++ b/pkg/flux/helmrelease.go @@ -44,6 +44,18 @@ func (obj helmReleaseAdapter) setSuspended() { obj.HelmRelease.Spec.Suspend = true } +func (obj helmReleaseAdapter) setUnsuspended() { + obj.HelmRelease.Spec.Suspend = false +} + +func (obj helmReleaseAdapter) getObservedGeneration() int64 { + return obj.HelmRelease.Status.ObservedGeneration +} + +func (obj helmReleaseAdapter) isStatic() bool { + return false +} + func (obj helmReleaseAdapter) lastHandledReconcileRequest() string { return obj.Status.GetLastHandledReconcileRequest() } @@ -67,3 +79,7 @@ func (h helmReleaseListAdapter) len() int { func (a helmReleaseListAdapter) item(i int) suspendable { return &helmReleaseAdapter{&a.HelmReleaseList.Items[i]} } + +func (a helmReleaseListAdapter) resumeItem(i int) resumable { + return &helmReleaseAdapter{&a.HelmReleaseList.Items[i]} +} diff --git a/pkg/flux/kustomization.go b/pkg/flux/kustomization.go index 681409b..b06385b 100644 --- a/pkg/flux/kustomization.go +++ b/pkg/flux/kustomization.go @@ -44,6 +44,18 @@ func (obj kustomizationAdapter) setSuspended() { obj.Kustomization.Spec.Suspend = true } +func (obj kustomizationAdapter) setUnsuspended() { + obj.Kustomization.Spec.Suspend = false +} + +func (obj kustomizationAdapter) getObservedGeneration() int64 { + return obj.Kustomization.Status.ObservedGeneration +} + +func (obj kustomizationAdapter) isStatic() bool { + return false +} + func (obj kustomizationAdapter) lastHandledReconcileRequest() string { return obj.Status.GetLastHandledReconcileRequest() } @@ -67,3 +79,7 @@ func (a kustomizationListAdapter) len() int { func (a kustomizationListAdapter) item(i int) suspendable { return &kustomizationAdapter{&a.KustomizationList.Items[i]} } + +func (a kustomizationListAdapter) resumeItem(i int) resumable { + return &kustomizationAdapter{&a.KustomizationList.Items[i]} +} diff --git a/pkg/flux/resume.go b/pkg/flux/resume.go new file mode 100644 index 0000000..29a316e --- /dev/null +++ b/pkg/flux/resume.go @@ -0,0 +1,295 @@ +/* +Copyright 2020 The Flux 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. +Original version: https://github.com/fluxcd/flux2/blob/437a94367784541695fa68deba7a52b188d97ea8/cmd/flux/resume.go +*/ + +package flux + +import ( + "context" + "fmt" + "time" + + helmv2beta1 "github.com/fluxcd/helm-controller/api/v2beta1" + kustomizationv1 "github.com/fluxcd/kustomize-controller/api/v1" + "github.com/fluxcd/pkg/apis/meta" + "github.com/fluxcd/pkg/runtime/object" + "github.com/fluxcd/pkg/runtime/patch" + sourcev1 "github.com/fluxcd/source-controller/api/v1" + sourcev1beta2 "github.com/fluxcd/source-controller/api/v1beta2" + "github.com/sirupsen/logrus" + apimeta "k8s.io/apimachinery/pkg/api/meta" + "k8s.io/apimachinery/pkg/runtime" + "k8s.io/apimachinery/pkg/runtime/schema" + "k8s.io/apimachinery/pkg/types" + "k8s.io/apimachinery/pkg/util/wait" + "k8s.io/client-go/rest" + kstatus "sigs.k8s.io/cli-utils/pkg/kstatus/status" + "sigs.k8s.io/controller-runtime/pkg/client" +) + +// objectStatusType is the type of object in terms of status when computing the +// readiness of an object. Readiness check method depends on the type of object. +// For a dynamic object, Ready status condition is considered only for the +// latest generation of the object. For a static object that don't have any +// condition, the object generation is not considered. +type objectStatusType int + +const ( + objectStatusDynamic objectStatusType = iota + objectStatusStatic +) + +type resumeCommand struct { + kind string + groupVersion schema.GroupVersion + list listResumable +} + +type listResumable interface { + asClientList() client.ObjectList + len() int + resumeItem(i int) resumable +} + +type resumable interface { + asClientObject() client.Object + deepCopyClientObject() client.Object + GetGeneration() int64 + getObservedGeneration() int64 + setUnsuspended() + isStatic() bool + successMessage() string +} + +func NewResumeCommand(resource string) *resumeCommand { + switch resource { + case "kustomization": + return &resumeCommand{ + kind: kustomizationv1.KustomizationKind, + groupVersion: kustomizationv1.GroupVersion, + list: kustomizationListAdapter{&kustomizationv1.KustomizationList{}}, + } + case "helmrelease": + return &resumeCommand{ + kind: helmv2beta1.HelmReleaseKind, + groupVersion: helmv2beta1.GroupVersion, + list: helmReleaseListAdapter{&helmv2beta1.HelmReleaseList{}}, + } + case sourcev1.GitRepositoryKind: + return &resumeCommand{ + kind: sourcev1.GitRepositoryKind, + groupVersion: sourcev1.GroupVersion, + list: gitRepositoryListAdapter{&sourcev1.GitRepositoryList{}}, + } + case sourcev1beta2.OCIRepositoryKind: + return &resumeCommand{ + kind: sourcev1beta2.OCIRepositoryKind, + groupVersion: sourcev1beta2.GroupVersion, + list: ociRepositoryListAdapter{&sourcev1beta2.OCIRepositoryList{}}, + } + case sourcev1beta2.BucketKind: + return &resumeCommand{ + kind: sourcev1beta2.BucketKind, + groupVersion: sourcev1beta2.GroupVersion, + list: bucketListAdapter{&sourcev1beta2.BucketList{}}, + } + } + + return nil +} + +func (r *resumeCommand) Run(config *rest.Config, namespace, name string) { + scheme := runtime.NewScheme() + sourcev1.AddToScheme(scheme) + sourcev1beta2.AddToScheme(scheme) + kustomizationv1.AddToScheme(scheme) + helmv2beta1.AddToScheme(scheme) + + kubeClient, err := client.NewWithWatch(config, client.Options{ + Scheme: scheme, + }) + if err != nil { + logrus.Errorf("kubernetes client initialization failed: %s", err) + return + } + + listOpts := []client.ListOption{ + client.InNamespace(namespace), + client.MatchingFields{ + "metadata.name": name, + }, + } + + obj, err := r.patch(context.TODO(), kubeClient, listOpts, namespace) + if err != nil { + if err == ErrNoObjectsFound { + logrus.Errorf("%s %s not found in %s namespace", r.kind, name, namespace) + } else { + logrus.Errorf("failed suspending %s %s in %s namespace: %s", r.kind, name, namespace, err.Error()) + } + } + r.reconcile(kubeClient, obj, namespace) +} + +// Patches resumable object by setting status to unsuspended. +// Returns a resumable that have been patched and any error encountered during patching. +func (r resumeCommand) patch(ctx context.Context, kubeClient client.WithWatch, listOpts []client.ListOption, namespace string) (resumable, error) { + if err := kubeClient.List(ctx, r.list.asClientList(), listOpts...); err != nil { + return nil, err + } + + if r.list.len() == 0 { + logrus.Errorf("no %s objects found in %s namespace", r.kind, namespace) + return nil, nil + } + + var resumables []resumable + for i := 0; i < r.list.len(); i++ { + obj := r.list.resumeItem(i) + logrus.Infof("resuming %s %s in %s namespace", r.kind, obj.asClientObject().GetName(), namespace) + + patch := client.MergeFrom(obj.deepCopyClientObject()) + obj.setUnsuspended() + if err := kubeClient.Patch(ctx, obj.asClientObject(), patch); err != nil { + return nil, err + } + + resumables = append(resumables, obj) + + logrus.Infof("%s resumed", r.kind) + } + + return resumables[0], nil +} + +// Waits for resumable object to be reconciled and returns the object and any error encountered while waiting. +func (r resumeCommand) reconcile(kubeClient client.WithWatch, res resumable, namespace string) { + namespacedName := types.NamespacedName{ + Name: res.asClientObject().GetName(), + Namespace: namespace, + } + + logrus.Infof("waiting for %s reconciliation", r.kind) + + readyConditionFunc := isObjectReadyConditionFunc(kubeClient, namespacedName, res.asClientObject()) + if res.isStatic() { + readyConditionFunc = isStaticObjectReadyConditionFunc(kubeClient, namespacedName, res.asClientObject()) + } + + if err := wait.PollUntilContextTimeout(context.TODO(), 2*time.Second, 5*time.Minute, true, readyConditionFunc); err != nil { + logrus.Error(err) + return + } + + logrus.Infof("%s %s reconciliation completed", r.kind, res.asClientObject().GetName()) + logrus.Infof(res.successMessage()) +} + +// isObjectReady determines if an object is ready using the kstatus.Compute() +// result. statusType helps differenciate between static and dynamic objects to +// accurately check the object's readiness. A dynamic object may have some extra +// considerations depending on the object. +func isObjectReady(obj client.Object, statusType objectStatusType) (bool, error) { + observedGen, err := object.GetStatusObservedGeneration(obj) + if err != nil && err != object.ErrObservedGenerationNotFound { + return false, err + } + + if statusType == objectStatusDynamic { + // Object not reconciled yet. + if observedGen < 1 { + return false, nil + } + + cobj, ok := obj.(meta.ObjectWithConditions) + if !ok { + return false, fmt.Errorf("unable to get conditions from object") + } + + if c := apimeta.FindStatusCondition(cobj.GetConditions(), meta.ReadyCondition); c != nil { + // Ensure that the ready condition is for the latest generation of + // the object. + // NOTE: Some APIs like ImageUpdateAutomation and HelmRelease don't + // support per condition observed generation yet. Per condition + // observed generation for them are always zero. + // There are two strategies used across different object kinds to + // check the latest ready condition: + // - check that the ready condition's generation matches the + // object's generation. + // - check that the observed generation of the object in the + // status matches the object's generation. + // + // TODO: Once ImageUpdateAutomation and HelmRelease APIs have per + // condition observed generation, remove the object's observed + // generation and object's generation check (the second condition + // below). Also, try replacing this readiness check function with + // fluxcd/pkg/ssa's ResourceManager.Wait(), which uses kstatus + // internally to check readiness of the objects. + if c.ObservedGeneration != 0 && c.ObservedGeneration != obj.GetGeneration() { + return false, nil + } + if c.ObservedGeneration == 0 && observedGen != obj.GetGeneration() { + return false, nil + } + } else { + return false, nil + } + } + + u, err := patch.ToUnstructured(obj) + if err != nil { + return false, err + } + result, err := kstatus.Compute(u) + if err != nil { + return false, err + } + switch result.Status { + case kstatus.CurrentStatus: + return true, nil + case kstatus.InProgressStatus: + return false, nil + default: + return false, fmt.Errorf(result.Message) + } +} + +// isObjectReadyConditionFunc returns a wait.ConditionFunc to be used with +// wait.Poll* while polling for an object with dynamic status to be ready. +func isObjectReadyConditionFunc(kubeClient client.Client, namespaceName types.NamespacedName, obj client.Object) wait.ConditionWithContextFunc { + return func(ctx context.Context) (bool, error) { + err := kubeClient.Get(ctx, namespaceName, obj) + if err != nil { + return false, err + } + + return isObjectReady(obj, objectStatusDynamic) + } +} + +// isStaticObjectReadyConditionFunc returns a wait.ConditionFunc to be used with +// wait.Poll* while polling for an object with static or no status to be +// ready. +func isStaticObjectReadyConditionFunc(kubeClient client.Client, namespaceName types.NamespacedName, obj client.Object) wait.ConditionWithContextFunc { + return func(ctx context.Context) (bool, error) { + err := kubeClient.Get(ctx, namespaceName, obj) + if err != nil { + return false, err + } + + return isObjectReady(obj, objectStatusStatic) + } +} diff --git a/pkg/flux/source.go b/pkg/flux/source.go index db9d343..888404d 100644 --- a/pkg/flux/source.go +++ b/pkg/flux/source.go @@ -46,6 +46,18 @@ func (obj gitRepositoryAdapter) setSuspended() { obj.GitRepository.Spec.Suspend = true } +func (obj gitRepositoryAdapter) setUnsuspended() { + obj.GitRepository.Spec.Suspend = false +} + +func (obj gitRepositoryAdapter) getObservedGeneration() int64 { + return obj.GitRepository.Status.ObservedGeneration +} + +func (obj gitRepositoryAdapter) isStatic() bool { + return false +} + func (obj gitRepositoryAdapter) lastHandledReconcileRequest() string { return obj.Status.GetLastHandledReconcileRequest() } @@ -70,6 +82,10 @@ func (a gitRepositoryListAdapter) item(i int) suspendable { return &gitRepositoryAdapter{&a.GitRepositoryList.Items[i]} } +func (a gitRepositoryListAdapter) resumeItem(i int) resumable { + return &gitRepositoryAdapter{&a.GitRepositoryList.Items[i]} +} + type ociRepositoryAdapter struct { *sourcev1beta2.OCIRepository } @@ -90,6 +106,18 @@ func (obj ociRepositoryAdapter) setSuspended() { obj.OCIRepository.Spec.Suspend = true } +func (obj ociRepositoryAdapter) setUnsuspended() { + obj.OCIRepository.Spec.Suspend = false +} + +func (obj ociRepositoryAdapter) getObservedGeneration() int64 { + return obj.OCIRepository.Status.ObservedGeneration +} + +func (obj ociRepositoryAdapter) isStatic() bool { + return false +} + func (obj ociRepositoryAdapter) lastHandledReconcileRequest() string { return obj.Status.GetLastHandledReconcileRequest() } @@ -114,6 +142,10 @@ func (a ociRepositoryListAdapter) item(i int) suspendable { return &ociRepositoryAdapter{&a.OCIRepositoryList.Items[i]} } +func (a ociRepositoryListAdapter) resumeItem(i int) resumable { + return &ociRepositoryAdapter{&a.OCIRepositoryList.Items[i]} +} + type bucketAdapter struct { *sourcev1beta2.Bucket } @@ -134,6 +166,18 @@ func (obj bucketAdapter) setSuspended() { obj.Bucket.Spec.Suspend = true } +func (obj bucketAdapter) setUnsuspended() { + obj.Bucket.Spec.Suspend = false +} + +func (obj bucketAdapter) getObservedGeneration() int64 { + return obj.Bucket.Status.ObservedGeneration +} + +func (obj bucketAdapter) isStatic() bool { + return false +} + func (obj bucketAdapter) lastHandledReconcileRequest() string { return obj.Status.GetLastHandledReconcileRequest() } @@ -157,3 +201,7 @@ func (a bucketListAdapter) len() int { func (a bucketListAdapter) item(i int) suspendable { return &bucketAdapter{&a.BucketList.Items[i]} } + +func (a bucketListAdapter) resumeItem(i int) resumable { + return &bucketAdapter{&a.BucketList.Items[i]} +} diff --git a/pkg/flux/suspend.go b/pkg/flux/suspend.go index e3db6ba..b0ea73f 100644 --- a/pkg/flux/suspend.go +++ b/pkg/flux/suspend.go @@ -101,7 +101,6 @@ func (s *suspendCommand) Run(config *rest.Config, namespace, name string) { kustomizationv1.AddToScheme(scheme) helmv2beta1.AddToScheme(scheme) - //TODO log.SetLogger(...) was never called; logs will not be displayed. kubeClient, err := client.NewWithWatch(config, client.Options{ Scheme: scheme, }) diff --git a/web/src/HelmRelease.jsx b/web/src/HelmRelease.jsx index f7dedae..85efde7 100644 --- a/web/src/HelmRelease.jsx +++ b/web/src/HelmRelease.jsx @@ -35,12 +35,17 @@ export function HelmRelease(props) {
-
+
+
-
+
+
-
+
+