2019-01-12 04:58:27 +00:00
|
|
|
/*
|
|
|
|
Copyright 2014 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 resource
|
|
|
|
|
|
|
|
import (
|
2020-03-26 21:07:15 +00:00
|
|
|
"context"
|
2021-07-02 08:43:15 +00:00
|
|
|
"fmt"
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2021-07-02 08:43:15 +00:00
|
|
|
apierrors "k8s.io/apimachinery/pkg/api/errors"
|
2019-01-12 04:58:27 +00:00
|
|
|
"k8s.io/apimachinery/pkg/api/meta"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2021-07-02 08:43:15 +00:00
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
"k8s.io/apimachinery/pkg/fields"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
|
|
"k8s.io/apimachinery/pkg/types"
|
|
|
|
"k8s.io/apimachinery/pkg/watch"
|
|
|
|
)
|
|
|
|
|
|
|
|
var metadataAccessor = meta.NewAccessor()
|
|
|
|
|
|
|
|
// Helper provides methods for retrieving or mutating a RESTful
|
|
|
|
// resource.
|
|
|
|
type Helper struct {
|
|
|
|
// The name of this resource as the server would recognize it
|
|
|
|
Resource string
|
|
|
|
// A RESTClient capable of mutating this resource.
|
|
|
|
RESTClient RESTClient
|
|
|
|
// True if the resource type is scoped to namespaces
|
|
|
|
NamespaceScoped bool
|
2020-03-26 21:07:15 +00:00
|
|
|
// If true, then use server-side dry-run to not persist changes to storage
|
|
|
|
// for verbs and resources that support server-side dry-run.
|
|
|
|
//
|
|
|
|
// Note this should only be used against an apiserver with dry-run enabled,
|
|
|
|
// and on resources that support dry-run. If the apiserver or the resource
|
|
|
|
// does not support dry-run, then the change will be persisted to storage.
|
|
|
|
ServerDryRun bool
|
2020-08-10 17:43:49 +00:00
|
|
|
|
|
|
|
// FieldManager is the name associated with the actor or entity that is making
|
|
|
|
// changes.
|
|
|
|
FieldManager string
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewHelper creates a Helper from a ResourceMapping
|
|
|
|
func NewHelper(client RESTClient, mapping *meta.RESTMapping) *Helper {
|
|
|
|
return &Helper{
|
|
|
|
Resource: mapping.Resource.Resource,
|
|
|
|
RESTClient: client,
|
|
|
|
NamespaceScoped: mapping.Scope.Name() == meta.RESTScopeNameNamespace,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
// DryRun, if true, will use server-side dry-run to not persist changes to storage.
|
|
|
|
// Otherwise, changes will be persisted to storage.
|
|
|
|
func (m *Helper) DryRun(dryRun bool) *Helper {
|
|
|
|
m.ServerDryRun = dryRun
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
2020-08-10 17:43:49 +00:00
|
|
|
// WithFieldManager sets the field manager option to indicate the actor or entity
|
|
|
|
// that is making changes in a create or update operation.
|
|
|
|
func (m *Helper) WithFieldManager(fieldManager string) *Helper {
|
|
|
|
m.FieldManager = fieldManager
|
|
|
|
return m
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Helper) Get(namespace, name string) (runtime.Object, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
req := m.RESTClient.Get().
|
|
|
|
NamespaceIfScoped(namespace, m.NamespaceScoped).
|
|
|
|
Resource(m.Resource).
|
|
|
|
Name(name)
|
2020-03-26 21:07:15 +00:00
|
|
|
return req.Do(context.TODO()).Get()
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2020-08-10 17:43:49 +00:00
|
|
|
func (m *Helper) List(namespace, apiVersion string, options *metav1.ListOptions) (runtime.Object, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
req := m.RESTClient.Get().
|
|
|
|
NamespaceIfScoped(namespace, m.NamespaceScoped).
|
|
|
|
Resource(m.Resource).
|
|
|
|
VersionedParams(options, metav1.ParameterCodec)
|
2020-03-26 21:07:15 +00:00
|
|
|
return req.Do(context.TODO()).Get()
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2021-07-02 08:43:15 +00:00
|
|
|
// FollowContinue handles the continue parameter returned by the API server when using list
|
|
|
|
// chunking. To take advantage of this, the initial ListOptions provided by the consumer
|
|
|
|
// should include a non-zero Limit parameter.
|
|
|
|
func FollowContinue(initialOpts *metav1.ListOptions,
|
|
|
|
listFunc func(metav1.ListOptions) (runtime.Object, error)) error {
|
|
|
|
opts := initialOpts
|
|
|
|
for {
|
|
|
|
list, err := listFunc(*opts)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
nextContinueToken, _ := metadataAccessor.Continue(list)
|
|
|
|
if len(nextContinueToken) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
opts.Continue = nextContinueToken
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// EnhanceListError augments errors typically returned by List operations with additional context,
|
|
|
|
// making sure to retain the StatusError type when applicable.
|
|
|
|
func EnhanceListError(err error, opts metav1.ListOptions, subj string) error {
|
|
|
|
if apierrors.IsResourceExpired(err) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if apierrors.IsBadRequest(err) || apierrors.IsNotFound(err) {
|
|
|
|
if se, ok := err.(*apierrors.StatusError); ok {
|
|
|
|
// modify the message without hiding this is an API error
|
|
|
|
if len(opts.LabelSelector) == 0 && len(opts.FieldSelector) == 0 {
|
|
|
|
se.ErrStatus.Message = fmt.Sprintf("Unable to list %q: %v", subj,
|
|
|
|
se.ErrStatus.Message)
|
|
|
|
} else {
|
|
|
|
se.ErrStatus.Message = fmt.Sprintf(
|
|
|
|
"Unable to find %q that match label selector %q, field selector %q: %v", subj,
|
|
|
|
opts.LabelSelector,
|
|
|
|
opts.FieldSelector, se.ErrStatus.Message)
|
|
|
|
}
|
|
|
|
return se
|
|
|
|
}
|
|
|
|
if len(opts.LabelSelector) == 0 && len(opts.FieldSelector) == 0 {
|
|
|
|
return fmt.Errorf("Unable to list %q: %v", subj, err)
|
|
|
|
}
|
|
|
|
return fmt.Errorf("Unable to find %q that match label selector %q, field selector %q: %v",
|
|
|
|
subj, opts.LabelSelector, opts.FieldSelector, err)
|
|
|
|
}
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
func (m *Helper) Watch(namespace, apiVersion string, options *metav1.ListOptions) (watch.Interface, error) {
|
|
|
|
options.Watch = true
|
|
|
|
return m.RESTClient.Get().
|
|
|
|
NamespaceIfScoped(namespace, m.NamespaceScoped).
|
|
|
|
Resource(m.Resource).
|
|
|
|
VersionedParams(options, metav1.ParameterCodec).
|
2020-03-26 21:07:15 +00:00
|
|
|
Watch(context.TODO())
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Helper) WatchSingle(namespace, name, resourceVersion string) (watch.Interface, error) {
|
|
|
|
return m.RESTClient.Get().
|
|
|
|
NamespaceIfScoped(namespace, m.NamespaceScoped).
|
|
|
|
Resource(m.Resource).
|
|
|
|
VersionedParams(&metav1.ListOptions{
|
|
|
|
ResourceVersion: resourceVersion,
|
|
|
|
Watch: true,
|
|
|
|
FieldSelector: fields.OneTermEqualSelector("metadata.name", name).String(),
|
|
|
|
}, metav1.ParameterCodec).
|
2020-03-26 21:07:15 +00:00
|
|
|
Watch(context.TODO())
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Helper) Delete(namespace, name string) (runtime.Object, error) {
|
|
|
|
return m.DeleteWithOptions(namespace, name, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Helper) DeleteWithOptions(namespace, name string, options *metav1.DeleteOptions) (runtime.Object, error) {
|
2020-03-26 21:07:15 +00:00
|
|
|
if options == nil {
|
|
|
|
options = &metav1.DeleteOptions{}
|
|
|
|
}
|
|
|
|
if m.ServerDryRun {
|
|
|
|
options.DryRun = []string{metav1.DryRunAll}
|
|
|
|
}
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
return m.RESTClient.Delete().
|
|
|
|
NamespaceIfScoped(namespace, m.NamespaceScoped).
|
|
|
|
Resource(m.Resource).
|
|
|
|
Name(name).
|
|
|
|
Body(options).
|
2020-03-26 21:07:15 +00:00
|
|
|
Do(context.TODO()).
|
2019-01-12 04:58:27 +00:00
|
|
|
Get()
|
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
func (m *Helper) Create(namespace string, modify bool, obj runtime.Object) (runtime.Object, error) {
|
|
|
|
return m.CreateWithOptions(namespace, modify, obj, nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Helper) CreateWithOptions(namespace string, modify bool, obj runtime.Object, options *metav1.CreateOptions) (runtime.Object, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
if options == nil {
|
|
|
|
options = &metav1.CreateOptions{}
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
if m.ServerDryRun {
|
|
|
|
options.DryRun = []string{metav1.DryRunAll}
|
|
|
|
}
|
2020-08-10 17:43:49 +00:00
|
|
|
if m.FieldManager != "" {
|
|
|
|
options.FieldManager = m.FieldManager
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
if modify {
|
|
|
|
// Attempt to version the object based on client logic.
|
|
|
|
version, err := metadataAccessor.ResourceVersion(obj)
|
|
|
|
if err != nil {
|
|
|
|
// We don't know how to clear the version on this object, so send it to the server as is
|
|
|
|
return m.createResource(m.RESTClient, m.Resource, namespace, obj, options)
|
|
|
|
}
|
|
|
|
if version != "" {
|
|
|
|
if err := metadataAccessor.SetResourceVersion(obj, ""); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.createResource(m.RESTClient, m.Resource, namespace, obj, options)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Helper) createResource(c RESTClient, resource, namespace string, obj runtime.Object, options *metav1.CreateOptions) (runtime.Object, error) {
|
|
|
|
return c.Post().
|
|
|
|
NamespaceIfScoped(namespace, m.NamespaceScoped).
|
|
|
|
Resource(resource).
|
|
|
|
VersionedParams(options, metav1.ParameterCodec).
|
|
|
|
Body(obj).
|
2020-03-26 21:07:15 +00:00
|
|
|
Do(context.TODO()).
|
2019-01-12 04:58:27 +00:00
|
|
|
Get()
|
|
|
|
}
|
2019-04-07 17:07:55 +00:00
|
|
|
func (m *Helper) Patch(namespace, name string, pt types.PatchType, data []byte, options *metav1.PatchOptions) (runtime.Object, error) {
|
2019-01-12 04:58:27 +00:00
|
|
|
if options == nil {
|
2019-04-07 17:07:55 +00:00
|
|
|
options = &metav1.PatchOptions{}
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
if m.ServerDryRun {
|
|
|
|
options.DryRun = []string{metav1.DryRunAll}
|
|
|
|
}
|
2020-08-10 17:43:49 +00:00
|
|
|
if m.FieldManager != "" {
|
|
|
|
options.FieldManager = m.FieldManager
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
return m.RESTClient.Patch(pt).
|
|
|
|
NamespaceIfScoped(namespace, m.NamespaceScoped).
|
|
|
|
Resource(m.Resource).
|
|
|
|
Name(name).
|
|
|
|
VersionedParams(options, metav1.ParameterCodec).
|
|
|
|
Body(data).
|
2020-03-26 21:07:15 +00:00
|
|
|
Do(context.TODO()).
|
2019-01-12 04:58:27 +00:00
|
|
|
Get()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Helper) Replace(namespace, name string, overwrite bool, obj runtime.Object) (runtime.Object, error) {
|
|
|
|
c := m.RESTClient
|
2020-03-26 21:07:15 +00:00
|
|
|
var options = &metav1.UpdateOptions{}
|
|
|
|
if m.ServerDryRun {
|
|
|
|
options.DryRun = []string{metav1.DryRunAll}
|
|
|
|
}
|
2020-08-10 17:43:49 +00:00
|
|
|
if m.FieldManager != "" {
|
|
|
|
options.FieldManager = m.FieldManager
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
// Attempt to version the object based on client logic.
|
|
|
|
version, err := metadataAccessor.ResourceVersion(obj)
|
|
|
|
if err != nil {
|
|
|
|
// We don't know how to version this object, so send it to the server as is
|
2020-03-26 21:07:15 +00:00
|
|
|
return m.replaceResource(c, m.Resource, namespace, name, obj, options)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
if version == "" && overwrite {
|
|
|
|
// Retrieve the current version of the object to overwrite the server object
|
2020-03-26 21:07:15 +00:00
|
|
|
serverObj, err := c.Get().NamespaceIfScoped(namespace, m.NamespaceScoped).Resource(m.Resource).Name(name).Do(context.TODO()).Get()
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
// The object does not exist, but we want it to be created
|
2020-03-26 21:07:15 +00:00
|
|
|
return m.replaceResource(c, m.Resource, namespace, name, obj, options)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
serverVersion, err := metadataAccessor.ResourceVersion(serverObj)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err := metadataAccessor.SetResourceVersion(obj, serverVersion); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
return m.replaceResource(c, m.Resource, namespace, name, obj, options)
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
func (m *Helper) replaceResource(c RESTClient, resource, namespace, name string, obj runtime.Object, options *metav1.UpdateOptions) (runtime.Object, error) {
|
|
|
|
return c.Put().
|
|
|
|
NamespaceIfScoped(namespace, m.NamespaceScoped).
|
|
|
|
Resource(resource).
|
|
|
|
Name(name).
|
|
|
|
VersionedParams(options, metav1.ParameterCodec).
|
|
|
|
Body(obj).
|
|
|
|
Do(context.TODO()).
|
|
|
|
Get()
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|