diff --git a/experiments/compositions/composition/internal/controller/expander_reconciler.go b/experiments/compositions/composition/internal/controller/expander_reconciler.go index 45819f4d00..20c2ffb721 100644 --- a/experiments/compositions/composition/internal/controller/expander_reconciler.go +++ b/experiments/compositions/composition/internal/controller/expander_reconciler.go @@ -169,13 +169,20 @@ func (r *ExpanderReconciler) Reconcile(ctx context.Context, req ctrl.Request) (c logger.Error(err, "unable to update Plan status") } }() + expanderDebugLogsEnabled := false + _, exist := inputcr.GetAnnotations()["composition-expander-debug-logs"] + if exist { + expanderDebugLogsEnabled = true + logger.Info("annotation'composition-expander-debug-logs' is turned on.") + } + logger.Info("annotation'composition-expander-debug-logs' is turned off.") // ------------------- EVALUATION SECTION ----------------------- stagesEvaluated := []string{} values := map[string]interface{}{} planUpdated := false waitRequested := false - for _, expander := range compositionCR.Spec.Expanders { + for i, expander := range compositionCR.Spec.Expanders { logger = loggerCR.WithName(expander.Name).WithName("Expand") uri, ev, reason, err := r.getExpanderValue(ctx, expander.Version, expander.Type) @@ -188,6 +195,10 @@ func (r *ExpanderReconciler) Reconcile(ctx context.Context, req ctrl.Request) (c logger.Info("Got valid expander uri", "uri", uri) + if expanderDebugLogsEnabled { + // TODO(@xiaoweim): Log ressource UID in the future if possible + r.Recorder.Event(&inputcr, "Normal", fmt.Sprintf("Running expander stage %d: %s", i, expander.Name), fmt.Sprintf("version: %d", inputcr.GetGeneration())) + } if ev.Spec.Type == compositionv1alpha1.ExpanderTypeJob { reason, err := r.runJob(ctx, logger, &inputcr, expander.Name, planNN.Name, uri, ev.Spec.ImageRegistry) if err != nil { @@ -195,7 +206,7 @@ func (r *ExpanderReconciler) Reconcile(ctx context.Context, req ctrl.Request) (c return ctrl.Result{}, err } } else { - newValues, updated, reason, err := r.evaluateAndSavePlan(ctx, logger, &inputcr, values, expander, planNN, ev, uri) + newValues, updated, reason, err := r.evaluateAndSavePlan(ctx, logger, &inputcr, values, expander, planNN, ev, uri, expanderDebugLogsEnabled) _, iswaitErr := err.(*EvaluateWaitError) if iswaitErr { newStatus.AppendWaitingCondition(expander.Name, err.Error(), reason) @@ -324,6 +335,16 @@ func (r *ExpanderReconciler) Reconcile(ctx context.Context, req ctrl.Request) (c newStatus.ClearCondition(compositionv1alpha1.Ready) message := fmt.Sprintf("Evaluated stages: %s \n Applied stages: %s", strings.Join(stagesEvaluated, ", "), strings.Join(stagesApplied, ", ")) newStatus.AppendCondition(compositionv1alpha1.Ready, metav1.ConditionFalse, message, "PendingStages") + + if expanderDebugLogsEnabled { + // TODO(@xiaoweim): Log ressource UID in the future if possible + r.Recorder.Event(&inputcr, "Normal", fmt.Sprintf("Finished expander stage %d: %s", index, expander), fmt.Sprintf("version: %d, resource count: %d", inputcr.GetGeneration(), applier.Count())) + for i, resourceStatus := range newStatus.Stages[expander].LastApplied { + logger.Info("Resource %d\n", i) + logger.Info("Resource name: %s, resource namespace: %s, resource gvk: %s %s %s, resource status: %s", + resourceStatus.Name, resourceStatus.Namespace, resourceStatus.Group, resourceStatus.Version, resourceStatus.Kind, resourceStatus.Health) + } + } } // Inject plan.Ready Condition with list of expanders @@ -413,7 +434,7 @@ func (r *ExpanderReconciler) runJob(ctx context.Context, logger logr.Logger, func (r *ExpanderReconciler) evaluateAndSavePlan(ctx context.Context, logger logr.Logger, cr *unstructured.Unstructured, values map[string]interface{}, expander compositionv1alpha1.Expander, - planNN types.NamespacedName, ev *compositionv1alpha1.ExpanderVersion, grpcService string) (map[string]interface{}, bool, string, error) { + planNN types.NamespacedName, ev *compositionv1alpha1.ExpanderVersion, grpcService string, expanderDebugLogEnabled bool) (map[string]interface{}, bool, string, error) { // Set up a connection to the server. updated := false @@ -488,16 +509,18 @@ func (r *ExpanderReconciler) evaluateAndSavePlan(ctx context.Context, logger log logger.Error(err, "failed to marshall Getter Values") return values, updated, "MarshallValuesFailed", err } - + evaluateRequest := &pb.EvaluateRequest{ + Config: configBytes, + Context: contextBytes, + Facade: facadeBytes, + Resource: r.InputGVR.Resource, + Value: valuesBytes, + } expanderClient := pb.NewExpanderClient(conn) - result, err := expanderClient.Evaluate(ctx, - &pb.EvaluateRequest{ - Config: configBytes, - Context: contextBytes, - Facade: facadeBytes, - Resource: r.InputGVR.Resource, - Value: valuesBytes, - }) + if expanderDebugLogEnabled { + logger.Info(fmt.Sprintf("expander version: %d, sending expander request: %v", cr.GetGeneration(), evaluateRequest)) + } + result, err := expanderClient.Evaluate(ctx, evaluateRequest) if err != nil { logger.Error(err, "expander.Evaluate() Failed", "expander", expander.Name) return values, updated, "EvaluateError", err @@ -512,6 +535,10 @@ func (r *ExpanderReconciler) evaluateAndSavePlan(ctx context.Context, logger log err = fmt.Errorf("Evaluate Failed: %s", result.Error.Message) return values, updated, "EvaluateStatusFailed", err } + if expanderDebugLogEnabled { + logger.Info(fmt.Sprintf("expander version: %d, sent expander request: %v, received results: %v", cr.GetGeneration(), evaluateRequest, result)) + r.Recorder.Event(cr, "Normal", fmt.Sprintf("Expander stage %s evaluation completed", expander.Name), fmt.Sprintf("version: %d, request: %v, result: %v", cr.GetGeneration(), evaluateRequest, result)) + } // Write to Plan object // Re-read the Plan CR to load the expanded manifests diff --git a/experiments/compositions/composition/tests/data/TestSimpleCompositionExpanderLoggingEnabled/input.yaml b/experiments/compositions/composition/tests/data/TestSimpleCompositionExpanderLoggingEnabled/input.yaml new file mode 100644 index 0000000000..1dfcc6d231 --- /dev/null +++ b/experiments/compositions/composition/tests/data/TestSimpleCompositionExpanderLoggingEnabled/input.yaml @@ -0,0 +1,161 @@ +# Copyright 2024 Google LLC +# +# 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. + + +apiVersion: apiextensions.k8s.io/v1 +kind: CustomResourceDefinition +metadata: + name: pconfigs.facade.foocorp.com +spec: + group: facade.foocorp.com + names: + kind: PConfig + listKind: PConfigList + plural: pconfigs + singular: pconfig + scope: Namespaced + versions: + - name: v1alpha1 + schema: + openAPIV3Schema: + description: Schema for the pconfig + properties: + apiVersion: + description: api-version of api + type: string + kind: + description: gvk Kind + type: string + metadata: + type: object + spec: + description: PConfig spec + properties: + project: + type: string + required: + - project + type: object + status: + description: PConfig status + type: object + type: object + served: true + storage: true + subresources: + status: {} +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRole +metadata: + name: composition-facade-foocorp +rules: +- apiGroups: + - facade.foocorp.com + resources: + - '*' + verbs: + - get + - list + - patch + - update + - watch + - create + - delete +- apiGroups: + - facade.foocorp.com + resources: + - "*/status" + verbs: + - get + - update +- apiGroups: ["*"] + resources: ["*"] + verbs: ["*"] +--- +apiVersion: rbac.authorization.k8s.io/v1 +kind: ClusterRoleBinding +metadata: + name: composition-facade-foocorp +roleRef: + apiGroup: rbac.authorization.k8s.io + kind: ClusterRole + name: composition-facade-foocorp +subjects: +- kind: ServiceAccount + name: composition-controller-manager + namespace: composition-system +--- +apiVersion: composition.google.com/v1alpha1 +kind: GetterConfiguration +metadata: + name: projectvalues + namespace: default +spec: + valuesFrom: + - name: commondata + resourceRef: + group: "" + version: v1 + kind: ConfigMap + resource: configmaps + name: common-config + fieldRef: + - path: ".data" + as: data +--- +apiVersion: composition.google.com/v1alpha1 +kind: Composition +metadata: + name: projectconfigmap + namespace: default +spec: + inputAPIGroup: pconfigs.facade.foocorp.com + expanders: + - type: jinja2 + version: v0.0.1 + name: common + template: | + apiVersion: v1 + kind: ConfigMap + metadata: + name: common-config + namespace: {{ pconfigs.metadata.namespace }} + labels: + createdby: "composition-namespaceconfigmap" + data: + key1: value1 +--- +apiVersion: v1 +kind: Namespace +metadata: + name: team-a +--- +apiVersion: composition.google.com/v1alpha1 +kind: Context +metadata: + name: context + namespace: team-a +spec: + project: proj-a +--- +apiVersion: facade.foocorp.com/v1alpha1 +kind: PConfig +metadata: + name: team-a-config + namespace: team-a + annotations: + composition-expander-debug-logs: "true" +spec: + project: proj-a \ No newline at end of file diff --git a/experiments/compositions/composition/tests/data/TestSimpleCompositionExpanderLoggingEnabled/output.yaml b/experiments/compositions/composition/tests/data/TestSimpleCompositionExpanderLoggingEnabled/output.yaml new file mode 100644 index 0000000000..bb1f9313a9 --- /dev/null +++ b/experiments/compositions/composition/tests/data/TestSimpleCompositionExpanderLoggingEnabled/output.yaml @@ -0,0 +1,23 @@ +# Copyright 2024 Google LLC +# +# 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. + +apiVersion: v1 +kind: ConfigMap +metadata: + name: common-config + namespace: team-a + labels: + createdby: "composition-namespaceconfigmap" +data: + key1: value1 \ No newline at end of file diff --git a/experiments/compositions/composition/tests/testcases/simple_test.go b/experiments/compositions/composition/tests/testcases/simple_test.go index ecfb0fd456..69192ebba8 100644 --- a/experiments/compositions/composition/tests/testcases/simple_test.go +++ b/experiments/compositions/composition/tests/testcases/simple_test.go @@ -372,3 +372,14 @@ func TestSimpleExpanderVersionInvalid(t *testing.T) { condition = utils.GetErrorCondition("VersionNotFound", "") s.C.MustHaveCondition(plan, condition, scenario.CompositionReconcileTimeout) } + +func TestSimpleCompositionExpanderLoggingEnabled(t *testing.T) { + s := scenario.NewBasic(t) + defer s.Cleanup() + s.Setup() + + plan := utils.GetPlanObj("team-a", "pconfigs-team-a-config") + condition := utils.GetWaitingCondition("EvaluateStatusWait", "") + s.C.MustNotHaveCondition(plan, condition, 3*scenario.CompositionReconcileTimeout) + s.VerifyOutputExists() +}