Skip to content

Commit

Permalink
feat: add optional logging for expander stages
Browse files Browse the repository at this point in the history
  • Loading branch information
xiaoweim committed Jun 28, 2024
1 parent 148ffb5 commit b64a07a
Show file tree
Hide file tree
Showing 4 changed files with 234 additions and 12 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -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)
Expand All @@ -188,14 +195,18 @@ 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 {
newStatus.AppendErrorCondition(expander.Name, err.Error(), reason)
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)
Expand Down Expand Up @@ -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
Expand Down Expand Up @@ -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

Expand Down Expand Up @@ -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
Expand All @@ -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
Expand Down
Original file line number Diff line number Diff line change
@@ -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
Original file line number Diff line number Diff line change
@@ -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
Original file line number Diff line number Diff line change
Expand Up @@ -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()
}

0 comments on commit b64a07a

Please sign in to comment.