crossplane-runtime/pkg/fieldpath/merge_test.go

214 lines
5.2 KiB
Go

/*
Copyright 2021 The Crossplane 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.
*/
package fieldpath
import (
"fmt"
"reflect"
"strings"
"testing"
"github.com/google/go-cmp/cmp"
"k8s.io/apimachinery/pkg/util/json"
xpv1 "github.com/crossplane/crossplane-runtime/apis/common/v1"
"github.com/crossplane/crossplane-runtime/pkg/test"
)
func TestMergeValue(t *testing.T) {
const (
pathTest = "a"
valSrc = "e1-from-source"
valSrc2 = "e1-from-source-2"
valDst = "e1-from-destination"
)
formatArr := func(arr []string) string {
return fmt.Sprintf(`{"%s": ["%s"]}`, pathTest, strings.Join(arr, `", "`))
}
formatMap := func(val string) string {
return fmt.Sprintf(`{"%s": {"%s": "%s"}}`, pathTest, pathTest, val)
}
appendArr := func(dst, src []string) []string {
return reflect.AppendSlice(reflect.ValueOf(dst), reflect.ValueOf(src)).Interface().([]string)
}
arrSrc := []string{valSrc}
fnMapSrc := func() map[string]any {
return map[string]any{pathTest: valSrc}
}
arrDst := []string{valDst}
fnMapDst := func() map[string]any {
return map[string]any{pathTest: map[string]any{pathTest: valDst}}
}
valFalse, valTrue := false, true
type fields struct {
object map[string]any
}
type args struct {
path string
value any
mo *xpv1.MergeOptions
}
type want struct {
serialized string
err error
}
tests := map[string]struct {
reason string
fields fields
args args
want want
}{
"MergeArrayNoMergeOptions": {
reason: "If no merge options are given, default is to override an array",
fields: fields{
object: map[string]any{
pathTest: valDst,
},
},
args: args{
path: pathTest,
value: arrSrc,
},
want: want{
serialized: formatArr(arrSrc),
},
},
"MergeArrayNoAppend": {
reason: "If MergeOptions.AppendSlice is false, an array should be overridden when merging",
fields: fields{
object: map[string]any{
pathTest: arrDst,
},
},
args: args{
path: pathTest,
value: arrSrc,
mo: &xpv1.MergeOptions{
AppendSlice: &valFalse,
},
},
want: want{
serialized: formatArr(arrSrc),
},
},
"MergeArrayAppend": {
reason: "If MergeOptions.AppendSlice is true, dst array should be merged with the src array",
fields: fields{
object: map[string]any{
pathTest: arrDst,
},
},
args: args{
path: pathTest,
value: arrSrc,
mo: &xpv1.MergeOptions{
AppendSlice: &valTrue,
},
},
want: want{
serialized: formatArr(appendArr(arrDst, arrSrc)),
},
},
"MergeArrayAppendDuplicate": {
reason: "If MergeOptions.AppendSlice is true, dst array should be merged with the src array not allowing duplicates",
fields: fields{
object: map[string]any{
pathTest: []string{valDst, valSrc},
},
},
args: args{
path: pathTest,
value: []string{valSrc, valSrc2},
mo: &xpv1.MergeOptions{
AppendSlice: &valTrue,
},
},
want: want{
serialized: formatArr([]string{valDst, valSrc, valSrc2}),
},
},
"MergeMapNoMergeOptions": {
reason: "If no merge options are given, default is to override a map key",
fields: fields{
object: fnMapDst(),
},
args: args{
path: pathTest,
value: fnMapSrc(),
},
want: want{
serialized: formatMap(valSrc),
},
},
"MergeMapNoKeep": {
reason: "If MergeOptions.KeepMapValues is false, a map key should be overridden",
fields: fields{
object: fnMapDst(),
},
args: args{
path: pathTest,
value: fnMapSrc(),
mo: &xpv1.MergeOptions{
KeepMapValues: &valFalse,
},
},
want: want{
serialized: formatMap(valSrc),
},
},
"MergeMapKeep": {
reason: "If MergeOptions.KeepMapValues is true, a dst map key should preserve its value",
fields: fields{
object: fnMapDst(),
},
args: args{
path: pathTest,
value: fnMapSrc(),
mo: &xpv1.MergeOptions{
KeepMapValues: &valTrue,
},
},
want: want{
serialized: formatMap(valDst),
},
},
}
for name, tc := range tests {
t.Run(name, func(t *testing.T) {
want := make(map[string]any)
if err := json.Unmarshal([]byte(tc.want.serialized), &want); err != nil {
t.Fatalf("Test case error: Unable to unmarshall JSON doc: %v", err)
}
p := &Paved{
object: tc.fields.object,
}
err := p.MergeValue(tc.args.path, tc.args.value, tc.args.mo)
if diff := cmp.Diff(tc.want.err, err, test.EquateErrors()); diff != "" {
t.Fatalf("\np.MergeValue(%s, %v): %s: -want error, +got error:\n%s",
tc.args.path, tc.args.value, tc.reason, diff)
}
if diff := cmp.Diff(want, p.object); diff != "" {
t.Fatalf("\np.MergeValue(%s, %v): %s: -want, +got:\n%s",
tc.args.path, tc.args.value, tc.reason, diff)
}
})
}
}