OOTB hybris allows to automatically retry business process action after some time. To do this is enough to set delay in RetryLaterException and throw it. But there is no OOTB implementation of limiting amount of such retries in business processes.
Seems like to implement such behavior is enough to:
Extend BusinessProcess with custom integer attribute, which will hold current retry iteration:
Unfortunately it would not work and after debugging you could see that retry attribute is always 0 and is not persisted to db. This is OOTB behaviour of process execution in DefaultBusinessProcessService. All actions are executed within transactions. If action execution throws error, transaction would be reverted and no changes would be persisted to DB. So retry attribute for BusinessProcessModel will never change its value.
According to DefaultTaskExecutionStrategy#run coulbe used isRollBack value of RetryLaterException to skip exception throwing, so Transaction#finishExecute will not rollback changes, what will allow to save amount of retries on BusinessProcessModel.
publicclassExampleActionextendsAbstractSimpleDecisionAction<OrderProcessModel>{@ResourceprivateModelServicemodelService;@OverridepublicTransitionexecuteAction(finalOrderProcessModelprocess){if(process.getRetry()<=MAX_RETRIES){process.setRetry(process.getRetry()+1);modelService.save(process);RetryLaterExceptionex=newRetryLaterException("Retry ");ex.setDelay(1000);ex.setRollBack(false);// Will skip transaction rollback on exceptionthrowex;}else{thrownewIllegalStateException("Fail process.");}}}
Keep in mind that you retry method for RetryLaterException can be set up. By default, is used RetryLaterException.Method.EXPONENTIAL, which will use 2 to the power of retries with some random deviation. It means that code above would be executed in around 1 sec, 2 sec, 4 sec, 8 sec etc.
To manually control time of retries could be used RetryLaterException.Method.LINEAR:
1
2
3
4
5
6
7
8
9
// Will retry in 10 min, in 20 min, in 30 min etc ..intdelayInMinutes=businessProcessModel.getRetry()*10;RetryLaterExceptionex=newRetryLaterException();ex.setDelay(delayInMinutes*60*1000);ex.setMethod(RetryLaterException.Method.LINEAR);
UPD 2020-09-03
One more way to bypass not saving BusinessProcessModel on exception throw. For that we should close current transaction, save changes in BusinessProcessModel separate transaction, and create one more transaction, which would be roll backed by DefaultBusinessProcessService.
publicclassExampleActionextendsAbstractSimpleDecisionAction<OrderProcessModel>{@ResourceprivateModelServicemodelService;@OverridepublicTransitionexecuteAction(finalOrderProcessModelprocess){modelService.refresh(process);if(process.getRetry()<=MAX_RETRIES){// On exception throw in action hybris transaction manager will rollback everything// But we need to store amount of retries, that`s why transaction is rollbacked here// and new transaction is started and commited// Rollback action transactionTransactioncurrentTransaction=Transaction.current();if(currentTransaction.isRunning()){currentTransaction.rollback();}// Create and commit transaction to persist amount of retriescurrentTransaction.begin();process.setRetry(process.getRetry()+1);modelService.save(process);currentTransaction.commit();// Create new transaction, which would be rollbacked by task servicecurrentTransaction.begin();RetryLaterExceptionex=newRetryLaterException("Retry ");ex.setDelay(1000);throwex;}else{thrownewIllegalStateException("Fail process.");}}}