use*_*276 1 java unit-testing mockito spring-webflux spring-data-r2dbc
我为某些服务编写单元测试,但我不明白为什么我模拟的某些方法不调用。我尝试测试testDeleteDeviceWhenDeviceNotFound()方法并且通过了,但是当我尝试时testDeleteDeviceSuccess()遇到问题
@Service
@Slf4j
@AllArgsConstructor
public class DeviceService {
private DeviceRepository deviceRepository;
private SyncSender syncSender;
public Mono<Void> deleteDevice(long deviceId) {
return deviceRepository
.findById(deviceId)
.switchIfEmpty(Mono.error(new NotFoundException()))
.flatMap(existingDevice -> deviceRepository
.delete(existingDevice)
.then(syncSender.sendDeviceDelete(existingDevice.getDeviceId()))
);
}
Run Code Online (Sandbox Code Playgroud)
和我的测试
@ExtendWith(MockitoExtension.class)
class DeviceServiceTest {
@Mock
private DeviceRepository deviceRepository;
@Mock
private SyncSender syncSender;
@InjectMocks
private DeviceService deviceService;
@Test
@DisplayName("Test deleteDevice when NotFoundException")
void testDeleteDeviceWhenDeviceNotFound() {
long deviceId = 100L;
Mockito.when(deviceRepository.findById(deviceId)).thenReturn(Mono.empty());
Mono<Void> mono = deviceService.deleteDevice(deviceId);
StepVerifier
.create(mono)
.expectErrorMatches(throwable -> throwable instanceof NotFoundException)
.verify();
}
@Test
@DisplayName("Test deleteDevice success")
void testDeleteDeviceSuccess() {
DeviceModel deviceModel = createDeviceModel();
deviceModel.setDeviceId(100L);
Mockito.when(deviceRepository.findById(deviceModel.getDeviceId())).thenReturn(Mono.just(deviceModel));
Mockito.when(syncSender.sendDeviceDelete(anyLong())).thenReturn(Mono.empty());
Mockito.when(deviceRepository.delete(any(DeviceModel.class))).thenReturn(Mono.empty());
deviceService.deleteDevice(deviceModel.getDeviceId());
}
}
Run Code Online (Sandbox Code Playgroud)
junit 的例外情况是,在测试执行期间,被测试的代码中从未实现过存根方法之一
org.mockito.exceptions.misusing.UnnecessaryStubbingException:
Unnecessary stubbings detected.
Clean & maintainable test code requires zero unnecessary code.
Following stubbings are unnecessary (click to navigate to relevant line of code):
1. -> at com.ecosoft.dms.device.vending.service.DeviceServiceTest.testDeleteDeviceSuccess(DeviceServiceTest.java:172)
2. -> at com.ecosoft.dms.device.vending.service.DeviceServiceTest.testDeleteDeviceSuccess(DeviceServiceTest.java:173)
Please remove unnecessary stubbings or use 'lenient' strictness. More info: javadoc for UnnecessaryStubbingException class.
Run Code Online (Sandbox Code Playgroud)
当我尝试调试模式时,我发现我返回了这一步。为什么?
.flatMap(existingDevice -> deviceRepository
Run Code Online (Sandbox Code Playgroud)
在失败的测试中,您有:
deviceService.deleteDevice(deviceModel.getDeviceId());
Run Code Online (Sandbox Code Playgroud)
这会创建一个单声道,但永远不会启动底层异步任务。测试完成,Mockito 正确报告从未调用存根方法。
您在其他测试中使用了 StepVerifier,在我看来,您也需要在这里使用它来触发底层异步操作
StepVerifier.create(deviceService.deleteDevice(deviceModel.getDeviceId()))
.verifyComplete();
Run Code Online (Sandbox Code Playgroud)
检查更简单的例子:
@Test
void testMonoNotRun() {
Mono.fromCallable(() -> {
System.out.println("Callable run");
return 1;
});
}
@Test
void testMonoRunBySubscribe() {
Mono.fromCallable(() -> {
System.out.println("Callable run");
return 1;
}).subscribe();
}
@Test
void testMonoRunByStepVerify() {
StepVerifier.create(Mono.fromCallable(() -> {
System.out.println("Callable run");
return 1;
})).verifyComplete();
}
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
5570 次 |
| 最近记录: |