Que mettre en données JTA-source de persistence.xml?
Question
Quelle valeur dois-je placer dans <jta-data-source>
de mon persistence.xml
?
Dans le panneau d'administration 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
? Je l'ai trouvé beaucoup de variantes dans le net, comme: "jdbc/abcDS"
, "java:/abcDS"
, "abcDS"
. Laquelle est la bonne? Et est-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");
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)
[...]
La solution
Le problème est que Persistence.createEntityManagerFactory("abc")
est l'API « do it yourself » et ne tire pas profit du conteneur EJB intégré. Vous pouvez obtenir un conteneur EntityManager
géré dans votre cas de test très facilement.
Tout comme le jndi lié / datasource question que je vous recommandons de vérifier les exemples dans la section examples.zip . Ils sont tous conçus pour prendre la lutte de commencer.
Voici un extrait de l'exemple de testcase-injection
qui montre comment vous pouvez obtenir un EntityManager et d'autres choses du conteneur pour être utilisé 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
Alors, annoter votre cas de test avec @org.apache.openejb.api.LocalClient
et utiliser les annotations standards JavaEE 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 <non-jta-data-source>
vide et OpenEJB saurez toujours quoi faire.
Mais par souci d'exhaustivité, voici comment a défini cette application particulière du 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();
}
}