Skip to content

Commit 163b40c

Browse files
csvirimetacosm
andcommitted
improve: dependent configuration improvements - context independent (#2389)
Signed-off-by: Attila Mészáros <[email protected]> Signed-off-by: Chris Laprun <[email protected]> Co-authored-by: Chris Laprun <[email protected]>
1 parent 6f4a0c9 commit 163b40c

File tree

102 files changed

+1067
-902
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

102 files changed

+1067
-902
lines changed

caffeine-bounded-cache-support/src/test/java/io/javaoperatorsdk/operator/processing/event/source/cache/sample/AbstractTestReconciler.java

+4-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ public List<EventSource> prepareEventSources(
7777
boundedItemStore(new KubernetesClientBuilder().build(),
7878
ConfigMap.class, Duration.ofMinutes(1), 1); // setting max size for testing purposes
7979

80-
var es = new InformerEventSource<>(InformerConfiguration.from(ConfigMap.class, context)
80+
var es = new InformerEventSource<>(InformerConfiguration.from(ConfigMap.class, primaryClass())
8181
.withItemStore(boundedItemStore)
8282
.withSecondaryToPrimaryMapper(
8383
Mappers.fromOwnerReferences(context.getPrimaryResourceClass(),
@@ -104,4 +104,7 @@ public static <R extends HasMetadata> BoundedItemStore<R> boundedItemStore(
104104
.build();
105105
return CaffeineBoundedItemStores.boundedItemStore(client, rClass, cache);
106106
}
107+
108+
protected abstract Class<P> primaryClass();
109+
107110
}

caffeine-bounded-cache-support/src/test/java/io/javaoperatorsdk/operator/processing/event/source/cache/sample/clusterscope/BoundedCacheClusterScopeTestReconciler.java

+4
Original file line numberDiff line numberDiff line change
@@ -7,4 +7,8 @@
77
public class BoundedCacheClusterScopeTestReconciler extends
88
AbstractTestReconciler<BoundedCacheClusterScopeTestCustomResource> {
99

10+
@Override
11+
protected Class<BoundedCacheClusterScopeTestCustomResource> primaryClass() {
12+
return BoundedCacheClusterScopeTestCustomResource.class;
13+
}
1014
}

caffeine-bounded-cache-support/src/test/java/io/javaoperatorsdk/operator/processing/event/source/cache/sample/namespacescope/BoundedCacheTestReconciler.java

+4
Original file line numberDiff line numberDiff line change
@@ -7,4 +7,8 @@
77
public class BoundedCacheTestReconciler
88
extends AbstractTestReconciler<BoundedCacheTestCustomResource> {
99

10+
@Override
11+
protected Class<BoundedCacheTestCustomResource> primaryClass() {
12+
return BoundedCacheTestCustomResource.class;
13+
}
1014
}

docs/documentation/v5-0-migration.md

-65
This file was deleted.

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

+9-2
Original file line numberDiff line numberDiff line change
@@ -17,6 +17,7 @@
1717
import io.javaoperatorsdk.operator.OperatorException;
1818
import io.javaoperatorsdk.operator.ReconcilerUtils;
1919
import io.javaoperatorsdk.operator.api.config.Utils.Configurator;
20+
import io.javaoperatorsdk.operator.api.config.dependent.DependentResourceConfigurationResolver;
2021
import io.javaoperatorsdk.operator.api.config.dependent.DependentResourceSpec;
2122
import io.javaoperatorsdk.operator.api.config.workflow.WorkflowSpec;
2223
import io.javaoperatorsdk.operator.api.reconciler.Constants;
@@ -194,7 +195,7 @@ public boolean handleExceptionsInReconciler() {
194195
@SuppressWarnings({"unchecked", "rawtypes"})
195196
private static List<DependentResourceSpec> dependentResources(
196197
Workflow annotation,
197-
ControllerConfiguration<?> parent) {
198+
ControllerConfiguration<?> controllerConfiguration) {
198199
final var dependents = annotation.dependents();
199200

200201

@@ -213,7 +214,7 @@ private static List<DependentResourceSpec> dependentResources(
213214
"A DependentResource named '" + dependentName + "' already exists: " + spec);
214215
}
215216

216-
final var name = parent.getName();
217+
final var name = controllerConfiguration.getName();
217218

218219
var eventSourceName = dependent.useEventSourceWithName();
219220
eventSourceName = Constants.NO_VALUE_SET.equals(eventSourceName) ? null : eventSourceName;
@@ -225,6 +226,12 @@ private static List<DependentResourceSpec> dependentResources(
225226
Utils.instantiate(dependent.deletePostcondition(), Condition.class, context),
226227
Utils.instantiate(dependent.activationCondition(), Condition.class, context),
227228
eventSourceName);
229+
230+
// extract potential configuration
231+
DependentResourceConfigurationResolver.configureSpecFromConfigured(spec,
232+
controllerConfiguration,
233+
dependentType);
234+
228235
specsMap.put(dependentName, spec);
229236
}
230237
return specsMap.values().stream().toList();

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

+3-2
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@
66

77
import io.fabric8.kubernetes.api.model.HasMetadata;
88
import io.javaoperatorsdk.operator.ReconcilerUtils;
9+
import io.javaoperatorsdk.operator.api.config.dependent.DependentResourceSpec;
910
import io.javaoperatorsdk.operator.api.config.workflow.WorkflowSpec;
1011
import io.javaoperatorsdk.operator.api.reconciler.MaxReconciliationInterval;
1112
import io.javaoperatorsdk.operator.processing.event.rate.LinearRateLimiter;
@@ -71,7 +72,6 @@ default Optional<Duration> maxReconciliationInterval() {
7172
return Optional.of(Duration.ofHours(MaxReconciliationInterval.DEFAULT_INTERVAL));
7273
}
7374

74-
@SuppressWarnings("unused")
7575
ConfigurationService getConfigurationService();
7676

7777
@SuppressWarnings("unchecked")
@@ -86,7 +86,7 @@ default Class<P> getResourceClass() {
8686

8787
@SuppressWarnings("unused")
8888
default Set<String> getEffectiveNamespaces() {
89-
return ResourceConfiguration.super.getEffectiveNamespaces(getConfigurationService());
89+
return ResourceConfiguration.super.getEffectiveNamespaces(this);
9090
}
9191

9292
/**
@@ -100,4 +100,5 @@ default String fieldManager() {
100100
return getName();
101101
}
102102

103+
<C> C getConfigurationFor(DependentResourceSpec<?, P, C> spec);
103104
}

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

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@
1919
import static io.javaoperatorsdk.operator.api.reconciler.Constants.DEFAULT_NAMESPACES_SET;
2020
import static io.javaoperatorsdk.operator.api.reconciler.Constants.WATCH_CURRENT_NAMESPACE_SET;
2121

22-
@SuppressWarnings({"rawtypes", "unused"})
22+
@SuppressWarnings({"rawtypes", "unused", "UnusedReturnValue"})
2323
public class ControllerConfigurationOverrider<R extends HasMetadata> {
2424

2525
private String finalizer;

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

+10-5
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,6 @@
99

1010
import io.fabric8.kubernetes.api.model.HasMetadata;
1111
import io.fabric8.kubernetes.client.informers.cache.ItemStore;
12-
import io.javaoperatorsdk.operator.api.config.dependent.DependentResourceConfigurationProvider;
1312
import io.javaoperatorsdk.operator.api.config.dependent.DependentResourceSpec;
1413
import io.javaoperatorsdk.operator.api.config.workflow.WorkflowSpec;
1514
import io.javaoperatorsdk.operator.api.reconciler.Reconciler;
@@ -22,8 +21,7 @@
2221
@SuppressWarnings("rawtypes")
2322
public class ResolvedControllerConfiguration<P extends HasMetadata>
2423
extends DefaultResourceConfiguration<P>
25-
implements io.javaoperatorsdk.operator.api.config.ControllerConfiguration<P>,
26-
DependentResourceConfigurationProvider {
24+
implements io.javaoperatorsdk.operator.api.config.ControllerConfiguration<P> {
2725

2826
private final String name;
2927
private final boolean generationAware;
@@ -168,8 +166,15 @@ public ConfigurationService getConfigurationService() {
168166
}
169167

170168
@Override
171-
public Object getConfigurationFor(DependentResourceSpec spec) {
172-
return configurations.get(spec);
169+
@SuppressWarnings("unchecked")
170+
public <C> C getConfigurationFor(DependentResourceSpec<?, P, C> spec) {
171+
// first check if there's an overridden configuration at the controller level
172+
var config = configurations.get(spec);
173+
if (config == null) {
174+
// if not, check the spec for configuration
175+
config = spec.getConfiguration().orElse(null);
176+
}
177+
return (C) config;
173178
}
174179

175180
@Override

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

+3-2
Original file line numberDiff line numberDiff line change
@@ -111,11 +111,12 @@ static Set<String> ensureValidNamespaces(Collection<String> namespaces) {
111111
*
112112
* @return a Set of namespace names the associated controller will watch
113113
*/
114-
default Set<String> getEffectiveNamespaces(ConfigurationService configurationService) {
114+
default Set<String> getEffectiveNamespaces(ControllerConfiguration<?> controllerConfiguration) {
115115
var targetNamespaces = getNamespaces();
116116
if (watchCurrentNamespace()) {
117117
final String namespace =
118-
configurationService.getKubernetesClient().getConfiguration().getNamespace();
118+
controllerConfiguration.getConfigurationService().getKubernetesClient().getConfiguration()
119+
.getNamespace();
119120
if (namespace == null) {
120121
throw new OperatorException(
121122
"Couldn't retrieve the currently connected namespace. Make sure it's correctly set in your ~/.kube/config file, using, e.g. 'kubectl config set-context <your context> --namespace=<your namespace>'");

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

+3-4
Original file line numberDiff line numberDiff line change
@@ -3,10 +3,9 @@
33
import java.lang.annotation.Annotation;
44

55
import io.javaoperatorsdk.operator.api.config.ControllerConfiguration;
6-
import io.javaoperatorsdk.operator.api.reconciler.dependent.managed.DependentResourceConfigurator;
76

8-
public interface ConfigurationConverter<A extends Annotation, C, D extends DependentResourceConfigurator<? extends C>> {
7+
public interface ConfigurationConverter<A extends Annotation, C> {
98

10-
C configFrom(A configAnnotation, ControllerConfiguration<?> parentConfiguration,
11-
Class<D> originatingClass);
9+
C configFrom(A configAnnotation, DependentResourceSpec<?, ?, C> spec,
10+
ControllerConfiguration<?> parentConfiguration);
1211
}

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

+7-28
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,6 @@
88
import io.javaoperatorsdk.operator.api.config.ControllerConfiguration;
99
import io.javaoperatorsdk.operator.api.config.Utils;
1010
import io.javaoperatorsdk.operator.api.reconciler.dependent.DependentResource;
11-
import io.javaoperatorsdk.operator.api.reconciler.dependent.managed.DependentResourceConfigurator;
1211

1312
@SuppressWarnings({"rawtypes", "unchecked"})
1413
public class DependentResourceConfigurationResolver {
@@ -20,39 +19,18 @@ private DependentResourceConfigurationResolver() {}
2019
private static final Map<Class<? extends ConfigurationConverter>, ConfigurationConverter> knownConverters =
2120
new HashMap<>();
2221

23-
public static <C extends ControllerConfiguration<? extends HasMetadata>> void configure(
24-
DependentResource dependentResource, DependentResourceSpec spec, C parentConfiguration) {
25-
if (dependentResource instanceof DependentResourceConfigurator configurator) {
26-
final var config = configurationFor(spec, parentConfiguration);
27-
configurator.configureWith(config);
28-
}
29-
}
30-
31-
public static <C extends ControllerConfiguration<? extends HasMetadata>> Object configurationFor(
32-
DependentResourceSpec spec, C parentConfiguration) {
3322

34-
// first check if the parent configuration has potentially already resolved the configuration
35-
if (parentConfiguration instanceof DependentResourceConfigurationProvider provider) {
36-
final var configuration = provider.getConfigurationFor(spec);
37-
if (configuration != null) {
38-
return configuration;
39-
}
40-
}
41-
42-
// find Configured-annotated class if it exists
43-
return extractConfigurationFromConfigured(spec.getDependentResourceClass(),
44-
parentConfiguration);
45-
}
46-
47-
public static <C extends ControllerConfiguration<? extends HasMetadata>> Object extractConfigurationFromConfigured(
48-
Class<? extends DependentResource> dependentResourceClass, C parentConfiguration) {
23+
public static <C extends ControllerConfiguration<?>> void configureSpecFromConfigured(
24+
DependentResourceSpec spec,
25+
C parentConfiguration,
26+
Class<? extends DependentResource> dependentResourceClass) {
4927
var converterAnnotationPair = converters.get(dependentResourceClass);
5028

5129
Annotation configAnnotation;
5230
if (converterAnnotationPair == null) {
5331
var configuredClassPair = getConfigured(dependentResourceClass);
5432
if (configuredClassPair == null) {
55-
return null;
33+
return;
5634
}
5735

5836
// check if we already have a converter registered for the found Configured annotated class
@@ -76,7 +54,8 @@ public static <C extends ControllerConfiguration<? extends HasMetadata>> Object
7654

7755
// always called even if the annotation is null so that implementations can provide default
7856
// values
79-
return converter.configFrom(configAnnotation, parentConfiguration, dependentResourceClass);
57+
final var config = converter.configFrom(configAnnotation, spec, parentConfiguration);
58+
spec.setNullableConfiguration(config);
8059
}
8160

8261
private static ConfiguredClassPair getConfigured(

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

+12-9
Original file line numberDiff line numberDiff line change
@@ -8,23 +8,17 @@
88
import io.javaoperatorsdk.operator.api.reconciler.dependent.DependentResource;
99
import io.javaoperatorsdk.operator.processing.dependent.workflow.Condition;
1010

11-
public class DependentResourceSpec<R, P extends HasMetadata> {
11+
public class DependentResourceSpec<R, P extends HasMetadata, C> {
1212

1313
private final Class<? extends DependentResource<R, P>> dependentResourceClass;
14-
1514
private final String name;
16-
1715
private final Set<String> dependsOn;
18-
1916
private final Condition<?, ?> readyCondition;
20-
2117
private final Condition<?, ?> reconcileCondition;
22-
2318
private final Condition<?, ?> deletePostCondition;
24-
2519
private final Condition<?, ?> activationCondition;
26-
2720
private final String useEventSourceWithName;
21+
private C nullableConfiguration;
2822

2923
public DependentResourceSpec(Class<? extends DependentResource<R, P>> dependentResourceClass,
3024
String name, Set<String> dependsOn, Condition<?, ?> readyCondition,
@@ -62,7 +56,7 @@ public boolean equals(Object o) {
6256
if (o == null || getClass() != o.getClass()) {
6357
return false;
6458
}
65-
DependentResourceSpec<?, ?> that = (DependentResourceSpec<?, ?>) o;
59+
DependentResourceSpec<?, ?, ?> that = (DependentResourceSpec<?, ?, ?>) o;
6660
return name.equals(that.name);
6761
}
6862

@@ -98,4 +92,13 @@ public Condition getActivationCondition() {
9892
public Optional<String> getUseEventSourceWithName() {
9993
return Optional.ofNullable(useEventSourceWithName);
10094
}
95+
96+
public Optional<C> getConfiguration() {
97+
return Optional.ofNullable(nullableConfiguration);
98+
}
99+
100+
protected void setNullableConfiguration(C configuration) {
101+
this.nullableConfiguration = configuration;
102+
}
103+
101104
}

0 commit comments

Comments
 (0)