Как использовать Atomicos Transaction Essentials с Hibernate >= 4.3
Я переключился с Hibernate 4.2 на Hibernate 4.3, и мой проект больше не работает. Я получаю
HibernateException: невозможно найти текущую транзакцию JTA
когда я делаю
Session s = sessionFactory.getCurrentSession();
Я понял, что org.hibernate.transaction.TransactionManagerLookup больше не существует. Он был удален в Hibernate 4.3. Как мне изменить текущую конфигурацию?
<hibernate-configuration>
<session-factory>
<property name="connection.datasource">testDS</property>
<property name="current_session_context_class">jta</property>
<property name="transaction.manager_lookup_class">com.atomikos.icatch.jta.hibernate3.TransactionManagerLookup</property>
<property name="transaction.factory_class">org.hibernate.transaction.CMTTransactionFactory</property>
<property name="connection.release_mode">auto</property>
<property name="cache.provider_class">org.hibernate.cache.NoCacheProvider</property>
<property name="hibernate.hbm2ddl.auto">create-drop</property>
<property name="hibernate.show_sql">true</property>
<mapping class="test.entities.Person" />
<mapping class="test.entities.CreditCard" />
<mapping class="test.entities.ExampleRevEntity" />
</session-factory>
Ответы
Ответ 1
В Hibernate 4.3 длинный устаревший TransactionManagerLookup удален. Теперь провайдер JTA должен реализовать org.hibernate.engine.transaction.jta.platform.spi.JtaPlatform. Абстрактная реализация платформы JTA уже доступна в Hibernate, а именно org.hibernate.engine.transaction.jta.platform.internal.AbstractJtaPlatform. Используя это, довольно просто написать JTA Platform для Atomikos:
package test;
import javax.transaction.TransactionManager;
import javax.transaction.UserTransaction;
import org.hibernate.engine.transaction.jta.platform.internal.AbstractJtaPlatform;
import com.atomikos.icatch.jta.UserTransactionManager;
public class AtomikosJtaPlatform extends AbstractJtaPlatform {
private static final long serialVersionUID = 1L;
private UserTransactionManager utm;
public AtomikosJtaPlatform() {
utm = new UserTransactionManager();
}
@Override
protected TransactionManager locateTransactionManager() {
return utm;
}
@Override
protected UserTransaction locateUserTransaction() {
return utm;
}
}
Кроме того, имя класса реализации платформы должно быть добавлено в конфигурацию спящего режима:
<property name="hibernate.transaction.jta.platform">test.AtomikosJtaPlatform</property>
Ответ 2
Чтобы использовать платформу JTA для Hibernate с Spring написать и скомпилировать этот код
package my.domain.spring.hibernate.jta;
import javax.transaction.TransactionManager;
import javax.transaction.UserTransaction;
import org.hibernate.engine.transaction.jta.platform.internal.AbstractJtaPlatform;
import org.springframework.transaction.jta.JtaTransactionManager;
import org.springframework.util.Assert;
@SuppressWarnings("serial")
public class SpringJtaPlatformAdapter extends AbstractJtaPlatform {
private static TransactionManager sTransactionManager;
private static UserTransaction sUserTransaction;
@Override
protected TransactionManager locateTransactionManager() {
Assert.notNull(sTransactionManager, "TransactionManager has not been setted");
return sTransactionManager;
}
@Override
protected UserTransaction locateUserTransaction() {
Assert.notNull(sUserTransaction, "UserTransaction has not been setted");
return sUserTransaction;
}
public void setJtaTransactionManager(JtaTransactionManager jtaTransactionManager) {
sTransactionManager = jtaTransactionManager.getTransactionManager();
sUserTransaction = jtaTransactionManager.getUserTransaction();
}
public void setTransactionManager(TransactionManager transactionManager) {
sTransactionManager = transactionManager;
}
public void setUserTransaction(UserTransaction userTransaction) {
sUserTransaction = userTransaction;
}
}
Добавьте в spring -configuration
<bean id="txObjcoreManager" class="org.springframework.transaction.jta.JtaTransactionManager">
<property name="transactionManager">
<bean class="com.atomikos.icatch.jta.UserTransactionManager" init-method="init" destroy-method="close">
<property name="forceShutdown" value="true" />
</bean>
</property>
<property name="userTransaction">
<bean class="com.atomikos.icatch.jta.UserTransactionImp">
<property name="transactionTimeout" value="300" />
</bean>
</property>
</bean>
<bean id="springJtaPlatformAdapter" class="my.domain.spring.hibernate.jta.SpringJtaPlatformAdapter">
<property name="jtaTransactionManager" ref="txObjcoreManager" />
</bean>
Не забудьте добавить зависимость
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean"
depends-on="springJtaPlatformAdapter">
И, наконец, измените конфигурацию спящего режима, такую как hibernate.transaction.jta.platform=my.domain.spring.hibernate.jta.SpringJtaPlatformAdapter
Ответ 3
Некоторые подсказки для пользователей Spring - просто используйте эту реализацию, если вы настроите материал с помощью factory bean:
public class AtomikosPlatform extends AbstractJtaPlatform {
private static final long serialVersionUID = -1L;
@Override
protected TransactionManager locateTransactionManager() {
return new J2eeTransactionManager();
}
@Override
protected UserTransaction locateUserTransaction() {
return new J2eeUserTransaction();
}
}
Ответ 4
Можете ли вы попробовать установить свойство jtaTransactionManager для org.springframework.orm.hibernate4.LocalSessionFactoryBean для Spring JtaTransactionManager? У меня есть аналогичная проблема, но она решена этим. Кстати, HibernateTemplate вернулся на Spring 4.0.1. Хотя это не рекомендуется, но я люблю его использовать. Это помогло позаботиться о многом. (Я использую Spring 4.0.5 + Hibernate 4.3.5 + Atomikos 3.9.3)
<bean id="atomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager" init-method="init" destroy-method="close" depends-on="userTransactionService">
<property name="forceShutdown" value="true" />
</bean>
<bean id="atomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp" depends-on="userTransactionService">
<property name="transactionTimeout" value="180" />
</bean>
<bean id="JtaTransactionManager" class="org.springframework.transaction.jta.JtaTransactionManager">
<property name="transactionManager" ref="atomikosTransactionManager" />
<property name="userTransaction" ref="atomikosUserTransaction" />
<property name="allowCustomIsolationLevels" value="true"></property>
</bean>
<bean id="rentSessionFactory" class="org.springframework.orm.hibernate4.LocalSessionFactoryBean">
<property name="dataSource"><ref bean="rentXADataSource" /></property>
<property name="mappingLocations" value="classpath:com/kj/model/web/*.hbm.xml"/>
<property name="hibernateProperties">
<props>
<prop key="hibernate.dialect">org.hibernate.dialect.SQLServerDialect</prop>
<prop key="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</prop>
<prop key="hibernate.show_sql">false</prop>
<prop key="hibernate.transaction.factory_class">org.hibernate.transaction.CMTTransactionFactory</prop>
<prop key="hibernate.cache.use_query_cache">true</prop>
<prop key="hibernate.cache.use_second_level_cache">true</prop>
<prop key="hibernate.format_sql">false</prop>
<prop key="hibernate.bytecode.use_reflection_optimizer">true</prop>
</props>
</property>
<property name="jtaTransactionManager" ref="JtaTransactionManager"></property>
</bean>
Ответ 5
Ниже приведен альтернативный подход, который работает с конфигурацией Spring. Это отличается от подхода Антона, то есть не полагается на метод экземпляра, который записывает в статическое поле (которое обычно считается плохой практикой).
@SuppressWarnings("serial")
public class AtomikosJtaPlatform extends AbstractJtaPlatform {
private static TransactionManager transactionManager;
private static UserTransaction userTransaction;
public static void factoryInit(TransactionManager transactionManager, UserTransaction userTransaction) {
AtomikosJtaPlatform.transactionManager = transactionManager;
AtomikosJtaPlatform.userTransaction = userTransaction;
}
@Override
protected TransactionManager locateTransactionManager() {
return transactionManager;
}
@Override
protected UserTransaction locateUserTransaction() {
return userTransaction;
}
}
Затем в конфигурации Spring:
<bean id="AtomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager" init-method="init"
destroy-method="close">
<property name="startupTransactionService" value="false" />
<property name="forceShutdown" value="false" />
</bean>
<bean id="AtomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp">
<property name="transactionTimeout" value="300" />
</bean>
<!-- AtomikosJtaPlatform is created by Hibernate using reflection. This ensures it uses our Spring configured beans -->
<bean id="JtaPlatformInitializer" class="org.springframework.beans.factory.config.MethodInvokingBean">
<property name="targetClass" value="com.mycompany.a.b.AtomikosJtaPlatform" />
<property name="targetMethod" value="factoryInit" />
<property name="arguments">
<list>
<ref bean="AtomikosTransactionManager" />
<ref bean="AtomikosUserTransaction" />
</list>
</property>
</bean>
<bean id="emf" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean" depends-on="JtaPlatformInitializer">
<property name="jpaProperties">
<props>
<prop key="hibernate.transaction.jta.platform">com.mycompnay.a.b.AtomikosJtaPlatform</prop>
...