java - 使用模拟服务进行 H2 数据库测试会出现空指针异常

标签 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 为空,我不断收到 NullPointer 异常。 它甚至不会被 mock 。

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

我非常感谢任何帮助!

谢谢!

<小时/>

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)
<小时/>

好吧,我发现缺少了什么...最初,我的测试类上有以下注释:

@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列表中......

最佳答案

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

@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仅外部服务。然后在外部服务上设置模拟行为,即被模拟。但是您似乎正在模拟被测试的服务,仅 Autowiring 存储库,然后在外部服务上设置模拟行为;但您没有向我们展示您如何设置外部服务 - 也许它根本没有设置,这就是它为空的原因。

我认为这会帮助您将测试类减少到足够小,以便将其完整发布到此处。当您这样做时,您很可能会找到解释。

关于java - 使用模拟服务进行 H2 数据库测试会出现空指针异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54889355/

相关文章:

java - Android操作栏,更改颜色并添加搜索功能

java - 用于覆盖 XML 定义的 Bean 注释 - Spring

java - java spring 为具有相同属性的多个不同 POJO 创建自定义 validator

java - 如何模拟具有两种不同实现的接口(interface)

java - PreparedStatementCallback 覆盖率的 Junit 测试用例

java - 乔比 : How to properly unit test a route that returns different content depending on MediaType?

java - CXF项目抛出java.lang.NoClassDefFoundError : javax/ws/rs/NotFoundException

java - 来自模拟 bean 的方法无法正常工作

java - 如何从 OnCreate 方法外部更改小部件的可见性? Java Android Studio

java - jms queuebrowser # getEnumeration 总是获取为空