JTA EntityManager не может использовать getTransaction() [Spring + Hibernate + EntityManager]
Я использую Spring + JPA + Hibernate + EntityManager для общения с базой данных. Я получаю сообщение об ошибке "JTA EntityManager не может использовать getTransaction()". Пожалуйста, предоставьте ваши идеи и помогите мне решить проблему.
beans.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans default-autowire="byName"
... xmlns definitions...
xsi:schemaLocation="...">
<context:component-scan base-package="com.mycompany.myproject" />
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean" />
<bean class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" />
<bean id="myDAO" class="com.mycompany.myproject.dao.myDAO" />
<bean id="transactionManager" class="org.springframework.orm.jpa.JpaTransactionManager">
<property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>
<tx:annotation-driven />
</beans>
persistence.xml
<?xml version="1.0" encoding="UTF-8"?>
<persistence ... xmlns definitions xsi:schemaLocation="..." version="1.0">
<persistence-unit name="TEST_DS">
<provider>org.hibernate.ejb.HibernatePersistence</provider>
<jta-data-source>java:/TEST_DS</jta-data-source>
<class>com.twinspires.exchange.model.Test</class>
<properties>
<property name="hibernate.archive.autodetection" value="class, hbm" />
<property name="hibernate.dialect" value="org.hibernate.dialect.MySQLDialect" />
<property name="hibernate.hbm2ddl.auto" value="validate" /> <!-- create-drop update -->
<property name="hibernate.cache.use_query_cache" value="true" />
<property name="hibernate.show_sql" value="true" />
<property name="hibernate.cache.provider_class" value="org.hibernate.cache.HashtableCacheProvider" />
<property name="hibernate.transaction.manager_lookup_class" value="org.hibernate.transaction.JBossTransactionManagerLookup"/>
</properties>
</persistence-unit>
</persistence>
Трассировка стека исключений (выдержка)
15:47:43,340 INFO [STDOUT] DEBUG: org.springframework.transaction.annotation.AnnotationTransactionAttributeSource - Adding transactional method 'getName' with attribute: PROPAGATION_REQUIRED,ISOLATION_DEFAULT; ''
15:47:43,343 INFO [STDOUT] DEBUG: org.springframework.beans.factory.support.DefaultListableBeanFactory - Returning cached instance of singleton bean 'transactionManager'
15:47:43,356 INFO [STDOUT] DEBUG: org.springframework.orm.jpa.JpaTransactionManager - Creating new transaction with name [com.twinspires.exchange.dao.PicDAO.getName]: PROPAGATION_REQUIRED,ISOLATION_DEFAULT; ''
15:47:44,114 INFO [STDOUT] DEBUG: org.springframework.orm.jpa.JpaTransactionManager - Opened new EntityManager [org.hibernate.ejb.EntityManagerImpl@c629e5] for JPA transaction
15:47:44,124 INFO [STDOUT] DEBUG: org.springframework.orm.jpa.JpaTransactionManager - Could not rollback EntityManager after failed transaction begin
15:47:44,125 INFO [STDOUT] java.lang.IllegalStateException: A JTA EntityManager cannot use getTransaction()
15:47:44,125 INFO [STDOUT] at org.hibernate.ejb.AbstractEntityManagerImpl.getTransaction(AbstractEntityManagerImpl.java:818)
15:47:44,126 INFO [STDOUT] at org.springframework.orm.jpa.JpaTransactionManager.closeEntityManagerAfterFailedBegin(JpaTransactionManager.java:412)
15:47:44,127 INFO [STDOUT] at org.springframework.orm.jpa.JpaTransactionManager.doBegin(JpaTransactionManager.java:381)
15:47:44,128 INFO [STDOUT] at org.springframework.transaction.support.AbstractPlatformTransactionManager.getTransaction(AbstractPlatformTransactionManager.java:371)
15:47:44,129 INFO [STDOUT] at org.springframework.transaction.interceptor.TransactionAspectSupport.createTransactionIfNecessary(TransactionAspectSupport.java:335)
15:47:44,129 INFO [STDOUT] at org.springframework.transaction.interceptor.TransactionInterceptor.invoke(TransactionInterceptor.java:105)
15:47:44,130 INFO [STDOUT] at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172)
15:47:44,131 INFO [STDOUT] at org.springframework.aop.framework.JdkDynamicAopProxy.invoke(JdkDynamicAopProxy.java:202)
15:47:44,131 INFO [STDOUT] at $Proxy175.getName(Unknown Source)
Folowing мой класс дао:-
public class MyDao implements IMyDao {
@PersistenceContext(unitName = "TEST_DS")
private EntityManager entityManager;
@Transactional
public String getName() {
final Query query = entityManager.createQuery("from TestTable");
final Object obj = query.getResultList().get(0);
return obj == null ? "Empty" : (String) obj;
}
}
Ваша помощь высоко ценится.
2 ответа
Удалите источник данных jta из файла persitence.xml, настройте источник данных как bean-компонент j2ee: jdni-lookup и вставьте его в LocalContainerEntityManagerFactoryBean.
Удалить это из persistence.xml
<jta-data-source>java:/TEST_DS</jta-data-source>
Настройте тип транзакции для локального ресурса
<persistence-unit name="TEST_DS" transaction-type="RESOURCE_LOCAL">
Перейдите на ваш beans.xml
<j2ee:jndi-lookup id="dataSource" jndi-name="java:/TEST_DS"/>
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
<property name="dataSource" ref="dataSource"/>
</bean>
Прежде всего, я хочу сказать вам, что у меня была такая же проблема. Узнав о том, что я нашел эту страницу с вашим постом и ответом "gkamal".
Я думаю, что ответ приходит к тому, что если вы пытаетесь использовать "глобальную" транзакцию JTA в своем приложении, и она не работает, то не используйте ее, используйте вместо нее "локальную" транзакцию JDBC.
Но если вам нужно использовать глобальную транзакцию JTA, вы должны использовать ее.
Ну, я собираюсь дать вам решение, которое я нашел в моем исследовании.
Есть свойства, которые зависят от сервера приложений. Я использую Glassfish, и это решение отлично работает, но я думаю, что вы используете JBOSS (из-за значения, которое вы используете в jta-data-source в вашем файле persistence.xml), я напишу значения для JBOSS AS, но в JBOSS Поскольку я не докажу это, я просто докажу это только в стеклянной рыбе.
В вашем файле persistence.xml вы должны указать другое свойство:
<property name="hibernate.transaction.jta.platform" value="org.hibernate.service.jta.platform.internal.JBossStandAloneJtaPlatform"/>
Значение этого свойства зависит от сервера AP.
В вашем файле beans.xml вы должны взглянуть на фабрику менеджера сущностей и менеджер транзакций. Более того, вы должны указать в своем web.xml, какие постоянные единицы вы используете в своем приложении.
beans.xml
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
<property name="persistenceUnitName" value="TEST_DS"/>
</bean>
В вашем менеджере транзакций вы используете фабрику менеджера сущностей, но вам не нужно этого делать. Вы должны указать менеджер транзакций и пользовательскую транзакцию, используемую вашим сервером приложений, таким образом
<bean id="txManager" class="org.springframework.transaction.jta.JtaTransactionManager">
<property name="transactionManagerName" value="java:appserver/TransactionManager"/>
<property name="userTransactionName" value="java:comp/UserTransaction"/>
</bean>
Значения этих свойств зависят от сервера приложений (я использую Glassfish). Я думаю, что вы должны использовать значения для Jboss AP:
<property name="transactionManagerName" value="java:/TransactionManager"/>
<property name="userTransactionName" value="UserTransaction"/>
Но я этого не доказал.
Наконец, в glassfish мне нужно поместить в beans.xml (и я не знаю почему) следующий bean-компонент (я думаю, что в jboss ap это не обязательно, но вы можете доказать это)
<bean class="org.springframework.orm.jpa.support.PersistenceAnnotationBeanPostProcessor" >
<property name="persistenceUnits">
<map>
<entry key="TEST_DS" value="persistence/TEST_DS"/>
</map>
</property>
</bean>
В вашем файле web.xml необходимо указать единицы измерения сохраняемости, которые будет использовать ваше приложение.
<persistence-unit-ref>
<persistence-unit-ref-name>persistence/TEST_DS</persistence-unit-ref-name>
<persistence-unit-name>TEST_DS</persistence-unit-name>
</persistence-unit-ref>
Я надеюсь, что решение поможет вам. Меня устраивает.