Правильный способ автоматического подключения Hibernate-сессии в тесте Spring Transaction JUnit - PullRequest
11 голосов
/ 01 октября 2009

Этот вопрос похож на предыдущий один . Я пытаюсь @Autowire сеанс Hibernate в одном из моих тестов Spring-JUnit-Transactional, но получаю следующее исключение:

java.lang.IllegalStateException: No Hibernate Session bound to thread, and configuration does not allow creation of non-transactional ...

Вот мой класс JUnit:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"/applicationContext.xml"})
@TransactionConfiguration(transactionManager="transactionManager")
@Transactional
public class MyTest {
    @Qualifier("session")
    @Autowired
    private Session session;

    @Test
    public void testSomething() {
        session.get(User.class, "me@here.com");
    }
}

Каждый работает нормально, если я @Autowire a SessionFactory и получаю мой Session программно (вместо определения его в Spring XML) примерно так:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations={"/applicationContext.xml"})
@TransactionConfiguration(transactionManager="transactionManager")
@Transactional
public class MyTest{    
    @Qualifier("sessionFactory")
    @Autowired
    private SessionFactory sessionFactory;

    @Test
    public void testSomething() {
    Session session = SessionFactoryUtils.getSession(sessionFactory, false);
        session.get(User.class, "me@here.com");
    }
}

Тем не менее, я могу заставить мой оригинальный пример работать, если я определю свой Session в своем Spring XML с помощью <aop:scoped-proxy /> следующим образом:

<?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:tx="http://www.springframework.org/schema/tx"
        xmlns:aop="http://www.springframework.org/schema/aop"
        xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-2.5.xsd
        http://www.springframework.org/schema/tx
        http://www.springframework.org/schema/tx/spring-tx-2.5.xsd
        http://www.springframework.org/schema/aop 
        http://www.springframework.org/schema/aop/spring-aop-2.5.xsd
        ">

    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" destroy-method="close">
        ...
    </bean>

    <bean id="sessionFactory" class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
        <property name="dataSource" ref="dataSource" />
        <property name="configLocation"><value>classpath:/hibernate.cfg.xml</value></property>
        <property name="configurationClass">
            <value>org.hibernate.cfg.AnnotationConfiguration</value>
        </property>
    </bean>

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

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

    <bean id="session" class="org.springframework.orm.hibernate3.SessionFactoryUtils" factory-method="getSession" scope="prototype">
        <constructor-arg ref="sessionFactory" />
        <constructor-arg value="false" />
        <!-- This is seems to be needed to get rid of the 'No Hibernate Session' error' -->
        <aop:scoped-proxy />
    </bean>
</beans>

У меня такой вопрос: почему <aop:scoped-proxy /> необходим, учитывая, что в моем модульном тесте должен быть только один контекст транзакции, ограниченный потоками? Что является правильным способом определения моего Hibernate Session бина?

Ответы [ 2 ]

6 голосов
/ 01 октября 2009

SessionFactoryUtils.getSession () так же хорош, как и любой другой способ получения Session. Он делает то же, что и HibernateDaoSupport.getSession ().

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

Добавляя прокси-сервер scoped, он проксирует Сеанс и внедряет его, чтобы не вводить фактический сеанс заранее (до начала транзакции), а только извлекает его и выполняет вызовы после запуска теста, когда это действительно необходимо позвонить против него.

4 голосов
/ 01 октября 2009

Я думаю, что "правильный" способ - это внедрение SessionFactory и программное извлечение сеанса из него. Причина, по которой вы получаете исключение, заключается в документированном поведении SessionFactoryUtils.getSession():

Получить Hibernate Session для данного SessionFactory. В курсе и будет вернуть любой существующий соответствующий Сессия привязана к текущему потоку, например при использовании HibernateTransactionManager. Будет в противном случае создайте новую сессию «allowCreate» имеет значение true.

Поскольку с текущей транзакцией ничто не связывает сеанс, происходит сбой.

Я бы предложил использовать HibernateTemplate - определите один из них в своем контексте и включите его в свой тест. HibernateTemplate выполняет большинство операций, аналогичных сеансу войны, но выполняет бит обработки сеанса за вас. Вы должны просто быть в состоянии сделать:

hibernateTemplate.get(User.class, "me@here.com");
...