
本文旨在解决在使用 WebClient 进行请求时,对 doOnError 方法进行单元测试时遇到的问题。通常,直接抛出 WebClientResponseException 无法触发 doOnError 中的异常处理逻辑。本文将介绍如何使用 Mono.error 正确地模拟异常,从而确保 doOnError 能够被有效测试。
在使用 Spring WebClient 进行外部服务调用时,doOnError 方法用于处理请求过程中发生的异常。为了保证代码的健壮性,我们需要对 doOnError 逻辑进行单元测试。然而,直接在 Mockito 中 thenThrow 抛出 WebClientResponseException 往往无法触发 doOnError 中的异常处理。这是因为 WebClient 内部使用了 Reactor 响应式编程模型,异常需要通过 Mono.error 传递才能被正确处理。
下面通过一个示例来说明如何正确地测试 doOnError。
示例代码:
假设有如下代码片段,使用 WebClient 发送请求,并在 doOnError 中将 WebClientResponseException 转换为自定义异常 InvalidRequestException:
import org.springframework.web.reactive.function.client.WebClient;
import org.springframework.web.reactive.function.client.WebClientResponseException;
import reactor.core.publisher.Mono;
public class SomeService {
private final WebClient client;
public SomeService(WebClient client) {
this.client = client;
}
public String getResponse(String requestBody) {
return client.post()
.bodyValue(requestBody)
.retrieve()
.bodyToMono(String.class)
.doOnError(
WebClientResponseException.class,
err -> {
// 模拟一些处理逻辑
System.out.println("Error occurred: " + err.getResponseBodyAsString());
throw new InvalidRequestException(err.getResponseBodyAsString());
})
.block();
}
// 自定义异常
public static class InvalidRequestException extends RuntimeException {
public InvalidRequestException(String message) {
super(message);
}
}
}错误的单元测试示例:
以下是一个错误的单元测试示例,直接使用 thenThrow 抛出 WebClientResponseException:
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.junit.jupiter.MockitoExtension;
import org.springframework.web.reactive.function.client.WebClient;
import org.springframework.web.reactive.function.client.WebClient.RequestHeadersSpec;
import org.springframework.web.reactive.function.client.WebClient.RequestBodySpec;
import org.springframework.web.reactive.function.client.WebClient.ResponseSpec;
import org.springframework.web.reactive.function.client.WebClientResponseException;
import reactor.core.publisher.Mono;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.when;
@ExtendWith(MockitoExtension.class)
public class SomeServiceTest {
@Mock
private WebClient webClientMock;
@Mock
private RequestBodySpec requestBodyUriMock;
@Mock
private RequestHeadersSpec requestHeadersMock;
@Mock
private ResponseSpec responseMock;
@InjectMocks
private SomeService someService;
@Test
public void testGetResponse_doOnError() {
String requestBody = "test request";
String expectedMessage = "Bad Request";
WebClientResponseException thrownException = new WebClientResponseException(400, expectedMessage, null, null, null);
when(webClientMock.post()).thenReturn(requestBodyUriMock);
when(requestBodyUriMock.bodyValue(any())).thenReturn(requestHeadersMock);
when(requestHeadersMock.retrieve()).thenReturn(responseMock);
// 错误的方式:直接 thenThrow
when(responseMock.bodyToMono(String.class)).thenThrow(thrownException);
InvalidRequestException exception = assertThrows(InvalidRequestException.class, () -> {
someService.getResponse(requestBody);
});
assertEquals(expectedMessage, exception.getMessage());
}
}在这个例子中,尽管我们期望 InvalidRequestException 被抛出,但实际上抛出的是 WebClientResponseException。
正确的单元测试示例:
使用 Mono.error 包装异常:
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.ExtendWith;
import org.mockito.InjectMocks;
import org.mockito.Mock;
import org.mockito.junit.jupiter.MockitoExtension;
import org.springframework.web.reactive.function.client.WebClient;
import org.springframework.web.reactive.function.client.WebClient.RequestHeadersSpec;
import org.springframework.web.reactive.function.client.WebClient.RequestBodySpec;
import org.springframework.web.reactive.function.client.WebClient.ResponseSpec;
import org.springframework.web.reactive.function.client.WebClientResponseException;
import reactor.core.publisher.Mono;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertThrows;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.when;
@ExtendWith(MockitoExtension.class)
public class SomeServiceTest {
@Mock
private WebClient webClientMock;
@Mock
private RequestBodySpec requestBodyUriMock;
@Mock
private RequestHeadersSpec requestHeadersMock;
@Mock
private ResponseSpec responseMock;
@InjectMocks
private SomeService someService;
@Test
public void testGetResponse_doOnError() {
String requestBody = "test request";
String expectedMessage = "Bad Request";
WebClientResponseException thrownException = new WebClientResponseException(400, expectedMessage, null, null, null);
when(webClientMock.post()).thenReturn(requestBodyUriMock);
when(requestBodyUriMock.bodyValue(any())).thenReturn(requestHeadersMock);
when(requestHeadersMock.retrieve()).thenReturn(responseMock);
// 正确的方式:使用 Mono.error 包装异常
when(responseMock.bodyToMono(String.class)).thenReturn(Mono.error(thrownException));
InvalidRequestException exception = assertThrows(InvalidRequestException.class, () -> {
someService.getResponse(requestBody);
});
assertEquals(expectedMessage, exception.getMessage());
}
}在这个修正后的例子中,我们使用 Mono.error(thrownException) 包装了 WebClientResponseException。这样,doOnError 方法就能正确捕获异常,并执行相应的处理逻辑,最终抛出 InvalidRequestException。
总结与注意事项:
通过本文的介绍,你应该能够正确地对 WebClient 的 doOnError 方法进行单元测试,从而提高代码的质量和可靠性。
以上就是使用 Mono.error 进行 WebClient 异常处理单元测试的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号