OpenJPA是否适用于Glassfish?

时间:2011-07-17 04:44:18

标签: jpa glassfish ejb openjpa

是否有人成功使用OpenJPA和Glassfish?

我正在尝试将OpenJPA 2.1与Glassfish 3.1 Open Source一起使用。我按照说明将两者整合在一起 - > http://weblogs.java.net/blog/ss141213/archive/2006/07/using_openjpa_a.html

我在Eclipse Indigo中有一个非常简单的EJB项目,其中包含以下内容:

  • com.rares.test.Person - @Entity
  • com.rares.test.PersonManager - interface
  • com.rares.test.PersonDao - @Stateless

但是,当我尝试部署时,我的Person @Entity上会出现ClassNotFoundException。投诉似乎是在我的PersonDao中实现的创建方法的Person parm(参见下面的所有代码)。

我在persistence.xml中没有指定提供程序就尝试了同一个项目,项目工作正常(能够将Person @Entity持久保存到MySql中的PERSON表)。如果我没有指定提供者,我想我正在使用EclipseLink(如果我错了,请纠正我)。这让我相信我没有正确配置OpenJPA和Glassfish。

堆栈跟踪

Caused by: java.lang.IllegalArgumentException: java.lang.ClassNotFoundException: com.rares.test.Person
    at serp.util.Strings.toClass(Strings.java:164)
    at serp.util.Strings.toClass(Strings.java:108)
    at serp.bytecode.BCClass.getType(BCClass.java:566)
    at org.apache.openjpa.enhance.PCEnhancer.<init>(PCEnhancer.java:283)
    at org.apache.openjpa.enhance.PCEnhancer.<init>(PCEnhancer.java:254)
    at org.apache.openjpa.enhance.PCClassFileTransformer.transform0(PCClassFileTransformer.java:144)
    at org.apache.openjpa.enhance.PCClassFileTransformer.transform(PCClassFileTransformer.java:124)
    at org.apache.openjpa.persistence.PersistenceProviderImpl$ClassTransformerImpl.transform(PersistenceProviderImpl.java:294)
    at org.glassfish.persistence.jpa.ServerProviderContainerContractInfo$1.transform(ServerProviderContainerContractInfo.java:98)
    at com.sun.enterprise.loader.ASURLClassLoader.findClass(ASURLClassLoader.java:742)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:306)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:247)
    at java.lang.Class.getDeclaredMethods0(Native Method)
    at java.lang.Class.privateGetDeclaredMethods(Class.java:2427)
    at java.lang.Class.privateGetPublicMethods(Class.java:2547)
    at java.lang.Class.getMethods(Class.java:1410)
    at com.sun.enterprise.deployment.EjbDescriptor.addAllInterfaceMethodsIn(EjbDescriptor.java:2327)
    at com.sun.enterprise.deployment.EjbDescriptor.getLocalRemoteBusinessMethodDescriptors(EjbDescriptor.java:2290)
    ... 40 more

com.rares.test.Person

@Entity
@Table (name="PERSON")
public class Person implements Serializable {
    private static final long serialVersionUID = 3707476467775531463L;

    @Id
    @GeneratedValue (strategy=GenerationType.IDENTITY)
    @Column private Long id;
    @Column private String name;

com.rares.test.PersonManager

public interface PersonManager {
    void create (com.rares.test.Person p);
}

com.rares.test.PersonDao

@Stateless
public class PersonDao implements PersonManager {
    @PersistenceContext (unitName="RarePersistUnit")
    protected EntityManager mgr;

    @Override
    public void create(com.rares.test.Person p) {
        mgr.persist(p); 
    }

}

的persistence.xml

<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="RarePersistUnit">
        <provider>org.apache.openjpa.persistence.PersistenceProviderImpl</provider>
        <jta-data-source>jdbc/RaresMySql</jta-data-source>
        <class>com.rares.test.Person</class>        
        <properties>

        </properties>
    </persistence-unit>
</persistence>

1 个答案:

答案 0 :(得分:12)

注意:下面假设 OpenJPA 2.1 Glassfish 3.1 。两种版本的结果可能不同。

抛出的异常是OpenJPA在运行时无法执行bytecode enchancement的典型情况。字节码增强可以在构建时或运行时完成。获得运行时支持的一个更好的选择是使用javaagent但需要一些怪异的配置:

