Почему транзакции не откатываются при использовании SpringJUnit4ClassRunner / MySQL / Spring / Hibernate - PullRequest
4 голосов
/ 13 мая 2010

Я выполняю модульное тестирование и ожидаю, что все данные, сохраненные в базе данных MySQL, будут откатаны ... но это не так. Данные фиксируются, хотя мой журнал показывал, что произошел откат. Я боролся с этим пару дней, поэтому мои настройки немного изменились, вот мои текущие настройки.

LoginDAOTest.java:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"file:web/WEB-INF/applicationContext-test.xml", "file:web/WEB-INF/dispatcher-servlet-test.xml"})
@TransactionConfiguration(transactionManager = "transactionManager", defaultRollback = true)
public class UserServiceTest {

  private UserService userService;

  @Test
  public void should_return_true_when_user_is_logged_in ()
          throws Exception
  {
    String[] usernames = {"a","b","c","d"};

    for (String username : usernames)
    {
      userService.logUserIn(username);
      assertThat(userService.isUserLoggedIn(username), is(equalTo(true)));
    }
  }

ApplicationContext-Text.xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:p="http://www.springframework.org/schema/p"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
       http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
       http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">

  <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
          <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
          <property name="url" value="jdbc:mysql://localhost:3306/******"/>
          <property name="username" value="*****"/>
          <property name="password" value="*****"/>
  </bean>

  <tx:annotation-driven transaction-manager="transactionManager"/>

  <bean id="userService" class="Service.UserService">
    <property name="userDAO" ref="userDAO"/>
  </bean>

  <bean id="userDAO" class="DAO.UserDAO">
    <property name="hibernateTemplate" ref="hibernateTemplate"/>
  </bean>

  <bean id="sessionFactory" class="org.springframework.orm.hibernate3.LocalSessionFactoryBean">
    <property name="dataSource" ref="dataSource"/>
    <property name="mappingResources">
      <list>
        <value>/himapping/User.hbm.xml</value>
        <value>/himapping/setup.hbm.xml</value>
        <value>/himapping/UserHistory.hbm.xml</value>
      </list>
    </property>
    <property name="hibernateProperties">
      <props>
        <prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>
        <prop key="hibernate.show_sql">true</prop>
      </props>
    </property>
  </bean>

  <bean id="transactionManager" class="org.springframework.orm.hibernate3.HibernateTransactionManager"
          p:sessionFactory-ref="sessionFactory"/>

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

</beans>

Я читал об этой проблеме и уже проверил, чтобы таблицы базы данных MySQL были настроены для использования InnoDB. Также я смог успешно реализовать откат транзакций за пределами моего пакета тестирования. Так что это должно быть какая-то неправильная установка с моей стороны.

Любая помощь будет принята с благодарностью:)

Ответы [ 5 ]

12 голосов
/ 16 мая 2010

Проблема оказалась в том, что соединение автоматически фиксировалось ДО того, как транзакция могла быть откатана. Мне пришлось изменить bean-компонент dataSource, чтобы он включал свойство defaultAutoCommit:

<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
  <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
  <property name="url" value="jdbc:mysql://localhost:3306/test"/>
  <property name="username" value="root"/>
  <property name="password" value="Ecosim07"/>
  <property name="defaultAutoCommit" value="false" /> 
</bean>
5 голосов
/ 20 мая 2011

Для меня defaultAutoCommit и @Transactional не помогли. Мне пришлось изменить тип БД на InnoDB

1 голос
/ 31 мая 2014

Это должно быть использовано

@TransactionConfiguration(transactionManager = "transactionManager", defaultRollback = true)
@TestExecutionListeners({ TransactionalTestExecutionListener.class })
@Transactional

TransactionalTestExecutionListener содержит isRollback (), который выполняет откат транзакция после метода испытаний.

0 голосов
/ 19 июля 2011

Еще один способ решить вашу проблему:

Вместо использования:

<prop key="hibernate.dialect">org.hibernate.dialect.MySQLDialect</prop>

, которая по умолчанию создает таблицу MyISAM, следовательно, не поддерживает транзакции

Попробуйте использовать

<prop key="hibernate.dialect">org.hibernate.dialect.MySQLInnoDBDialect</prop>

, который создает таблицы InnoDB и, таким образом, поддерживает транзакции.

0 голосов
/ 14 мая 2010

Я надеюсь, что я прав, и что это просто. Вам не хватает аннотации @Transactional в вашем тестовом классе. Это означает, что сам метод теста не выполняется в транзакции, и поэтому откатывать нечего. Надеюсь, это поможет.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...