Тестирование Spring Repository - не удается правильно вызвать загрузку и удаление EntityManager - PullRequest
0 голосов
/ 25 февраля 2012

Добрый день, дорогие укладчики,

В настоящий момент я действительно борюсь с базовым тестовым классом фиктивного DAO.Несмотря на то, что я вижу, что сохраняются и находят методы введенного EntityManager, наконец, запускает SQL-запросы, мои утверждения продолжают сбой.

Во-первых, я использую тестовую конфигурацию:

<!-- test -->
<persistence-unit name="eSporxPersistenceTestUnit" transaction-type="RESOURCE_LOCAL">
    <provider>org.hibernate.ejb.HibernatePersistence</provider>
    <class>...CoalmineCanary</class>
    <properties>
        <property name="hibernate.dialect" value="org.hibernate.dialect.H2Dialect" />
        <property name="hibernate.hbm2ddl.auto" value="create-drop" />
        <property name="hibernate.ejb.naming_strategy" value="org.hibernate.cfg.ImprovedNamingStrategy" />
        <property name="hibernate.connection.charSet" value="UTF-8" />
    </properties>
</persistence-unit>

... вместе с контекстом приложения, используемым в тестах:

<!-- data source definition -->
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close">
    <property name="driverClassName" value="${database.driverClassName}" />
    <property name="url" value="${database.url}" />
    <property name="username" value="${database.username}" />
    <property name="password" value="${database.password}" />
</bean>

<!-- entity manager -->
<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="dataSource" ref="dataSource" />
    <property name="persistenceUnitName" value="eSporxPersistenceTestUnit" />
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
            <property name="generateDdl" value="true" />
            <property name="showSql" value="true" />
            <property name="databasePlatform" value="${database.dialect}" />
        </bean>
    </property>
</bean>

<!-- transaction manager -->
<bean class="org.springframework.orm.jpa.JpaTransactionManager" id="transactionManager">
    <property name="entityManagerFactory" ref="entityManagerFactory" />
</bean>

<tx:annotation-driven transaction-manager="transactionManager" />

Мой DAO работает со следующим объектом:

@Entity
public class CoalmineCanary {

@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private int id;
private String text;

public int getId() {
    return id;
}

public String getText() {
    return this.text;
}

public void setText(String text) {
    this.text = text;
}
}

... и с самим DAO:

@Repository
public class CoalmineCanaryRepository {

@PersistenceContext
private EntityManager entityManager;

@Transactional
public void delete(CoalmineCanary entity) {
    entityManager.remove(entity);
}

@Transactional
public CoalmineCanary findById(final int id) {
    return entityManager.find(CoalmineCanary.class, id);
}

@Transactional
public void save(CoalmineCanary entity) {
    entityManager.persist(entity);
}
}

Мой тестовый класс состоит только из 2 очень простых тестов:

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:/META-INF/spring/testApplicationContext.xml")
public class CoalmineCanaryRepositoryTest {

@Autowired
private CoalmineCanaryRepository canaryRepository;

private CoalmineCanary coalmineCanary;

@Before
public void setup() {
    coalmineCanary = new CoalmineCanary();
    coalmineCanary.setText("pokus");
    canaryRepository.save(coalmineCanary);
}

@Test
public void when_loaded_then_entity_is_retrieved() {
    CoalmineCanary managedCanary = canaryRepository.findById(1);
    assertThat(managedCanary).isNotNull();
    assertThat(managedCanary.getText()).isEqualTo("pokus");
}

@Test
public void when_removed_then_entity_not_retrievable() {
    CoalmineCanary managedCanary = canaryRepository.findById(1);
    assertThat(managedCanary).isNotNull();
    canaryRepository.delete(managedCanary);
    CoalmineCanary canary = canaryRepository.findById(1);
    assertThat(canary).isNull();
}
}

Вот вывод консоли при запуске тестов:

