Нет доступных транзакционных EntityManager

Я новичок в мире jpa и spring, и в настоящее время я выполняю модульное тестирование простого метода, но продолжаю получать это сообщение об ошибке только тогда, когда я запускаю свой тестовый класс в режиме модульного тестирования:

java.lang.IllegalStateException: No transactional EntityManager available
at org.springframework.orm.jpa.SharedEntityManagerCreator$SharedEntityManagerInvocationHandler.invoke(SharedEntityManagerCreator.java:223)
at $Proxy19.unwrap(Unknown Source)
at com.gemstone.integration.PersonDao.getPersonByUserNamePassword(PersonDao.java:59)
at com.gemstone.integration.PersonDaoTest.getPersonByUserNamePassword_Exist(PersonDaoTest.java:37)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(Unknown Source)
at java.lang.reflect.Method.invoke(Unknown Source)
at org.junit.runners.model.FrameworkMethod$1.runReflectiveCall(FrameworkMethod.java:44)
at org.junit.internal.runners.model.ReflectiveCallable.run(ReflectiveCallable.java:15)
at org.junit.runners.model.FrameworkMethod.invokeExplosively(FrameworkMethod.java:41)
at org.junit.internal.runners.statements.InvokeMethod.evaluate(InvokeMethod.java:20)
at org.springframework.test.context.junit4.statements.RunBeforeTestMethodCallbacks.evaluate(RunBeforeTestMethodCallbacks.java:74)
at org.springframework.test.context.junit4.statements.RunAfterTestMethodCallbacks.evaluate(RunAfterTestMethodCallbacks.java:82)
at org.springframework.test.context.junit4.statements.SpringRepeat.evaluate(SpringRepeat.java:72)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.runChild(SpringJUnit4ClassRunner.java:240)
at org.junit.runners.BlockJUnit4ClassRunner.runChild(BlockJUnit4ClassRunner.java:49)
at org.junit.runners.ParentRunner$3.run(ParentRunner.java:193)
at org.junit.runners.ParentRunner$1.schedule(ParentRunner.java:52)
at org.junit.runners.ParentRunner.runChildren(ParentRunner.java:191)
at org.junit.runners.ParentRunner.access$000(ParentRunner.java:42)
at org.junit.runners.ParentRunner$2.evaluate(ParentRunner.java:184)
at org.springframework.test.context.junit4.statements.RunBeforeTestClassCallbacks.evaluate(RunBeforeTestClassCallbacks.java:61)
at org.springframework.test.context.junit4.statements.RunAfterTestClassCallbacks.evaluate(RunAfterTestClassCallbacks.java:70)
at org.junit.runners.ParentRunner.run(ParentRunner.java:236)
at org.springframework.test.context.junit4.SpringJUnit4ClassRunner.run(SpringJUnit4ClassRunner.java:180)
at org.eclipse.jdt.internal.junit4.runner.JUnit4TestReference.run(JUnit4TestReference.java:50)
at org.eclipse.jdt.internal.junit.runner.TestExecution.run(TestExecution.java:38)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:467)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.runTests(RemoteTestRunner.java:683)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.run(RemoteTestRunner.java:390)
at org.eclipse.jdt.internal.junit.runner.RemoteTestRunner.main(RemoteTestRunner.java:197)

Однако код работает нормально, когда я запускаю свой проект, он правильно извлекает данные (не в режиме модульного тестирования).

Пожалуйста, найдите ниже мой метод, на котором будет основан модульный тест:

@PersistenceContext
private EntityManager entityManager;
@SuppressWarnings("unchecked")
public boolean getPersonByUserNamePassword(String firsName, String password) {

    String hql = "from Person p where p.firstName = :firsName and p.password =:password";
    Session mysession = entityManager.unwrap(Session.class);

    Query query = mysession.createQuery(hql);
    query.setParameter("firsName", firsName);
    query.setParameter("password", password);
    List<Person> results = query.list();

    if (results != null && results.size() > 0) {

        return true;
    }
    return false;

}

Мой модульный тест ниже:

@TestExecutionListeners({ DependencyInjectionTestExecutionListener.class })
@RunWith(SpringJUnit4ClassRunner.class)

@ContextConfiguration(locations = { "classpath:/Intcontext.xml" })

public class PersonDaoTest {

    @Autowired
    PersonDao personDao;

    @Transactional
    @Test
    public void getPersonByUserNamePassword_Exist() {

        String firsName = "Andy";
        String password = "123";

        boolean isUserExists = personDao.getPersonByUserNamePassword(firsName,
                password);

        Assert.assertTrue(isUserExists);
    }

}

Любые идеи, почему я получаю сообщение об ошибке Нет транзакционного EntityManager, пожалуйста? :(

заранее спасибо.


person user1999453    schedule 23.01.2013    source источник
comment
Вы пытались удалить @TestExecutionListeners? Также покажите, как вы настраиваете JPA, менеджер транзакций и т. д.   -  person axtavt    schedule 23.01.2013


Ответы (3)


Объявив

@TestExecutionListeners({ DependencyInjectionTestExecutionListener.class })

вы полностью (хотя и непреднамеренно) отключили поддержку тестовых транзакций. Причина в том, что вы исключили TransactionalTestExecutionListener из списка объявленных слушателей.

Обратите внимание, однако, что TransactionalTestExecutionListener по умолчанию объявляется прозрачным. Таким образом, когда вы удаляете объявление @TestExecutionListeners из своего тестового класса, TransactionalTestExecutionListener снова включается.

Конечно, вы можете найти подробности в Справочном руководстве Spring здесь.

С уважением,

Сэм (автор Spring TestContext Framework;))

person Sam Brannen    schedule 23.01.2013

Я думаю, что EntityManager имеет значение null. Попробуйте так, может быть, это сработает

private EntityManager entityManager = null;

@PersistenceContext
public void setEntityManager(EntityManager entityManager) {
    this.entityManager = entityManager;
}
person 9ine    schedule 23.01.2013
comment
Спасибо за ответ, как указано ранее @axtavt, удаление TestExecutionListeners решило мою проблему. Но я не знаю, почему. :( Не могли бы вы рассказать нам, почему аннотация TestExecutionListeners вызвала указанную ранее ошибку? - person user1999453; 23.01.2013

У меня возникла аналогичная проблема, когда я попробовал модульные тесты с IntelliJ, хотя он работает с использованием cli из maven-surefire-plugin

оказывается, что сборка IntelliJ использовала компилятор javac, и хотя в pox.xml указано использовать компилятор ajc, сообщается об ошибке для IntelliJ по адресу IDEA-135483

что сделало @Transactional неработающим и дало вам No transactional EntityManager available

если это ваш случай, вам нужно изменить компилятор Java, чтобы использовать ajc вместо javac и post-compile weave mode в структуре проекта

person Basheer AL-MOMANI    schedule 13.04.2020