【问题标题】:Use DataNucleus with Glassfish 3.x在 Glassfish 3.x 中使用 DataNucleus
【发布时间】:2012-11-21 20:56:03
【问题描述】:

在 Glassfish AS 上使用 DataNucleus 作为 ORM 框架时遇到问题。实际上我正在尝试使用连接到 MongoDb。我预计这里会出现问题,因为每个人都知道 nosql 数据库不是标准的一部分,但是我在使用 DataNucleus 时遇到了问题。那是我没想到的。我对 DataNucleus 和 MongoDB 都不熟悉,很遗憾在 WEB 上找不到任何相关信息,所以我决定一步一步来。

我创建了一个使用 DataNucleus 和 MongoDB 并使用 Maven 构建的小型命令行应用程序。相关代码如下

persistece.xml

<?xml version="1.0" encoding="UTF-8"?>
<persistence version="1.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_1_0.xsd">
    <persistence-unit name="just_test">
        <provider>org.datanucleus.api.jpa.PersistenceProviderImpl</provider>

        <properties>
            <property name="datanucleus.ConnectionURL" value="mongodb:/justtest"/>
            <property name="datanucleus.autoCreateSchema" value="true"/>
            <property name="datanucleus.validateTables" value="false"/>
            <property name="datanucleus.validateConstraints" value="false"/>
        </properties>
    </persistence-unit>
</persistence>

持久化实体的代码

import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.EntityTransaction;
import javax.persistence.Persistence;


public class App {
    public static void main(String[] args) {
        EntityManagerFactory emf = Persistence.createEntityManagerFactory("just_test");

        // Create entity manager
        EntityManager em = null;
        String id;
        try {
            em = emf.createEntityManager();
            EntityTransaction tx = em.getTransaction();
            TestEntity testEntity = new TestEntity();
            tx.begin();
            em.persist(testEntity);
            em.flush();
            tx.commit();
            id = testEntity.getId();
        } finally {
            if (null != em) {
                em.close();
            }
        }

        try {
            em = emf.createEntityManager();
            TestEntity testEntity = em.find(TestEntity.class, id);
            System.out.println(testEntity);
        } finally {
            if (null != em) {
                em.close();
            }
        }
    }

pom.xml

...
<dependencies>
    <dependency>
        <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>3.8.1</version>
        <scope>test</scope>
    </dependency>
    <dependency>
        <groupId>org.mongodb</groupId>
        <artifactId>mongo-java-driver</artifactId>
        <version>[2.9.3, )</version>
        <!--<scope>provided</scope>-->
    </dependency>
    <dependency>
        <groupId>org.datanucleus</groupId>
        <artifactId>datanucleus-core</artifactId>
        <version>[2.9, )</version>
    </dependency>
    <dependency>
        <groupId>org.datanucleus</groupId>
        <artifactId>datanucleus-mongodb</artifactId>
        <version>[2.9, )</version>
    </dependency>
    <dependency>
        <groupId>org.datanucleus</groupId>
        <artifactId>datanucleus-api-jpa</artifactId>
        <version>[3.0, )</version>
    </dependency>
    <dependency>
        <groupId>org.datanucleus</groupId>
        <artifactId>datanucleus-api-jdo</artifactId>
        <version>[3.0, )</version>
    </dependency>
    <dependency>
        <groupId>org.apache.geronimo.specs</groupId>
        <artifactId>geronimo-jpa_2.0_spec</artifactId>
        <version>[1.0, )</version>
    </dependency>
    <dependency>
        <groupId>javax.jdo</groupId>
        <artifactId>jdo2-api</artifactId>
        <version>2.2</version>
    </dependency>
    <dependency>
        <groupId>javax.transaction</groupId>
        <artifactId>transaction-api</artifactId>
        <version>[1.1-rev-1, )</version>
    </dependency>
    <dependency>
        <groupId>javax.persistence</groupId>
        <artifactId>persistence-api</artifactId>
        <version>1.0b</version>
    </dependency>

</dependencies>

<build>
    <plugins>
        <plugin>
            <groupId>org.datanucleus</groupId>
            <artifactId>maven-datanucleus-plugin</artifactId>
            <version>3.1.0-m3</version>
            <configuration>
                <log4jConfiguration>${basedir}/log4j.properties</log4jConfiguration>
                <verbose>true</verbose>
            </configuration>
            <executions>
                <execution>
                    <phase>process-classes</phase>
                    <goals>
                        <goal>enhance</goal>
                    </goals>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>
...

此代码运行良好。

一旦我尝试部署一个使用相同代码的应用程序,首先发生的事情是 GF 无法找到某些类。这些类虽然在一个 .ear 文件中。我不得不将几个类放到 GF_HOME/glassfish/domains//lib 目录中,让 GF 能够找到这些类,但最后我以异常结束了

Caused by: javax.persistence.PersistenceException: No Persistence provider for EntityManager named just_test

值得一提的是,org.datanucleus.api.jpa.PersistenceProviderImpl 类在我也放在域的 lib 目录中的 datanucleus-api-jpa 文件中。

我也尝试使用 CDI 进行如下操作

@PersistenceContext(unitName = "just_test")
protected EntityManager entityManager;

@Override
public void persist(TestEntity testEntity) {
    entityManager.persist(testEntity);
}

这个方法也没有成功。我有一个例外

java.lang.ClassCastException: Cannot cast org.datanucleus.api.jpa.PersistenceProviderImpl to javax.persistence.spi.PersistenceProvider

任何想法或文章和/或文档的链接将不胜感激。谢谢

下面是我尝试在 Glassfish 上部署的应用程序的依赖项列表。我必须将所有“提供”范围的依赖项放入 Glassfish 的 lib 目录中

<dependency>
    <groupId>org.datanucleus</groupId>
    <artifactId>datanucleus-core</artifactId>
    <version>[2.9, )</version>
    <scope>provided</scope>
</dependency>
<dependency>
    <groupId>org.datanucleus</groupId>
    <artifactId>datanucleus-mongodb</artifactId>
    <version>[2.9, )</version>
</dependency>
<dependency>
    <groupId>org.datanucleus</groupId>
    <artifactId>datanucleus-api-jpa</artifactId>
    <version>[3.0, )</version>
    <scope>provided</scope>
</dependency>
<dependency>
    <groupId>org.apache.geronimo.specs</groupId>
    <artifactId>geronimo-jpa_2.0_spec</artifactId>
    <version>[1.0, )</version>
    <scope>provided</scope>
</dependency>
<dependency>
    <groupId>org.datanucleus</groupId>
    <artifactId>datanucleus-api-jdo</artifactId>
    <version>[3.0, )</version>
</dependency>
<dependency>
    <groupId>javax.jdo</groupId>
    <artifactId>jdo2-api</artifactId>
    <version>2.2</version>
</dependency>

【问题讨论】:

