/* Copyright 2024 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 gentype import ( "context" json "encoding/json" "fmt" "time" metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" "k8s.io/apimachinery/pkg/runtime" types "k8s.io/apimachinery/pkg/types" watch "k8s.io/apimachinery/pkg/watch" rest "k8s.io/client-go/rest" "k8s.io/client-go/util/consistencydetector" "k8s.io/client-go/util/watchlist" "k8s.io/klog/v2" ) // objectWithMeta matches objects implementing both runtime.Object and metav1.Object. type objectWithMeta interface { runtime.Object metav1.Object } // namedObject matches comparable objects implementing GetName(); it is intended for use with apply declarative configurations. type namedObject interface { comparable GetName() *string } // Client represents a client, optionally namespaced, with no support for lists or apply declarative configurations. type Client[T objectWithMeta] struct { resource string client rest.Interface namespace string // "" for non-namespaced clients newObject func() T parameterCodec runtime.ParameterCodec } // ClientWithList represents a client with support for lists. type ClientWithList[T objectWithMeta, L runtime.Object] struct { *Client[T] alsoLister[T, L] } // ClientWithApply represents a client with support for apply declarative configurations. type ClientWithApply[T objectWithMeta, C namedObject] struct { *Client[T] alsoApplier[T, C] } // ClientWithListAndApply represents a client with support for lists and apply declarative configurations. type ClientWithListAndApply[T objectWithMeta, L runtime.Object, C namedObject] struct { *Client[T] alsoLister[T, L] alsoApplier[T, C] } // Helper types for composition type alsoLister[T objectWithMeta, L runtime.Object] struct { client *Client[T] newList func() L } type alsoApplier[T objectWithMeta, C namedObject] struct { client *Client[T] } // NewClient constructs a client, namespaced or not, with no support for lists or apply. // Non-namespaced clients are constructed by passing an empty namespace (""). func NewClient[T objectWithMeta]( resource string, client rest.Interface, parameterCodec runtime.ParameterCodec, namespace string, emptyObjectCreator func() T, ) *Client[T] { return &Client[T]{ resource: resource, client: client, parameterCodec: parameterCodec, namespace: namespace, newObject: emptyObjectCreator, } } // NewClientWithList constructs a namespaced client with support for lists. func NewClientWithList[T objectWithMeta, L runtime.Object]( resource string, client rest.Interface, parameterCodec runtime.ParameterCodec, namespace string, emptyObjectCreator func() T, emptyListCreator func() L, ) *ClientWithList[T, L] { typeClient := NewClient[T](resource, client, parameterCodec, namespace, emptyObjectCreator) return &ClientWithList[T, L]{ typeClient, alsoLister[T, L]{typeClient, emptyListCreator}, } } // NewClientWithApply constructs a namespaced client with support for apply declarative configurations. func NewClientWithApply[T objectWithMeta, C namedObject]( resource string, client rest.Interface, parameterCodec runtime.ParameterCodec, namespace string, emptyObjectCreator func() T, ) *ClientWithApply[T, C] { typeClient := NewClient[T](resource, client, parameterCodec, namespace, emptyObjectCreator) return &ClientWithApply[T, C]{ typeClient, alsoApplier[T, C]{typeClient}, } } // NewClientWithListAndApply constructs a client with support for lists and applying declarative configurations. func NewClientWithListAndApply[T objectWithMeta, L runtime.Object, C namedObject]( resource string, client rest.Interface, parameterCodec runtime.ParameterCodec, namespace string, emptyObjectCreator func() T, emptyListCreator func() L, ) *ClientWithListAndApply[T, L, C] { typeClient := NewClient[T](resource, client, parameterCodec, namespace, emptyObjectCreator) return &ClientWithListAndApply[T, L, C]{ typeClient, alsoLister[T, L]{typeClient, emptyListCreator}, alsoApplier[T, C]{typeClient}, } } // GetClient returns the REST interface. func (c *Client[T]) GetClient() rest.Interface { return c.client } // GetNamespace returns the client's namespace, if any. func (c *Client[T]) GetNamespace() string { return c.namespace } // Get takes name of the resource, and returns the corresponding object, and an error if there is any. func (c *Client[T]) Get(ctx context.Context, name string, options metav1.GetOptions) (T, error) { result := c.newObject() err := c.client.Get(). NamespaceIfScoped(c.namespace, c.namespace != ""). Resource(c.resource). Name(name). VersionedParams(&options, c.parameterCodec). Do(ctx). Into(result) return result, err } // List takes label and field selectors, and returns the list of resources that match those selectors. func (l *alsoLister[T, L]) List(ctx context.Context, opts metav1.ListOptions) (L, error) { if watchListOptions, hasWatchListOptionsPrepared, watchListOptionsErr := watchlist.PrepareWatchListOptionsFromListOptions(opts); watchListOptionsErr != nil { klog.Warningf("Failed preparing watchlist options for $.type|resource$, falling back to the standard LIST semantics, err = %v", watchListOptionsErr) } else if hasWatchListOptionsPrepared { result, err := l.watchList(ctx, watchListOptions) if err == nil { consistencydetector.CheckWatchListFromCacheDataConsistencyIfRequested(ctx, "watchlist request for "+l.client.resource, l.list, opts, result) return result, nil } klog.Warningf("The watchlist request for %s ended with an error, falling back to the standard LIST semantics, err = %v", l.client.resource, err) } result, err := l.list(ctx, opts) if err == nil { consistencydetector.CheckListFromCacheDataConsistencyIfRequested(ctx, "list request for "+l.client.resource, l.list, opts, result) } return result, err } func (l *alsoLister[T, L]) list(ctx context.Context, opts metav1.ListOptions) (L, error) { list := l.newList() var timeout time.Duration if opts.TimeoutSeconds != nil { timeout = time.Duration(*opts.TimeoutSeconds) * time.Second } err := l.client.client.Get(). NamespaceIfScoped(l.client.namespace, l.client.namespace != ""). Resource(l.client.resource). VersionedParams(&opts, l.client.parameterCodec). Timeout(timeout). Do(ctx). Into(list) return list, err } // watchList establishes a watch stream with the server and returns the list of resources. func (l *alsoLister[T, L]) watchList(ctx context.Context, opts metav1.ListOptions) (result L, err error) { var timeout time.Duration if opts.TimeoutSeconds != nil { timeout = time.Duration(*opts.TimeoutSeconds) * time.Second } result = l.newList() err = l.client.client.Get(). NamespaceIfScoped(l.client.namespace, l.client.namespace != ""). Resource(l.client.resource). VersionedParams(&opts, l.client.parameterCodec). Timeout(timeout). WatchList(ctx). Into(result) return } // Watch returns a watch.Interface that watches the requested resources. func (c *Client[T]) Watch(ctx context.Context, opts metav1.ListOptions) (watch.Interface, error) { var timeout time.Duration if opts.TimeoutSeconds != nil { timeout = time.Duration(*opts.TimeoutSeconds) * time.Second } opts.Watch = true return c.client.Get(). NamespaceIfScoped(c.namespace, c.namespace != ""). Resource(c.resource). VersionedParams(&opts, c.parameterCodec). Timeout(timeout). Watch(ctx) } // 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. func (c *Client[T]) Create(ctx context.Context, obj T, opts metav1.CreateOptions) (T, error) { result := c.newObject() err := c.client.Post(). NamespaceIfScoped(c.namespace, c.namespace != ""). Resource(c.resource). VersionedParams(&opts, c.parameterCodec). Body(obj). Do(ctx). Into(result) return result, err } // 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. func (c *Client[T]) Update(ctx context.Context, obj T, opts metav1.UpdateOptions) (T, error) { result := c.newObject() err := c.client.Put(). NamespaceIfScoped(c.namespace, c.namespace != ""). Resource(c.resource). Name(obj.GetName()). VersionedParams(&opts, c.parameterCodec). Body(obj). Do(ctx). Into(result) return result, err } // UpdateStatus updates the status subresource of a resource. Returns the server's representation of the resource, and an error, if there is any. func (c *Client[T]) UpdateStatus(ctx context.Context, obj T, opts metav1.UpdateOptions) (T, error) { result := c.newObject() err := c.client.Put(). NamespaceIfScoped(c.namespace, c.namespace != ""). Resource(c.resource). Name(obj.GetName()). SubResource("status"). VersionedParams(&opts, c.parameterCodec). Body(obj). Do(ctx). Into(result) return result, err } // Delete takes name of the resource and deletes it. Returns an error if one occurs. func (c *Client[T]) Delete(ctx context.Context, name string, opts metav1.DeleteOptions) error { return c.client.Delete(). NamespaceIfScoped(c.namespace, c.namespace != ""). Resource(c.resource). Name(name). Body(&opts). Do(ctx). Error() } // DeleteCollection deletes a collection of objects. func (l *alsoLister[T, L]) DeleteCollection(ctx context.Context, opts metav1.DeleteOptions, listOpts metav1.ListOptions) error { var timeout time.Duration if listOpts.TimeoutSeconds != nil { timeout = time.Duration(*listOpts.TimeoutSeconds) * time.Second } return l.client.client.Delete(). NamespaceIfScoped(l.client.namespace, l.client.namespace != ""). Resource(l.client.resource). VersionedParams(&listOpts, l.client.parameterCodec). Timeout(timeout). Body(&opts). Do(ctx). Error() } // Patch applies the patch and returns the patched resource. func (c *Client[T]) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts metav1.PatchOptions, subresources ...string) (T, error) { result := c.newObject() err := c.client.Patch(pt). NamespaceIfScoped(c.namespace, c.namespace != ""). Resource(c.resource). Name(name). SubResource(subresources...). VersionedParams(&opts, c.parameterCodec). Body(data). Do(ctx). Into(result) return result, err } // Apply takes the given apply declarative configuration, applies it and returns the applied resource. func (a *alsoApplier[T, C]) Apply(ctx context.Context, obj C, opts metav1.ApplyOptions) (T, error) { result := a.client.newObject() if obj == *new(C) { return *new(T), fmt.Errorf("object provided to Apply must not be nil") } patchOpts := opts.ToPatchOptions() data, err := json.Marshal(obj) if err != nil { return *new(T), err } if obj.GetName() == nil { return *new(T), fmt.Errorf("obj.Name must be provided to Apply") } err = a.client.client.Patch(types.ApplyPatchType). NamespaceIfScoped(a.client.namespace, a.client.namespace != ""). Resource(a.client.resource). Name(*obj.GetName()). VersionedParams(&patchOpts, a.client.parameterCodec). Body(data). Do(ctx). Into(result) return result, err } // Apply takes the given apply declarative configuration, applies it to the status subresource and returns the applied resource. func (a *alsoApplier[T, C]) ApplyStatus(ctx context.Context, obj C, opts metav1.ApplyOptions) (T, error) { if obj == *new(C) { return *new(T), fmt.Errorf("object provided to Apply must not be nil") } patchOpts := opts.ToPatchOptions() data, err := json.Marshal(obj) if err != nil { return *new(T), err } if obj.GetName() == nil { return *new(T), fmt.Errorf("obj.Name must be provided to Apply") } result := a.client.newObject() err = a.client.client.Patch(types.ApplyPatchType). NamespaceIfScoped(a.client.namespace, a.client.namespace != ""). Resource(a.client.resource). Name(*obj.GetName()). SubResource("status"). VersionedParams(&patchOpts, a.client.parameterCodec). Body(data). Do(ctx). Into(result) return result, err }