mirror of https://github.com/fluxcd/cli-utils.git
201 lines
5.7 KiB
Go
201 lines
5.7 KiB
Go
/*
|
|
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 client
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"k8s.io/apimachinery/pkg/api/meta"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
"k8s.io/apimachinery/pkg/types"
|
|
"k8s.io/client-go/dynamic"
|
|
)
|
|
|
|
// NewForConfig returns a new Client using the provided config and Options.
|
|
// The returned client reads *and* writes directly from the server
|
|
// (it doesn't use object caches). It understands how to work with
|
|
// normal types (both custom resources and aggregated/built-in resources),
|
|
// as well as unstructured types.
|
|
//
|
|
// In the case of normal types, the scheme will be used to look up the
|
|
// corresponding group, version, and kind for the given type. In the
|
|
// case of unstrctured types, the group, version, and kind will be extracted
|
|
// from the corresponding fields on the object.
|
|
func NewForConfig(dynamicClient dynamic.Interface, mapper meta.RESTMapper) (Client, error) {
|
|
c := &client{
|
|
client: dynamicClient,
|
|
restMapper: mapper,
|
|
}
|
|
|
|
return c, nil
|
|
}
|
|
|
|
var _ Client = &client{}
|
|
|
|
// client is a client.Client that reads and writes directly from/to an API server. It lazily initializes
|
|
// new clients at the time they are used, and caches the client.
|
|
type client struct {
|
|
client dynamic.Interface
|
|
restMapper meta.RESTMapper
|
|
}
|
|
|
|
// Create implements client.Client
|
|
func (uc *client) Create(_ context.Context, obj runtime.Object, options metav1.CreateOptions) error {
|
|
u, ok := obj.(*unstructured.Unstructured)
|
|
if !ok {
|
|
return fmt.Errorf("unstructured client did not understand object: %T", obj)
|
|
}
|
|
r, err := uc.getResourceInterface(u.GroupVersionKind(), u.GetNamespace())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
i, err := r.Create(u, options)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
u.Object = i.Object
|
|
return nil
|
|
}
|
|
|
|
// Update implements client.Client
|
|
func (uc *client) Update(_ context.Context, obj runtime.Object, options metav1.UpdateOptions) error {
|
|
u, ok := obj.(*unstructured.Unstructured)
|
|
if !ok {
|
|
return fmt.Errorf("unstructured client did not understand object: %T", obj)
|
|
}
|
|
r, err := uc.getResourceInterface(u.GroupVersionKind(), u.GetNamespace())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
i, err := r.Update(u, options)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
u.Object = i.Object
|
|
return nil
|
|
}
|
|
|
|
// Delete implements client.Client
|
|
func (uc *client) Delete(_ context.Context, obj runtime.Object, options *metav1.DeleteOptions) error {
|
|
u, ok := obj.(*unstructured.Unstructured)
|
|
if !ok {
|
|
return fmt.Errorf("unstructured client did not understand object: %T", obj)
|
|
}
|
|
r, err := uc.getResourceInterface(u.GroupVersionKind(), u.GetNamespace())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
err = r.Delete(u.GetName(), options)
|
|
return err
|
|
}
|
|
|
|
// Patch implements client.Client
|
|
func (uc *client) Patch(_ context.Context, obj runtime.Object, patch Patch, options metav1.PatchOptions) error {
|
|
u, ok := obj.(*unstructured.Unstructured)
|
|
if !ok {
|
|
return fmt.Errorf("unstructured client did not understand object: %T", obj)
|
|
}
|
|
r, err := uc.getResourceInterface(u.GroupVersionKind(), u.GetNamespace())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
data, err := patch.Data(obj)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
i, err := r.Patch(u.GetName(), patch.Type(), data, options)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
u.Object = i.Object
|
|
return nil
|
|
}
|
|
|
|
// Get implements client.Client
|
|
func (uc *client) Get(_ context.Context, key types.NamespacedName, obj runtime.Object) error {
|
|
u, ok := obj.(*unstructured.Unstructured)
|
|
if !ok {
|
|
return fmt.Errorf("unstructured client did not understand object: %T", obj)
|
|
}
|
|
r, err := uc.getResourceInterface(u.GroupVersionKind(), key.Namespace)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
i, err := r.Get(key.Name, metav1.GetOptions{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
u.Object = i.Object
|
|
return nil
|
|
}
|
|
|
|
// List implements client.Client
|
|
func (uc *client) List(_ context.Context, obj runtime.Object, namespace string, options metav1.ListOptions) error {
|
|
u, ok := obj.(*unstructured.UnstructuredList)
|
|
if !ok {
|
|
return fmt.Errorf("unstructured client did not understand object: %T", obj)
|
|
}
|
|
gvk := u.GroupVersionKind()
|
|
if strings.HasSuffix(gvk.Kind, "List") {
|
|
gvk.Kind = gvk.Kind[:len(gvk.Kind)-4]
|
|
}
|
|
r, err := uc.getResourceInterface(gvk, namespace)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
i, err := r.List(options)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
u.Items = i.Items
|
|
u.Object = i.Object
|
|
return nil
|
|
}
|
|
|
|
func (uc *client) UpdateStatus(_ context.Context, obj runtime.Object) error {
|
|
u, ok := obj.(*unstructured.Unstructured)
|
|
if !ok {
|
|
return fmt.Errorf("unstructured client did not understand object: %T", obj)
|
|
}
|
|
r, err := uc.getResourceInterface(u.GroupVersionKind(), u.GetNamespace())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
i, err := r.UpdateStatus(u, metav1.UpdateOptions{})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
u.Object = i.Object
|
|
return nil
|
|
}
|
|
|
|
func (uc *client) getResourceInterface(gvk schema.GroupVersionKind, ns string) (dynamic.ResourceInterface, error) {
|
|
mapping, err := uc.restMapper.RESTMapping(gvk.GroupKind(), gvk.Version)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if mapping.Scope.Name() == meta.RESTScopeNameRoot {
|
|
return uc.client.Resource(mapping.Resource), nil
|
|
}
|
|
return uc.client.Resource(mapping.Resource).Namespace(ns), nil
|
|
}
|