使用Spring、Hibernate和Spring事务支持为单元测试设置数据库

5

我希望在单元测试中测试dao层与service层的集成。因此,我需要在我的数据库(hsql)中设置一些数据。为了这个设置,我需要在测试用例开始时拥有自己的事务,以确保所有设置都真正地提交到数据库中,然后才能开始我的测试用例。

所以这就是我想要实现的:

// NotTranactional
public void doTest {
  // transaction begins
  setup database
  // commit transaction

  service.doStuff() // doStuff is annotated @Transactional(propagation=Propagation.REQUIRED)
}

这是我的代码,但它无法正常工作:

<bean id="dataSource"
    class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close"
    p:driverClassName="org.hsqldb.jdbcDriver"
    p:url="jdbc:hsqldb:mem:posmail" p:username="sa"
    p:password="" />

<bean id="sessionFactory"
    class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
    <property name="dataSource" ref="dataSource" />
    <property name="annotatedClasses">
        <list>        
            <value>de.diandan.asynch.modell.receipt.Position</value>
            <value>de.diandan.asynch.modell.receipt.Receipt</value>
        </list>
    </property>    

    <property name="hibernateProperties">
        <props>
            <prop key="hibernate.connection.useUnicode">true</prop>
            <prop key="hibernate.connection.characterEncoding">UTF-8</prop>
            <prop key="hibernate.connection.charSet">UTF-8</prop>
            <prop key="hibernate.dialect">org.hibernate.dialect.HSQLDialect</prop>
            <prop key="hibernate.show_sql">false</prop>
            <prop key="hibernate.hbm2ddl.auto">update</prop>
            <!-- Enable Hibernate's automatic session context management 
            <prop key="current_session_context_class">thread</prop> -->

            <!-- Disable the second-level cache  -->
            <prop key="cache.provider_class">org.hibernate.cache.NoCacheProvider</prop>

            <prop key="hibernate.show_sql">false</prop>

        </props>
    </property>
</bean>

<bean id="transactionManager" 
      class="org.springframework.orm.hibernate3.HibernateTransactionManager">
    <property name="sessionFactory" ref="sessionFactory"/>
</bean>

我的测试用例

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"/asynchUnit.xml"})
@DirtiesContext(classMode=ClassMode.AFTER_EACH_TEST_METHOD)
public class ReceiptServiceTest implements ApplicationContextAware {
  @Autowired(required=true)
  private UserHome userHome;

  private ApplicationContext context;

  @Before
  @Transactional(propagation=Propagation.REQUIRED)
  public void init() throws Exception {
    User user = InitialCreator.createUser();
    userHome.persist(user);
  }

  @Test
  public void testDoSomething() {
    ...
  }
}

导致此异常的原因是:
org.hibernate.HibernateException: No Hibernate Session bound to thread, and configuration does not allow creation of non-transactional one here
    at org.springframework.orm.hibernate3.SpringSessionContext.currentSession(SpringSessionContext.java:63)
    at org.hibernate.impl.SessionFactoryImpl.getCurrentSession(SessionFactoryImpl.java:687)
    at de.diandan.asynch.modell.GenericHome.getSession(GenericHome.java:40)
    at de.diandan.asynch.modell.GenericHome.persist(GenericHome.java:53)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
    at java.lang.reflect.Method.invoke(Method.java:597)
    at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:318)
    at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:196)
    at $Proxy28.persist(Unknown Source)
    at de.diandan.asynch.service.ReceiptServiceTest.init(ReceiptServiceTest.java:63)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
    at java.lang.reflect.Method.invoke(Method.java:597)
    at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
    at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
    at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
    at org.junit.internal.runners.statements.RunBefores.evaluate(RunBefores.java:27)
    at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:74)
    at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:83)
    at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:72)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:231)
    at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:50)
    at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
    at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
    at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
    at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
    at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
    at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
    at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:71)
    at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
    at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:174)
    at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:49)
    at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
    at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)

