From 784a9dfc30dbce41008c006828b6fc524151865b Mon Sep 17 00:00:00 2001
From: Philippe Scorsolini
Date: Fri, 29 Mar 2024 07:41:44 +0000
Subject: [PATCH] fix: return fatal on required envconfig missing
Signed-off-by: Philippe Scorsolini
---
fn.go | 7 +++--
fn_test.go | 85 +++++++++++++++++++++++++++++++++++++++++++++++++++---
2 files changed, 86 insertions(+), 6 deletions(-)
diff --git a/fn.go b/fn.go
index 1d2b633..edd11d1 100644
--- a/fn.go
+++ b/fn.go
@@ -174,8 +174,11 @@ func verifyAndSortExtras(in *v1beta1.Input, extraResources map[string][]resource
}
switch extraResource.GetType() {
case v1beta1.ResourceSourceTypeReference:
- if len(resources) == 0 && in.Spec.Policy.IsResolutionPolicyOptional() {
- continue
+ if len(resources) == 0 {
+ if in.Spec.Policy.IsResolutionPolicyOptional() {
+ continue
+ }
+ return nil, errors.Errorf("Required extra resource %q not found", extraResName)
}
if len(resources) > 1 {
return nil, errors.Errorf("expected exactly one extra resource %q, got %d", extraResName, len(resources))
diff --git a/fn_test.go b/fn_test.go
index 0a8959b..dea92d8 100644
--- a/fn_test.go
+++ b/fn_test.go
@@ -6,7 +6,7 @@ import (
"github.com/google/go-cmp/cmp"
"github.com/google/go-cmp/cmp/cmpopts"
- "google.golang.org/protobuf/testing/protocmp"
+ "google.golang.org/protobuf/encoding/protojson"
"google.golang.org/protobuf/types/known/durationpb"
"google.golang.org/protobuf/types/known/structpb"
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
@@ -53,7 +53,7 @@ func TestRunFunction(t *testing.T) {
},
},
Input: resource.MustStructJSON(`{
- "apiVersion": "template.fn.crossplane.io/v1beta1",
+ "apiVersion": "extra-resources.fn.crossplane.io/v1beta1",
"kind": "Input",
"spec": {
"extraResources": [
@@ -291,7 +291,7 @@ func TestRunFunction(t *testing.T) {
},
},
Input: resource.MustStructJSON(`{
- "apiVersion": "template.fn.crossplane.io/v1beta1",
+ "apiVersion": "extra-resources.fn.crossplane.io/v1beta1",
"kind": "Input",
"spec": {
"extraResources": [
@@ -489,6 +489,68 @@ func TestRunFunction(t *testing.T) {
},
},
},
+ "RequestEnvironmentConfigsNotFoundRequired": {
+ reason: "The Function should return fatal if a required EnvironmentConfig is not found",
+ args: args{
+ req: &fnv1beta1.RunFunctionRequest{
+ Meta: &fnv1beta1.RequestMeta{Tag: "hello"},
+ Observed: &fnv1beta1.State{
+ Composite: &fnv1beta1.Resource{
+ Resource: resource.MustStructJSON(`{
+ "apiVersion": "test.crossplane.io/v1alpha1",
+ "kind": "XR",
+ "metadata": {
+ "name": "my-xr"
+ }
+ }`),
+ },
+ },
+ ExtraResources: map[string]*fnv1beta1.Resources{
+ "environment-config-0": {
+ Items: []*fnv1beta1.Resource{},
+ },
+ },
+ Input: resource.MustStructJSON(`{
+ "apiVersion": "extra-resources.fn.crossplane.io/v1beta1",
+ "kind": "Input",
+ "spec": {
+ "extraResources": [
+ {
+ "type": "Reference",
+ "into": "obj-0",
+ "kind": "EnvironmentConfig",
+ "apiVersion": "apiextensions.crossplane.io/v1alpha1",
+ "ref": {
+ "name": "my-env-config"
+ }
+ }
+ ]
+ }
+ }`),
+ },
+ },
+ want: want{
+ rsp: &fnv1beta1.RunFunctionResponse{
+ Meta: &fnv1beta1.ResponseMeta{Tag: "hello", Ttl: durationpb.New(response.DefaultTTL)},
+ Results: []*fnv1beta1.Result{
+ {
+ Severity: fnv1beta1.Severity_SEVERITY_FATAL,
+ },
+ },
+ Requirements: &fnv1beta1.Requirements{
+ ExtraResources: map[string]*fnv1beta1.ResourceSelector{
+ "obj-0": {
+ ApiVersion: "apiextensions.crossplane.io/v1alpha1",
+ Kind: "EnvironmentConfig",
+ Match: &fnv1beta1.ResourceSelector_MatchName{
+ MatchName: "my-env-config",
+ },
+ },
+ },
+ },
+ },
+ },
+ },
}
for name, tc := range cases {
@@ -496,7 +558,22 @@ func TestRunFunction(t *testing.T) {
f := &Function{log: logging.NewNopLogger()}
rsp, err := f.RunFunction(tc.args.ctx, tc.args.req)
- if diff := cmp.Diff(tc.want.rsp, rsp, protocmp.Transform()); diff != "" {
+ diff := cmp.Diff(tc.want.rsp, rsp, cmpopts.AcyclicTransformer("toJsonWithoutResultMessages", func(r *fnv1beta1.RunFunctionResponse) []byte {
+ // We don't care about messages.
+ // cmptopts.IgnoreField wasn't working with protocmp.Transform
+ // We can't split this to another transformer as
+ // transformers are applied not in order but as soon as they
+ // match the type, which are walked from the root (RunFunctionResponse).
+ for _, result := range r.GetResults() {
+ result.Message = ""
+ }
+ out, err := protojson.Marshal(r)
+ if err != nil {
+ t.Fatalf("cannot marshal %T to JSON: %s", r, err)
+ }
+ return out
+ }))
+ if diff != "" {
t.Errorf("%s\nf.RunFunction(...): -want rsp, +got rsp:\n%s", tc.reason, diff)
}