-
Notifications
You must be signed in to change notification settings - Fork 1.3k
/
getter.go
259 lines (222 loc) · 7.89 KB
/
getter.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
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
/*
Copyright 2020 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 conditions
import (
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"sigs.k8s.io/controller-runtime/pkg/client"
clusterv1 "sigs.k8s.io/cluster-api/api/v1beta1"
)
// Getter interface defines methods that a Cluster API object should implement in order to
// use the conditions package for getting conditions.
type Getter interface {
client.Object
// GetConditions returns the list of conditions for a cluster API object.
GetConditions() clusterv1.Conditions
}
// Get returns the condition with the given type, if the condition does not exists,
// it returns nil.
func Get(from Getter, t clusterv1.ConditionType) *clusterv1.Condition {
conditions := from.GetConditions()
if conditions == nil {
return nil
}
for _, condition := range conditions {
if condition.Type == t {
return &condition
}
}
return nil
}
// Has returns true if a condition with the given type exists.
func Has(from Getter, t clusterv1.ConditionType) bool {
return Get(from, t) != nil
}
// IsTrue is true if the condition with the given type is True, otherwise it return false
// if the condition is not True or if the condition does not exist (is nil).
func IsTrue(from Getter, t clusterv1.ConditionType) bool {
if c := Get(from, t); c != nil {
return c.Status == corev1.ConditionTrue
}
return false
}
// IsFalse is true if the condition with the given type is False, otherwise it return false
// if the condition is not False or if the condition does not exist (is nil).
func IsFalse(from Getter, t clusterv1.ConditionType) bool {
if c := Get(from, t); c != nil {
return c.Status == corev1.ConditionFalse
}
return false
}
// IsUnknown is true if the condition with the given type is Unknown or if the condition
// does not exist (is nil).
func IsUnknown(from Getter, t clusterv1.ConditionType) bool {
if c := Get(from, t); c != nil {
return c.Status == corev1.ConditionUnknown
}
return true
}
// GetReason returns a nil safe string of Reason for the condition with the given type.
func GetReason(from Getter, t clusterv1.ConditionType) string {
if c := Get(from, t); c != nil {
return c.Reason
}
return ""
}
// GetMessage returns a nil safe string of Message.
func GetMessage(from Getter, t clusterv1.ConditionType) string {
if c := Get(from, t); c != nil {
return c.Message
}
return ""
}
// GetSeverity returns the condition Severity or nil if the condition
// does not exist (is nil).
func GetSeverity(from Getter, t clusterv1.ConditionType) *clusterv1.ConditionSeverity {
if c := Get(from, t); c != nil {
return &c.Severity
}
return nil
}
// GetLastTransitionTime returns the condition Severity or nil if the condition
// does not exist (is nil).
func GetLastTransitionTime(from Getter, t clusterv1.ConditionType) *metav1.Time {
if c := Get(from, t); c != nil {
return &c.LastTransitionTime
}
return nil
}
// summary returns a Ready condition with the summary of all the conditions existing
// on an object. If the object does not have other conditions, no summary condition is generated.
func summary(from Getter, options ...MergeOption) *clusterv1.Condition {
conditions := from.GetConditions()
mergeOpt := &mergeOptions{}
for _, o := range options {
o(mergeOpt)
}
// Identifies the conditions in scope for the Summary by taking all the existing conditions except Ready,
// or, if a list of conditions types is specified, only the conditions the condition in that list.
conditionsInScope := make([]localizedCondition, 0, len(conditions))
for i := range conditions {
c := conditions[i]
if c.Type == clusterv1.ReadyCondition {
continue
}
if mergeOpt.conditionTypes != nil {
found := false
for _, t := range mergeOpt.conditionTypes {
if c.Type == t {
found = true
break
}
}
if !found {
continue
}
}
conditionsInScope = append(conditionsInScope, localizedCondition{
Condition: &c,
Getter: from,
})
}
// If it is required to add a step counter only if a subset of condition exists, check if the conditions
// in scope are included in this subset or not.
if mergeOpt.addStepCounterIfOnlyConditionTypes != nil {
for _, c := range conditionsInScope {
found := false
for _, t := range mergeOpt.addStepCounterIfOnlyConditionTypes {
if c.Type == t {
found = true
break
}
}
if !found {
mergeOpt.addStepCounter = false
break
}
}
}
// If it is required to add a step counter, determine the total number of conditions defaulting
// to the selected conditions or, if defined, to the total number of conditions type to be considered.
if mergeOpt.addStepCounter {
mergeOpt.stepCounter = len(conditionsInScope)
if mergeOpt.conditionTypes != nil {
mergeOpt.stepCounter = len(mergeOpt.conditionTypes)
}
if mergeOpt.addStepCounterIfOnlyConditionTypes != nil {
mergeOpt.stepCounter = len(mergeOpt.addStepCounterIfOnlyConditionTypes)
}
}
return merge(conditionsInScope, clusterv1.ReadyCondition, mergeOpt)
}
// mirrorOptions allows to set options for the mirror operation.
type mirrorOptions struct {
fallbackTo *bool
fallbackReason string
fallbackSeverity clusterv1.ConditionSeverity
fallbackMessage string
}
// MirrorOptions defines an option for mirroring conditions.
type MirrorOptions func(*mirrorOptions)
// WithFallbackValue specify a fallback value to use in case the mirrored condition does not exists;
// in case the fallbackValue is false, given values for reason, severity and message will be used.
func WithFallbackValue(fallbackValue bool, reason string, severity clusterv1.ConditionSeverity, message string) MirrorOptions {
return func(c *mirrorOptions) {
c.fallbackTo = &fallbackValue
c.fallbackReason = reason
c.fallbackSeverity = severity
c.fallbackMessage = message
}
}
// mirror mirrors the Ready condition from a dependent object into the target condition;
// if the Ready condition does not exists in the source object, no target conditions is generated.
func mirror(from Getter, targetCondition clusterv1.ConditionType, options ...MirrorOptions) *clusterv1.Condition {
mirrorOpt := &mirrorOptions{}
for _, o := range options {
o(mirrorOpt)
}
condition := Get(from, clusterv1.ReadyCondition)
if mirrorOpt.fallbackTo != nil && condition == nil {
switch *mirrorOpt.fallbackTo {
case true:
condition = TrueCondition(targetCondition)
case false:
condition = FalseCondition(targetCondition, mirrorOpt.fallbackReason, mirrorOpt.fallbackSeverity, mirrorOpt.fallbackMessage)
}
}
if condition != nil {
condition.Type = targetCondition
}
return condition
}
// Aggregates all the Ready condition from a list of dependent objects into the target object;
// if the Ready condition does not exists in one of the source object, the object is excluded from
// the aggregation; if none of the source object have ready condition, no target conditions is generated.
func aggregate(from []Getter, targetCondition clusterv1.ConditionType, options ...MergeOption) *clusterv1.Condition {
conditionsInScope := make([]localizedCondition, 0, len(from))
for i := range from {
condition := Get(from[i], clusterv1.ReadyCondition)
conditionsInScope = append(conditionsInScope, localizedCondition{
Condition: condition,
Getter: from[i],
})
}
mergeOpt := &mergeOptions{
addStepCounter: true,
stepCounter: len(from),
}
for _, o := range options {
o(mergeOpt)
}
return merge(conditionsInScope, targetCondition, mergeOpt)
}