Project Reactor async在发生错误时重试发送电子邮件

Але*_*нов 2 java project-reactor

我需要在用户注册后发送一些数据.我想在主线程中首次尝试,但如果有任何错误,我想以10分钟的间隔重试5次.

@Override
public void sendRegisterInfo(MailData data) {
    Mono.just(data)
        .doOnNext(this::send)
        .doOnError(ex -> logger.warn("Main queue {}", ex.getMessage()))
        .doOnSuccess(d -> logger.info("Send mail to {}", d.getRecipient()))
        .onErrorResume(ex -> retryQueue(data))
        .subscribe();
}

private Mono<MailData> retryQueue(MailData data) {
    return Mono.just(data)
               .delayElement(Duration.of(10, ChronoUnit.MINUTES))
               .doOnNext(this::send)
               .doOnError(ex -> logger.warn("Retry queue {}", ex.getMessage()))
               .doOnSuccess(d -> logger.info("Send mail to {}", d.getRecipient()))
               .retry(5)
               .subscribe();
}
Run Code Online (Sandbox Code Playgroud)

有用.但是我有一些问题:

  1. 我是否纠正了在doOnNext功能上进行操作?
  2. 用于delayElement在执行之间延迟是否正确?
  3. 等待延迟时线程是否被阻止?
  4. 什么是对错误进行重试并在其间进行延迟的最佳实践?

Sim*_*slé 6

  1. doOnXXX记录很好.但对于实际的元素处理,您必须更喜欢使用flatMap而不是doOnNext(假设您的处理是异步的/可以转换为返回Flux/ Mono).

  2. 这是对的.另一种方法是转换代码并从a开始Flux.interval,但这里delayElement是更好的IMO.

  3. 延迟在单独的线程/调度程序上运行(默认情况下Schedulers.parallel()),因此不会阻塞主线程.

  4. Retryreactor-extra插件中实际上有一个专门用于这种用例的构建器:https://github.com/reactor/reactor-addons/blob/master/reactor-extra/src/main/java/reactor/retry/Retry.java