Powermockito / Java - 正在测试的间谍类以验证私有方法调用

时间:2016-11-11 16:55:20

标签: java unit-testing mocking mockito stubbing

我正在使用Mockito / PowerMockito在java中编写单元测试,但是在我正在进行的测试中,我无法摆脱这个UnfinishedStubbingException。

我试图测试的方法是私有的,所以我使用WhiteBoxImpl来调用该方法。在我调用的方法中,可能会在被测试的类中调用另一个私有方法(称为pm2)。我想验证pm2永远不会被调用,所以我为被测试的类做了一个间谍,并且验证pm2是never()被调用。

到目前为止,这个测试总是抛出一个UnfinishedStubbingException,但我无法弄清楚我的测试中哪个部分Powermockito不喜欢。我有另一个(工作)测试,操作非常相似,除了我不需要验证像pm2这样的方法的行为。所以在这个工作案例中,我不需要为被测试的类创建一个间谍。我相信我的问题在某种程度上与间谍有关,但我不知道如何在没有它的情况下测试我想要测试的内容。

这就是我现在所拥有的:

@Mock(name = "BO")
BO BOMock;
@Mock(name = "DAO")
DAOI DAOMock;

@InjectMocks
ServiceImpl service;

@Test
public void unitTest(){
    MessageObject msg = new MessageObject();
    Record recordMock = mock(Record.class);
    MetaData metaDataMock = mock(MetaData.class);

    doNothing().when(DAOMock).doAction(any(Param1.class), anyInt());
    when(DAOMock.doOtherAction(any(Param1.class), eq(msg.getId()))).thenReturn(recordMock);
    when(BOMock.getMetaData(anyInt(), anyInt()).thenReturn(metaDataMock);

    ServiceImpl spy = PowerMockito.spy(this.service);
    PowerMockito.doReturn(new Long(10)).when(spy, "checkDelay", recordMock, msg, metaDataMock);

    Whitebox.invokeMethod(spy, "process", msg);
    verify(recordMock, never()).getStatus();
}

这是我正在测试的ServiceImpl类中的方法:

private BO BO = new BO();
private DAOI DAO = new DAO();

private void process(Message msg) {
    try {
        DAO.doAction(new Param1.class, msg.getId());
    } catch(Exception e) {
        logger.error("some message");
        return;
    }

    Record record = null;

    try {

        int intParam1 = msg.getId();
        int intParam2 = msg.getDifferentId();

        MetaData metaData = BO.getMetaData(intParam1, intParam2);

        record = DAO.loadRecord(new Param1(), msg.getId());

        // checkDelay is a private method in ServiceImpl.java
        long delayForMinutes = checkDelay(record, msg, metaData);
        if(delayForMinutes > 0) {
            // Control should reach here
            logger.debug("some message");
            return;
        }

        // Control should not reach here

        if(Record != null && Record.getStatus() != CREATED) {
            logger.debug("some message");
            return;
        }

        // Perform various actions

    } catch(Exception e) {
        // Perform other various actions
    }
}

当我运行此测试时,我得到一个UnfinishedStubbingException。堆栈跟踪顶部的行是:

DAO.doAction(new Param1.class, msg.getId());

错误消息提供以下提示:

E.g. thenReturn() may be missing.
Examples of correct stubbing:
    when(mock.isOk()).thenReturn(true);
    when(mock.isOk()).thenThrow(exception);
    doThrow(exception).when(mock).someVoidMethod();
Hints:
 1. missing thenReturn()
 2. you are trying to stub a final method, you naughty developer!
 3: you are stubbing the behaviour of another mock inside before 'thenReturn' instruction if completed

但我似乎无法弄清楚它们中的任何一种如何适用于我的情况。有谁知道幕后发生了什么导致这个错误?

谢谢

1 个答案:

答案 0 :(得分:0)

问题在于

when(DAOMock.doOtherAction(any(Param1.class), eq(msg.getId()))).loadRecord(recordMock);

此处似乎没有thenthenReturnthenThrow。您始终需要使用其中一个when