Camel + JBoss + JPA:TransactionRequiredException:尽管有JTA数据源

时间:2015-10-28 20:36:42

标签: jpa jboss transactions apache-camel eclipselink

我有一个基于Apache Camel的应用程序,它运行在JBoss AS 7.1应用程序服务器上。

在一个Camel路由中,我让Camel读取XML文件,使用JAXB将它们解组为JPA实体对象,并使用Camel jpa://组件将它们写入数据库。

但是,每当将实体发送到JPA端点时,Camel都会给我以下TransactionRequiredException:

javax.persistence.TransactionRequiredException: joinTransaction has been called on a resource-local EntityManager which is unable to register for a JTA transaction.
    at org.eclipse.persistence.internal.jpa.transaction.EntityTransactionWrapper.registerIfRequired(EntityTransactionWrapper.java:91)
    at org.eclipse.persistence.internal.jpa.EntityManagerImpl.joinTransaction(EntityManagerImpl.java:2092)
    at org.springframework.orm.jpa.ExtendedEntityManagerCreator$ExtendedEntityManagerInvocationHandler.doJoinTransaction(ExtendedEntityManagerCreator.java:360)
    at org.springframework.orm.jpa.ExtendedEntityManagerCreator$ExtendedEntityManagerInvocationHandler.invoke(ExtendedEntityManagerCreator.java:327)
    at com.sun.proxy.$Proxy48.joinTransaction(Unknown Source)
    at org.apache.camel.component.jpa.JpaProducer$1.doInTransaction(JpaProducer.java:64)
    at org.springframework.transaction.support.TransactionTemplate.execute(TransactionTemplate.java:131)
    at org.apache.camel.component.jpa.JpaProducer.process(JpaProducer.java:61)
    at org.apache.camel.util.AsyncProcessorConverterHelper$ProcessorToAsyncProcessorBridge.process(AsyncProcessorConverterHelper.java:61)
    at org.apache.camel.processor.SendProcessor$2.doInAsyncProducer(SendProcessor.java:152)
    at org.apache.camel.impl.ProducerCache.doInAsyncProducer(ProducerCache.java:304)
    at org.apache.camel.processor.SendProcessor.process(SendProcessor.java:147)
    at org.apache.camel.management.InstrumentationProcessor.process(InstrumentationProcessor.java:72)
    at org.apache.camel.processor.RedeliveryErrorHandler.process(RedeliveryErrorHandler.java:424)
    at org.apache.camel.processor.CamelInternalProcessor.process(CamelInternalProcessor.java:191)
    at org.apache.camel.processor.Pipeline.process(Pipeline.java:118)
    at org.apache.camel.processor.Pipeline.process(Pipeline.java:80)
    at org.apache.camel.processor.CamelInternalProcessor.process(CamelInternalProcessor.java:191)
    at org.apache.camel.component.direct.DirectProducer.process(DirectProducer.java:51)
    at org.apache.camel.processor.SendProcessor.process(SendProcessor.java:120)
    at org.apache.camel.management.InstrumentationProcessor.process(InstrumentationProcessor.java:72)
    at org.apache.camel.processor.CamelInternalProcessor.process(CamelInternalProcessor.java:191)
    at org.apache.camel.processor.RedeliveryErrorHandler.process(RedeliveryErrorHandler.java:424)
    at org.apache.camel.processor.CamelInternalProcessor.process(CamelInternalProcessor.java:191)
    at org.apache.camel.util.AsyncProcessorHelper.process(AsyncProcessorHelper.java:105)
    at org.apache.camel.processor.MulticastProcessor.doProcessParallel(MulticastProcessor.java:735)
    at org.apache.camel.processor.MulticastProcessor.access$200(MulticastProcessor.java:82)
    at org.apache.camel.processor.MulticastProcessor$1.call(MulticastProcessor.java:303)
    at org.apache.camel.processor.MulticastProcessor$1.call(MulticastProcessor.java:288)
    at java.util.concurrent.FutureTask.run(FutureTask.java:262)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1145)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:615)
    at java.lang.Thread.run(Thread.java:745)

我不明白的是,JPA坚持认为我的EntityManager是资源本地的。在我的persistence.xml中,我将数据源定义为JTA:

的persistence.xml:

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="2.0" xmlns="http://java.sun.com/xml/ns/persistence" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd">
  <persistence-unit name="foobar">
    <provider>org.eclipse.persistence.jpa.PersistenceProvider</provider>
    <jta-data-source>java:jboss/datasources/myDS</jta-data-source>
    <mapping-file>META-INF/eclipselink-orm.xml</mapping-file>
    <!-- ... lots of entity classes omitted ... -->
    <properties>
      <property name="eclipselink.weaving" value="static"/>
      <property name="eclipselink.ddl-generation" value="create-or-extend-tables"/>
      <property name="eclipselink.ddl-generation.output-mode" value="database"/>
      <property name="eclipselink.logging.parameters" value="true"/>
    </properties>
  </persistence-unit>
</persistence>

我的applicationContext.xml中的相关部分是:

<!-- snip -->

        <import resource="classpath:/datasource.xml"/>

        <tx:annotation-driven />

        <bean id="jpa" class="org.apache.camel.component.jpa.JpaComponent">
            <property name="entityManagerFactory" ref="entityManagerFactory"/>
            <property name="transactionManager" ref="transactionManager"/>
        </bean>

        <camelContext xmlns="http://camel.apache.org/schema/spring" id="foobar-context" useMDCLogging="true">

            <!-- ... most routes omitted ... --> 

            <camel:route id="readXML">
                <camel:from uri="file://C:/Data/xmldir"/>               
                <camel:to uri="direct:persistProduct"/>
            </camel:route>

            <camel:route id="persistProduct">
                <camel:from uri="direct:persistProduct"/>
                <camel:unmarshal ref="tnxJAXB"/>
                <camel:to uri="jpa:com.foo.bar.Product"/>
            </camel:route>

            <!-- ... a lot of other unrelated stuff omitted ... -->

        </camelContext>

<!-- snip --> 

为了测试性,数据源的内容在另一个文件中:

datasource.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:context="http://www.springframework.org/schema/context" 
    xmlns:p="http://www.springframework.org/schema/p"
    xmlns:mvc="http://www.springframework.org/schema/mvc"
    xmlns:jee="http://www.springframework.org/schema/jee"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xsi:schemaLocation="
       http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
       http://camel.apache.org/schema/spring http://camel.apache.org/schema/spring/camel-spring.xsd
       http://www.springframework.org/schema/jee http://www.springframework.org/schema/jee/spring-jee-2.0.xsd
       http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.0.xsd
       http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.2.xsd
    ">

    <jee:jndi-lookup id="myDS" jndi-name="java:jboss/datasources/myDS" />

    <bean id="transactionManager" class="org.springframework.transaction.jta.JtaTransactionManager"/>

    <bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
        <property name="persistenceUnitName" value="foobar"/>
    </bean>

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

</beans>

如果我的在线研究是彻底的,那么这一切都应该是正确的,交易管理应该有效。 但事实并非如此。 我做错了什么?

我正在使用的软件包版本:

  • Camel 2.14.3
  • Spring 4.1.6.RELEASE
  • Eclipselink 2.6.0-M3
  • JBoss AS 7.1.1-Final

先感谢大家!

1 个答案:

答案 0 :(得分:0)

当您不启动事务时,通常会发生此异常。记住,事务需要在服务级别而不是在存储库级别启动。由于服务和存储库代码不可用,因此无法提供更多帮助。