Feb 25, 2012 2:41:39 PM org.springframework.test.context.TestContextManager retrieveTestExecutionListeners
INFO: @TestExecutionListeners is not present for class [class tv.esporx.framework.CoalmineCanaryRepositoryTest]: using defaults.
Feb 25, 2012 2:41:39 PM org.springframework.beans.factory.xml.XmlBeanDefinitionReader loadBeanDefinitions
INFO: Loading XML bean definitions from class path resource [META-INF/spring/testApplicationContext.xml]
Feb 25, 2012 2:41:39 PM org.springframework.context.support.AbstractApplicationContext prepareRefresh
INFO: Refreshing org.springframework.context.support.GenericApplicationContext@c4fe76: startup date [Sat Feb 25 14:41:39 CET 2012]; root of context hierarchy
Feb 25, 2012 2:41:40 PM org.springframework.core.io.support.PropertiesLoaderSupport loadProperties
INFO: Loading properties file from file [/usr/local/projects/esporx/esporx-webapp/target/test-classes/META-INF/spring/datasource.properties]
Feb 25, 2012 2:41:40 PM org.springframework.core.io.support.PropertiesLoaderSupport loadProperties
INFO: Loading properties file from file [/usr/local/projects/esporx/esporx-webapp/target/classes/META-INF/spring/datasource.properties]
Feb 25, 2012 2:41:40 PM org.springframework.beans.factory.support.DefaultListableBeanFactory preInstantiateSingletons
INFO: Pre-instantiating singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@5eb489: defining beans [coalmineCanaryRepository,org.springframework.context.annotation.internalConfigurationAnnotationProcessor,org.springframework.context.annotation.internalAutowiredAnnotationProcessor,org.springframework.context.annotation.internalRequiredAnnotationProcessor,org.springframework.context.annotation.internalCommonAnnotationProcessor,org.springframework.context.annotation.internalPersistenceAnnotationProcessor,org.springframework.beans.factory.config.PropertyPlaceholderConfigurer#0,validator,dataSource,entityManagerFactory,transactionManager,org.springframework.aop.config.internalAutoProxyCreator,org.springframework.transaction.annotation.AnnotationTransactionAttributeSource#0,org.springframework.transaction.interceptor.TransactionInterceptor#0,org.springframework.transaction.config.internalTransactionAdvisor,org.springframework.context.annotation.ConfigurationClassPostProcessor$ImportAwareBeanPostProcessor#0]; root of factory hierarchy
Feb 25, 2012 2:41:40 PM org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean createNativeEntityManagerFactory
INFO: Building JPA container EntityManagerFactory for persistence unit 'eSporxPersistenceTestUnit'
Feb 25, 2012 2:41:41 PM org.hibernate.annotations.common.Version <clinit>
INFO: HCANN000001: Hibernate Commons Annotations {4.0.1.Final}
Feb 25, 2012 2:41:41 PM org.hibernate.Version logVersion
INFO: HHH000412: Hibernate Core {4.0.1.Final}
Feb 25, 2012 2:41:41 PM org.hibernate.cfg.Environment <clinit>
INFO: HHH000206: hibernate.properties not found
Feb 25, 2012 2:41:41 PM org.hibernate.cfg.Environment buildBytecodeProvider
INFO: HHH000021: Bytecode provider name : javassist
Feb 25, 2012 2:41:41 PM org.hibernate.ejb.Ejb3Configuration configure
INFO: HHH000204: Processing PersistenceUnitInfo [
    name: eSporxPersistenceTestUnit
    ...]
Feb 25, 2012 2:41:41 PM org.hibernate.service.jdbc.connections.internal.ConnectionProviderInitiator instantiateExplicitConnectionProvider
INFO: HHH000130: Instantiating explicit connection provider: org.hibernate.ejb.connection.InjectedDataSourceConnectionProvider
Feb 25, 2012 2:41:41 PM org.hibernate.dialect.Dialect <init>
INFO: HHH000400: Using dialect: org.hibernate.dialect.MySQLDialect
Feb 25, 2012 2:41:41 PM org.hibernate.engine.transaction.internal.TransactionFactoryInitiator initiateService
INFO: HHH000268: Transaction strategy: org.hibernate.engine.transaction.internal.jdbc.JdbcTransactionFactory
Feb 25, 2012 2:41:41 PM org.hibernate.hql.internal.ast.ASTQueryTranslatorFactory <init>
INFO: HHH000397: Using ASTQueryTranslatorFactory
SLF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder".
SLF4J: Defaulting to no-operation (NOP) logger implementation
SLF4J: See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.
Feb 25, 2012 2:41:42 PM org.hibernate.tool.hbm2ddl.SchemaUpdate execute
INFO: HHH000228: Running hbm2ddl schema update
Feb 25, 2012 2:41:42 PM org.hibernate.tool.hbm2ddl.SchemaUpdate execute
INFO: HHH000102: Fetching database metadata
Feb 25, 2012 2:41:42 PM org.hibernate.tool.hbm2ddl.SchemaUpdate execute
INFO: HHH000396: Updating schema
Feb 25, 2012 2:41:42 PM org.hibernate.tool.hbm2ddl.TableMetadata <init>
INFO: HHH000261: Table found: esporx.coalmine_canary
Feb 25, 2012 2:41:42 PM org.hibernate.tool.hbm2ddl.TableMetadata <init>
INFO: HHH000037: Columns: [id, text]
Feb 25, 2012 2:41:42 PM org.hibernate.tool.hbm2ddl.TableMetadata <init>
INFO: HHH000108: Foreign keys: []
Feb 25, 2012 2:41:42 PM org.hibernate.tool.hbm2ddl.TableMetadata <init>
INFO: HHH000126: Indexes: [primary]
Feb 25, 2012 2:41:42 PM org.hibernate.tool.hbm2ddl.SchemaUpdate execute
INFO: HHH000232: Schema update complete
Hibernate: insert into coalmine_canary (text) values (?)
Hibernate: select coalmineca0_.id as id0_0_, coalmineca0_.text as text0_0_ from coalmine_canary coalmineca0_ where coalmineca0_.id=?
Hibernate: insert into coalmine_canary (text) values (?)
Hibernate: select coalmineca0_.id as id0_0_, coalmineca0_.text as text0_0_ from coalmine_canary coalmineca0_ where coalmineca0_.id=?
Feb 25, 2012 2:41:42 PM org.springframework.context.support.AbstractApplicationContext doClose
INFO: Closing org.springframework.context.support.GenericApplicationContext@c4fe76: startup date [Sat Feb 25 14:41:39 CET 2012]; root of context hierarchy
Feb 25, 2012 2:41:42 PM org.springframework.beans.factory.support.DefaultSingletonBeanRegistry destroySingletons
INFO: Destroying singletons in org.springframework.beans.factory.support.DefaultListableBeanFactory@5eb489: defining beans [coalmineCanaryRepository,org.springframework.context.annotation.internalConfigurationAnnotationProcessor,org.springframework.context.annotation.internalAutowiredAnnotationProcessor,org.springframework.context.annotation.internalRequiredAnnotationProcessor,org.springframework.context.annotation.internalCommonAnnotationProcessor,org.springframework.context.annotation.internalPersistenceAnnotationProcessor,org.springframework.beans.factory.config.PropertyPlaceholderConfigurer#0,validator,dataSource,entityManagerFactory,transactionManager,org.springframework.aop.config.internalAutoProxyCreator,org.springframework.transaction.annotation.AnnotationTransactionAttributeSource#0,org.springframework.transaction.interceptor.TransactionInterceptor#0,org.springframework.transaction.config.internalTransactionAdvisor,org.springframework.context.annotation.ConfigurationClassPostProcessor$ImportAwareBeanPostProcessor#0]; root of factory hierarchy
Feb 25, 2012 2:41:42 PM org.springframework.orm.jpa.AbstractEntityManagerFactoryBean destroy
INFO: Closing JPA EntityManagerFactory for persistence unit 'eSporxPersistenceTestUnit'

Что происходит сейчасявляется то, что findById (1) всегда возвращает ноль.Я пытался многое изменить, но все еще не смог пройти тесты ...

Заранее спасибо за помощь!

Рольф

1 Ответ

2 голосов
/ 25 февраля 2012

Весь написанный вами код в порядке, но тесты нуждаются в некоторой настройке,

private CoalmineCanary coalmineCanary;

@Before
public void setup() {
    coalmineCanary = new CoalmineCanary();
    coalmineCanary.setText("pokus");
    coalmineCanary = canaryRepository.save(coalmineCanary);

    if(coalmineCanary.getId() == 0){
       fail("Could not insert the Canary!");
    }
}

@Test
public void when_loaded_then_entity_is_retrieved() {
    CoalmineCanary managedCanary = canaryRepository.findById(coalmineCanary.getId());
    assertThat(managedCanary).isNotNull();
    assertThat(managedCanary.getText()).isEqualTo("pokus");
}

Ваши тесты могут быть неудачными, потому что нет никакой гарантии, что идентификатор объектов будетбыть 1 .Когда объекты сохранены, hibernate обновляет id значением из БД и возвращает объект обратно.Если вы хотите использовать это, вы должны вернуть это для метода вызывающего.

Здесь, в вашем методе тестирования вместо жестко закодированного идентификатора, вы должны использовать идентификатор сохраненного компонента, чтобы тесты могли пройти.Также, пусть тесты не пройдут, если id не был установлен в методе setup().

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...