|
| 1 | +/* |
| 2 | +Copyright 2024 The Kubernetes Authors. |
| 3 | +
|
| 4 | +Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | +you may not use this file except in compliance with the License. |
| 6 | +You may obtain a copy of the License at |
| 7 | +
|
| 8 | + http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | +
|
| 10 | +Unless required by applicable law or agreed to in writing, software |
| 11 | +distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | +See the License for the specific language governing permissions and |
| 14 | +limitations under the License. |
| 15 | +*/ |
| 16 | + |
| 17 | +package gentype |
| 18 | + |
| 19 | +import ( |
| 20 | + "context" |
| 21 | + json "encoding/json" |
| 22 | + "fmt" |
| 23 | + |
| 24 | + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" |
| 25 | + labels "k8s.io/apimachinery/pkg/labels" |
| 26 | + "k8s.io/apimachinery/pkg/runtime" |
| 27 | + "k8s.io/apimachinery/pkg/runtime/schema" |
| 28 | + types "k8s.io/apimachinery/pkg/types" |
| 29 | + watch "k8s.io/apimachinery/pkg/watch" |
| 30 | + testing "k8s.io/client-go/testing" |
| 31 | +) |
| 32 | + |
| 33 | +// FakeClient represents a fake client |
| 34 | +type FakeClient[T objectWithMeta] struct { |
| 35 | + *testing.Fake |
| 36 | + ns string |
| 37 | + resource schema.GroupVersionResource |
| 38 | + kind schema.GroupVersionKind |
| 39 | + newObject func() T |
| 40 | +} |
| 41 | + |
| 42 | +// FakeClientWithList represents a fake client with support for lists. |
| 43 | +type FakeClientWithList[T objectWithMeta, L runtime.Object] struct { |
| 44 | + *FakeClient[T] |
| 45 | + alsoFakeLister[T, L] |
| 46 | +} |
| 47 | + |
| 48 | +// FakeClientWithApply represents a fake client with support for apply declarative configurations. |
| 49 | +type FakeClientWithApply[T objectWithMeta, C namedObject] struct { |
| 50 | + *FakeClient[T] |
| 51 | + alsoFakeApplier[T, C] |
| 52 | +} |
| 53 | + |
| 54 | +// FakeClientWithListAndApply represents a fake client with support for lists and apply declarative configurations. |
| 55 | +type FakeClientWithListAndApply[T objectWithMeta, L runtime.Object, C namedObject] struct { |
| 56 | + *FakeClient[T] |
| 57 | + alsoFakeLister[T, L] |
| 58 | + alsoFakeApplier[T, C] |
| 59 | +} |
| 60 | + |
| 61 | +// Helper types for composition |
| 62 | +type alsoFakeLister[T objectWithMeta, L runtime.Object] struct { |
| 63 | + client *FakeClient[T] |
| 64 | + newList func() L |
| 65 | + copyListMeta func(L, L) |
| 66 | + getItems func(L) []T |
| 67 | + setItems func(L, []T) |
| 68 | +} |
| 69 | + |
| 70 | +type alsoFakeApplier[T objectWithMeta, C namedObject] struct { |
| 71 | + client *FakeClient[T] |
| 72 | +} |
| 73 | + |
| 74 | +// NewFakeClient constructs a fake client, namespaced or not, with no support for lists or apply. |
| 75 | +// Non-namespaced clients are constructed by passing an empty namespace (""). |
| 76 | +func NewFakeClient[T objectWithMeta]( |
| 77 | + fake *testing.Fake, namespace string, resource schema.GroupVersionResource, kind schema.GroupVersionKind, emptyObjectCreator func() T, |
| 78 | +) *FakeClient[T] { |
| 79 | + return &FakeClient[T]{fake, namespace, resource, kind, emptyObjectCreator} |
| 80 | +} |
| 81 | + |
| 82 | +// NewFakeClientWithList constructs a namespaced client with support for lists. |
| 83 | +func NewFakeClientWithList[T runtime.Object, L runtime.Object]( |
| 84 | + fake *testing.Fake, namespace string, resource schema.GroupVersionResource, kind schema.GroupVersionKind, emptyObjectCreator func() T, |
| 85 | + emptyListCreator func() L, listMetaCopier func(L, L), itemGetter func(L) []T, itemSetter func(L, []T), |
| 86 | +) *FakeClientWithList[T, L] { |
| 87 | + fakeClient := NewFakeClient[T](fake, namespace, resource, kind, emptyObjectCreator) |
| 88 | + return &FakeClientWithList[T, L]{ |
| 89 | + fakeClient, |
| 90 | + alsoFakeLister[T, L]{fakeClient, emptyListCreator, listMetaCopier, itemGetter, itemSetter}, |
| 91 | + } |
| 92 | +} |
| 93 | + |
| 94 | +// NewFakeClientWithApply constructs a namespaced client with support for apply declarative configurations. |
| 95 | +func NewFakeClientWithApply[T runtime.Object, C namedObject]( |
| 96 | + fake *testing.Fake, namespace string, resource schema.GroupVersionResource, kind schema.GroupVersionKind, emptyObjectCreator func() T, |
| 97 | +) *FakeClientWithApply[T, C] { |
| 98 | + fakeClient := NewFakeClient[T](fake, namespace, resource, kind, emptyObjectCreator) |
| 99 | + return &FakeClientWithApply[T, C]{ |
| 100 | + fakeClient, |
| 101 | + alsoFakeApplier[T, C]{fakeClient}, |
| 102 | + } |
| 103 | +} |
| 104 | + |
| 105 | +// NewFakeClientWithListAndApply constructs a client with support for lists and applying declarative configurations. |
| 106 | +func NewFakeClientWithListAndApply[T runtime.Object, L runtime.Object, C namedObject]( |
| 107 | + fake *testing.Fake, namespace string, resource schema.GroupVersionResource, kind schema.GroupVersionKind, emptyObjectCreator func() T, |
| 108 | + emptyListCreator func() L, listMetaCopier func(L, L), itemGetter func(L) []T, itemSetter func(L, []T), |
| 109 | +) *FakeClientWithListAndApply[T, L, C] { |
| 110 | + fakeClient := NewFakeClient[T](fake, namespace, resource, kind, emptyObjectCreator) |
| 111 | + return &FakeClientWithListAndApply[T, L, C]{ |
| 112 | + fakeClient, |
| 113 | + alsoFakeLister[T, L]{fakeClient, emptyListCreator, listMetaCopier, itemGetter, itemSetter}, |
| 114 | + alsoFakeApplier[T, C]{fakeClient}, |
| 115 | + } |
| 116 | +} |
| 117 | + |
| 118 | +// Get takes name of a resource, and returns the corresponding object, and an error if there is any. |
| 119 | +func (c *FakeClient[T]) Get(ctx context.Context, name string, options metav1.GetOptions) (T, error) { |
| 120 | + emptyResult := c.newObject() |
| 121 | + |
| 122 | + obj, err := c.Fake. |
| 123 | + Invokes(testing.NewGetActionWithOptions(c.resource, c.ns, name, options), emptyResult) |
| 124 | + if obj == nil { |
| 125 | + return emptyResult, err |
| 126 | + } |
| 127 | + return obj.(T), err |
| 128 | +} |
| 129 | + |
| 130 | +func ToPointerSlice[T any](src []T) []*T { |
| 131 | + if src == nil { |
| 132 | + return nil |
| 133 | + } |
| 134 | + result := make([]*T, len(src)) |
| 135 | + for i := range src { |
| 136 | + result[i] = &src[i] |
| 137 | + } |
| 138 | + return result |
| 139 | +} |
| 140 | + |
| 141 | +func FromPointerSlice[T any](src []*T) []T { |
| 142 | + if src == nil { |
| 143 | + return nil |
| 144 | + } |
| 145 | + result := make([]T, len(src)) |
| 146 | + for i := range src { |
| 147 | + result[i] = *src[i] |
| 148 | + } |
| 149 | + return result |
| 150 | +} |
| 151 | + |
| 152 | +// List takes label and field selectors, and returns the list of ClusterRoles that match those selectors. |
| 153 | +func (l *alsoFakeLister[T, L]) List(ctx context.Context, opts metav1.ListOptions) (result L, err error) { |
| 154 | + emptyResult := l.newList() |
| 155 | + obj, err := l.client.Fake. |
| 156 | + Invokes(testing.NewListActionWithOptions(l.client.resource, l.client.kind, l.client.ns, opts), emptyResult) |
| 157 | + if obj == nil { |
| 158 | + return emptyResult, err |
| 159 | + } |
| 160 | + |
| 161 | + label, _, _ := testing.ExtractFromListOptions(opts) |
| 162 | + if label == nil { |
| 163 | + label = labels.Everything() |
| 164 | + } |
| 165 | + list := l.newList() |
| 166 | + l.copyListMeta(list, obj.(L)) |
| 167 | + var items []T |
| 168 | + for _, item := range l.getItems(obj.(L)) { |
| 169 | + if label.Matches(labels.Set(item.GetLabels())) { |
| 170 | + items = append(items, item) |
| 171 | + } |
| 172 | + } |
| 173 | + l.setItems(list, items) |
| 174 | + return list, err |
| 175 | +} |
| 176 | + |
| 177 | +// Watch returns a watch.Interface that watches the requested resources. |
| 178 | +func (c *FakeClient[T]) Watch(ctx context.Context, opts metav1.ListOptions) (watch.Interface, error) { |
| 179 | + return c.Fake. |
| 180 | + InvokesWatch(testing.NewWatchActionWithOptions(c.resource, c.ns, opts)) |
| 181 | +} |
| 182 | + |
| 183 | +// Create takes the representation of a resource and creates it. Returns the server's representation of the resource, and an error, if there is any. |
| 184 | +func (c *FakeClient[T]) Create(ctx context.Context, resource T, opts metav1.CreateOptions) (result T, err error) { |
| 185 | + emptyResult := c.newObject() |
| 186 | + obj, err := c.Fake. |
| 187 | + Invokes(testing.NewCreateActionWithOptions(c.resource, c.ns, resource, opts), emptyResult) |
| 188 | + if obj == nil { |
| 189 | + return emptyResult, err |
| 190 | + } |
| 191 | + return obj.(T), err |
| 192 | +} |
| 193 | + |
| 194 | +// Update takes the representation of a resource and updates it. Returns the server's representation of the resource, and an error, if there is any. |
| 195 | +func (c *FakeClient[T]) Update(ctx context.Context, resource T, opts metav1.UpdateOptions) (result T, err error) { |
| 196 | + emptyResult := c.newObject() |
| 197 | + obj, err := c.Fake. |
| 198 | + Invokes(testing.NewUpdateActionWithOptions(c.resource, c.ns, resource, opts), emptyResult) |
| 199 | + if obj == nil { |
| 200 | + return emptyResult, err |
| 201 | + } |
| 202 | + return obj.(T), err |
| 203 | +} |
| 204 | + |
| 205 | +// UpdateStatus updates the resource's status and returns the updated resource. |
| 206 | +func (c *FakeClient[T]) UpdateStatus(ctx context.Context, resource T, opts metav1.UpdateOptions) (result T, err error) { |
| 207 | + emptyResult := c.newObject() |
| 208 | + obj, err := c.Fake. |
| 209 | + Invokes(testing.NewUpdateSubresourceActionWithOptions(c.resource, "status", c.ns, resource, opts), emptyResult) |
| 210 | + |
| 211 | + if obj == nil { |
| 212 | + return emptyResult, err |
| 213 | + } |
| 214 | + return obj.(T), err |
| 215 | +} |
| 216 | + |
| 217 | +// Delete deletes the resource matching the given name. Returns an error if one occurs. |
| 218 | +func (c *FakeClient[T]) Delete(ctx context.Context, name string, opts metav1.DeleteOptions) error { |
| 219 | + _, err := c.Fake. |
| 220 | + Invokes(testing.NewDeleteActionWithOptions(c.resource, c.ns, name, opts), c.newObject()) |
| 221 | + return err |
| 222 | +} |
| 223 | + |
| 224 | +// DeleteCollection deletes a collection of objects. |
| 225 | +func (l *alsoFakeLister[T, L]) DeleteCollection(ctx context.Context, opts metav1.DeleteOptions, listOpts metav1.ListOptions) error { |
| 226 | + _, err := l.client.Fake. |
| 227 | + Invokes(testing.NewDeleteCollectionActionWithOptions(l.client.resource, l.client.ns, opts, listOpts), l.newList()) |
| 228 | + return err |
| 229 | +} |
| 230 | + |
| 231 | +// Patch applies the patch and returns the patched resource. |
| 232 | +func (c *FakeClient[T]) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts metav1.PatchOptions, subresources ...string) (result T, err error) { |
| 233 | + emptyResult := c.newObject() |
| 234 | + obj, err := c.Fake. |
| 235 | + Invokes(testing.NewPatchSubresourceActionWithOptions(c.resource, c.ns, name, pt, data, opts, subresources...), emptyResult) |
| 236 | + if obj == nil { |
| 237 | + return emptyResult, err |
| 238 | + } |
| 239 | + return obj.(T), err |
| 240 | +} |
| 241 | + |
| 242 | +// Apply takes the given apply declarative configuration, applies it and returns the applied resource. |
| 243 | +func (a *alsoFakeApplier[T, C]) Apply(ctx context.Context, configuration C, opts metav1.ApplyOptions) (result T, err error) { |
| 244 | + if configuration == *new(C) { |
| 245 | + return *new(T), fmt.Errorf("configuration provided to Apply must not be nil") |
| 246 | + } |
| 247 | + data, err := json.Marshal(configuration) |
| 248 | + if err != nil { |
| 249 | + return *new(T), err |
| 250 | + } |
| 251 | + name := configuration.GetName() |
| 252 | + if name == nil { |
| 253 | + return *new(T), fmt.Errorf("configuration.Name must be provided to Apply") |
| 254 | + } |
| 255 | + emptyResult := a.client.newObject() |
| 256 | + obj, err := a.client.Fake. |
| 257 | + Invokes(testing.NewPatchSubresourceActionWithOptions(a.client.resource, a.client.ns, *name, types.ApplyPatchType, data, opts.ToPatchOptions()), emptyResult) |
| 258 | + if obj == nil { |
| 259 | + return emptyResult, err |
| 260 | + } |
| 261 | + return obj.(T), err |
| 262 | +} |
| 263 | + |
| 264 | +// ApplyStatus applies the given apply declarative configuration to the resource's status and returns the updated resource. |
| 265 | +func (a *alsoFakeApplier[T, C]) ApplyStatus(ctx context.Context, configuration C, opts metav1.ApplyOptions) (result T, err error) { |
| 266 | + if configuration == *new(C) { |
| 267 | + return *new(T), fmt.Errorf("configuration provided to Apply must not be nil") |
| 268 | + } |
| 269 | + data, err := json.Marshal(configuration) |
| 270 | + if err != nil { |
| 271 | + return *new(T), err |
| 272 | + } |
| 273 | + name := configuration.GetName() |
| 274 | + if name == nil { |
| 275 | + return *new(T), fmt.Errorf("configuration.Name must be provided to Apply") |
| 276 | + } |
| 277 | + emptyResult := a.client.newObject() |
| 278 | + obj, err := a.client.Fake. |
| 279 | + Invokes(testing.NewPatchSubresourceActionWithOptions(a.client.resource, a.client.ns, *name, types.ApplyPatchType, data, opts.ToPatchOptions(), "status"), emptyResult) |
| 280 | + |
| 281 | + if obj == nil { |
| 282 | + return emptyResult, err |
| 283 | + } |
| 284 | + return obj.(T), err |
| 285 | +} |
| 286 | + |
| 287 | +func (c *FakeClient[T]) Namespace() string { |
| 288 | + return c.ns |
| 289 | +} |
| 290 | + |
| 291 | +func (c *FakeClient[T]) Kind() schema.GroupVersionKind { |
| 292 | + return c.kind |
| 293 | +} |
| 294 | + |
| 295 | +func (c *FakeClient[T]) Resource() schema.GroupVersionResource { |
| 296 | + return c.resource |
| 297 | +} |
0 commit comments