-
Notifications
You must be signed in to change notification settings - Fork 0
/
verber.go
161 lines (135 loc) · 5.08 KB
/
verber.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
// Copyright 2017 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 (
"fmt"
"github.com/kubernetes/dashboard/src/app/backend/api"
clientapi "github.com/kubernetes/dashboard/src/app/backend/client/api"
"k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/apimachinery/pkg/runtime"
restclient "k8s.io/client-go/rest"
)
// resourceVerber is a struct responsible for doing common verb operations on resources, like
// DELETE, PUT, UPDATE.
type resourceVerber struct {
client RESTClient
extensionsClient RESTClient
appsClient RESTClient
batchClient RESTClient
betaBatchClient RESTClient
autoscalingClient RESTClient
storageClient RESTClient
}
func (verber *resourceVerber) getRESTClientByType(clientType api.ClientType) RESTClient {
switch clientType {
case api.ClientTypeExtensionClient:
return verber.extensionsClient
case api.ClientTypeAppsClient:
return verber.appsClient
case api.ClientTypeBatchClient:
return verber.batchClient
case api.ClientTypeBetaBatchClient:
return verber.betaBatchClient
case api.ClientTypeAutoscalingClient:
return verber.autoscalingClient
case api.ClientTypeStorageClient:
return verber.storageClient
default:
return verber.client
}
}
// RESTClient is an interface for REST operations used in this file.
type RESTClient interface {
Delete() *restclient.Request
Put() *restclient.Request
Get() *restclient.Request
}
// NewResourceVerber creates a new resource verber that uses the given client for performing operations.
func NewResourceVerber(client, extensionsClient, appsClient,
batchClient, betaBatchClient, autoscalingClient, storageClient RESTClient) clientapi.ResourceVerber {
return &resourceVerber{client, extensionsClient, appsClient,
batchClient, betaBatchClient, autoscalingClient, storageClient}
}
// Delete deletes the resource of the given kind in the given namespace with the given name.
func (verber *resourceVerber) Delete(kind string, namespaceSet bool, namespace string, name string) error {
resourceSpec, ok := api.KindToAPIMapping[kind]
if !ok {
return fmt.Errorf("Unknown resource kind: %s", kind)
}
if namespaceSet != resourceSpec.Namespaced {
if namespaceSet {
return fmt.Errorf("Set namespace for not-namespaced resource kind: %s", kind)
} else {
return fmt.Errorf("Set no namespace for namespaced resource kind: %s", kind)
}
}
client := verber.getRESTClientByType(resourceSpec.ClientType)
// Do cascade delete by default, as this is what users typically expect.
defaultPropagationPolicy := v1.DeletePropagationForeground
defaultDeleteOptions := &v1.DeleteOptions{
PropagationPolicy: &defaultPropagationPolicy,
}
req := client.Delete().Resource(resourceSpec.Resource).Name(name).Body(defaultDeleteOptions)
if resourceSpec.Namespaced {
req.Namespace(namespace)
}
return req.Do().Error()
}
// Put puts new resource version of the given kind in the given namespace with the given name.
func (verber *resourceVerber) Put(kind string, namespaceSet bool, namespace string, name string,
object *runtime.Unknown) error {
resourceSpec, ok := api.KindToAPIMapping[kind]
if !ok {
return fmt.Errorf("Unknown resource kind: %s", kind)
}
if namespaceSet != resourceSpec.Namespaced {
if namespaceSet {
return fmt.Errorf("Set namespace for not-namespaced resource kind: %s", kind)
} else {
return fmt.Errorf("Set no namespace for namespaced resource kind: %s", kind)
}
}
client := verber.getRESTClientByType(resourceSpec.ClientType)
req := client.Put().
Resource(resourceSpec.Resource).
Name(name).
SetHeader("Content-Type", "application/json").
Body([]byte(object.Raw))
if resourceSpec.Namespaced {
req.Namespace(namespace)
}
return req.Do().Error()
}
// Get gets the resource of the given kind in the given namespace with the given name.
func (verber *resourceVerber) Get(kind string, namespaceSet bool, namespace string, name string) (runtime.Object, error) {
resourceSpec, ok := api.KindToAPIMapping[kind]
if !ok {
return nil, fmt.Errorf("Unknown resource kind: %s", kind)
}
if namespaceSet != resourceSpec.Namespaced {
if namespaceSet {
return nil, fmt.Errorf("Set namespace for not-namespaced resource kind: %s", kind)
} else {
return nil, fmt.Errorf("Set no namespace for namespaced resource kind: %s", kind)
}
}
client := verber.getRESTClientByType(resourceSpec.ClientType)
result := &runtime.Unknown{}
req := client.Get().Resource(resourceSpec.Resource).Name(name).SetHeader("Accept", "application/json")
if resourceSpec.Namespaced {
req.Namespace(namespace)
}
err := req.Do().Into(result)
return result, err
}