Skip to content

Commit a3e2c24

Browse files
committed
feat: configurable SSA per Dependent Resource (#2045)
Signed-off-by: Attila Mészáros <[email protected]>
1 parent 8608e12 commit a3e2c24

File tree

5 files changed

+49
-5
lines changed

5 files changed

+49
-5
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
package io.javaoperatorsdk.operator.processing.dependent.kubernetes;
2+
3+
/**
4+
* A replacement for {@link Boolean}, which can't be used in annotations.
5+
*/
6+
public enum BooleanWithUndefined {
7+
TRUE, FALSE, UNDEFINED;
8+
9+
Boolean asBoolean() {
10+
switch (this) {
11+
case TRUE:
12+
return Boolean.TRUE;
13+
case FALSE:
14+
return Boolean.FALSE;
15+
default:
16+
return null;
17+
}
18+
}
19+
}

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

+13-1
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@
2525
* namespace is specified then the controller will monitor the namespaces configured for the
2626
* controller.
2727
*
28-
* @return the list of namespaces this controller monitors
28+
* @return the array of namespaces this controller monitors
2929
*/
3030
String[] namespaces() default {Constants.SAME_AS_CONTROLLER};
3131

@@ -76,4 +76,16 @@
7676
* Creates the resource only if did not exist before, this applies only if SSA is used.
7777
*/
7878
boolean createResourceOnlyIfNotExistingWithSSA() default KubernetesDependentResourceConfig.DEFAULT_CREATE_RESOURCE_ONLY_IF_NOT_EXISTING_WITH_SSA;
79+
80+
/**
81+
* Determines whether to use SSA (Server-Side Apply) for this dependent. If SSA is used, the
82+
* dependent resource will only be created if it did not exist before. Default value is
83+
* {@link BooleanWithUndefined#UNDEFINED}, which specifies that the behavior with respect to SSA
84+
* is inherited from the global configuration.
85+
*
86+
* @return {@code true} if SSA is enabled, {@code false} if SSA is disabled,
87+
* {@link BooleanWithUndefined#UNDEFINED} if the SSA behavior should be inherited from the
88+
* global configuration
89+
*/
90+
BooleanWithUndefined useSSA() default BooleanWithUndefined.UNDEFINED;
7991
}

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

+3-1
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ public KubernetesDependentResourceConfig<R> configFrom(KubernetesDependent confi
3434
OnDeleteFilter<? extends HasMetadata> onDeleteFilter = null;
3535
GenericFilter<? extends HasMetadata> genericFilter = null;
3636
ResourceDiscriminator<?, ?> resourceDiscriminator = null;
37+
Boolean useSSA = null;
3738
if (configAnnotation != null) {
3839
if (!Arrays.equals(KubernetesDependent.DEFAULT_NAMESPACES, configAnnotation.namespaces())) {
3940
namespaces = Set.of(configAnnotation.namespaces());
@@ -58,10 +59,11 @@ public KubernetesDependentResourceConfig<R> configFrom(KubernetesDependent confi
5859
context);
5960
createResourceOnlyIfNotExistingWithSSA =
6061
configAnnotation.createResourceOnlyIfNotExistingWithSSA();
62+
useSSA = configAnnotation.useSSA().asBoolean();
6163
}
6264

6365
return new KubernetesDependentResourceConfig(namespaces, labelSelector, configuredNS,
6466
createResourceOnlyIfNotExistingWithSSA,
65-
resourceDiscriminator, onAddFilter, onUpdateFilter, onDeleteFilter, genericFilter);
67+
resourceDiscriminator, useSSA, onAddFilter, onUpdateFilter, onDeleteFilter, genericFilter);
6668
}
6769
}

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

+5-2
Original file line numberDiff line numberDiff line change
@@ -190,8 +190,10 @@ protected void addMetadata(boolean forMatch, R actualResource, final R target, P
190190
}
191191

192192
private boolean useSSA(Context<P> context) {
193-
return context.getControllerConfiguration().getConfigurationService()
194-
.ssaBasedCreateUpdateMatchForDependentResources();
193+
Optional<Boolean> useSSAConfig =
194+
configuration().flatMap(KubernetesDependentResourceConfig::useSSA);
195+
return useSSAConfig.orElse(context.getControllerConfiguration().getConfigurationService()
196+
.ssaBasedCreateUpdateMatchForDependentResources());
195197
}
196198

197199
@Override
@@ -206,6 +208,7 @@ public void deleteTargetResource(P primary, R resource, String key, Context<P> c
206208
client.resource(resource).delete();
207209
}
208210

211+
@SuppressWarnings("unused")
209212
protected Resource<R> prepare(R desired, P primary, String actionName) {
210213
log.debug("{} target resource with type: {}, with id: {}",
211214
actionName,

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

+9-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
package io.javaoperatorsdk.operator.processing.dependent.kubernetes;
22

3+
import java.util.Optional;
34
import java.util.Set;
45

56
import io.javaoperatorsdk.operator.api.reconciler.Constants;
@@ -20,6 +21,7 @@ public class KubernetesDependentResourceConfig<R> {
2021
private boolean namespacesWereConfigured = false;
2122
private boolean createResourceOnlyIfNotExistingWithSSA;
2223
private ResourceDiscriminator<R, ?> resourceDiscriminator;
24+
private Boolean useSSA;
2325

2426
private OnAddFilter<R> onAddFilter;
2527

@@ -36,6 +38,7 @@ public KubernetesDependentResourceConfig(Set<String> namespaces,
3638
boolean configuredNS,
3739
boolean createResourceOnlyIfNotExistingWithSSA,
3840
ResourceDiscriminator<R, ?> resourceDiscriminator,
41+
Boolean useSSA,
3942
OnAddFilter<R> onAddFilter,
4043
OnUpdateFilter<R> onUpdateFilter,
4144
OnDeleteFilter<R> onDeleteFilter, GenericFilter<R> genericFilter) {
@@ -48,12 +51,13 @@ public KubernetesDependentResourceConfig(Set<String> namespaces,
4851
this.onDeleteFilter = onDeleteFilter;
4952
this.genericFilter = genericFilter;
5053
this.resourceDiscriminator = resourceDiscriminator;
54+
this.useSSA = useSSA;
5155
}
5256

5357
public KubernetesDependentResourceConfig(Set<String> namespaces, String labelSelector) {
5458
this(namespaces, labelSelector, true, DEFAULT_CREATE_RESOURCE_ONLY_IF_NOT_EXISTING_WITH_SSA,
5559
null, null, null,
56-
null, null);
60+
null, null, null);
5761
}
5862

5963
public KubernetesDependentResourceConfig<R> setLabelSelector(String labelSelector) {
@@ -105,4 +109,8 @@ protected void setNamespaces(Set<String> namespaces) {
105109
this.namespaces = namespaces;
106110
}
107111
}
112+
113+
public Optional<Boolean> useSSA() {
114+
return Optional.ofNullable(useSSA);
115+
}
108116
}

0 commit comments

Comments
 (0)