使用模拟服务进行H2数据库测试会导致nullpointer异常

时间:2019-02-26 15:50:54

标签: java spring mockito h2 springrunner

我正在研究一个基于Spring Java的项目。 为了进行测试,我实现了一个H2内存数据库,该数据库已通过xml文档填充了数据。 我希望测试一种通过传入数据报更新数据库记录的方法。 首先,此方法检查到达的数据报中是否有在数据库中找到的任何相关数据(这里是存储库调用),然后调用具有给定id的另一个外部服务,该外部服务提供了需要放入数据库,而不是已经保存的数据库。 因此,首先有一个存储库调用,然后是对外部服务的第二个调用。 当我想测试此功能时,我使用以下注释:

@RunWith(SpringRunner.class)
@SpringBootTest
@DatabaseSetup("classpath:datasets/FileWhichContainsDbRecords.xml")

然后我使用以下方法模拟外部服务:

@MockBean
private ServiceName myService;

并使用以下命令调用Spring存储库:

@Autowired
private RepositoryName repositoryName;

并具有以下测试代码:

@Test
@Transactional
public void testCase() {
    GivenDg dg = Factory.createDg("id1", Collections.singletonList(Factory.createAnotherObject("id2", OPERATION_TYPE_MODIFY, Collections.singletonList("channel"))));

    MyObject myObjectSavedInDB = repositoryName.findByProvidedIds("id1", "id2").get();

    when(serviceName.getData(anyString(), anyString(), anyString(), any())).thenReturn(
            Factory.createActualData(false, false, false, "HUF", Collections.singletonList("channels")));

    myService.updateFunction(dg);

    Optional<MyObject> optResult = repositoryName.findByProvidedIds("id1", "id2");
    assertTrue(optResult.isPresent());
    assertNotSame(optResult.get().isA(), myObjectSavedInDB.isA());
}

因此,我模拟了ServiceName服务,并提供了希望在调用此函数时得到的响应,但是由于ServiceName为null,所以我一直收到NullPointer异常。 它甚至没有被嘲笑。

是否无法同时测试存储库和模拟服务? 我在做什么错了?

我将非常感谢您的帮助!

谢谢!


enter image description here


java.lang.NullPointerException
at ---------)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:50)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:12)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:47)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:17)
at org.springframework.test.context.junit4.statements.RunBeforeTestExecutionCallbacks.evaluate(RunBeforeTestExecutionCallbacks.java:74)
at org.springframework.test.context.junit4.statements.RunAfterTestExecutionCallbacks.evaluate(RunAfterTestExecutionCallbacks.java:84)
at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:75)
at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:86)
at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:84)
at org.junit.runners.ParentRunner.runLeaf(ParentRunner.java:325)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:251)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:97)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:290)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:71)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:288)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:58)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:268)
at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
at org.junit.runners.ParentRunner.run(ParentRunner.java:363)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:190)
at org.junit.runner.JUnitCore.run(JUnitCore.java:137)
at com.intellij.junit4.JUnit4IdeaTestRunner.startRunnerWithArgs(JUnit4IdeaTestRunner.java:68)
at com.intellij.rt.execution.junit.IdeaTestRunner$Repeater.startRunnerWithArgs(IdeaTestRunner.java:47)
at com.intellij.rt.execution.junit.JUnitStarter.prepareStreamsAndStart(JUnitStarter.java:242)
at com.intellij.rt.execution.junit.JUnitStarter.main(JUnitStarter.java:70)

好吧,我发现了所缺少的东西……最初,我在Test类上有以下注释:

@RunWith(SpringRunner.class)
@SpringBootTest()
@DatabaseSetup("classpath:datasets/FileWhichContainsDbRecords.xml.xml")
@ContextConfiguration(classes = {TestRepositoryConfig.class})
@TestExecutionListeners({DependencyInjectionTestExecutionListener.class,
    DirtiesContextTestExecutionListener.class,
    TransactionalTestExecutionListener.class,
    DbUnitTestExecutionListener.class})
@DirtiesContext()
@Transactional

我必须将 MockitoTestExecutionListener.class 添加到TestExecutionListener的列表中。...

1 个答案:

答案 0 :(得分:1)

您发布的代码中缺少某些内容。我希望看到:

@RunWith(SpringRunner.class)
@SpringBootTest
@DatabaseSetup("classpath:datasets/FileWhichContainsDbRecords.xml")
class MyTest {
    @Autowired private WidgetRepository repository;
    @Autowired private ServiceUnderTest service;
    @MockBean private ExternalService externalService;

    @Test
    @Transactional
    public void testWidgetService() {
        // ...
        when(externalService.someCall()).thenReturn(Factory.whatever());
        // ...
    }
}

换句话说,我希望看到您@Autowire 两者您正在测试的服务和存储库,以及@MockBean仅是外部服务。然后在模拟的外部服务上设置模拟行为。但是,您似乎在嘲笑被测服务,仅自动装配存储库,然后在外部服务上设置模拟行为。但您没有向我们展示如何设置外部服务-也许根本没有设置,这就是为什么它为空的原因。

我认为这将帮助您将测试类减少到足以在此处完整发布的内容。您可能会在这样做的同时找到解释。