From 4f7743fa4fe822a0bfb2fe8a7adb1716ad86e6a1 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Charles-Edouard=20Br=C3=A9t=C3=A9ch=C3=A9?= Date: Wed, 4 Oct 2023 00:10:31 +0200 Subject: [PATCH] fix: custom marshalling for Any (#11) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Signed-off-by: Charles-Edouard Brétéché --- config/crds/json.kyverno.io_policies.yaml | 6 +--- pkg/apis/v1alpha1/any.go | 22 +++++++++--- pkg/apis/v1alpha1/context_entry.go | 4 ++- pkg/apis/v1alpha1/variable.go | 25 ------------- pkg/apis/v1alpha1/variable_test.go | 39 --------------------- pkg/apis/v1alpha1/zz_generated.deepcopy.go | 5 +-- pkg/data/crds/json.kyverno.io_policies.yaml | 6 +--- pkg/policy/load.go | 19 ++-------- testdata/payload-yaml/policy.yaml | 3 +- testdata/pod-all-latest/policy.yaml | 6 ++-- testdata/pod-no-latest/policy.yaml | 3 +- testdata/tf-plan/policy.yaml | 3 +- 12 files changed, 30 insertions(+), 111 deletions(-) delete mode 100644 pkg/apis/v1alpha1/variable.go delete mode 100644 pkg/apis/v1alpha1/variable_test.go diff --git a/config/crds/json.kyverno.io_policies.yaml b/config/crds/json.kyverno.io_policies.yaml index 4b102c8d..184801c3 100644 --- a/config/crds/json.kyverno.io_policies.yaml +++ b/config/crds/json.kyverno.io_policies.yaml @@ -94,11 +94,7 @@ spec: variable: description: Variable defines an arbitrary JMESPath context variable that can be defined inline. - properties: - value: - description: Value is any arbitrary object. - x-kubernetes-preserve-unknown-fields: true - type: object + x-kubernetes-preserve-unknown-fields: true required: - name type: object diff --git a/pkg/apis/v1alpha1/any.go b/pkg/apis/v1alpha1/any.go index 751c3153..a1a10e9c 100644 --- a/pkg/apis/v1alpha1/any.go +++ b/pkg/apis/v1alpha1/any.go @@ -1,18 +1,16 @@ package v1alpha1 import ( + "encoding/json" + "github.com/jinzhu/copier" ) -type Value = interface{} - // +k8s:deepcopy-gen=false type Any struct { - // TODO: this is needed to workaround a bug in api machinery code - // https://kubernetes.slack.com/archives/C0EG7JC6T/p1696331287543159 // +kubebuilder:pruning:PreserveUnknownFields // +kubebuilder:validation:Schemaless - Value `json:",inline"` + Value interface{} `json:",inline"` } func (in *Any) DeepCopyInto(out *Any) { @@ -20,3 +18,17 @@ func (in *Any) DeepCopyInto(out *Any) { panic("deep copy failed") } } + +func (a *Any) MarshalJSON() ([]byte, error) { + return json.Marshal(a.Value) +} + +func (a *Any) UnmarshalJSON(data []byte) error { + var v interface{} + err := json.Unmarshal(data, &v) + if err != nil { + return err + } + a.Value = v + return nil +} diff --git a/pkg/apis/v1alpha1/context_entry.go b/pkg/apis/v1alpha1/context_entry.go index 664d7b0d..e13d89d5 100644 --- a/pkg/apis/v1alpha1/context_entry.go +++ b/pkg/apis/v1alpha1/context_entry.go @@ -6,7 +6,9 @@ type ContextEntry struct { Name string `json:"name"` // Variable defines an arbitrary JMESPath context variable that can be defined inline. - Variable *Variable `json:"variable,omitempty"` + // +kubebuilder:pruning:PreserveUnknownFields + // +kubebuilder:validation:Schemaless + Variable Any `json:"variable,omitempty"` // ImageRegistry defines requests to an OCI/Docker V2 registry to fetch image details. ImageRegistry *ImageRegistry `json:"imageRegistry,omitempty"` diff --git a/pkg/apis/v1alpha1/variable.go b/pkg/apis/v1alpha1/variable.go deleted file mode 100644 index 1c6bb749..00000000 --- a/pkg/apis/v1alpha1/variable.go +++ /dev/null @@ -1,25 +0,0 @@ -package v1alpha1 - -import ( - "github.com/jinzhu/copier" -) - -// Variable defines an arbitrary JMESPath context variable that can be defined inline. -// +k8s:deepcopy-gen=false -type Variable struct { - // Value is any arbitrary object. - // +kubebuilder:pruning:PreserveUnknownFields - // +kubebuilder:validation:Schemaless - Value interface{} `json:"value,omitempty"` -} - -func (in *Variable) DeepCopy() *Variable { - if in == nil { - return nil - } - out := &Variable{} - if err := copier.Copy(out, in); err != nil { - panic("deep copy failed") - } - return out -} diff --git a/pkg/apis/v1alpha1/variable_test.go b/pkg/apis/v1alpha1/variable_test.go deleted file mode 100644 index 07719da2..00000000 --- a/pkg/apis/v1alpha1/variable_test.go +++ /dev/null @@ -1,39 +0,0 @@ -package v1alpha1 - -import ( - "reflect" - "testing" - - "github.com/stretchr/testify/assert" -) - -func TestVariable_DeepCopy(t *testing.T) { - tests := []struct { - name string - in *Variable - }{{ - name: "nil", - in: &Variable{nil}, - }, { - name: "nil", - in: nil, - }, { - name: "int", - in: &Variable{42}, - }, { - name: "string", - in: &Variable{"foo"}, - }, { - name: "slice", - in: &Variable{[]interface{}{42, "string"}}, - }} - for _, tt := range tests { - t.Run(tt.name, func(t *testing.T) { - if got := tt.in.DeepCopy(); !reflect.DeepEqual(got, tt.in) { - t.Errorf("Variable.DeepCopy() = %v, want %v", got, tt.in) - } else if tt.in != nil { - assert.False(t, got == tt.in) - } - }) - } -} diff --git a/pkg/apis/v1alpha1/zz_generated.deepcopy.go b/pkg/apis/v1alpha1/zz_generated.deepcopy.go index f104bb7a..1efc09f7 100644 --- a/pkg/apis/v1alpha1/zz_generated.deepcopy.go +++ b/pkg/apis/v1alpha1/zz_generated.deepcopy.go @@ -50,10 +50,7 @@ func (in Assertions) DeepCopy() Assertions { // DeepCopyInto is an autogenerated deepcopy function, copying the receiver, writing into out. in must be non-nil. func (in *ContextEntry) DeepCopyInto(out *ContextEntry) { *out = *in - if in.Variable != nil { - in, out := &in.Variable, &out.Variable - *out = (*in).DeepCopy() - } + in.Variable.DeepCopyInto(&out.Variable) if in.ImageRegistry != nil { in, out := &in.ImageRegistry, &out.ImageRegistry *out = new(ImageRegistry) diff --git a/pkg/data/crds/json.kyverno.io_policies.yaml b/pkg/data/crds/json.kyverno.io_policies.yaml index 4b102c8d..184801c3 100644 --- a/pkg/data/crds/json.kyverno.io_policies.yaml +++ b/pkg/data/crds/json.kyverno.io_policies.yaml @@ -94,11 +94,7 @@ spec: variable: description: Variable defines an arbitrary JMESPath context variable that can be defined inline. - properties: - value: - description: Value is any arbitrary object. - x-kubernetes-preserve-unknown-fields: true - type: object + x-kubernetes-preserve-unknown-fields: true required: - name type: object diff --git a/pkg/policy/load.go b/pkg/policy/load.go index ee188fee..dfbaac7e 100644 --- a/pkg/policy/load.go +++ b/pkg/policy/load.go @@ -9,10 +9,9 @@ import ( "github.com/kyverno/kyverno-json/pkg/apis/v1alpha1" "github.com/kyverno/kyverno-json/pkg/data" fileinfo "github.com/kyverno/kyverno-json/pkg/utils/file-info" + "github.com/kyverno/kyverno/cmd/cli/kubectl-kyverno/resource/convert" "github.com/kyverno/kyverno/cmd/cli/kubectl-kyverno/resource/loader" yamlutils "github.com/kyverno/kyverno/pkg/utils/yaml" - "k8s.io/apimachinery/pkg/apis/meta/v1/unstructured" - "k8s.io/apimachinery/pkg/runtime" "k8s.io/apimachinery/pkg/runtime/schema" "sigs.k8s.io/kubectl-validate/pkg/openapiclient" ) @@ -81,9 +80,7 @@ func Parse(content []byte) ([]*v1alpha1.Policy, error) { } switch gvk { case policy_v1alpha1: - // TODO: don't use kyverno's convert for now to workaround the bug in api machinery code - // https://kubernetes.slack.com/archives/C0EG7JC6T/p1696331287543159 - policy, err := To[v1alpha1.Policy](untyped) + policy, err := convert.To[v1alpha1.Policy](untyped) if err != nil { return nil, err } @@ -94,15 +91,3 @@ func Parse(content []byte) ([]*v1alpha1.Policy, error) { } return policies, nil } - -func Into[T any](untyped unstructured.Unstructured, result *T) error { - return runtime.DefaultUnstructuredConverter.FromUnstructured(untyped.UnstructuredContent(), result) -} - -func To[T any](untyped unstructured.Unstructured) (*T, error) { - var result T - if err := Into(untyped, &result); err != nil { - return nil, err - } - return &result, nil -} diff --git a/testdata/payload-yaml/policy.yaml b/testdata/payload-yaml/policy.yaml index 222d49b7..9bb5ab89 100644 --- a/testdata/payload-yaml/policy.yaml +++ b/testdata/payload-yaml/policy.yaml @@ -12,8 +12,7 @@ spec: context: - name: tags variable: - value: - Team: Kyverno + Team: Kyverno validate: message: Bucket `{{ resource.name }}` ({{ resource.address }}) does not have the required tags {{ to_string($tags) }} assert: diff --git a/testdata/pod-all-latest/policy.yaml b/testdata/pod-all-latest/policy.yaml index c6f1d430..e365e491 100644 --- a/testdata/pod-all-latest/policy.yaml +++ b/testdata/pod-all-latest/policy.yaml @@ -7,11 +7,9 @@ spec: - name: pod-no-latest context: - name: tag - variable: - value: latest + variable: latest - name: tag - variable: - value: (concat(':', $tag)) + variable: (concat(':', $tag)) match: any: - resource: diff --git a/testdata/pod-no-latest/policy.yaml b/testdata/pod-no-latest/policy.yaml index fbaffc96..4019f749 100644 --- a/testdata/pod-no-latest/policy.yaml +++ b/testdata/pod-no-latest/policy.yaml @@ -7,8 +7,7 @@ spec: - name: pod-no-latest context: - name: tag - variable: - value: :latest + variable: :latest match: any: - resource: diff --git a/testdata/tf-plan/policy.yaml b/testdata/tf-plan/policy.yaml index 222d49b7..9bb5ab89 100644 --- a/testdata/tf-plan/policy.yaml +++ b/testdata/tf-plan/policy.yaml @@ -12,8 +12,7 @@ spec: context: - name: tags variable: - value: - Team: Kyverno + Team: Kyverno validate: message: Bucket `{{ resource.name }}` ({{ resource.address }}) does not have the required tags {{ to_string($tags) }} assert: