Я создаю приложение Java SE Gradle, основанное на Hibernate как мой ORM. Мой план состоит в том, чтобы использовать weld-se
, чтобы иметь возможность использовать аннотации CDI для инъекций EntityManagers
во всем приложении.
Основываясь на общем вспомогательном классе HibernateUtil
, найденном в документации Hibernate, я перешел к интерфейсам JPA и добавил аннотации @Produces
для предоставления методов производителя (я также добавил пустой META-INF/beans.xml
):
package dao;
import javax.enterprise.inject.Disposes;
import javax.enterprise.inject.Produces;
import javax.persistence.EntityManager;
import javax.persistence.EntityManagerFactory;
import javax.persistence.Persistence;
public class HibernateUtil {
private static final EntityManagerFactory emf = buildEntityManagerFactory();
private static EntityManagerFactory buildEntityManagerFactory() {
try {
return Persistence.createEntityManagerFactory("persistenceUnit");
} catch (Throwable ex) {
System.err.println("Initial EntityManagerFactory creation failed." + ex);
throw new ExceptionInInitializerError(ex);
}
}
@Produces
public static EntityManager createEntityManager() {
return emf.createEntityManager();
}
public static void closeEntityManager(@Disposes EntityManager em) {
System.out.println("Closing EM");
try {
em.close();
} catch (Throwable t) {
t.printStackTrace();
}
}
}
Однако, когда я пытаюсь использовать аннотацию @Inject
в поле, Weld не может разрешить правильный метод производителя и вместо этого создает исключение:
Исключение в потоке "main" org.jboss.weld.exceptions.UnsatisfiedResolutionException: WELD-001308: Не удалось разрешить любой beans для типа: class app.DemoApplication; Квалификаторы: [@javax.enterprise.inject.Any()] на org.jboss.weld.bean.builtin.InstanceImpl.get(InstanceImpl.java:101) at app.Main.main(Main.java:14)
Код нарушения создается через контейнер Weld для поддержки CDI и невероятно прост:
package app;
import javax.inject.Inject;
import javax.persistence.EntityManager;
public class DemoApplication {
@Inject private EntityManager em;
public void run() {
try {
em.getTransaction().begin();
System.out.println("Inside transaction");
} catch (Throwable t) {
t.printStackTrace();
} finally {
em.getTransaction().rollback();
em.close();
}
}
}
Я пропустил здесь очевидный момент? Как я могу заставить Weld обнаружить метод производителя для ввода моих зависимостей?
Я собрал минимальный проект, воспроизводящий мою проблему на Github. Спасибо за любые полезные предложения!:)
Обновление 2015-05-18:
Похоже, я неправильно понял сообщение об ошибке. Фактически, Weld даже не разрешает DemoApplication
bean, что заставляет меня поверить, что что-то не так с процессом обнаружения bean. После обновления моей зависимости от сварки до только что выпущенной версии 3.0.0.Alpha8 (см. Связанное репозиторию Github) я смог заставить приложение работать, вручную сообщив Weld о моем beans в Main.java
:
final Weld weld = new Weld()
.enableDiscovery()
.addPackage(false, HibernateUtil.class)
.addPackage(false, DemoApplication.class);
Тем не менее, любые предложения относительно того, почему beans не открываются автоматически, несмотря на наличие пустого META-INF/beans.xml
, высоко оценены!
Обновление 2015-05-19:
Тайна распущена, см. мой собственный ответ ниже. Я изменил название вопроса, чтобы отразить фактический характер проблемы.