  • 您的 CLASSPATH 中似乎有 JPA1 jar (javax.persistence) 和 JPA2 jar (geronimo-jpa_2.0_spec)。如果我是你,我会先决定要使用哪个版本的 JPA。显然那里使用的 DN 版本完全实现了 JPA2
  • @DataNucleus 感谢您的评论。之所以存在这种依赖关系,是因为我首先找到了 JPA1 的示例,然后修改了代码以使用 JPA2。我删除了不必要的依赖,但没有帮助。我有相同的堆栈跟踪。我还修改了我的帖子并将我用于 GF 的整个依赖项列表。
  • 当 DN 版本实现 JDO 3.0 时,出于某种原因,您拥有 JDO 2.2。 Glassfish 将在某处拥有自己的 JPA.jar,因此它可能不想要另一个 JPA.jar 或感到困惑。事实是“org.datanucleus.api.jpa.PersistenceProviderImpl”确实完全实现了 javax.persistence.spi.PersistenceProvider 所以它是一个类加载的东西
  • 看来是类加载问题。问题是为什么。
  • 例如,我不需要采取任何特殊步骤来使用 Hibernate。你能推荐任何与之相关的文件吗?我在这里查看datanucleus.org/products/accessplatform_2_2/guides/jpa/…,但还没有找到解决方案。

标签: glassfish-3 datanucleus


【解决方案1】:

我设法解决了。我必须修改依赖项,如下所示

<properties>
    <endorsed.dir>${project.build.directory}/endorsed</endorsed.dir>
    <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    <datanucleus.version>3.1.3</datanucleus.version>
</properties>

<dependencies>
    <dependency>
        <groupId>org.datanucleus</groupId>
        <artifactId>datanucleus-core</artifactId>
        <version>${datanucleus.version}</version>
        <scope>provided</scope>
    </dependency>
    <dependency>
        <groupId>org.datanucleus</groupId>
        <artifactId>datanucleus-mongodb</artifactId>
        <version>${datanucleus.version}</version>
        <scope>provided</scope>
    </dependency>
    <dependency>
        <groupId>org.datanucleus</groupId>
        <artifactId>datanucleus-api-jpa</artifactId>
        <version>${datanucleus.version}</version>
        <scope>provided</scope>
    </dependency>
    <dependency>
        <groupId>org.datanucleus</groupId>
        <artifactId>datanucleus-api-jdo</artifactId>
        <version>${datanucleus.version}</version>
        <scope>provided</scope>
    </dependency>
    <dependency>
        <groupId>javax.jdo</groupId>
        <artifactId>jdo2-api</artifactId>
        <version>2.2</version>
        <scope>provided</scope>
    </dependency>
</dependencies>

我还必须将所有具有“提供”范围的 jar 放入 GF_HOME/glassfish/domains//lib 目录。我不知道为什么我必须这样做。很高兴对此有答案。希望这可以为想要将 Datanucleus 与 Glassfish 一起使用的人节省时间

【讨论】:

    猜你喜欢
    • 2011-12-24
    • 1970-01-01
    • 2013-04-19
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2011-08-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多