JMockit:bean 的模拟集合不适用于 Stream API

JMockit: mocked collection of beans doesn't work with Stream API

我有一个简单的方法需要使用 JMockit 进行测试,但似乎使用 Stream API 需要额外的模拟,但我不明白我到底需要添加什么。

使用for-loop

有效的方法
private final List<ValueProcessor> valueProcessors;

public List<Response> proc(Entry value) {
    List<Response> badResponses = new ArrayList<>();
    for (ValueProcessor processor : valueProcessors) {
        List<Response> responses = processor.process(value);
        if (!CollectionUtils.isEmpty(responses)) {
                badResponses.addAll(responses);
        }
    }
    return badResponses;
}

使用 Stream 的相同代码 API:

public List<Response> proc(Entry value) {
    return valueProcessors.stream()
            .map(processor -> processor.process(value))
            .filter(collection -> !CollectionUtils.isEmpty(collection))
            .flatMap(Collection::stream)           
            .collect(Collectors.toList());
}

我想用JMockit测试代码,所以我有下面的测试

@RunWith(JMockit.class)
public class ProcessorTest {
    @Injectable
    private List<ValueProcessor> valueProcessors = Collections.singletonList(processor -> Collections.singletonList(new Response()));
    @Tested
    private Processor processor;

    @Test
    public void testProc() {
        assertThat(processor.proc(null)).hasSize(1);
    }
}

for-loop 方法的测试总是通过,Stream 方法总是失败。我该如何解决?

谢谢!

据我了解,当您使用 @Injectable 创建 valueProcessors 的列表时,您正在创建列表的模拟。

这意味着上述列表的方法将被模拟,以便它们默认为 return "empty" 值(如果未通过 Expectation 块指定任何内容)。

所以首先,非流代码,因为列表中没有调用任何方法,所以没有问题。

但是,在第二个使用流的代码中,对 valueProcessors.stream() 的调用执行 "empty" 模拟行为,即 return 是一个空流,因此,其他所有内容都为空也是。

我想出的解决方案是创建测试实例,并向其传递一个 "real" 列表,而不是模拟列表:

@RunWith(JMockit.class)
public class WhosebugTest {

    // empty injectable just so that the processor can be created
    @Injectable
    private List<ValueProcessor> valueProcessors;

    // availableDuringSetup = true to be able to modify the processor
    @Tested(availableDuringSetup = true)
    private Processor processor;

    @Before
    public void setUp() throws Exception {
        // create new instance of the processor (now the valueProcessors list is a real list)
        processor = new Processor(Collections.singletonList(processor -> Collections.singletonList(new Response())));
    }

    @Test
    public void testProc() {
        assertThat(processor.proc(null), hasSize(1));
    }

}