2010-10-29 3 views
29

Quelle valeur dois-je placer dans <jta-data-source> de mon persistence.xml?Que mettre dans jta-data-source de persistence.xml?

Dans le panneau d'administration de glassfish, j'ai créé un nom de source de données "abcDS". Dans mon jndi.properties (à l'intérieur src/test/resources) je définissais comme ceci:

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

Que dois-je mettre en persistence.xml? J'ai trouvé beaucoup de variantes dans le Net, comme: "jdbc/abcDS", "java:/abcDS", "abcDS". Lequel a raison? Et y a-t-il une règle pour cela? Je comprends que c'est lié à JNDI, mais ...

Je suis en train de créer EMF dans mon test unitaire:

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

C'est ce que je veux dans le journal:

[...] 
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) 
[...] 

Répondre

37

Le problème est que Persistence.createEntityManagerFactory("abc") est l'API «faites-le vous-même» et ne tire pas parti du conteneur Embedded EJB. Vous pouvez obtenir un conteneur géré EntityManager très facilement dans votre test.

Tout comme avec la question jndi/datasource connexe, je vous recommande de consulter les exemples dans le examples.zip. Ils sont tous conçus pour prendre le départ.

Voici un extrait de l'exemple testcase-injection qui montre comment vous pouvez obtenir un EntityManager et d'autres éléments du conteneur pour une utilisation dans un test.

Tout d'abord, ajouter un vide ejb-jar.xml ou application client.xml à votre test pour activer la recherche de votre code de test:

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

Ensuite, annotez votre scénario de test avec @org.apache.openejb.api.LocalClient et utilisez les annotations JavaEE standard pour l'injection réelle.

@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); 
    } 

Comme movieDatabase est la seule source de données que nous avons configuration, OpenEJB attribuera automatiquement DataSource à votre unité de persistance sans qu'il soit nécessaire de modifier votre persistence.xml. Vous pouvez même laisser le <jta-data-source> ou le <non-jta-data-source> vide et OpenEJB saura toujours quoi faire.

Mais par souci d'exhaustivité, voici comment cette application particulière a défini le 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> 

Ensuite, la partie amusante, en utilisant tous ensemble dans les tests

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(); 
    } 
} 
Questions connexes