  • 它要求您在Glassfish域配置文件中指定javaagent(通过在jvm-options元素下指定其他java-config元素),
  • 并且还将类路径从默认值${com.sun.aas.installRoot}/modules/glassfish.jar修改为包含commons-lang-2.4.jar(我没有这么做,因为它会导致非常脆弱的设置)。

可以在运行时使用的其他选项非常不稳定,使用Serp作为字节码增强器的选择是部署时抛出异常的原因。显然,由于使用了不正确的类加载器来定位类,部署时间增强无法找到持久化上下文中使用的类。在我的例子中,Glassfish EarClassLoaderEarLibClassLoader类加载器用于在两个单独的调用中加载类,两个调用都失败并显示以下消息(堆栈跟踪在这里是无关紧要的):

WARNING: LDR5207: ASURLClassLoader EarLibClassLoader : 
doneCalled = true
doneSnapshot = ASURLClassLoader.done() called ON EarLibClassLoader : 
urlSet = []
doneCalled = false 
 Parent -> org.glassfish.internal.api.DelegatingClassLoader@10e3c8c
 AT Sun Jul 17 13:27:54 IST 2011 
 BY :java.lang.Throwable: printStackTraceToString
    at com.sun.enterprise.util.Print.printStackTraceToString(Print.java:639)
    at com.sun.enterprise.loader.ASURLClassLoader.done(ASURLClassLoader.java:211)
    ...

...

WARNING: LDR5207: ASURLClassLoader EarClassLoader : 
doneCalled = true
doneSnapshot = ASURLClassLoader.done() called ON EarClassLoader : 
urlSet = [URLEntry : file:/C:/glassfish3/glassfish/domains/domain1/eclipseApps/app-ear/app-jsf-0.0.1-SNAPSHOT_war/WEB-INF/classes/, URLEntry : file:/C:/glassfish3/glassfish/domains/domain1/eclipseApps/app-ear/app-jsf-0.0.1-SNAPSHOT_war/WEB-INF/lib/commons-fileupload-1.2.1.jar, URLEntry : file:/C:/glassfish3/glassfish/domains/domain1/eclipseApps/app-ear/app-jsf-0.0.1-SNAPSHOT_war/WEB-INF/lib/commons-io-1.4.jar, URLEntry : file:/C:/glassfish3/glassfish/domains/domain1/eclipseApps/app-ear/app-jsf-0.0.1-SNAPSHOT_war/WEB-INF/lib/primefaces-3.0.M2.jar, URLEntry : file:/C:/glassfish3/glassfish/domains/domain1/generated/ejb/app-ear/app-jsf-0.0.1-SNAPSHOT_war/, URLEntry : file:/C:/glassfish3/glassfish/domains/domain1/eclipseApps/app-ear/app-ejb-0.0.1-SNAPSHOT_jar/, URLEntry : file:/C:/glassfish3/glassfish/domains/domain1/generated/ejb/app-ear/app-ejb-0.0.1-SNAPSHOT_jar]
doneCalled = false 
 Parent -> org.glassfish.internal.api.DelegatingClassLoader@1a3fe65
 AT Sun Jul 17 13:27:54 IST 2011 
 BY :java.lang.Throwable: printStackTraceToString
    at com.sun.enterprise.util.Print.printStackTraceToString(Print.java:639)
    at com.sun.enterprise.loader.ASURLClassLoader.done(ASURLClassLoader.java:211)
    ...

...

显然,由于某些未知原因,EJB的生成类区域在运行时未包含JPA实体类,导致无法在部署时找到类。失败的最可能原因是,虽然实体类虽然打包在EJB模块中,但它们本身可能没有放在生成的类目录中;只有带注释的EJB类可能已被放置在那里。

唯一合理的选择是使用构建时间增强,这是使用Maven POM中的以下配置正确完成的:

 <build>
     <plugins>
         <plugin>
             <groupId>org.apache.maven.plugins</groupId>
             <artifactId>maven-antrun-plugin</artifactId>
             <version>1.6</version>
                 <executions>
                     <execution>
                         <phase>process-classes</phase>
                         <configuration>
                             <tasks>
                                 <taskdef name="openjpac" classname="org.apache.openjpa.ant.PCEnhancerTask" classpathref="maven.compile.classpath"/>
                                 <openjpac>
                                     <classpath refid="maven.compile.classpath"/>
                                 </openjpac>
                             </tasks>
                         </configuration>
                         <goals>
                             <goal>run</goal>
                         </goals>
                     </execution>
                 </executions>
             </plugin>
         </plugins>
    </pluginManagement>
</build>

以上配置源自enhancing classes with Maven上的OpenJPA文档。请注意,我没有使用openjpa-maven-plugin,因为OpenJPA的2.2.0版本在撰写本文时仅作为快照提供。

当然,上述任何一项都需要在Glassfish 3.1中安装OpenJPA 2.1,这是通过将以下JAR复制到${com.sun.aas.installRoot}/glassfish/lib(例如,C:/glassfishv3/glassfish/lib)来完成的,而不是复制到${com.sun.aas.instanceRoot}/lib的旧博客文章中提供的建议(例如,C:/glassfishv3/glassfish/domains/domain1/lib

  • 公地BeanUtils的-1.8.3.jar
  • 公地集合-3.2.1.jar
  • 公地DBCP-1.4.jar
  • 公地琅2.4.jar
  • 共享记录-1.0.4.jar
  • 公地池1.5.4.jar
  • SERP-1.13.1.jar
  • OpenJPA的-2.1.0.jar

显然,将这​​些JAR放在${com.sun.aas.instanceRoot}/lib导致Maven Glassfish插件无法部署应用程序。

OpenJPA 2.1发行版中存在的其他JAR(geronimo-*derby-*org.apache.bval*)由Glassfish 3.1提供,作为Java EE 6 SDK,Java DB实现或Derby客户端,以及JSR 303 bean验证框架(Hibernate Validator)。