-
Notifications
You must be signed in to change notification settings - Fork 1.4k
/
Copy pathconversion.go
257 lines (219 loc) · 8.53 KB
/
conversion.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
/*
Copyright 2019 The Kubernetes 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 conversion implements conversion utilities.
package conversion
import (
"context"
"math/rand"
"sort"
"strings"
"testing"
"github.com/google/go-cmp/cmp"
fuzz "github.com/google/gofuzz"
"github.com/onsi/gomega"
"github.com/pkg/errors"
corev1 "k8s.io/api/core/v1"
"k8s.io/apimachinery/pkg/api/apitesting/fuzzer"
apiequality "k8s.io/apimachinery/pkg/api/equality"
metafuzzer "k8s.io/apimachinery/pkg/apis/meta/fuzzer"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
runtimeserializer "k8s.io/apimachinery/pkg/runtime/serializer"
"k8s.io/apimachinery/pkg/util/json"
"k8s.io/client-go/kubernetes/scheme"
ctrl "sigs.k8s.io/controller-runtime"
"sigs.k8s.io/controller-runtime/pkg/client"
"sigs.k8s.io/controller-runtime/pkg/conversion"
clusterv1 "sigs.k8s.io/cluster-api/api/v1beta1"
"sigs.k8s.io/cluster-api/util"
)
const (
// DataAnnotation is the annotation that conversion webhooks
// use to retain the data in case of down-conversion from the hub.
DataAnnotation = "cluster.x-k8s.io/conversion-data"
)
var (
contract = clusterv1.GroupVersion.String()
)
// UpdateReferenceAPIContract takes a client and object reference, queries the API Server for
// the Custom Resource Definition and looks which one is the stored version available.
//
// The object passed as input is modified in place if an updated compatible version is found.
// NOTE: In case CRDs are named incorrectly, this func is using an APIReader instead of the regular client to list CRDs
// to avoid implicitly creating an informer for CRDs which would lead to high memory consumption.
func UpdateReferenceAPIContract(ctx context.Context, c client.Client, apiReader client.Reader, ref *corev1.ObjectReference) error {
log := ctrl.LoggerFrom(ctx)
gvk := ref.GroupVersionKind()
metadata, err := util.GetGVKMetadata(ctx, c, gvk)
if err != nil {
log.Info("Cannot retrieve CRD with metadata only client, falling back to slower listing", "err", err.Error())
// Fallback to slower and more memory intensive method to get the full CRD.
crd, err := util.GetCRDWithContract(ctx, apiReader, gvk, contract)
if err != nil {
return err
}
metadata = &metav1.PartialObjectMetadata{
TypeMeta: crd.TypeMeta,
ObjectMeta: crd.ObjectMeta,
}
}
chosen, err := getLatestAPIVersionFromContract(metadata)
if err != nil {
return err
}
// Modify the GroupVersionKind with the new version.
if gvk.Version != chosen {
gvk.Version = chosen
ref.SetGroupVersionKind(gvk)
}
return nil
}
func getLatestAPIVersionFromContract(metadata metav1.Object) (string, error) {
labels := metadata.GetLabels()
// If there is no label, return early without changing the reference.
supportedVersions, ok := labels[contract]
if !ok || supportedVersions == "" {
return "", errors.Errorf("cannot find any versions matching contract %q for GVK %v as contract version label(s) are either missing or empty", contract, metadata.GetName())
}
// Pick the latest version in the slice and validate it.
kubeVersions := util.KubeAwareAPIVersions(strings.Split(supportedVersions, "_"))
sort.Sort(kubeVersions)
return kubeVersions[len(kubeVersions)-1], nil
}
// MarshalData stores the source object as json data in the destination object annotations map.
// It ignores the metadata of the source object.
func MarshalData(src metav1.Object, dst metav1.Object) error {
u, err := runtime.DefaultUnstructuredConverter.ToUnstructured(src)
if err != nil {
return err
}
delete(u, "metadata")
data, err := json.Marshal(u)
if err != nil {
return err
}
annotations := dst.GetAnnotations()
if annotations == nil {
annotations = map[string]string{}
}
annotations[DataAnnotation] = string(data)
dst.SetAnnotations(annotations)
return nil
}
// UnmarshalData tries to retrieve the data from the annotation and unmarshals it into the object passed as input.
func UnmarshalData(from metav1.Object, to interface{}) (bool, error) {
annotations := from.GetAnnotations()
data, ok := annotations[DataAnnotation]
if !ok {
return false, nil
}
if err := json.Unmarshal([]byte(data), to); err != nil {
return false, err
}
delete(annotations, DataAnnotation)
from.SetAnnotations(annotations)
return true, nil
}
// GetFuzzer returns a new fuzzer to be used for testing.
func GetFuzzer(scheme *runtime.Scheme, funcs ...fuzzer.FuzzerFuncs) *fuzz.Fuzzer {
funcs = append([]fuzzer.FuzzerFuncs{
metafuzzer.Funcs,
func(_ runtimeserializer.CodecFactory) []interface{} {
return []interface{}{
// Custom fuzzer for metav1.Time pointers which weren't
// fuzzed and always resulted in `nil` values.
// This implementation is somewhat similar to the one provided
// in the metafuzzer.Funcs.
func(input *metav1.Time, c fuzz.Continue) {
if input != nil {
var sec, nsec uint32
c.Fuzz(&sec)
c.Fuzz(&nsec)
fuzzed := metav1.Unix(int64(sec), int64(nsec)).Rfc3339Copy()
input.Time = fuzzed.Time
}
},
}
},
}, funcs...)
return fuzzer.FuzzerFor(
fuzzer.MergeFuzzerFuncs(funcs...),
rand.NewSource(rand.Int63()), //nolint:gosec
runtimeserializer.NewCodecFactory(scheme),
)
}
// FuzzTestFuncInput contains input parameters
// for the FuzzTestFunc function.
type FuzzTestFuncInput struct {
Scheme *runtime.Scheme
Hub conversion.Hub
HubAfterMutation func(conversion.Hub)
Spoke conversion.Convertible
SpokeAfterMutation func(convertible conversion.Convertible)
SkipSpokeAnnotationCleanup bool
FuzzerFuncs []fuzzer.FuzzerFuncs
}
// FuzzTestFunc returns a new testing function to be used in tests to make sure conversions between
// the Hub version of an object and an older version aren't lossy.
func FuzzTestFunc(input FuzzTestFuncInput) func(*testing.T) {
if input.Scheme == nil {
input.Scheme = scheme.Scheme
}
return func(t *testing.T) {
t.Helper()
t.Run("spoke-hub-spoke", func(t *testing.T) {
g := gomega.NewWithT(t)
fuzzer := GetFuzzer(input.Scheme, input.FuzzerFuncs...)
for i := 0; i < 10000; i++ {
// Create the spoke and fuzz it
spokeBefore := input.Spoke.DeepCopyObject().(conversion.Convertible)
fuzzer.Fuzz(spokeBefore)
// First convert spoke to hub
hubCopy := input.Hub.DeepCopyObject().(conversion.Hub)
g.Expect(spokeBefore.ConvertTo(hubCopy)).To(gomega.Succeed())
// Convert hub back to spoke and check if the resulting spoke is equal to the spoke before the round trip
spokeAfter := input.Spoke.DeepCopyObject().(conversion.Convertible)
g.Expect(spokeAfter.ConvertFrom(hubCopy)).To(gomega.Succeed())
// Remove data annotation eventually added by ConvertFrom for avoiding data loss in hub-spoke-hub round trips
// NOTE: There are use case when we want to skip this operation, e.g. if the spoke object does not have ObjectMeta (e.g. kubeadm types).
if !input.SkipSpokeAnnotationCleanup {
metaAfter := spokeAfter.(metav1.Object)
delete(metaAfter.GetAnnotations(), DataAnnotation)
}
if input.SpokeAfterMutation != nil {
input.SpokeAfterMutation(spokeAfter)
}
g.Expect(apiequality.Semantic.DeepEqual(spokeBefore, spokeAfter)).To(gomega.BeTrue(), cmp.Diff(spokeBefore, spokeAfter))
}
})
t.Run("hub-spoke-hub", func(t *testing.T) {
g := gomega.NewWithT(t)
fuzzer := GetFuzzer(input.Scheme, input.FuzzerFuncs...)
for i := 0; i < 10000; i++ {
// Create the hub and fuzz it
hubBefore := input.Hub.DeepCopyObject().(conversion.Hub)
fuzzer.Fuzz(hubBefore)
// First convert hub to spoke
dstCopy := input.Spoke.DeepCopyObject().(conversion.Convertible)
g.Expect(dstCopy.ConvertFrom(hubBefore)).To(gomega.Succeed())
// Convert spoke back to hub and check if the resulting hub is equal to the hub before the round trip
hubAfter := input.Hub.DeepCopyObject().(conversion.Hub)
g.Expect(dstCopy.ConvertTo(hubAfter)).To(gomega.Succeed())
if input.HubAfterMutation != nil {
input.HubAfterMutation(hubAfter)
}
g.Expect(apiequality.Semantic.DeepEqual(hubBefore, hubAfter)).To(gomega.BeTrue(), cmp.Diff(hubBefore, hubAfter))
}
})
}
}