0

我正在通过使端点一次反应一个来将 Spring Boot 应用程序转换为使用 webflux。大多数(Spock)测试运行良好,但我在测试中自动连接 JPA 存储库以插入数据的测试却没有。当资源(我们正在测试的)从中读取时,存储库总是空的。我用不同的存储库进行了数十次这样的测试,它们都有相同的问题。

这是一个测试示例(我们正在测试的资源只是 findById 并从存储库返回示例):

@SpringBootTest
@AutoConfigureWebTestClient(timeout = "60000")
@Transactional
class PaymentControllerIntegrationTest extends Specification {

    @Autowired
    WebTestClient client

    @Autowired
    PaymentRepository repo

    def "GET /example/{id} returns correct example"() {
        given:
        def needle = new Example(id: 1L)
        def haystack = repo.saveAll([needle, new Example(id: 2L), new Example(id: 3L)])

        when:
        def response = client.get().uri(EXAMPLE_URL, [id: needle.id.toString()]).exchange()

        then:
        response.expectStatus().isOk()
        response.returnResult(ExampleResponse.class).getResponseBody().blockLast().id == needle.id
    }

当我在控制器中放置断点并执行 findAll() 时,存储库始终为空。

  • 我尝试改用 TestEntityManager.persistAndFlush
  • 我尝试使用 repo.saveAllAndFlush 代替
  • 我尝试过自动装配 ApplicationContext ,然后从中构建 WebTestClient ,但我从来没有让自动装配工作

我目前最好的猜测是测试设置不正确(应用程序上下文?)所以测试中的存储库与应用程序中的存储库不同。

4

1 回答 1

1

当使用TestWebClient(或TestRestTemplate)时,您实际上是在向您启动的服务器发出一个真实的 HTTP 请求。此请求在不同的线程中处理,因此使用新事务。

您的测试也是事务性的,但数据尚未提交,另一个事务只能读取已提交的数据(或者您需要将隔离级别设置为,READ_UNCOMMITTED但这可能不是您应该或不想做的事情)。

使用时,MockMvc您正在用模拟实例替换实际容器,它使用MockHttpServletRequest在同一个线程中执行(因此重用相同的事务并可以看到数据)。

要解决这个问题,请让您的测试不是事务性的,然后清理数据。然而,这会影响测试的性能(因为提交然后删除比在测试后回滚事务更慢)。

于 2021-10-01T06:47:10.317 回答