438 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			438 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Go
		
	
	
	
/*
 | 
						|
Copyright 2018 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 dynamic
 | 
						|
 | 
						|
import (
 | 
						|
	"context"
 | 
						|
	"fmt"
 | 
						|
	"net/http"
 | 
						|
 | 
						|
	"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/apimachinery/pkg/watch"
 | 
						|
	"k8s.io/client-go/rest"
 | 
						|
)
 | 
						|
 | 
						|
type DynamicClient struct {
 | 
						|
	client rest.Interface
 | 
						|
}
 | 
						|
 | 
						|
var _ Interface = &DynamicClient{}
 | 
						|
 | 
						|
// ConfigFor returns a copy of the provided config with the
 | 
						|
// appropriate dynamic client defaults set.
 | 
						|
func ConfigFor(inConfig *rest.Config) *rest.Config {
 | 
						|
	config := rest.CopyConfig(inConfig)
 | 
						|
	config.AcceptContentTypes = "application/json"
 | 
						|
	config.ContentType = "application/json"
 | 
						|
	config.NegotiatedSerializer = basicNegotiatedSerializer{} // this gets used for discovery and error handling types
 | 
						|
	if config.UserAgent == "" {
 | 
						|
		config.UserAgent = rest.DefaultKubernetesUserAgent()
 | 
						|
	}
 | 
						|
	return config
 | 
						|
}
 | 
						|
 | 
						|
// New creates a new DynamicClient for the given RESTClient.
 | 
						|
func New(c rest.Interface) *DynamicClient {
 | 
						|
	return &DynamicClient{client: c}
 | 
						|
}
 | 
						|
 | 
						|
// NewForConfigOrDie creates a new DynamicClient for the given config and
 | 
						|
// panics if there is an error in the config.
 | 
						|
func NewForConfigOrDie(c *rest.Config) *DynamicClient {
 | 
						|
	ret, err := NewForConfig(c)
 | 
						|
	if err != nil {
 | 
						|
		panic(err)
 | 
						|
	}
 | 
						|
	return ret
 | 
						|
}
 | 
						|
 | 
						|
// NewForConfig creates a new dynamic client or returns an error.
 | 
						|
// NewForConfig is equivalent to NewForConfigAndClient(c, httpClient),
 | 
						|
// where httpClient was generated with rest.HTTPClientFor(c).
 | 
						|
func NewForConfig(inConfig *rest.Config) (*DynamicClient, error) {
 | 
						|
	config := ConfigFor(inConfig)
 | 
						|
 | 
						|
	httpClient, err := rest.HTTPClientFor(config)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return NewForConfigAndClient(config, httpClient)
 | 
						|
}
 | 
						|
 | 
						|
// NewForConfigAndClient creates a new dynamic client for the given config and http client.
 | 
						|
// Note the http client provided takes precedence over the configured transport values.
 | 
						|
func NewForConfigAndClient(inConfig *rest.Config, h *http.Client) (*DynamicClient, error) {
 | 
						|
	config := ConfigFor(inConfig)
 | 
						|
	// for serializing the options
 | 
						|
	config.GroupVersion = &schema.GroupVersion{}
 | 
						|
	config.APIPath = "/if-you-see-this-search-for-the-break"
 | 
						|
 | 
						|
	restClient, err := rest.RESTClientForConfigAndClient(config, h)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return &DynamicClient{client: restClient}, nil
 | 
						|
}
 | 
						|
 | 
						|
type dynamicResourceClient struct {
 | 
						|
	client    *DynamicClient
 | 
						|
	namespace string
 | 
						|
	resource  schema.GroupVersionResource
 | 
						|
}
 | 
						|
 | 
						|
func (c *DynamicClient) Resource(resource schema.GroupVersionResource) NamespaceableResourceInterface {
 | 
						|
	return &dynamicResourceClient{client: c, resource: resource}
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) Namespace(ns string) ResourceInterface {
 | 
						|
	ret := *c
 | 
						|
	ret.namespace = ns
 | 
						|
	return &ret
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) Create(ctx context.Context, obj *unstructured.Unstructured, opts metav1.CreateOptions, subresources ...string) (*unstructured.Unstructured, error) {
 | 
						|
	outBytes, err := runtime.Encode(unstructured.UnstructuredJSONScheme, obj)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	name := ""
 | 
						|
	if len(subresources) > 0 {
 | 
						|
		accessor, err := meta.Accessor(obj)
 | 
						|
		if err != nil {
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
		name = accessor.GetName()
 | 
						|
		if len(name) == 0 {
 | 
						|
			return nil, fmt.Errorf("name is required")
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace, name); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	result := c.client.client.
 | 
						|
		Post().
 | 
						|
		AbsPath(append(c.makeURLSegments(name), subresources...)...).
 | 
						|
		SetHeader("Content-Type", runtime.ContentTypeJSON).
 | 
						|
		Body(outBytes).
 | 
						|
		SpecificallyVersionedParams(&opts, dynamicParameterCodec, versionV1).
 | 
						|
		Do(ctx)
 | 
						|
	if err := result.Error(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	retBytes, err := result.Raw()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	uncastObj, err := runtime.Decode(unstructured.UnstructuredJSONScheme, retBytes)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return uncastObj.(*unstructured.Unstructured), nil
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) Update(ctx context.Context, obj *unstructured.Unstructured, opts metav1.UpdateOptions, subresources ...string) (*unstructured.Unstructured, error) {
 | 
						|
	accessor, err := meta.Accessor(obj)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	name := accessor.GetName()
 | 
						|
	if len(name) == 0 {
 | 
						|
		return nil, fmt.Errorf("name is required")
 | 
						|
	}
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace, name); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	outBytes, err := runtime.Encode(unstructured.UnstructuredJSONScheme, obj)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	result := c.client.client.
 | 
						|
		Put().
 | 
						|
		AbsPath(append(c.makeURLSegments(name), subresources...)...).
 | 
						|
		SetHeader("Content-Type", runtime.ContentTypeJSON).
 | 
						|
		Body(outBytes).
 | 
						|
		SpecificallyVersionedParams(&opts, dynamicParameterCodec, versionV1).
 | 
						|
		Do(ctx)
 | 
						|
	if err := result.Error(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	retBytes, err := result.Raw()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	uncastObj, err := runtime.Decode(unstructured.UnstructuredJSONScheme, retBytes)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return uncastObj.(*unstructured.Unstructured), nil
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) UpdateStatus(ctx context.Context, obj *unstructured.Unstructured, opts metav1.UpdateOptions) (*unstructured.Unstructured, error) {
 | 
						|
	accessor, err := meta.Accessor(obj)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	name := accessor.GetName()
 | 
						|
	if len(name) == 0 {
 | 
						|
		return nil, fmt.Errorf("name is required")
 | 
						|
	}
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace, name); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	outBytes, err := runtime.Encode(unstructured.UnstructuredJSONScheme, obj)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	result := c.client.client.
 | 
						|
		Put().
 | 
						|
		AbsPath(append(c.makeURLSegments(name), "status")...).
 | 
						|
		SetHeader("Content-Type", runtime.ContentTypeJSON).
 | 
						|
		Body(outBytes).
 | 
						|
		SpecificallyVersionedParams(&opts, dynamicParameterCodec, versionV1).
 | 
						|
		Do(ctx)
 | 
						|
	if err := result.Error(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
 | 
						|
	retBytes, err := result.Raw()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	uncastObj, err := runtime.Decode(unstructured.UnstructuredJSONScheme, retBytes)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return uncastObj.(*unstructured.Unstructured), nil
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) Delete(ctx context.Context, name string, opts metav1.DeleteOptions, subresources ...string) error {
 | 
						|
	if len(name) == 0 {
 | 
						|
		return fmt.Errorf("name is required")
 | 
						|
	}
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace, name); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	deleteOptionsByte, err := runtime.Encode(deleteOptionsCodec.LegacyCodec(schema.GroupVersion{Version: "v1"}), &opts)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	result := c.client.client.
 | 
						|
		Delete().
 | 
						|
		AbsPath(append(c.makeURLSegments(name), subresources...)...).
 | 
						|
		SetHeader("Content-Type", runtime.ContentTypeJSON).
 | 
						|
		Body(deleteOptionsByte).
 | 
						|
		Do(ctx)
 | 
						|
	return result.Error()
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) DeleteCollection(ctx context.Context, opts metav1.DeleteOptions, listOptions metav1.ListOptions) error {
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace); err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	deleteOptionsByte, err := runtime.Encode(deleteOptionsCodec.LegacyCodec(schema.GroupVersion{Version: "v1"}), &opts)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
 | 
						|
	result := c.client.client.
 | 
						|
		Delete().
 | 
						|
		AbsPath(c.makeURLSegments("")...).
 | 
						|
		SetHeader("Content-Type", runtime.ContentTypeJSON).
 | 
						|
		Body(deleteOptionsByte).
 | 
						|
		SpecificallyVersionedParams(&listOptions, dynamicParameterCodec, versionV1).
 | 
						|
		Do(ctx)
 | 
						|
	return result.Error()
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) Get(ctx context.Context, name string, opts metav1.GetOptions, subresources ...string) (*unstructured.Unstructured, error) {
 | 
						|
	if len(name) == 0 {
 | 
						|
		return nil, fmt.Errorf("name is required")
 | 
						|
	}
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace, name); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	result := c.client.client.Get().AbsPath(append(c.makeURLSegments(name), subresources...)...).SpecificallyVersionedParams(&opts, dynamicParameterCodec, versionV1).Do(ctx)
 | 
						|
	if err := result.Error(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	retBytes, err := result.Raw()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	uncastObj, err := runtime.Decode(unstructured.UnstructuredJSONScheme, retBytes)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return uncastObj.(*unstructured.Unstructured), nil
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) List(ctx context.Context, opts metav1.ListOptions) (*unstructured.UnstructuredList, error) {
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	result := c.client.client.Get().AbsPath(c.makeURLSegments("")...).SpecificallyVersionedParams(&opts, dynamicParameterCodec, versionV1).Do(ctx)
 | 
						|
	if err := result.Error(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	retBytes, err := result.Raw()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	uncastObj, err := runtime.Decode(unstructured.UnstructuredJSONScheme, retBytes)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if list, ok := uncastObj.(*unstructured.UnstructuredList); ok {
 | 
						|
		return list, nil
 | 
						|
	}
 | 
						|
 | 
						|
	list, err := uncastObj.(*unstructured.Unstructured).ToList()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return list, nil
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) Watch(ctx context.Context, opts metav1.ListOptions) (watch.Interface, error) {
 | 
						|
	opts.Watch = true
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return c.client.client.Get().AbsPath(c.makeURLSegments("")...).
 | 
						|
		SpecificallyVersionedParams(&opts, dynamicParameterCodec, versionV1).
 | 
						|
		Watch(ctx)
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) Patch(ctx context.Context, name string, pt types.PatchType, data []byte, opts metav1.PatchOptions, subresources ...string) (*unstructured.Unstructured, error) {
 | 
						|
	if len(name) == 0 {
 | 
						|
		return nil, fmt.Errorf("name is required")
 | 
						|
	}
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace, name); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	result := c.client.client.
 | 
						|
		Patch(pt).
 | 
						|
		AbsPath(append(c.makeURLSegments(name), subresources...)...).
 | 
						|
		Body(data).
 | 
						|
		SpecificallyVersionedParams(&opts, dynamicParameterCodec, versionV1).
 | 
						|
		Do(ctx)
 | 
						|
	if err := result.Error(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	retBytes, err := result.Raw()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	uncastObj, err := runtime.Decode(unstructured.UnstructuredJSONScheme, retBytes)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return uncastObj.(*unstructured.Unstructured), nil
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) Apply(ctx context.Context, name string, obj *unstructured.Unstructured, opts metav1.ApplyOptions, subresources ...string) (*unstructured.Unstructured, error) {
 | 
						|
	if len(name) == 0 {
 | 
						|
		return nil, fmt.Errorf("name is required")
 | 
						|
	}
 | 
						|
	if err := validateNamespaceWithOptionalName(c.namespace, name); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	outBytes, err := runtime.Encode(unstructured.UnstructuredJSONScheme, obj)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	accessor, err := meta.Accessor(obj)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	managedFields := accessor.GetManagedFields()
 | 
						|
	if len(managedFields) > 0 {
 | 
						|
		return nil, fmt.Errorf(`cannot apply an object with managed fields already set.
 | 
						|
		Use the client-go/applyconfigurations "UnstructructuredExtractor" to obtain the unstructured ApplyConfiguration for the given field manager that you can use/modify here to apply`)
 | 
						|
	}
 | 
						|
	patchOpts := opts.ToPatchOptions()
 | 
						|
 | 
						|
	result := c.client.client.
 | 
						|
		Patch(types.ApplyPatchType).
 | 
						|
		AbsPath(append(c.makeURLSegments(name), subresources...)...).
 | 
						|
		Body(outBytes).
 | 
						|
		SpecificallyVersionedParams(&patchOpts, dynamicParameterCodec, versionV1).
 | 
						|
		Do(ctx)
 | 
						|
	if err := result.Error(); err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	retBytes, err := result.Raw()
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	uncastObj, err := runtime.Decode(unstructured.UnstructuredJSONScheme, retBytes)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return uncastObj.(*unstructured.Unstructured), nil
 | 
						|
}
 | 
						|
func (c *dynamicResourceClient) ApplyStatus(ctx context.Context, name string, obj *unstructured.Unstructured, opts metav1.ApplyOptions) (*unstructured.Unstructured, error) {
 | 
						|
	return c.Apply(ctx, name, obj, opts, "status")
 | 
						|
}
 | 
						|
 | 
						|
func validateNamespaceWithOptionalName(namespace string, name ...string) error {
 | 
						|
	if msgs := rest.IsValidPathSegmentName(namespace); len(msgs) != 0 {
 | 
						|
		return fmt.Errorf("invalid namespace %q: %v", namespace, msgs)
 | 
						|
	}
 | 
						|
	if len(name) > 1 {
 | 
						|
		panic("Invalid number of names")
 | 
						|
	} else if len(name) == 1 {
 | 
						|
		if msgs := rest.IsValidPathSegmentName(name[0]); len(msgs) != 0 {
 | 
						|
			return fmt.Errorf("invalid resource name %q: %v", name[0], msgs)
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (c *dynamicResourceClient) makeURLSegments(name string) []string {
 | 
						|
	url := []string{}
 | 
						|
	if len(c.resource.Group) == 0 {
 | 
						|
		url = append(url, "api")
 | 
						|
	} else {
 | 
						|
		url = append(url, "apis", c.resource.Group)
 | 
						|
	}
 | 
						|
	url = append(url, c.resource.Version)
 | 
						|
 | 
						|
	if len(c.namespace) > 0 {
 | 
						|
		url = append(url, "namespaces", c.namespace)
 | 
						|
	}
 | 
						|
	url = append(url, c.resource.Resource)
 | 
						|
 | 
						|
	if len(name) > 0 {
 | 
						|
		url = append(url, name)
 | 
						|
	}
 | 
						|
 | 
						|
	return url
 | 
						|
}
 |