java Resilience4j RetryConfig.custom()不工作

olhwl3o2  于 2024-01-05  发布在  Java
关注(0)|答案(1)|浏览(217)

我正在尝试用RetryConfig实现一个测试。虽然我说了应该重试哪个异常,但什么也没发生。错误被抛出,测试完成。Fallback-method也没有触发。我的测试有什么问题?

TestConfig.java

  1. @TestConfiguration
  2. public class TestRetryConfig {
  3. @Autowired
  4. private RetryRegistry retryRegistry;
  5. @Bean("FeignRetry")
  6. public Retry retryWithCustomConfig() {
  7. RetryConfig customConfig = RetryConfig.custom()
  8. .maxAttempts(3)
  9. .waitDuration(Duration.ofSeconds(1))
  10. .retryExceptions(FeignException.class)
  11. .build();
  12. return retryRegistry.retry("feignException", customConfig);
  13. }
  14. }

字符串

TestClass.java

  1. @ExtendWith(SpringExtension.class)
  2. @ContextConfiguration(classes = {MyAdapter.class, InMemoryRetryRegistry.class, ServerInternalErrorVerifier.class, TestRetryConfig.class})
  3. class MyAdapterRetryTest {
  4. @Autowired
  5. private RetryRegistry registry;
  6. @Autowired
  7. private MyAdapter myAdapter;
  8. @Autowired
  9. private Retry retry; //i see here, that my retry has been injected correctly with defined bean-name
  10. @MockBean
  11. private MyClient myClient;
  12. @Test
  13. Request feignRequest =
  14. Request.create(Request.HttpMethod.POST, "", Map.of(), "".getBytes(StandardCharsets.UTF_8), null, null);
  15. FeignException.InternalServerError feignException =
  16. new FeignException.InternalServerError("", feignRequest, "".getBytes(StandardCharsets.UTF_8), null);
  17. when(myClient.sendPerson(any())).thenThrow(feignException); //here should retry be trigered
  18. String result = myAdapter.sendPerson(myRequest);
  19. assertThat(result).contains("was not send successfully");
  20. }


}

我的方法

  1. @Retry(name = "throwingFeignException", fallbackMethod = "logCommunicationFailure")
  2. public String sendPartner(MyRequest myRequest) {
  3. MyResponse response = myClient.sendPerson(myRequest);
  4. return "ok";
  5. }
  6. private String logCommunicationFailure(MyRequest myRequest, Exception exception) {
  7. return "fail";
  8. }

5uzkadbs

5uzkadbs1#

TL;DR:

简短的回答是:你需要用你的Retry retry对象和你提供的“My mix-Method”片段中的sendPartner()方法来调用Retry.decorateFunction()(这里有一个不一致的地方,因为你在测试中引用了sendPerson()--我认为你引用的是同一个方法,这只是一个拼写错误。

  1. assertThrows(
  2. FeignException.InternalServerError.class,
  3. () -> Retry.decorateFunction(retry, feignService::sendPartner).apply(person));
  4. verify(myClient, times(3)).sendPerson(any(Person.class));

字符串
这将执行用@Retry修饰的sendPartner()方法,并验证是否按照配置重试了3次。

Long Answer:A Working Example

FeignRetryConfiguration Bean

  1. @Configuration
  2. public class FeignRetryConfiguration {
  3. @Bean
  4. MyClient myClient() {
  5. return Feign.builder()
  6. .client(new Client.Default(null, null))
  7. .encoder(new Encoder.Default())
  8. .decoder(new Decoder.Default())
  9. .target(MyClient.class, "http://localhost:8080");
  10. }
  11. @Bean
  12. RetryConfig feignRetryConfig() {
  13. return RetryConfig.custom()
  14. .maxAttempts(3)
  15. .waitDuration(Duration.ofSeconds(1))
  16. .retryExceptions(FeignException.InternalServerError.class)
  17. .build();
  18. }
  19. @Bean("feignRetry")
  20. RetryRegistry retryRegistry() {
  21. return RetryRegistry.of(feignRetryConfig());
  22. }
  23. @Bean
  24. Retry retryFeign(@Autowired @Qualifier("feignRetry") RetryRegistry retryRegistry) {
  25. return retryRegistry.retry("throwingFeignException");
  26. }
  27. }

酒店FeignService Bean

  1. @Service
  2. public class FeignService {
  3. MyClient myClient;
  4. public FeignService(MyClient myClient) {
  5. this.myClient = myClient;
  6. }
  7. @Retry(name = "throwingFeignException", fallbackMethod = "logCommunicationFailure")
  8. public String sendPartner(Person person) {
  9. myClient.sendPerson(person);
  10. return "ok";
  11. }
  12. public String logCommunicationFailure(Person person, Exception exception) {
  13. return "fail";
  14. }
  15. }

MyClient

  1. public interface MyClient {
  2. @RequestLine("POST")
  3. @Headers("Content-Type: application/json")
  4. void sendPerson(Person person);
  5. }

  • Person是POJO -为简洁起见跳过 *
    实际单元测试
  1. @SpringBootTest
  2. class MyClientTest {
  3. @TestConfiguration
  4. static class MyClientTestConfiguration {
  5. @Bean
  6. FeignService feignService(@Autowired MyClient myClient) {
  7. return new FeignService(myClient);
  8. }
  9. }
  10. @MockBean
  11. MyClient myClient;
  12. @Autowired
  13. @Qualifier("retryFeign")
  14. Retry retry;
  15. @Autowired
  16. FeignService feignService;
  17. @Test
  18. void testSendPartnerShouldRetry() {
  19. Request request = Request.create(Request.HttpMethod.POST, "", Collections.emptyMap(), "".getBytes(StandardCharsets.UTF_8), null);
  20. doThrow(new FeignException.InternalServerError("internal error", request, "".getBytes(StandardCharsets.UTF_8), Collections.emptyMap()))
  21. .when(myClient).sendPerson(any(Person.class));
  22. Person person = new Person();
  23. person.setName("John Doe");
  24. person.setEmail("[email protected]");
  25. person.setPhone("123456789");
  26. assertThrows(FeignException.InternalServerError.class, () -> Retry.decorateFunction(retry, feignService::sendPartner).apply(person));
  27. verify(myClient, times(3)).sendPerson(any(Person.class));
  28. }
  29. }


当然,您需要扩展此代码以使其按预期工作,但希望它将演示如何实现它。

展开查看全部

相关问题