Skip to content

Commit 55d63c5

Browse files
csvirimetacosm
authored andcommitted
improve: remove ErrorStatusHandler interface (#2438)
Signed-off-by: Attila Mészáros <[email protected]>
1 parent 0ae3ad2 commit 55d63c5

File tree

16 files changed

+112
-114
lines changed

16 files changed

+112
-114
lines changed

docsy/content/en/docs/migration/v5-0-migration.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -60,4 +60,5 @@ description: Migrating from v4.7 to v5.0
6060
This also means, that `BulkDependentResource` now does not automatically implement `Creator` and `Deleter` as before.
6161
Make sure to implement those interfaces in your bulk dependent resources. You can use also the new helper interface, the
6262
[`CRUDBulkDependentResource`](https://github.com/operator-framework/java-operator-sdk/blob/main/operator-framework-core/src/main/java/io/javaoperatorsdk/operator/processing/dependent/CRUDBulkDependentResource.java)
63-
what also implement `BulkUpdater` interface.
63+
what also implement `BulkUpdater` interface.
64+
12. `ErrorStatusHandler` is deleted. Just delete the interface from your impl.

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

-28
This file was deleted.

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

+21
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ public class ErrorStatusUpdateControl<P extends HasMetadata>
1010

1111
private final P resource;
1212
private boolean noRetry = false;
13+
private final boolean defaultErrorProcessing;
1314

1415
public static <T extends HasMetadata> ErrorStatusUpdateControl<T> patchStatus(T resource) {
1516
return new ErrorStatusUpdateControl<>(resource);
@@ -19,8 +20,21 @@ public static <T extends HasMetadata> ErrorStatusUpdateControl<T> noStatusUpdate
1920
return new ErrorStatusUpdateControl<>(null);
2021
}
2122

23+
/**
24+
* No special processing of the error, the error will be thrown and default error handling will
25+
* apply
26+
*/
27+
public static <T extends HasMetadata> ErrorStatusUpdateControl<T> defaultErrorProcessing() {
28+
return new ErrorStatusUpdateControl<>(null, true);
29+
}
30+
2231
private ErrorStatusUpdateControl(P resource) {
32+
this(resource, false);
33+
}
34+
35+
private ErrorStatusUpdateControl(P resource, boolean defaultErrorProcessing) {
2336
this.resource = resource;
37+
this.defaultErrorProcessing = defaultErrorProcessing;
2438
}
2539

2640
/**
@@ -29,6 +43,9 @@ private ErrorStatusUpdateControl(P resource) {
2943
* @return ErrorStatusUpdateControl
3044
*/
3145
public ErrorStatusUpdateControl<P> withNoRetry() {
46+
if (defaultErrorProcessing) {
47+
throw new IllegalStateException("Cannot set no-retry for default error processing");
48+
}
3249
this.noRetry = true;
3350
return this;
3451
}
@@ -41,6 +58,10 @@ public boolean isNoRetry() {
4158
return noRetry;
4259
}
4360

61+
public boolean isDefaultErrorProcessing() {
62+
return defaultErrorProcessing;
63+
}
64+
4465
/**
4566
* If re-scheduled using this method, it is not considered as retry, it effectively cancels retry.
4667
*

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

+24
Original file line numberDiff line numberDiff line change
@@ -31,4 +31,28 @@ default List<EventSource> prepareEventSources(EventSourceContext<P> context) {
3131
return Collections.emptyList();
3232
}
3333

34+
/**
35+
* <p>
36+
* Reconciler can override this method in order to update the status sub-resource in the case an
37+
* exception in thrown. In that case {@link #updateErrorStatus(HasMetadata, Context, Exception)}
38+
* is called automatically.
39+
* <p>
40+
* The result of the method call is used to make a status update on the custom resource. This is
41+
* always a sub-resource update request, so no update on custom resource itself (like spec of
42+
* metadata) happens. Note that this update request will also produce an event, and will result in
43+
* a reconciliation if the controller is not generation aware.
44+
* <p>
45+
* Note that the scope of this feature is only the reconcile method of the reconciler, since there
46+
* should not be updates on custom resource after it is marked for deletion.
47+
*
48+
* @param resource to update the status on
49+
* @param context the current context
50+
* @param e exception thrown from the reconciler
51+
* @return the updated resource
52+
*/
53+
default ErrorStatusUpdateControl<P> updateErrorStatus(P resource, Context<P> context,
54+
Exception e) {
55+
return ErrorStatusUpdateControl.defaultErrorProcessing();
56+
}
57+
3458
}

operator-framework-core/src/main/java/io/javaoperatorsdk/operator/processing/event/ReconciliationDispatcher.java

+36-46
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,6 @@
2222
import io.javaoperatorsdk.operator.api.reconciler.Context;
2323
import io.javaoperatorsdk.operator.api.reconciler.DefaultContext;
2424
import io.javaoperatorsdk.operator.api.reconciler.DeleteControl;
25-
import io.javaoperatorsdk.operator.api.reconciler.ErrorStatusHandler;
2625
import io.javaoperatorsdk.operator.api.reconciler.RetryInfo;
2726
import io.javaoperatorsdk.operator.api.reconciler.UpdateControl;
2827
import io.javaoperatorsdk.operator.processing.Controller;
@@ -174,55 +173,46 @@ private PostExecutionControl<P> reconcileExecution(ExecutionScope<P> executionSc
174173
private PostExecutionControl<P> handleErrorStatusHandler(P resource, P originalResource,
175174
Context<P> context,
176175
Exception e) throws Exception {
177-
if (isErrorStatusHandlerPresent()) {
178-
try {
179-
RetryInfo retryInfo = context.getRetryInfo().orElseGet(() -> new RetryInfo() {
180-
@Override
181-
public int getAttemptCount() {
182-
return 0;
183-
}
184176

185-
@Override
186-
public boolean isLastAttempt() {
187-
// check also if the retry is limited to 0
188-
return retryConfigurationHasZeroAttempts ||
189-
controller.getConfiguration().getRetry() == null;
190-
}
191-
});
192-
((DefaultContext<P>) context).setRetryInfo(retryInfo);
193-
var errorStatusUpdateControl = ((ErrorStatusHandler<P>) controller.getReconciler())
194-
.updateErrorStatus(resource, context, e);
195-
196-
P updatedResource = null;
197-
if (errorStatusUpdateControl.getResource().isPresent()) {
198-
updatedResource = customResourceFacade
199-
.patchStatus(errorStatusUpdateControl.getResource().orElseThrow(), originalResource);
200-
}
201-
if (errorStatusUpdateControl.isNoRetry()) {
202-
PostExecutionControl<P> postExecutionControl;
203-
if (updatedResource != null) {
204-
postExecutionControl =
205-
PostExecutionControl.customResourceStatusPatched(updatedResource);
206-
} else {
207-
postExecutionControl = PostExecutionControl.defaultDispatch();
208-
}
209-
errorStatusUpdateControl.getScheduleDelay()
210-
.ifPresent(postExecutionControl::withReSchedule);
211-
return postExecutionControl;
212-
}
213-
} catch (RuntimeException ex) {
214-
log.error("Error during error status handling.", ex);
177+
RetryInfo retryInfo = context.getRetryInfo().orElseGet(() -> new RetryInfo() {
178+
@Override
179+
public int getAttemptCount() {
180+
return 0;
215181
}
216-
}
217-
throw e;
218-
}
219182

220-
private boolean isErrorStatusHandlerPresent() {
221-
return controller.getReconciler() instanceof ErrorStatusHandler;
222-
}
183+
@Override
184+
public boolean isLastAttempt() {
185+
// check also if the retry is limited to 0
186+
return retryConfigurationHasZeroAttempts ||
187+
controller.getConfiguration().getRetry() == null;
188+
}
189+
});
190+
((DefaultContext<P>) context).setRetryInfo(retryInfo);
191+
var errorStatusUpdateControl = controller.getReconciler()
192+
.updateErrorStatus(resource, context, e);
223193

224-
private P patchStatusGenerationAware(P resource, P originalResource) {
225-
return customResourceFacade.patchStatus(resource, originalResource);
194+
if (errorStatusUpdateControl.isDefaultErrorProcessing()) {
195+
throw e;
196+
}
197+
198+
P updatedResource = null;
199+
if (errorStatusUpdateControl.getResource().isPresent()) {
200+
updatedResource = customResourceFacade
201+
.patchStatus(errorStatusUpdateControl.getResource().orElseThrow(), originalResource);
202+
}
203+
if (errorStatusUpdateControl.isNoRetry()) {
204+
PostExecutionControl<P> postExecutionControl;
205+
if (updatedResource != null) {
206+
postExecutionControl =
207+
PostExecutionControl.customResourceStatusPatched(updatedResource);
208+
} else {
209+
postExecutionControl = PostExecutionControl.defaultDispatch();
210+
}
211+
errorStatusUpdateControl.getScheduleDelay()
212+
.ifPresent(postExecutionControl::withReSchedule);
213+
return postExecutionControl;
214+
}
215+
throw e;
226216
}
227217

228218
private PostExecutionControl<P> createPostExecutionControl(P updatedCustomResource,

operator-framework-core/src/test/java/io/javaoperatorsdk/operator/processing/event/ReconciliationDispatcherTest.java

+19-23
Original file line numberDiff line numberDiff line change
@@ -5,11 +5,11 @@
55
import java.util.Optional;
66
import java.util.concurrent.TimeUnit;
77
import java.util.function.BiFunction;
8+
import java.util.function.Supplier;
89

910
import org.junit.jupiter.api.BeforeEach;
1011
import org.junit.jupiter.api.Test;
1112
import org.mockito.ArgumentCaptor;
12-
import org.mockito.ArgumentMatcher;
1313
import org.mockito.ArgumentMatchers;
1414
import org.mockito.stubbing.Answer;
1515

@@ -28,7 +28,6 @@
2828
import io.javaoperatorsdk.operator.api.reconciler.Cleaner;
2929
import io.javaoperatorsdk.operator.api.reconciler.Context;
3030
import io.javaoperatorsdk.operator.api.reconciler.DeleteControl;
31-
import io.javaoperatorsdk.operator.api.reconciler.ErrorStatusHandler;
3231
import io.javaoperatorsdk.operator.api.reconciler.ErrorStatusUpdateControl;
3332
import io.javaoperatorsdk.operator.api.reconciler.Reconciler;
3433
import io.javaoperatorsdk.operator.api.reconciler.RetryInfo;
@@ -479,7 +478,7 @@ void callErrorStatusHandlerIfImplemented() {
479478
reconciler.reconcile = (r, c) -> {
480479
throw new IllegalStateException("Error Status Test");
481480
};
482-
reconciler.errorHandler = (r, ri, e) -> {
481+
reconciler.errorHandler = () -> {
483482
testCustomResource.getStatus().setConfigMapStatus(ERROR_MESSAGE);
484483
return ErrorStatusUpdateControl.patchStatus(testCustomResource);
485484
};
@@ -499,7 +498,7 @@ public boolean isLastAttempt() {
499498
}).setResource(testCustomResource));
500499

501500
verify(customResourceFacade, times(1)).patchStatus(eq(testCustomResource), any());
502-
verify(((ErrorStatusHandler) reconciler), times(1)).updateErrorStatus(eq(testCustomResource),
501+
verify(reconciler, times(1)).updateErrorStatus(eq(testCustomResource),
503502
any(), any());
504503
}
505504

@@ -510,15 +509,15 @@ void callErrorStatusHandlerEvenOnFirstError() {
510509
reconciler.reconcile = (r, c) -> {
511510
throw new IllegalStateException("Error Status Test");
512511
};
513-
reconciler.errorHandler = (r, ri, e) -> {
512+
reconciler.errorHandler = () -> {
514513
testCustomResource.getStatus().setConfigMapStatus(ERROR_MESSAGE);
515514
return ErrorStatusUpdateControl.patchStatus(testCustomResource);
516515
};
517516

518517
var postExecControl = reconciliationDispatcher.handleExecution(
519518
new ExecutionScope(null).setResource(testCustomResource));
520519
verify(customResourceFacade, times(1)).patchStatus(eq(testCustomResource), any());
521-
verify(((ErrorStatusHandler) reconciler), times(1)).updateErrorStatus(eq(testCustomResource),
520+
verify(reconciler, times(1)).updateErrorStatus(eq(testCustomResource),
522521
any(), any());
523522
assertThat(postExecControl.exceptionDuringExecution()).isTrue();
524523
}
@@ -529,15 +528,15 @@ void errorHandlerCanInstructNoRetryWithUpdate() {
529528
reconciler.reconcile = (r, c) -> {
530529
throw new IllegalStateException("Error Status Test");
531530
};
532-
reconciler.errorHandler = (r, ri, e) -> {
531+
reconciler.errorHandler = () -> {
533532
testCustomResource.getStatus().setConfigMapStatus(ERROR_MESSAGE);
534533
return ErrorStatusUpdateControl.patchStatus(testCustomResource).withNoRetry();
535534
};
536535

537536
var postExecControl = reconciliationDispatcher.handleExecution(
538537
new ExecutionScope(null).setResource(testCustomResource));
539538

540-
verify(((ErrorStatusHandler) reconciler), times(1)).updateErrorStatus(eq(testCustomResource),
539+
verify(reconciler, times(1)).updateErrorStatus(eq(testCustomResource),
541540
any(), any());
542541
verify(customResourceFacade, times(1)).patchStatus(eq(testCustomResource), any());
543542
assertThat(postExecControl.exceptionDuringExecution()).isFalse();
@@ -549,15 +548,15 @@ void errorHandlerCanInstructNoRetryNoUpdate() {
549548
reconciler.reconcile = (r, c) -> {
550549
throw new IllegalStateException("Error Status Test");
551550
};
552-
reconciler.errorHandler = (r, ri, e) -> {
551+
reconciler.errorHandler = () -> {
553552
testCustomResource.getStatus().setConfigMapStatus(ERROR_MESSAGE);
554553
return ErrorStatusUpdateControl.<TestCustomResource>noStatusUpdate().withNoRetry();
555554
};
556555

557556
var postExecControl = reconciliationDispatcher.handleExecution(
558557
new ExecutionScope(null).setResource(testCustomResource));
559558

560-
verify(((ErrorStatusHandler) reconciler), times(1)).updateErrorStatus(eq(testCustomResource),
559+
verify(reconciler, times(1)).updateErrorStatus(eq(testCustomResource),
561560
any(), any());
562561
verify(customResourceFacade, times(0)).patchStatus(eq(testCustomResource), any());
563562
assertThat(postExecControl.exceptionDuringExecution()).isFalse();
@@ -570,13 +569,13 @@ void errorStatusHandlerCanPatchResource() {
570569
throw new IllegalStateException("Error Status Test");
571570
};
572571
reconciler.errorHandler =
573-
(r, ri, e) -> ErrorStatusUpdateControl.patchStatus(testCustomResource);
572+
() -> ErrorStatusUpdateControl.patchStatus(testCustomResource);
574573

575574
reconciliationDispatcher.handleExecution(
576575
new ExecutionScope(null).setResource(testCustomResource));
577576

578577
verify(customResourceFacade, times(1)).patchStatus(eq(testCustomResource), any());
579-
verify(((ErrorStatusHandler) reconciler), times(1)).updateErrorStatus(eq(testCustomResource),
578+
verify(reconciler, times(1)).updateErrorStatus(eq(testCustomResource),
580579
any(), any());
581580
}
582581

@@ -592,16 +591,14 @@ void ifRetryLimitedToZeroMaxAttemptsErrorHandlerGetsCorrectLastAttempt() {
592591
reconciler.reconcile = (r, c) -> {
593592
throw new IllegalStateException("Error Status Test");
594593
};
595-
var mockErrorHandler = mock(ErrorStatusHandler.class);
596-
when(mockErrorHandler.updateErrorStatus(any(), any(), any()))
597-
.thenReturn(ErrorStatusUpdateControl.noStatusUpdate());
598-
reconciler.errorHandler = mockErrorHandler;
594+
595+
reconciler.errorHandler = () -> ErrorStatusUpdateControl.noStatusUpdate();
599596

600597
reconciliationDispatcher.handleExecution(
601598
new ExecutionScope(null).setResource(testCustomResource));
602599

603-
verify(mockErrorHandler, times(1)).updateErrorStatus(any(),
604-
ArgumentMatchers.argThat((ArgumentMatcher<Context<TestCustomResource>>) context -> {
600+
verify(reconciler, times(1)).updateErrorStatus(any(),
601+
ArgumentMatchers.argThat(context -> {
605602
var retryInfo = context.getRetryInfo().orElseThrow();
606603
return retryInfo.isLastAttempt();
607604
}), any());
@@ -651,7 +648,7 @@ void reSchedulesFromErrorHandler() {
651648
throw new IllegalStateException("Error Status Test");
652649
};
653650
reconciler.errorHandler =
654-
(r, ri, e) -> ErrorStatusUpdateControl.<TestCustomResource>noStatusUpdate()
651+
() -> ErrorStatusUpdateControl.<TestCustomResource>noStatusUpdate()
655652
.rescheduleAfter(delay);
656653

657654
var res = reconciliationDispatcher.handleExecution(
@@ -691,12 +688,11 @@ public <T extends HasMetadata> ExecutionScope<T> executionScopeWithCREvent(T res
691688
}
692689

693690
private class TestReconciler
694-
implements Reconciler<TestCustomResource>, Cleaner<TestCustomResource>,
695-
ErrorStatusHandler<TestCustomResource> {
691+
implements Reconciler<TestCustomResource>, Cleaner<TestCustomResource> {
696692

697693
private BiFunction<TestCustomResource, Context, UpdateControl<TestCustomResource>> reconcile;
698694
private BiFunction<TestCustomResource, Context, DeleteControl> cleanup;
699-
private ErrorStatusHandler<TestCustomResource> errorHandler;
695+
private Supplier<ErrorStatusUpdateControl> errorHandler;
700696

701697
@Override
702698
public UpdateControl<TestCustomResource> reconcile(TestCustomResource resource,
@@ -719,7 +715,7 @@ public DeleteControl cleanup(TestCustomResource resource, Context context) {
719715
public ErrorStatusUpdateControl<TestCustomResource> updateErrorStatus(
720716
TestCustomResource resource,
721717
Context<TestCustomResource> context, Exception e) {
722-
return errorHandler != null ? errorHandler.updateErrorStatus(resource, context, e)
718+
return errorHandler != null ? errorHandler.get()
723719
: ErrorStatusUpdateControl.noStatusUpdate();
724720
}
725721
}

operator-framework/src/test/java/io/javaoperatorsdk/operator/sample/dependentresourcecrossref/DependentResourceCrossRefReconciler.java

+1-2
Original file line numberDiff line numberDiff line change
@@ -21,8 +21,7 @@
2121
dependsOn = SECRET_NAME)})
2222
@ControllerConfiguration
2323
public class DependentResourceCrossRefReconciler
24-
implements Reconciler<DependentResourceCrossRefResource>,
25-
ErrorStatusHandler<DependentResourceCrossRefResource> {
24+
implements Reconciler<DependentResourceCrossRefResource> {
2625

2726
public static final String SECRET_NAME = "secret";
2827
private final AtomicInteger numberOfExecutions = new AtomicInteger(0);

0 commit comments

Comments
 (0)