【问题标题】:H2 database testing with mocked service gives nullpointer exception使用模拟服务进行 H2 数据库测试会产生空指针异常
【发布时间】:2019-02-26 15:50:54
【问题描述】:

我正在开发一个基于 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 为空,我不断收到 NullPointer 异常。 它甚至不会被嘲笑。

不能同时测试存储库和模拟服务吗? 我做错了什么?

非常感谢任何帮助!

谢谢!



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 添加到 TestExecutionListeners 列表中......

【问题讨论】:

  • serviceName和myService一样吗?
  • 是的,抱歉,打错了,但已更正。
  • 能贴一下空指针的stacktrace吗?
  • 我更新了我的帖子...谢谢!
  • 但是您删除了显示 NPE 发生位置的行!您能否将您的测试类减少到足够小,以便您可以在问题中发布整个内容以及完整的堆栈跟踪(如果需要,删除任何机密材料)?

标签: java spring mockito h2 springrunner


【解决方案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 只看到外部服务。然后在被模拟的外部服务上设置模拟行为。但是您似乎在模拟被测服务,仅自动装配存储库,然后在外部服务上设置模拟行为;但是您没有向我们展示您是如何设置外部服务的 - 也许它根本没有设置,这就是它为空的原因。

我认为它可以帮助您将测试课程减少到足够小的内容,以便在此处完整发布。您可能会在这样做时找到解释。

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2013-07-13
    • 2012-11-28
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多