Skip to content

Commit bf95051

Browse files
committed
fix: use SSA matcher to filter events
closes: operator-framework#2249 Signed-off-by: Steve Hawkins <[email protected]>
1 parent 7725ff4 commit bf95051

File tree

3 files changed

+66
-13
lines changed

3 files changed

+66
-13
lines changed

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/processing/dependent/kubernetes/SSABasedGenericKubernetesResourceMatcher.java

+22-6
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@
2323
import io.fabric8.kubernetes.api.model.apps.Deployment;
2424
import io.fabric8.kubernetes.api.model.apps.ReplicaSet;
2525
import io.fabric8.kubernetes.api.model.apps.StatefulSet;
26+
import io.fabric8.kubernetes.client.KubernetesClient;
2627
import io.fabric8.kubernetes.client.utils.KubernetesSerialization;
2728
import io.javaoperatorsdk.operator.OperatorException;
2829
import io.javaoperatorsdk.operator.api.reconciler.Context;
@@ -79,10 +80,14 @@ public static <L extends HasMetadata> SSABasedGenericKubernetesResourceMatcher<L
7980
private static final Logger log =
8081
LoggerFactory.getLogger(SSABasedGenericKubernetesResourceMatcher.class);
8182

82-
@SuppressWarnings("unchecked")
8383
public boolean matches(R actual, R desired, Context<?> context) {
84-
var optionalManagedFieldsEntry =
85-
checkIfFieldManagerExists(actual, context.getControllerConfiguration().fieldManager());
84+
return matches(
85+
actual, desired, context.getControllerConfiguration().fieldManager(), context.getClient(), false);
86+
}
87+
88+
@SuppressWarnings("unchecked")
89+
public boolean matches(R actual, R desired, String fieldManager, KubernetesClient client, boolean compareActual) {
90+
var optionalManagedFieldsEntry = checkIfFieldManagerExists(actual, fieldManager);
8691
// If no field is managed by our controller, that means the controller hasn't touched the
8792
// resource yet and the resource probably doesn't match the desired state. Not matching here
8893
// means that the resource will need to be updated and since this will be done using SSA, the
@@ -93,7 +98,7 @@ public boolean matches(R actual, R desired, Context<?> context) {
9398

9499
var managedFieldsEntry = optionalManagedFieldsEntry.orElseThrow();
95100

96-
var objectMapper = context.getClient().getKubernetesSerialization();
101+
var objectMapper = client.getKubernetesSerialization();
97102
var actualMap = objectMapper.convertValue(actual, Map.class);
98103
var desiredMap = objectMapper.convertValue(desired, Map.class);
99104
if (LoggingUtils.isNotSensitiveResource(desired)) {
@@ -108,7 +113,18 @@ public boolean matches(R actual, R desired, Context<?> context) {
108113
managedFieldsEntry.getFieldsV1().getAdditionalProperties(),
109114
objectMapper);
110115

111-
removeIrrelevantValues(desiredMap);
116+
if (compareActual) {
117+
var prunedDesired = new HashMap<String, Object>(actualMap.size());
118+
keepOnlyManagedFields(
119+
prunedDesired,
120+
desiredMap,
121+
managedFieldsEntry.getFieldsV1().getAdditionalProperties(),
122+
objectMapper);
123+
124+
desiredMap = prunedDesired;
125+
} else {
126+
removeIrrelevantValues(desiredMap);
127+
}
112128

113129
var matches = prunedActual.equals(desiredMap);
114130
if (!matches && log.isDebugEnabled() && LoggingUtils.isNotSensitiveResource(desired)) {
@@ -438,7 +454,7 @@ private static Map.Entry<Integer, Map<String, Object>> selectListEntryBasedOnKey
438454
return new AbstractMap.SimpleEntry<>(lastIndex, possibleTargets.get(0));
439455
}
440456

441-
private Optional<ManagedFieldsEntry> checkIfFieldManagerExists(R actual, String fieldManager) {
457+
public Optional<ManagedFieldsEntry> checkIfFieldManagerExists(R actual, String fieldManager) {
442458
var targetManagedFields =
443459
actual.getMetadata().getManagedFields().stream()
444460
// Only the apply operations are interesting for us since those were created properly be

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/processing/event/source/informer/InformerEventSource.java

+26-7
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,7 @@
1414
import io.fabric8.kubernetes.client.informers.ResourceEventHandler;
1515
import io.javaoperatorsdk.operator.api.config.informer.InformerEventSourceConfiguration;
1616
import io.javaoperatorsdk.operator.api.reconciler.EventSourceContext;
17+
import io.javaoperatorsdk.operator.processing.dependent.kubernetes.SSABasedGenericKubernetesResourceMatcher;
1718
import io.javaoperatorsdk.operator.processing.event.Event;
1819
import io.javaoperatorsdk.operator.processing.event.EventHandler;
1920
import io.javaoperatorsdk.operator.processing.event.ResourceID;
@@ -68,6 +69,8 @@ public class InformerEventSource<R extends HasMetadata, P extends HasMetadata>
6869
private final PrimaryToSecondaryIndex<R> primaryToSecondaryIndex;
6970
private final PrimaryToSecondaryMapper<P> primaryToSecondaryMapper;
7071
private final String id = UUID.randomUUID().toString();
72+
private final KubernetesClient client;
73+
private final String fieldManager;
7174

7275
public InformerEventSource(
7376
InformerEventSourceConfiguration<R> configuration, EventSourceContext<P> context) {
@@ -77,18 +80,20 @@ public InformerEventSource(
7780
context
7881
.getControllerConfiguration()
7982
.getConfigurationService()
80-
.parseResourceVersionsForEventFilteringAndCaching());
83+
.parseResourceVersionsForEventFilteringAndCaching(),
84+
context.getControllerConfiguration().fieldManager());
8185
}
8286

8387
InformerEventSource(InformerEventSourceConfiguration<R> configuration, KubernetesClient client) {
84-
this(configuration, client, false);
88+
this(configuration, client, false, "manager");
8589
}
8690

8791
@SuppressWarnings({"unchecked", "rawtypes"})
8892
private InformerEventSource(
8993
InformerEventSourceConfiguration<R> configuration,
9094
KubernetesClient client,
91-
boolean parseResourceVersions) {
95+
boolean parseResourceVersions,
96+
String fieldManager) {
9297
super(
9398
configuration.name(),
9499
configuration
@@ -112,6 +117,8 @@ private InformerEventSource(
112117
onUpdateFilter = informerConfig.getOnUpdateFilter();
113118
onDeleteFilter = informerConfig.getOnDeleteFilter();
114119
genericFilter = informerConfig.getGenericFilter();
120+
this.client = client;
121+
this.fieldManager = fieldManager;
115122
}
116123

117124
@Override
@@ -215,10 +222,22 @@ private boolean isEventKnownFromAnnotation(R newObject, R oldObject) {
215222
if (id.equals(parts[0])) {
216223
if (oldObject == null && parts.length == 1) {
217224
known = true;
218-
} else if (oldObject != null
219-
&& parts.length == 2
220-
&& oldObject.getMetadata().getResourceVersion().equals(parts[1])) {
221-
known = true;
225+
} else if (oldObject != null && parts.length == 2) {
226+
if (oldObject.getMetadata().getResourceVersion().equals(parts[1])) {
227+
known = true;
228+
} else {
229+
// if the resource version doesn't match, there's a chance that it's due
230+
// to a change that is not meaningful, but causes the matcher logic between
231+
// desired and newObject to see a change.
232+
// TODO: this likely should be conditional on useSSA, not just the presence of the
233+
// field manager
234+
var ssaMatcher = SSABasedGenericKubernetesResourceMatcher.getInstance();
235+
if (ssaMatcher.checkIfFieldManagerExists(newObject, fieldManager).isPresent()) {
236+
known = ssaMatcher.matches(oldObject, newObject, fieldManager, client, true);
237+
} else {
238+
// do we even need to worry about this case
239+
}
240+
}
222241
}
223242
}
224243
}

operator-framework-core/src/test/java/io/javaoperatorsdk/operator/processing/dependent/kubernetes/SSABasedGenericKubernetesResourceMatcherTest.java

+18
Original file line numberDiff line numberDiff line change
@@ -114,7 +114,25 @@ void addedLabelInDesiredMakesMatchFail() {
114114

115115
assertThat(matcher.matches(actualConfigMap, desiredConfigMap, mockedContext)).isFalse();
116116
}
117+
118+
@Test
119+
void compareActualSame() {
120+
var actualConfigMap = loadResource("configmap.empty-owner-reference.yaml", ConfigMap.class);
117121

122+
assertThat(matcher.matches(actualConfigMap, actualConfigMap, mockedContext.getControllerConfiguration().fieldManager(), mockedContext.getClient(), true)).isTrue();
123+
}
124+
125+
@Test
126+
void compareActualSameWithUnownedChange() {
127+
var actualConfigMap = loadResource("configmap.empty-owner-reference.yaml", ConfigMap.class);
128+
129+
var newConfigMap = loadResource("configmap.empty-owner-reference.yaml", ConfigMap.class);
130+
newConfigMap.getMetadata().setLabels(Map.of("newlabel", "val"));
131+
132+
assertThat(matcher.matches(newConfigMap, actualConfigMap, mockedContext.getControllerConfiguration().fieldManager(), mockedContext.getClient(), true)).isTrue();
133+
assertThat(matcher.matches(actualConfigMap, newConfigMap, mockedContext.getControllerConfiguration().fieldManager(), mockedContext.getClient(), true)).isTrue();
134+
}
135+
118136
@Test
119137
@SuppressWarnings("unchecked")
120138
void sortListItemsTest() {

0 commit comments

Comments
 (0)