6

我在测试多个服务的集成测试时遇到问题。我必须禁用事务性才能使测试正常工作,而不会出现任何与事务性相关的运行时错误。集成测试在单独运行时工作正常,但在与启用事务的其他测试一起运行时,它会产生以下运行时错误:

Running 48 integration tests... 43 of 48
Failure:  Tests the happy case flow of MyService.(MyServiceSpec)
org.springframework.transaction.HeuristicCompletionException: Heuristic completion: outcome state is rolled back; nested exception is org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has bee
n marked as rollback-only
Caused by: org.springframework.transaction.UnexpectedRollbackException: Transaction rolled back because it has been marked as rollback-only
        ... 4 more
Completed 43 integration tests, 1 failed in 0m 32s

我已经得出结论,运行时发生的原因是因为其他使用事务的集成测试,因为我通过成功运行所有禁用事务的测试来测试这一点;并且在使用启用事务的单个集成测试运行测试时失败。

如何在 Grails 中混合使用事务性和非事务性集成测试?

平台详情:

Grails-2.3.6 Windows 7 64 位。JDK v6。

4

2 回答 2

3

使用 Grails 2.4.3 遇到这个问题,经过一堆调试后发现org.springframework.orm.hiberante4.HibernateTransactionManager.doGetTransaction()它调用了TransactionSynchronizationManager.getResource(getSessionFactory()),如果有其他启用事务的测试,那么它将找到一个线程绑定的 SessionHolder,rollbackOnly 设置为 true(因为之前的测试回滚了) . 因此,当它第一次尝试提交事务时,它会看到这一点并给出UnexpectedRollbackException你所指示的。

我通过将以下内容放入标记为非事务性的测试的 setUp() 中解决了这个问题:

Holders.grailsApplication.mainContext.getBeansOfType(SessionFactory.class).each { beanName, sessionFactory ->

    SessionHolder sessionHolder = TransactionSynchronizationManager.getResource(sessionFactory)

    if (sessionHolder) {
         sessionHolder.clear()
    }
}
于 2015-08-22T00:58:57.013 回答
0

集成测试将执行以下操作

  • 开始交易
  • 运行测试
  • 回滚事务

通常,这将起作用并将数据库状态重置为测试前的状态。但是,如果您的测试逻辑将以特定方式处理事务,您将遇到问题。一个例子是在测试代码中创建您自己的事务,使用propagation=REQUIRES_NEW。无论您在该事务中做了什么,测试逻辑都无法回滚。

通常这样的代码会破坏测试之间的独立性。唯一真正安全的方法是让每个测试从一个空数据库开始并插入需要的内容......

于 2015-06-23T15:15:31.660 回答