org.hibernate.HibernateException: TransactionManagerLookup не указан при обновлении спящего режима

Я использую Spring 4.1.4, Hibernate 4.3.8, Atomikos 3.9.3, Java 8, Tomcat 8.

Я вижу указанное выше исключение в localhost.log при запуске своего сервера, но я не уверен, где настроить TransactionManagerLookup, кроме того места, где я его уже настраиваю. Этого не происходило до обновления спящего режима, поэтому, скорее всего, это проблема версии. Кто-нибудь может помочь, пожалуйста?

К вашему сведению: catalina.out не показывает ничего полезного. Только что:

SEVERE [localhost-startStop-1] org.apache.catalina.core.StandardContext.startInternal Error listenerStart
SEVERE [localhost-startStop-1] org.apache.catalina.core.StandardContext.startInternal Context [] startup failed due to previous errors

Мой appContext определяет:

<prop key="hibernate.transaction.factory_class">org.hibernate.engine.transaction.internal.jta.CMTTransactionFactory</prop>
<prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>

и полная трассировка стека:

22-Jan-2015 10:07:25.734 SEVERE [localhost-startStop-1] org.apache.catalina.core.StandardContext.listenerStart Exception sending context initialized event to listener instance of class com.my.app.web.InitializerListener
 org.hibernate.HibernateException: No TransactionManagerLookup specified
    at org.hibernate.context.internal.JTASessionContext.currentSession(JTASessionContext.java:85)
    at org.hibernate.internal.SessionFactoryImpl.getCurrentSession(SessionFactoryImpl.java:1014)
    at com.my.app.dao.AbstractMyDAO.currentSession(AbstractMyDAO.java:116)
    at com.my.app.dao.AbstractMyDAO.criteria(AbstractMyDAO.java:86)
    at com.my.app.dao.AbstractMyDAO.count(AbstractMyDAO.java:79)
    at com.my.app.initialize.MyInitializerImpl.initializeApplicaiton(MyInitializerImpl.java:69)
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
    at java.lang.reflect.Method.invoke(Method.java:483)
    at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:317)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.invokeJoinpoint(ReflectiveMethodInvocation.java:190)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:157)
    at org.springframework.transaction.interceptor.TransactionInterceptor$1.proceedWithInvocation(TransactionInterceptor.java:99)
    at org.springframework.transaction.interceptor.TransactionAspectSupport.invokeWithinTransaction(TransactionAspectSupport.java:281)
    at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:96)
    at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:179)
    at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:207)
    at com.sun.proxy.$Proxy220.initializeApplicaiton(Unknown Source)
    at com.my.app.web.MyInitializerListener.contextInitialized(MyInitializerListener.java:42)
    at org.apache.catalina.core.StandardContext.listenerStart(StandardContext.java:4772)
    at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5196)
    at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150)
    at org.apache.catalina.core.ContainerBase.addChildInternal(ContainerBase.java:725)
    at org.apache.catalina.core.ContainerBase.addChild(ContainerBase.java:701)
    at org.apache.catalina.core.StandardHost.addChild(StandardHost.java:714)
    at org.apache.catalina.startup.HostConfig.deployWAR(HostConfig.java:917)
    at org.apache.catalina.startup.HostConfig$DeployWar.run(HostConfig.java:1701)
    at java.util.concurrent.Executors$RunnableAdapter.call(Executors.java:511)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    at java.lang.Thread.run(Thread.java:745)

РЕДАКТИРОВАТЬ: менеджер транзакций Atomikos настроен как таковой

<bean id="atomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager" init-method="init" destroy-method="close">
    <property name="forceShutdown" value="true" />
    <!--  in secs -->
    <property name="transactionTimeout" value="300"/>
</bean>
<bean id="atomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp">
    <!--  in secs -->
    <property name="transactionTimeout" value="300" />
</bean>
<bean id="transactionManager" class="org.springframework.transaction.jta.JtaTransactionManager">
    <property name="transactionManager" ref="atomikosTransactionManager" />
    <property name="userTransaction" ref="atomikosUserTransaction" />
    <property name="transactionSynchronizationName" value="SYNCHRONIZATION_ON_ACTUAL_TRANSACTION" />
    <property name="allowCustomIsolationLevels" value="true"/>
</bean>

РЕДАКТИРОВАТЬ 2:

Думаю, здесь нужно немного прояснить ситуацию. У меня есть объект DAO с именем AbstractMyDAO (как видно из стека). В этом объекте определена фабрика сеансов

@Autowired
private SessionFactory mySessionFactory;

когда вызывается currentSession, выдается вышеуказанная ошибка, потому что hibernate не может найти поиск диспетчера транзакций, связанный с этой фабрикой сеансов.

public Session currentSession() {
    return mySessionFactory.getCurrentSession();
}

Эта фабрика сеансов определена в моем файле контекста приложения как таковая:

