Spring Retry with for loop

1k Views Asked by At

My RetryTemplate config:

@Configuration
@EnableRetry
public class RetryTemplateConfig {

    @Value("${spring.retry.attempts}")
    private int maxAttempts;

    @Value("${spring.retry.period}")
    private long backOffPeriod;

    @Bean
    public RetryTemplate retryTemplate() {
        SimpleRetryPolicy retryPolicy = new SimpleRetryPolicy();
        retryPolicy.setMaxAttempts(maxAttempts);
        FixedBackOffPolicy backOffPolicy = new FixedBackOffPolicy();
        backOffPolicy.setBackOffPeriod(backOffPeriod);
        RetryTemplate retryTemplate = new RetryTemplate();
        retryTemplate.setRetryPolicy(retryPolicy);
        retryTemplate.setBackOffPolicy(backOffPolicy);
        return retryTemplate;
    }
}

Scheduled method which invokes method that uses Retry:

@Scheduled(cron = "${schedule.cron.update.users}")
    public void sendToUsers() throws Exception {
        log.info("Scheduled sending for users started");
        try {
            mailSender.sendToUsers();
        } catch (MessagingException | IOException | TemplateException e) {
            log.error("Error occurred while sending email message to users: {}", e.toString());
        }
        log.info("Scheduled sending for users finished");
    }

Method in which i want to use RetryTemplate:

public void sendToUsers() throws Exception {
        String subject = reportGenerationService.getEmailSubjectForUser();
        Map<String, List<BadUtmMark>> utmMarksGroupedByEmail = userService.getUtmMarksGroupedByEmail(LocalDate.now());
        if (utmMarksGroupedByEmail.isEmpty()) {
            log.info("All's fine - no broken utm marks in database. Emails to users will not be send.");
        }
        for (Map.Entry<String, List<BadUtmMark>> pair : utmMarksGroupedByEmail.entrySet()) {
            retryTemplate.execute(retryContext -> {
                String emailTo = pair.getKey();
                List<BadUtmMark> badUtmMarks = pair.getValue();
                String report = reportGenerationService.getReportForUser(emailTo, badUtmMarks, template);
                MimeMessage mimeMessage = getMimeMessage(subject, report, Collections.singletonList(emailTo));
                log.info("Message will be sent to: {}; from: {}; with subject: {}", pair.getKey(), from, subject);
                mailSender.send(mimeMessage);
                return true;
            });
        }
    }

Expected behaviour: I want to send emails for 5 people. If error occurs, then try to send email to this user 5 more times, and then if retries exhausted, keep going and send email for next user in for loop.

Really behaviour: If error occurs, service will catch exception and stop looping.

If i'll move retry logic to this method:

@Scheduled(cron = "${schedule.cron.update.users}")
public void sendToUsers() throws Exception {
    log.info("Scheduled sending for users started");
    try {
        retryTemplate.execute(retryContext - > {
            log.warn("Sending email to users. Attempt: {}", retryContext.getRetryCount());
            mailSender.sendToUsers();
            return true;
        });
    } catch (MessagingException | IOException | TemplateException e) {
        log.error("Error occurred while sending email message to users: {}", e.toString());
    }
    log.info("Scheduled sending for users finished");
}

It works better, but still not what I expect. With this case if error occurs, service will try to send email 5 more times, but if retries exhausted, service will stop looping. So, if error occurs with one of the user, the service will try to send 5 more times for this user, and then will stop, ignoring other users in map.

But I want to do 5 retries for each email in my map. How can i do this?

1

There are 1 best solutions below

1
On BEST ANSWER

In your first version, use this execute instead...

    /**
     * Keep executing the callback until it either succeeds or the policy dictates that we
     * stop, in which case the recovery callback will be executed.
     *
     * @see RetryOperations#execute(RetryCallback, RecoveryCallback)
     * @param retryCallback the {@link RetryCallback}
     * @param recoveryCallback the {@link RecoveryCallback}
     * @throws TerminatedRetryException if the retry has been manually terminated by a
     * listener.
     */
    @Override
    public final <T, E extends Throwable> T execute(RetryCallback<T, E> retryCallback,
            RecoveryCallback<T> recoveryCallback) throws E {
        return doExecute(retryCallback, recoveryCallback, null);
    }
template.execute(context -> {
    ...
}, context -> {
    logger.error("Failed to send to ...");
});

If the callback exits normally, the failure is "recovered" and the exception not rethrown.