Skip to content

Commit 249b41f

Browse files
committed
feat: SSA based dependent resource matching and create/update (#1928)
1 parent bfdb07e commit 249b41f

File tree

35 files changed

+1552
-25
lines changed

35 files changed

+1552
-25
lines changed

cm.yaml

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
apiVersion: v1
2+
kind: ConfigMap
3+
metadata:
4+
name: test1
5+
namespace: default
6+
ownerReferences:
7+
- apiVersion: v1
8+
kind: ConfigMap
9+
name: kube-root-ca.crt
10+
uid: 1ef74cb4-dbbd-45ef-9caf-aa76186594ea
11+
data:
12+
key1: "val1"
13+
# key2: "val2"
14+

cm2.yaml

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
apiVersion: v1
2+
kind: ConfigMap
3+
metadata:
4+
name: test1
5+
namespace: default
6+
data:
7+
key3: "val3"
8+
9+

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/api/config/BaseConfigurationService.java

+7-1
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@
3333

3434
import com.fasterxml.jackson.databind.ObjectMapper;
3535

36+
import static io.javaoperatorsdk.operator.api.config.ControllerConfiguration.CONTROLLER_NAME_AS_FIELD_MANAGER;
3637
import static io.javaoperatorsdk.operator.api.reconciler.Constants.DEFAULT_NAMESPACES_SET;
3738

3839
public class BaseConfigurationService extends AbstractConfigurationService {
@@ -135,6 +136,10 @@ protected <P extends HasMetadata> ControllerConfiguration<P> configFor(Reconcile
135136
timeUnit = reconciliationInterval.timeUnit();
136137
}
137138

139+
final var dependentFieldManager =
140+
annotation.fieldManager().equals(CONTROLLER_NAME_AS_FIELD_MANAGER) ? name
141+
: annotation.fieldManager();
142+
138143
final var config = new ResolvedControllerConfiguration<P>(
139144
resourceClass, name, generationAware,
140145
associatedReconcilerClass, retry, rateLimiter,
@@ -152,7 +157,8 @@ protected <P extends HasMetadata> ControllerConfiguration<P> configFor(Reconcile
152157
io.javaoperatorsdk.operator.api.reconciler.ControllerConfiguration::labelSelector,
153158
Constants.NO_VALUE_SET),
154159
null,
155-
Utils.instantiate(annotation.itemStore(), ItemStore.class, context), this);
160+
Utils.instantiate(annotation.itemStore(), ItemStore.class, context), dependentFieldManager,
161+
this);
156162

157163
ResourceEventFilter<P> answer = deprecatedEventFilter(annotation);
158164
config.setEventFilter(answer != null ? answer : ResourceEventFilters.passthrough());

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/api/config/ConfigurationService.java

+24
Original file line numberDiff line numberDiff line change
@@ -264,4 +264,28 @@ static ConfigurationService newOverriddenConfigurationService(
264264
default ExecutorServiceManager getExecutorServiceManager() {
265265
return new ExecutorServiceManager(this);
266266
}
267+
268+
/**
269+
* Allows to revert to the 4.3 behavior when it comes to creating or updating Kubernetes Dependent
270+
* Resources when set to {@code false}. The default approach how these resources are
271+
* created/updated was change to use
272+
* <a href="https://kubernetes.io/docs/reference/using-api/server-side-apply/">Server-Side
273+
* Apply</a> (SSA) by default. Note that the legacy approach, and this setting, might be removed
274+
* in the future.
275+
*/
276+
default boolean ssaBasedCreateUpdateForDependentResources() {
277+
return true;
278+
}
279+
280+
/**
281+
* Allows to revert to the 4.3 generic matching algorithm for Kubernetes Dependent Resources when
282+
* set to {@code false}. Version 4.4 introduced a new generic matching algorithm for Kubernetes
283+
* Dependent Resources which is quite complex. As a consequence, we introduced this setting to
284+
* allow folks to revert to the previous matching algorithm if needed. Note, however, that the
285+
* legacy algorithm, and this setting, might be removed in the future.
286+
*/
287+
default boolean ssaBasedDefaultMatchingForDependentResources() {
288+
return true;
289+
}
290+
267291
}

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/api/config/ConfigurationServiceOverrider.java

+28
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,8 @@ public class ConfigurationServiceOverrider {
3232
private Boolean stopOnInformerErrorDuringStartup;
3333
private Duration cacheSyncTimeout;
3434
private ResourceClassResolver resourceClassResolver;
35+
private Boolean ssaBasedCreateUpdateForDependentResources;
36+
private Boolean ssaBasedDefaultMatchingForDependentResources;
3537

3638
ConfigurationServiceOverrider(ConfigurationService original) {
3739
this.original = original;
@@ -139,6 +141,18 @@ public ConfigurationServiceOverrider withResourceClassResolver(
139141
return this;
140142
}
141143

144+
public ConfigurationServiceOverrider withSSABasedCreateUpdateForDependentResources(
145+
boolean value) {
146+
this.ssaBasedCreateUpdateForDependentResources = value;
147+
return this;
148+
}
149+
150+
public ConfigurationServiceOverrider withSSABasedDefaultMatchingForDependentResources(
151+
boolean value) {
152+
this.ssaBasedDefaultMatchingForDependentResources = value;
153+
return this;
154+
}
155+
142156
public ConfigurationService build() {
143157
return new BaseConfigurationService(original.getVersion(), cloner, objectMapper) {
144158
@Override
@@ -248,6 +262,20 @@ public ResourceClassResolver getResourceClassResolver() {
248262
return resourceClassResolver != null ? resourceClassResolver
249263
: super.getResourceClassResolver();
250264
}
265+
266+
@Override
267+
public boolean ssaBasedCreateUpdateForDependentResources() {
268+
return ssaBasedCreateUpdateForDependentResources != null
269+
? ssaBasedCreateUpdateForDependentResources
270+
: super.ssaBasedCreateUpdateForDependentResources();
271+
}
272+
273+
@Override
274+
public boolean ssaBasedDefaultMatchingForDependentResources() {
275+
return ssaBasedDefaultMatchingForDependentResources != null
276+
? ssaBasedDefaultMatchingForDependentResources
277+
: super.ssaBasedDefaultMatchingForDependentResources();
278+
}
251279
};
252280
}
253281

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/api/config/ControllerConfiguration.java

+16
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,10 @@ public interface ControllerConfiguration<P extends HasMetadata> extends Resource
2222

2323
@SuppressWarnings("rawtypes")
2424
RateLimiter DEFAULT_RATE_LIMITER = LinearRateLimiter.deactivatedRateLimiter();
25+
/**
26+
* Will use the controller name as fieldManager if set.
27+
*/
28+
String CONTROLLER_NAME_AS_FIELD_MANAGER = "use_controller_name";
2529

2630
default String getName() {
2731
return ensureValidName(null, getAssociatedReconcilerClassName());
@@ -124,4 +128,16 @@ default Class<P> getResourceClass() {
124128
default Set<String> getEffectiveNamespaces() {
125129
return ResourceConfiguration.super.getEffectiveNamespaces(getConfigurationService());
126130
}
131+
132+
/**
133+
* Retrieves the name used to assign as field manager for
134+
* <a href="https://kubernetes.io/docs/reference/using-api/server-side-apply/">Server-Side
135+
* Apply</a> (SSA) operations. If unset, the sanitized controller name will be used.
136+
*
137+
* @return the name used as field manager for SSA operations
138+
*/
139+
default String fieldManager() {
140+
return getName();
141+
}
142+
127143
}

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/api/config/ControllerConfigurationOverrider.java

+9-1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ public class ControllerConfigurationOverrider<R extends HasMetadata> {
3939
private Map<DependentResourceSpec, Object> configurations;
4040
private ItemStore<R> itemStore;
4141
private String name;
42+
private String fieldManager;
4243

4344
private ControllerConfigurationOverrider(ControllerConfiguration<R> original) {
4445
this.finalizer = original.getFinalizerName();
@@ -54,6 +55,7 @@ private ControllerConfigurationOverrider(ControllerConfiguration<R> original) {
5455
this.original = original;
5556
this.rateLimiter = original.getRateLimiter();
5657
this.name = original.getName();
58+
this.fieldManager = original.fieldManager();
5759
}
5860

5961
public ControllerConfigurationOverrider<R> withFinalizer(String finalizer) {
@@ -168,6 +170,12 @@ public ControllerConfigurationOverrider<R> withName(String name) {
168170
return this;
169171
}
170172

173+
public ControllerConfigurationOverrider<R> withFieldManager(
174+
String dependentFieldManager) {
175+
this.fieldManager = dependentFieldManager;
176+
return this;
177+
}
178+
171179
public ControllerConfigurationOverrider<R> replacingNamedDependentResourceConfig(String name,
172180
Object dependentResourceConfig) {
173181

@@ -190,7 +198,7 @@ public ControllerConfiguration<R> build() {
190198
generationAware, original.getAssociatedReconcilerClassName(), retry, rateLimiter,
191199
reconciliationMaxInterval, onAddFilter, onUpdateFilter, genericFilter,
192200
original.getDependentResources(),
193-
namespaces, finalizer, labelSelector, configurations, itemStore,
201+
namespaces, finalizer, labelSelector, configurations, itemStore, fieldManager,
194202
original.getConfigurationService());
195203
overridden.setEventFilter(customResourcePredicate);
196204
return overridden;

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/api/config/ResolvedControllerConfiguration.java

+14-3
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@ public class ResolvedControllerConfiguration<P extends HasMetadata>
3232
private final Map<DependentResourceSpec, Object> configurations;
3333
private final ItemStore<P> itemStore;
3434
private final ConfigurationService configurationService;
35+
private final String fieldManager;
3536

3637
private ResourceEventFilter<P> eventFilter;
3738
private List<DependentResourceSpec> dependentResources;
@@ -44,7 +45,8 @@ public ResolvedControllerConfiguration(Class<P> resourceClass, ControllerConfigu
4445
other.genericFilter().orElse(null),
4546
other.getDependentResources(), other.getNamespaces(),
4647
other.getFinalizerName(), other.getLabelSelector(), Collections.emptyMap(),
47-
other.getItemStore().orElse(null), other.getConfigurationService());
48+
other.getItemStore().orElse(null), other.fieldManager(),
49+
other.getConfigurationService());
4850
}
4951

5052
public static Duration getMaxReconciliationInterval(long interval, TimeUnit timeUnit) {
@@ -72,10 +74,12 @@ public ResolvedControllerConfiguration(Class<P> resourceClass, String name,
7274
List<DependentResourceSpec> dependentResources,
7375
Set<String> namespaces, String finalizer, String labelSelector,
7476
Map<DependentResourceSpec, Object> configurations, ItemStore<P> itemStore,
77+
String fieldManager,
7578
ConfigurationService configurationService) {
7679
this(resourceClass, name, generationAware, associatedReconcilerClassName, retry, rateLimiter,
7780
maxReconciliationInterval, onAddFilter, onUpdateFilter, genericFilter,
78-
namespaces, finalizer, labelSelector, configurations, itemStore, configurationService);
81+
namespaces, finalizer, labelSelector, configurations, itemStore, fieldManager,
82+
configurationService);
7983
setDependentResources(dependentResources);
8084
}
8185

@@ -86,6 +90,7 @@ protected ResolvedControllerConfiguration(Class<P> resourceClass, String name,
8690
GenericFilter<? super P> genericFilter,
8791
Set<String> namespaces, String finalizer, String labelSelector,
8892
Map<DependentResourceSpec, Object> configurations, ItemStore<P> itemStore,
93+
String fieldManager,
8994
ConfigurationService configurationService) {
9095
super(resourceClass, namespaces, labelSelector, onAddFilter, onUpdateFilter, genericFilter,
9196
itemStore);
@@ -100,13 +105,14 @@ protected ResolvedControllerConfiguration(Class<P> resourceClass, String name,
100105
this.itemStore = itemStore;
101106
this.finalizer =
102107
ControllerConfiguration.ensureValidFinalizerName(finalizer, getResourceTypeName());
108+
this.fieldManager = fieldManager;
103109
}
104110

105111
protected ResolvedControllerConfiguration(Class<P> resourceClass, String name,
106112
Class<? extends Reconciler> reconcilerClas, ConfigurationService configurationService) {
107113
this(resourceClass, name, false, getAssociatedReconcilerClassName(reconcilerClas), null, null,
108114
null, null, null, null, null,
109-
null, null, null, null, configurationService);
115+
null, null, null, null, null, configurationService);
110116
}
111117

112118
@Override
@@ -183,4 +189,9 @@ public Object getConfigurationFor(DependentResourceSpec spec) {
183189
public Optional<ItemStore<P>> getItemStore() {
184190
return Optional.ofNullable(itemStore);
185191
}
192+
193+
@Override
194+
public String fieldManager() {
195+
return fieldManager;
196+
}
186197
}

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/api/reconciler/ControllerConfiguration.java

+13-2
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,8 @@
1717
import io.javaoperatorsdk.operator.processing.retry.GenericRetry;
1818
import io.javaoperatorsdk.operator.processing.retry.Retry;
1919

20+
import static io.javaoperatorsdk.operator.api.config.ControllerConfiguration.CONTROLLER_NAME_AS_FIELD_MANAGER;
21+
2022
@Inherited
2123
@Retention(RetentionPolicy.RUNTIME)
2224
@Target({ElementType.TYPE})
@@ -47,7 +49,7 @@
4749
* Specified which namespaces this Controller monitors for custom resources events. If no
4850
* namespace is specified then the controller will monitor all namespaces by default.
4951
*
50-
* @return the list of namespaces this controller monitors
52+
* @return the array of namespaces this controller monitors
5153
*/
5254
String[] namespaces() default Constants.WATCH_ALL_NAMESPACES;
5355

@@ -108,7 +110,7 @@ MaxReconciliationInterval maxReconciliationInterval() default @MaxReconciliation
108110
* Optional list of {@link Dependent} configurations which associate a resource type to a
109111
* {@link io.javaoperatorsdk.operator.api.reconciler.dependent.DependentResource} implementation
110112
*
111-
* @return the list of {@link Dependent} configurations
113+
* @return the array of {@link Dependent} configurations
112114
*/
113115
Dependent[] dependents() default {};
114116

@@ -129,4 +131,13 @@ MaxReconciliationInterval maxReconciliationInterval() default @MaxReconciliation
129131
Class<? extends RateLimiter> rateLimiter() default LinearRateLimiter.class;
130132

131133
Class<? extends ItemStore> itemStore() default ItemStore.class;
134+
135+
/**
136+
* Retrieves the name used to assign as field manager for
137+
* <a href="https://kubernetes.io/docs/reference/using-api/server-side-apply/">Server-Side
138+
* Apply</a> (SSA) operations. If unset, the sanitized controller name will be used.
139+
*
140+
* @return the name used as field manager for SSA operations
141+
*/
142+
String fieldManager() default CONTROLLER_NAME_AS_FIELD_MANAGER;
132143
}

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

+2-2
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,8 @@ public class GenericKubernetesResourceMatcher<R extends HasMetadata, P extends H
1919

2020
private static final String ADD = "add";
2121
private static final String OP = "op";
22-
private static final String METADATA_LABELS = "/metadata/labels";
23-
private static final String METADATA_ANNOTATIONS = "/metadata/annotations";
22+
public static final String METADATA_LABELS = "/metadata/labels";
23+
public static final String METADATA_ANNOTATIONS = "/metadata/annotations";
2424

2525
private static final String PATH = "path";
2626
private static final String[] EMPTY_ARRAY = {};

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

+39-9
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@ public abstract class KubernetesDependentResource<R extends HasMetadata, P exten
4444
private final boolean garbageCollected = this instanceof GarbageCollected;
4545
private KubernetesDependentResourceConfig<R> kubernetesDependentResourceConfig;
4646

47+
4748
@SuppressWarnings("unchecked")
4849
public KubernetesDependentResource(Class<R> resourceType) {
4950
super(resourceType);
@@ -128,16 +129,41 @@ protected R handleUpdate(R actual, R desired, P primary, Context<P> context) {
128129

129130
@SuppressWarnings("unused")
130131
public R create(R target, P primary, Context<P> context) {
131-
return prepare(target, primary, "Creating").create();
132+
if (!context.getControllerConfiguration().getConfigurationService()
133+
.ssaBasedCreateUpdateForDependentResources()) {
134+
return prepare(target, primary, "Creating").create();
135+
} else {
136+
return prepare(target, primary, "Creating")
137+
.fieldManager(context.getControllerConfiguration().fieldManager())
138+
.forceConflicts()
139+
.serverSideApply();
140+
}
132141
}
133142

134143
public R update(R actual, R target, P primary, Context<P> context) {
135-
var updatedActual = processor.replaceSpecOnActual(actual, target, context);
136-
return prepare(updatedActual, primary, "Updating").replace();
144+
if (!context.getControllerConfiguration().getConfigurationService()
145+
.ssaBasedCreateUpdateForDependentResources()) {
146+
var updatedActual = processor.replaceSpecOnActual(actual, target, context);
147+
return prepare(updatedActual, primary, "Updating").replace();
148+
} else {
149+
target.getMetadata().setResourceVersion(actual.getMetadata().getResourceVersion());
150+
return prepare(target, primary, "Updating")
151+
.fieldManager(context.getControllerConfiguration().fieldManager())
152+
.forceConflicts().serverSideApply();
153+
}
137154
}
138155

139156
public Result<R> match(R actualResource, P primary, Context<P> context) {
140-
return GenericKubernetesResourceMatcher.match(this, actualResource, primary, context, false);
157+
if (!context.getControllerConfiguration().getConfigurationService()
158+
.ssaBasedDefaultMatchingForDependentResources()) {
159+
return GenericKubernetesResourceMatcher.match(this, actualResource, primary, context, false);
160+
} else {
161+
final var desired = desired(primary, context);
162+
addReferenceHandlingMetadata(desired, primary);
163+
var matches = SSABasedGenericKubernetesResourceMatcher.getInstance().matches(actualResource,
164+
desired, context);
165+
return Result.computed(matches, desired);
166+
}
141167
}
142168

143169
@SuppressWarnings("unused")
@@ -164,11 +190,7 @@ protected Resource<R> prepare(R desired, P primary, String actionName) {
164190
desired.getClass(),
165191
ResourceID.fromResource(desired));
166192

167-
if (addOwnerReference()) {
168-
desired.addOwnerReference(primary);
169-
} else if (useDefaultAnnotationsToIdentifyPrimary()) {
170-
addDefaultSecondaryToPrimaryMapperAnnotations(desired, primary);
171-
}
193+
addReferenceHandlingMetadata(desired, primary);
172194

173195
if (desired instanceof Namespaced) {
174196
return client.resource(desired).inNamespace(desired.getMetadata().getNamespace());
@@ -177,6 +199,14 @@ protected Resource<R> prepare(R desired, P primary, String actionName) {
177199
}
178200
}
179201

202+
protected void addReferenceHandlingMetadata(R desired, P primary) {
203+
if (addOwnerReference()) {
204+
desired.addOwnerReference(primary);
205+
} else if (useDefaultAnnotationsToIdentifyPrimary()) {
206+
addDefaultSecondaryToPrimaryMapperAnnotations(desired, primary);
207+
}
208+
}
209+
180210
@Override
181211
@SuppressWarnings("unchecked")
182212
protected InformerEventSource<R, P> createEventSource(EventSourceContext<P> context) {

0 commit comments

Comments
 (0)