5
5
"fmt"
6
6
"maps"
7
7
"path"
8
- "slices"
9
8
"strings"
10
9
11
10
"github.com/aws/aws-sdk-go-v2/aws"
@@ -28,6 +27,9 @@ type NodeLabelController struct {
28
27
// Labels is a list of label keys to sync from the node to the cloud provider
29
28
Labels []string
30
29
30
+ // Annotations is a list of annotation keys to sync from the node to the cloud provider
31
+ Annotations []string
32
+
31
33
// Cloud is the cloud provider (aws or gcp)
32
34
Cloud string
33
35
}
@@ -54,8 +56,8 @@ func (r *NodeLabelController) SetupCloudProvider(ctx context.Context) error {
54
56
55
57
func (r * NodeLabelController ) SetupWithManager (mgr ctrl.Manager ) error {
56
58
// to reduce the number of API calls to AWS and GCP, filter out node events that
57
- // do not involve changes to the monitored label set (r.labels) .
58
- labelChangePredicate := predicate.Funcs {
59
+ // do not involve changes to the monitored label or annotation sets .
60
+ changePredicate := predicate.Funcs {
59
61
UpdateFunc : func (e event.UpdateEvent ) bool {
60
62
oldNode , ok := e .ObjectOld .(* corev1.Node )
61
63
if ! ok {
@@ -65,15 +67,15 @@ func (r *NodeLabelController) SetupWithManager(mgr ctrl.Manager) error {
65
67
if ! ok {
66
68
return false
67
69
}
68
- return shouldProcessNodeUpdate (oldNode , newNode , r .Labels )
70
+ return shouldProcessNodeUpdate (oldNode , newNode , r .Labels , r . Annotations )
69
71
},
70
72
71
73
CreateFunc : func (e event.CreateEvent ) bool {
72
74
node , ok := e .Object .(* corev1.Node )
73
75
if ! ok {
74
76
return false
75
77
}
76
- return shouldProcessNodeCreate (node , r .Labels )
78
+ return shouldProcessNodeCreate (node , r .Labels , r . Annotations )
77
79
},
78
80
79
81
DeleteFunc : func (e event.DeleteEvent ) bool {
@@ -87,21 +89,46 @@ func (r *NodeLabelController) SetupWithManager(mgr ctrl.Manager) error {
87
89
88
90
return ctrl .NewControllerManagedBy (mgr ).
89
91
For (& corev1.Node {}).
90
- WithEventFilter (labelChangePredicate ).
92
+ WithEventFilter (changePredicate ).
91
93
Complete (r )
92
94
}
93
95
94
96
// shouldProcessNodeUpdate determines if a node update event should trigger reconciliation
95
- // based on whether any monitored labels have changed.
96
- func shouldProcessNodeUpdate (oldNode , newNode * corev1.Node , monitoredLabels []string ) bool {
97
+ // based on whether any monitored labels or annotations have changed.
98
+ func shouldProcessNodeUpdate (oldNode , newNode * corev1.Node , monitoredLabels , monitoredAnnotations []string ) bool {
97
99
if oldNode == nil || newNode == nil {
98
100
return false
99
101
}
100
102
101
103
// Check if any monitored labels changed
102
104
for _ , k := range monitoredLabels {
103
- newVal , newExists := newNode .Labels [k ]
104
- oldVal , oldExists := oldNode .Labels [k ]
105
+ newVal , newExists := "" , false
106
+ oldVal , oldExists := "" , false
107
+
108
+ if newNode .Labels != nil {
109
+ newVal , newExists = newNode .Labels [k ]
110
+ }
111
+ if oldNode .Labels != nil {
112
+ oldVal , oldExists = oldNode .Labels [k ]
113
+ }
114
+
115
+ if newExists != oldExists || (newExists && newVal != oldVal ) {
116
+ return true
117
+ }
118
+ }
119
+
120
+ // Check if any monitored annotations changed
121
+ for _ , k := range monitoredAnnotations {
122
+ newVal , newExists := "" , false
123
+ oldVal , oldExists := "" , false
124
+
125
+ if newNode .Annotations != nil {
126
+ newVal , newExists = newNode .Annotations [k ]
127
+ }
128
+ if oldNode .Annotations != nil {
129
+ oldVal , oldExists = oldNode .Annotations [k ]
130
+ }
131
+
105
132
if newExists != oldExists || (newExists && newVal != oldVal ) {
106
133
return true
107
134
}
@@ -110,15 +137,27 @@ func shouldProcessNodeUpdate(oldNode, newNode *corev1.Node, monitoredLabels []st
110
137
}
111
138
112
139
// shouldProcessNodeCreate determines if a newly created node should trigger reconciliation
113
- // based on whether it has any of the monitored labels.
114
- func shouldProcessNodeCreate (node * corev1.Node , monitoredLabels []string ) bool {
140
+ // based on whether it has any of the monitored labels or annotations .
141
+ func shouldProcessNodeCreate (node * corev1.Node , monitoredLabels , monitoredAnnotations []string ) bool {
115
142
if node == nil {
116
143
return false
117
144
}
118
145
119
- for _ , k := range monitoredLabels {
120
- if _ , ok := node .Labels [k ]; ok {
121
- return true
146
+ // Check if node has any monitored labels
147
+ if node .Labels != nil {
148
+ for _ , k := range monitoredLabels {
149
+ if _ , ok := node .Labels [k ]; ok {
150
+ return true
151
+ }
152
+ }
153
+ }
154
+
155
+ // Check if node has any monitored annotations
156
+ if node .Annotations != nil {
157
+ for _ , k := range monitoredAnnotations {
158
+ if _ , ok := node .Annotations [k ]; ok {
159
+ return true
160
+ }
122
161
}
123
162
}
124
163
return false
@@ -139,31 +178,45 @@ func (r *NodeLabelController) Reconcile(ctx context.Context, req ctrl.Request) (
139
178
return ctrl.Result {}, nil
140
179
}
141
180
142
- labels := make (map [string ]string )
143
- for _ , k := range r .Labels {
144
- if value , exists := node .Labels [k ]; exists {
145
- labels [k ] = value
181
+ // Create a map for tags to sync with the cloud provider
182
+ tagsToSync := make (map [string ]string )
183
+
184
+ // First collect labels (may be overwritten by annotations with same key)
185
+ if node .Labels != nil {
186
+ for _ , k := range r .Labels {
187
+ if value , exists := node .Labels [k ]; exists {
188
+ tagsToSync [k ] = value
189
+ }
190
+ }
191
+ }
192
+
193
+ // Then collect annotations (will overwrite labels with same key)
194
+ if node .Annotations != nil {
195
+ for _ , k := range r .Annotations {
196
+ if value , exists := node .Annotations [k ]; exists {
197
+ tagsToSync [k ] = value
198
+ }
146
199
}
147
200
}
148
201
149
202
var err error
150
203
switch r .Cloud {
151
204
case "aws" :
152
- err = r .syncAWSTags (ctx , providerID , labels )
205
+ err = r .syncAWSTags (ctx , providerID , tagsToSync )
153
206
case "gcp" :
154
- err = r .syncGCPLabels (ctx , providerID , labels )
207
+ err = r .syncGCPLabels (ctx , providerID , tagsToSync )
155
208
}
156
209
157
210
if err != nil {
158
- logger .Error (err , "failed to sync labels " )
211
+ logger .Error (err , "failed to sync tags " )
159
212
return ctrl.Result {}, err
160
213
}
161
214
162
- logger .Info ("Successfully synced labels to cloud provider" , "labels " , labels )
215
+ logger .Info ("Successfully synced tags to cloud provider" , "tags " , tagsToSync )
163
216
return ctrl.Result {}, nil
164
217
}
165
218
166
- func (r * NodeLabelController ) syncAWSTags (ctx context.Context , providerID string , desiredLabels map [string ]string ) error {
219
+ func (r * NodeLabelController ) syncAWSTags (ctx context.Context , providerID string , desiredTags map [string ]string ) error {
167
220
instanceID := path .Base (providerID )
168
221
if instanceID == "" {
169
222
return fmt .Errorf ("invalid AWS provider ID format: %q" , providerID )
@@ -181,9 +234,19 @@ func (r *NodeLabelController) syncAWSTags(ctx context.Context, providerID string
181
234
return fmt .Errorf ("failed to fetch node's current AWS tags: %v" , err )
182
235
}
183
236
237
+ // Create a set of all monitored keys (both labels and annotations)
238
+ monitoredKeys := make (map [string ]bool )
239
+ for _ , k := range r .Labels {
240
+ monitoredKeys [k ] = true
241
+ }
242
+ for _ , k := range r .Annotations {
243
+ monitoredKeys [k ] = true
244
+ }
245
+
184
246
currentTags := make (map [string ]string )
185
247
for _ , tag := range result .Tags {
186
- if key := aws .ToString (tag .Key ); key != "" && slices .Contains (r .Labels , key ) {
248
+ key := aws .ToString (tag .Key )
249
+ if key != "" && monitoredKeys [key ] {
187
250
currentTags [key ] = aws .ToString (tag .Value )
188
251
}
189
252
}
@@ -192,7 +255,7 @@ func (r *NodeLabelController) syncAWSTags(ctx context.Context, providerID string
192
255
toDelete := make ([]types.Tag , 0 )
193
256
194
257
// find tags to add or update
195
- for k , v := range desiredLabels {
258
+ for k , v := range desiredTags {
196
259
if curr , exists := currentTags [k ]; ! exists || curr != v {
197
260
toAdd = append (toAdd , types.Tag {
198
261
Key : aws .String (k ),
@@ -203,8 +266,8 @@ func (r *NodeLabelController) syncAWSTags(ctx context.Context, providerID string
203
266
204
267
// find monitored tags to remove
205
268
for k := range currentTags {
206
- if slices . Contains ( r . Labels , k ) {
207
- if _ , exists := desiredLabels [k ]; ! exists {
269
+ if monitoredKeys [ k ] {
270
+ if _ , exists := desiredTags [k ]; ! exists {
208
271
toDelete = append (toDelete , types.Tag {
209
272
Key : aws .String (k ),
210
273
})
@@ -235,7 +298,7 @@ func (r *NodeLabelController) syncAWSTags(ctx context.Context, providerID string
235
298
return nil
236
299
}
237
300
238
- func (r * NodeLabelController ) syncGCPLabels (ctx context.Context , providerID string , desiredLabels map [string ]string ) error {
301
+ func (r * NodeLabelController ) syncGCPLabels (ctx context.Context , providerID string , desiredTags map [string ]string ) error {
239
302
project , zone , name , err := parseGCPProviderID (providerID )
240
303
if err != nil {
241
304
return fmt .Errorf ("failed to parse GCP provider ID: %v" , err )
@@ -251,23 +314,28 @@ func (r *NodeLabelController) syncGCPLabels(ctx context.Context, providerID stri
251
314
newLabels = make (map [string ]string )
252
315
}
253
316
317
+ // Create a set of all monitored keys (both labels and annotations)
318
+ allMonitoredKeys := make ([]string , 0 , len (r .Labels )+ len (r .Annotations ))
319
+ allMonitoredKeys = append (allMonitoredKeys , r .Labels ... )
320
+ allMonitoredKeys = append (allMonitoredKeys , r .Annotations ... )
321
+
254
322
// create a set of sanitized monitored keys for easy lookup
255
323
monitoredKeys := make (map [string ]string ) // sanitized -> original
256
- for _ , k := range r . Labels {
324
+ for _ , k := range allMonitoredKeys {
257
325
monitoredKeys [sanitizeKeyForGCP (k )] = k
258
326
}
259
327
260
328
// remove any existing monitored labels that are no longer desired
261
329
for k := range newLabels {
262
330
if orig , isMonitored := monitoredKeys [k ]; isMonitored {
263
- if _ , exists := desiredLabels [orig ]; ! exists {
331
+ if _ , exists := desiredTags [orig ]; ! exists {
264
332
delete (newLabels , k )
265
333
}
266
334
}
267
335
}
268
336
269
- // add or update desired labels
270
- for k , v := range desiredLabels {
337
+ // add or update desired tags
338
+ for k , v := range desiredTags {
271
339
newLabels [sanitizeKeyForGCP (k )] = sanitizeValueForGCP (v )
272
340
}
273
341
0 commit comments