14
14
import com .fasterxml .jackson .core .type .TypeReference ;
15
15
import com .fasterxml .jackson .databind .ObjectMapper ;
16
16
17
+ /**
18
+ * Matches the actual state on the server vs the desired state. Based on the managedFields of SSA.
19
+ *
20
+ * The ide of algorithm is basically trivial, we convert resources to Map/List composition.
21
+ * The actual resource (from the server) is pruned, all the fields which are not mentioed in managedFields
22
+ * of the target manager is removed. Some irrelevant fields are also removed from desired. And the
23
+ * two resulted Maps are compared for equality. The implementation is a bit nasty since have to deal with
24
+ * some specific cases of managedFields format.
25
+ *
26
+ * @param <R> matched resource type
27
+ */
17
28
// https://kubernetes.io/docs/reference/generated/kubernetes-api/v1.27/#fieldsv1-v1-meta
18
29
// https://github.com/kubernetes-sigs/structured-merge-diff
19
30
// https://docs.aws.amazon.com/eks/latest/userguide/kubernetes-field-management.html
@@ -69,7 +80,7 @@ public boolean matches(R actual, R desired, Context<?> context) {
69
80
70
81
removeIrrelevantValues (desiredMap );
71
82
72
- log .trace ("Pruned actual: \n {} \n desired: \n {} " , prunedActual , desiredMap );
83
+ log .debug ("Pruned actual: \n {} \n desired: \n {} " , prunedActual , desiredMap );
73
84
74
85
return prunedActual .equals (desiredMap );
75
86
} catch (JsonProcessingException e ) {
@@ -99,22 +110,24 @@ private void pruneActualAccordingManagedFields(Map<String, Object> result,
99
110
for (Map .Entry <String , Object > entry : managedFields .entrySet ()) {
100
111
String key = entry .getKey ();
101
112
if (key .startsWith (F_PREFIX )) {
102
- String keyInActual = key . substring ( 2 );
113
+ String keyInActual = keyWithoutPrefix ( key );
103
114
var managedFieldValue = entry .getValue ();
104
- // basically if we should travers further
105
115
if (isNestedValue (managedFieldValue )) {
106
116
var managedEntrySet = ((Map <String , Object >) managedFieldValue ).entrySet ();
107
117
118
+ // two special cases "k:" and "v:" prefixes
108
119
if (isListKeyEntrySet (managedEntrySet )) {
109
120
handleListKeyEntrySet (result , actualMap , objectMapper , keyInActual , managedEntrySet );
110
121
} else if (isSetValueField (managedEntrySet )) {
111
122
handleSetValues (result , actualMap , objectMapper , keyInActual , managedEntrySet );
112
123
} else {
124
+ // basically if we should travers further
113
125
fillResultsAndTraversFurther (result , actualMap , managedFields , objectMapper , key ,
114
126
keyInActual , managedFieldValue );
115
127
}
116
128
} else {
117
- // this should handle the case when the value is complex in the actual map (not just a simple value).
129
+ // this should handle the case when the value is complex in the actual map (not just a
130
+ // simple value).
118
131
result .put (keyInActual , actualMap .get (keyInActual ));
119
132
}
120
133
} else {
@@ -124,7 +137,6 @@ private void pruneActualAccordingManagedFields(Map<String, Object> result,
124
137
}
125
138
}
126
139
}
127
-
128
140
}
129
141
130
142
private void fillResultsAndTraversFurther (Map <String , Object > result ,
@@ -146,31 +158,38 @@ private static boolean isNestedValue(Object managedFieldValue) {
146
158
147
159
// list entries referenced by key, or when "k:" prefix is used
148
160
private void handleListKeyEntrySet (Map <String , Object > result , Map <String , Object > actualMap ,
149
- ObjectMapper objectMapper , String keyInActual , Set <Map .Entry <String , Object >> managedEntrySet )
150
- throws JsonProcessingException {
161
+ ObjectMapper objectMapper , String keyInActual , Set <Map .Entry <String , Object >> managedEntrySet ) {
151
162
var valueList = new ArrayList <>();
152
163
result .put (keyInActual , valueList );
153
- for (Map .Entry <String , Object > listEntry : managedEntrySet ) {
154
- var actualListEntry = selectListEntryBasedOnKey (listEntry .getKey ().substring (2 ),
155
- (List <Map <String , Object >>) actualMap .get (keyInActual ), objectMapper );
156
-
157
- if (listEntry .getValue () instanceof Map
158
- && ((Map <String , Object >) listEntry .getValue ()).isEmpty ()) {
159
- var map = new HashMap <String , Object >();
160
- valueList .add (map );
161
- pruneActualAccordingManagedFields (map , actualListEntry ,
162
- (Map <String , Object >) listEntry .getValue (), objectMapper );
163
- continue ;
164
- }
164
+ var actualValueList = (List <Map <String , Object >>) actualMap .get (keyInActual );
165
+
166
+ Map <Integer , Map <String , Object >> targetValuesByIndex = new HashMap <>();
167
+ Map <Integer , Map <String , Object >> mangedEntryByIndex = new HashMap <>();
165
168
169
+ for (Map .Entry <String , Object > listEntry : managedEntrySet ) {
166
170
if (DOT_KEY .equals (listEntry .getKey ())) {
167
171
continue ;
168
172
}
169
- var emptyResMapValue = new HashMap < String , Object >();
170
- valueList . add ( emptyResMapValue );
171
- pruneActualAccordingManagedFields ( emptyResMapValue , actualListEntry ,
172
- (Map <String , Object >) listEntry .getValue (), objectMapper );
173
+ var actualListEntry = selectListEntryBasedOnKey ( keyWithoutPrefix ( listEntry . getKey ()),
174
+ actualValueList , objectMapper );
175
+ targetValuesByIndex . put ( actualListEntry . getKey () , actualListEntry . getValue ());
176
+ mangedEntryByIndex . put ( actualListEntry . getKey (), (Map <String , Object >) listEntry .getValue ());
173
177
}
178
+
179
+ targetValuesByIndex .entrySet ()
180
+ .stream ()
181
+ // list is sorted according to the value in actual
182
+ .sorted (Map .Entry .comparingByKey ())
183
+ .forEach (e -> {
184
+ var emptyResMapValue = new HashMap <String , Object >();
185
+ valueList .add (emptyResMapValue );
186
+ try {
187
+ pruneActualAccordingManagedFields (emptyResMapValue , e .getValue (),
188
+ mangedEntryByIndex .get (e .getKey ()), objectMapper );
189
+ } catch (JsonProcessingException ex ) {
190
+ throw new IllegalStateException (ex );
191
+ }
192
+ });
174
193
}
175
194
176
195
// set values, the "v:" prefix
@@ -184,15 +203,14 @@ private static void handleSetValues(Map<String, Object> result, Map<String, Obje
184
203
if (DOT_KEY .equals (valueEntry .getKey ())) {
185
204
continue ;
186
205
}
187
-
188
206
Class <?> targetClass = null ;
189
207
List values = (List ) actualMap .get (keyInActual );
190
208
if (!(values .get (0 ) instanceof Map )) {
191
209
targetClass = values .get (0 ).getClass ();
192
210
}
193
211
194
212
var value =
195
- parseKeyValue (valueEntry .getKey (). substring ( 2 ), targetClass , objectMapper );
213
+ parseKeyValue (keyWithoutPrefix ( valueEntry .getKey ()), targetClass , objectMapper );
196
214
valueList .add (value );
197
215
}
198
216
}
@@ -223,21 +241,26 @@ private boolean isSetValueField(Set<Map.Entry<String, Object>> managedEntrySet)
223
241
private boolean isListKeyEntrySet (Set <Map .Entry <String , Object >> managedEntrySet ) {
224
242
var iterator = managedEntrySet .iterator ();
225
243
var managedFieldEntry = iterator .next ();
226
- // todo unit test
227
244
if (managedFieldEntry .getKey ().equals (DOT_KEY )) {
228
245
managedFieldEntry = iterator .next ();
229
246
}
230
247
return managedFieldEntry .getKey ().startsWith (K_PREFIX );
231
248
}
232
249
233
- private Map < String , Object > selectListEntryBasedOnKey (String key ,
250
+ private java . util . Map . Entry < Integer , Map < String , Object > > selectListEntryBasedOnKey (String key ,
234
251
List <Map <String , Object >> values ,
235
252
ObjectMapper objectMapper ) {
236
253
try {
237
254
Map <String , Object > ids = objectMapper .readValue (key , typeRef );
238
- var possibleTargets =
239
- values .stream ().filter (v -> v .entrySet ().containsAll (ids .entrySet ()))
240
- .collect (Collectors .toList ());
255
+ List <Map <String , Object >> possibleTargets = new ArrayList <>(1 );
256
+ int index = -1 ;
257
+ for (int i = 0 ; i < values .size (); i ++) {
258
+ var v = values .get (i );
259
+ if (v .entrySet ().containsAll (ids .entrySet ())) {
260
+ possibleTargets .add (v );
261
+ index = i ;
262
+ }
263
+ }
241
264
if (possibleTargets .isEmpty ()) {
242
265
throw new IllegalStateException (
243
266
"Cannot find list element for key:" + key + ", in map: " + values );
@@ -246,8 +269,23 @@ private Map<String, Object> selectListEntryBasedOnKey(String key,
246
269
throw new IllegalStateException (
247
270
"More targets found in list element for key:" + key + ", in map: " + values );
248
271
}
272
+ final var finalIndex = index ;
273
+ return new Map .Entry <>() {
274
+ @ Override
275
+ public Integer getKey () {
276
+ return finalIndex ;
277
+ }
278
+
279
+ @ Override
280
+ public Map <String , Object > getValue () {
281
+ return possibleTargets .get (0 );
282
+ }
249
283
250
- return possibleTargets .get (0 );
284
+ @ Override
285
+ public Map <String , Object > setValue (Map <String , Object > stringObjectMap ) {
286
+ throw new IllegalStateException ("should not be called" );
287
+ }
288
+ };
251
289
} catch (JsonProcessingException e ) {
252
290
throw new IllegalStateException (e );
253
291
}
@@ -276,4 +314,8 @@ private Optional<ManagedFieldsEntry> checkIfFieldManagerExists(R actual, String
276
314
return Optional .of (targetManagedFields .get (0 ));
277
315
}
278
316
317
+ private static String keyWithoutPrefix (String key ) {
318
+ return key .substring (2 );
319
+ }
320
+
279
321
}
0 commit comments