<bean id="mySessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">

    <property name="dataSource">
        <ref bean="myDataSource" />
    </property>
    <property name="annotatedClasses">
        <list>
            <value>some values... </value>
            ...
        </list>
    </property>
    <property name="hibernateProperties">
        <props>
            <prop key="hibernate.dialect">
                ${gst.hibernate.dialect}
            </prop>
            <prop key="query.factory_class">
                org.hibernate.hql.ast.ASTQueryTranslatorFactory
            </prop>
            <prop key="hibernate.max_fetch_depth">3</prop>
            <prop key="hibernate.show_sql">${my.hibernate.showsql}</prop>
            <prop key="hibernate.cache.use_query_cache">true</prop>
            <prop key="hibernate.cache.use_second_level_cache">true</prop>
            <prop key="hibernate.cache.use_structured_entries">true</prop>
            <prop key="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.SingletonEhCacheRegionFactory</prop>
            <prop key="net.sf.ehcache.configurationResourceName">${ehcache.my.persist.config}</prop>
            <prop key="hibernate.connection.isolation">3</prop>
            <prop key="connection.release_mode">auto</prop>
            <prop key="hibernate.current_session_context_class">jta</prop>
            <prop key="hibernate.transaction.factory_class">org.hibernate.engine.transaction.internal.jta.CMTTransactionFactory</prop>
            <prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>
            <prop key="hibernate.search.default.directory_provider">filesystem</prop>
            <prop key="hibernate.search.default.indexBase">/var/log/my/lucene/indexes</prop>
        </props>
    </property>
</bean>

Как видите, свойство hibernate.transaction.manager_lookup_class определяется, но не может быть найдено при создании компонента, и я понятия не имею, почему. Конфигурация изменилась?

РЕДАКТИРОВАТЬ 3: Когда я отлаживаю: final JtaPlatform jtaPlatform = factory().getServiceRegistry().getService(JtaPlatform.class);

Он возвращает NoJTAPlatform. Я думаю, это моя проблема.

РЕДАКТИРОВАТЬ 4: Кажется, нет ничего, что реализовало бы JtaPlatform, подходящую для tomcat, или я не умею?


person Treasa McCabe    schedule 22.01.2015    source источник


Ответы (3)


Убедитесь, что вы также настроили диспетчер транзакций Atomikos, чтобы TransactionManagerLookup мог найти UserTransaction соответственно:

<bean id="atomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager"
    init-method="init" destroy-method="close">
    <property name="transactionTimeout" value="300" />
</bean>
<bean id="atomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp">
    <property name="transactionTimeout" value="300" />
</bean>

<bean id="transactionManager"
    class="org.springframework.transaction.jta.JtaTransactionManager">
    <property name="transactionManager" ref="atomikosTransactionManager" />
    <property name="userTransaction" ref="atomikosUserTransaction" />
</bean>

Затем вы также должны использовать фабрику Spring EntityManager:

<bean id="entityManager"
    class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean"
    p:dataSource-ref="dataSource"
    p:persistenceXmlLocation="**/persistence.xml"
    p:persistenceUnitName="persistenceUnit" 
    depends-on="dataSource, transactionManager">
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
        </bean>
    </property>
    <property name="jpaProperties">
        <props>                   
            <prop key="hibernate.transaction.factory_class">org.hibernate.engine.transaction.internal.jta.CMTTransactionFactory</prop>                     
            <prop key="hibernate.transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</prop>
            ...
        </props>
    </property>
</bean>

Затем DAO должны внедрить EntityManager:

@PersistenceContext(unitName = "persistenceUnit")
private EntityManager entityManager;

Вместо прямого вызова currentSession(AbstractMyDAO):

at com.my.app.dao.AbstractMyDAO.currentSession(AbstractMyDAO.java:116)
person Vlad Mihalcea    schedule 22.01.2015
comment
Спасибо, у меня уже настроен диспетчер транзакций атомикоса (см. редактирование), но не диспетчер сущностей. Попробую и отчитаюсь :) - person Treasa McCabe; 22.01.2015
comment
Не могли бы вы объяснить мне, почему мне нужно использовать диспетчер сущностей? Мое приложение раньше работало без этого компонента. Проблема возникла только после перехода в спящий режим. Спасибо. - person Treasa McCabe; 23.01.2015
comment
Это связано с тем, что Hibernate перешел к реализации JPA, а не к использованию в качестве автономной инфраструктуры ORM. Вы по-прежнему можете использовать SessionFactory, но лучше использовать Spring LocalSessionFactoryBean, потому что вы также получаете автоматическое управление сеансом и соединением JDBC. - person Vlad Mihalcea; 23.01.2015
comment
Итак, предположим, я хочу по-прежнему использовать SessionFactory, а не LocalSessionFactoryBean. Вы хоть понимаете, что я сейчас делаю не так? - person Treasa McCabe; 23.01.2015
comment
У меня нет другой идеи. - person Vlad Mihalcea; 23.01.2015

Ответ на мой вопрос здесь.

«В Hibernate 4.3 был удален давно устаревший TransactionManagerLookup. Теперь поставщик JTA должен реализовать org.hibernate.engine.transaction.jta.platform.spi.JtaPlatform».

person Treasa McCabe    schedule 26.01.2015

Я решил это

в файл hibernate.cfg.xml

изменение

<property name="hibernate.current_session_context_class">thread</property>

на thread вместо jta .

Подробнее здесь

person Pipo    schedule 09.07.2018