Что положить в jta-data-source файла persistence.xml? - PullRequest
34 голосов
/ 29 октября 2010

Какое значение я должен поместить в <jta-data-source> моего persistence.xml?

В админ панели Glassfish я создал имя источника данных "abcDS". В моем jndi.properties (внутри src/test/resources) я определил это так:

[...]
abcDS=new://Resource?type=DataSource
abcDS.JdbcDriver=org.hsqldb.jdbcDriver
abcDS.JdbcUrl=jdbc:hsqldb:mem:testdb
abcDS.JtaManaged=true
[...]

Что мне поместить в persistence.xml? Я нашел много вариантов в сети, например: "jdbc/abcDS", "java:/abcDS", "abcDS". Какой из них прав? И есть ли какое-то правило для этого? Я понимаю, что это связано с JNDI, но ...

Я пытаюсь создать ЭДС в моем модульном тесте:

EntityManagerFactory emf = Persistence.createEntityManagerFactory("abc");

Вот что я получаю в журнале:

[...]
SEVERE: Could not find datasource: abcDS javax.naming.NameNotFoundException: 
    Name "abcDS" not found.
at org.apache.openejb.core.ivm.naming.IvmContext.federate(IvmContext.java:193)
at org.apache.openejb.core.ivm.naming.IvmContext.lookup(IvmContext.java:150)
at org.apache.openejb.core.ivm.naming.ContextWrapper.lookup(ContextWrapper.java:115)
at javax.naming.InitialContext.lookup(InitialContext.java:392)
[...]

1 Ответ

37 голосов
/ 29 октября 2010

Проблема в том, что Persistence.createEntityManagerFactory("abc") - это API "сделай сам", который не использует встроенный контейнер EJB.Вы можете легко получить контейнер с управлением EntityManager в своем тестовом примере.

Как и в случае с вопросом jndi / datasource, я рекомендую вам ознакомиться с примерами в examples.zip .Все они разработаны, чтобы не начинать борьбу с самого начала.

Вот фрагмент из примера testcase-injection, который показывает, как вы можете получить EntityManager и другие вещи из контейнера для использования в тесте.

Сначала добавьте в тест пустой файл ejb-jar.xml или application-client.xml, чтобы включить сканирование тестового кода:

  • src / test / resources / META-INF / application-client.xml

Затем аннотируйте свой тестовый набор с помощью @org.apache.openejb.api.LocalClient и используйте стандартные аннотации JavaEE для фактического внедрения.

@LocalClient
public class MoviesTest extends TestCase {

    @EJB
    private Movies movies;

    @Resource
    private UserTransaction userTransaction;

    @PersistenceContext
    private EntityManager entityManager;

    public void setUp() throws Exception {
        Properties p = new Properties();
        p.put(Context.INITIAL_CONTEXT_FACTORY, "org.apache.openejb.client.LocalInitialContextFactory");
        p.put("movieDatabase", "new://Resource?type=DataSource");
        p.put("movieDatabase.JdbcDriver", "org.hsqldb.jdbcDriver");
        p.put("movieDatabase.JdbcUrl", "jdbc:hsqldb:mem:moviedb");

        InitialContext initialContext = new InitialContext(p);

        // Here's the fun part
        initialContext.bind("inject", this);
    }

Как movieDatabaseявляется единственным источником данных, который мы настроили, OpenEJB автоматически назначит этот источник данных вашему постоянному устройству без необходимости изменять ваш файл persistence.xml.Вы даже можете оставить <jta-data-source> или <non-jta-data-source> пустым, и OpenEJB все равно будет знать, что делать.

Но для полноты, вот как это конкретное приложение определило persistence.xml

<persistence xmlns="http://java.sun.com/xml/ns/persistence" version="1.0">

  <persistence-unit name="movie-unit">
    <jta-data-source>movieDatabase</jta-data-source>
    <non-jta-data-source>movieDatabaseUnmanaged</non-jta-data-source>
    <class>org.superbiz.testinjection.Movie</class>

    <properties>
      <property name="openjpa.jdbc.SynchronizeMappings" value="buildSchema(ForeignKeys=true)"/>
    </properties>
  </persistence-unit>
</persistence>

Тогда самое интересное, используя все это вместе в тестах

public void test() throws Exception {

    userTransaction.begin();

    try {
        entityManager.persist(new Movie("Quentin Tarantino", "Reservoir Dogs", 1992));
        entityManager.persist(new Movie("Joel Coen", "Fargo", 1996));
        entityManager.persist(new Movie("Joel Coen", "The Big Lebowski", 1998));

        List<Movie> list = movies.getMovies();
        assertEquals("List.size()", 3, list.size());

        for (Movie movie : list) {
            movies.deleteMovie(movie);
        }

        assertEquals("Movies.getMovies()", 0, movies.getMovies().size());

    } finally {
        userTransaction.commit();
    }
}
...