我不知道在设置数据库时获取事务的正确方法。
我尝试过:
@Before
@Transactional(propagation=Propagation.REQUIRED)
public void setup() {
  setup database
}

-> Spring似乎不会在@Before注释的方法中启动事务。然而,这并不是我真正想要的,因为我的测试类中有很多方法需要稍微不同的设置,所以我需要几个初始化方法。

@Transactional(propagation=Propagation.REQUIRED)
public void setup() {
  setup database
}

public void doTest {
  init();

  service.doStuff() // doStuff is annotated @Transactional(propagation=Propagation.REQUIRED)
}

--> init似乎没有在事务中启动

我不想做的事情:

public void doTest {
  // doing my own transaction-handling
  setup database
  // doing my own transaction-handling

  service.doStuff() // doStuff is annotated @Transactional(propagation=Propagation.REQUIRED)
}

-->开始混合使用Spring事务处理和我的自己的事务处理,这似乎会变得非常麻烦。
@Transactional(propagation=Propagation.REQUIRED)
public void doTest {
  setup database
  service.doStuff()
}

--> 我希望尽可能真实地测试情况,因此我的服务应该从一个干净的会话开始,并且没有打开的事务。

那么为我的测试用例设置数据库的正确方法是什么?


请确保您的数据库支持事务。例如,MySQL在表格设置为InnoDB方言之前是不支持事务的。 - ŁukaszBachman
2个回答

4

添加以下注释即可解决问题:

@TransactionConfiguration(transactionManager="transactionManager", defaultRollback=false)
@Transactional

这是完整的配置结果:

@TransactionConfiguration(transactionManager="transactionManager", defaultRollback=false)
@Transactional
@RunWith(SpringJUnit4ClassRunner.class)
@ConnextConfiguration(locations={"/asynchUnit.xml"})
@DirtiesContext(classMode=ClassMode.AFTER_EACH_TEST_METHOD)
public class ReceiptServiceTest implements ApplicationContextAware {

ConnextConfiguration 应该是 ContextConfiguration;我自己无法编辑那个小错字 :) - orique

3

对于与spring有关的测试,您可以使用JUnit的spring runner:

@RunWith(SpringJUnit4ClassRunner.class)

这里记录了相关的文档:

http://static.springsource.org/spring/docs/3.0.0.M3/spring-framework-reference/html/ch10s03.html

你可能感兴趣的部分是,你可以使用注解对方法进行注释,如下所示:

@Rollback(false)
@Test
public void testProcessWithoutRollback() {
    // ...
}



@BeforeTransaction
public void beforeTransaction() {
    // logic to be executed before a transaction is started
}

并且

@AfterTransaction
public void afterTransaction() {
    // logic to be executed after a transaction has ended
}

您甚至可以使用类似以下的东西:
@NotTransactional 
@Test
public void testProcessWithoutTransaction() {
    // ...
}

当你想表示不希望对其应用事务时,可以使用@NotTransactional。但请注意该文档中的以下警告:

从Spring 3.0开始,@NotTransactional已被弃用,推荐将非事务测试方法移动到单独的(非事务性)测试类或@BeforeTransaction或@AfterTransaction方法中。作为在整个类上注释@Transactional的替代方案,考虑使用在各个方法上使用@Transactional进行注释;这样可以在同一个测试类中混合使用具有事务性和非事务性的方法,而无需使用@NotTransactional。


我已经在使用RunWith(SpringJUnit4ClassRunner.class),现在我将我的问题与示例代码和异常一起完成。其他的注释对我没有用,因为事务甚至还没有开始,所以BeforeTransaction和AfterTransaction都没有被调用。 - mibutec
你能提供Spring和Hibernate的配置吗? - Francisco Spaeth
我添加了Spring配置的有趣部分。但我认为原因在于堆栈跟踪中可见:ReceiptServiceTest.init周围没有代理。这是什么原因? - mibutec
1
顺便提一下:与此同时,@NotTransactional已被弃用。 - Arjan

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接