Интеграционное тестирование репозитория Spring с помощью DBUnit

Я следил за этой очень полезной публикацией в блоге Петри Кайнулайнена, чтобы написать интеграционные тесты для моего кода репозитория Spring, используя spring-data-jpa и DBUnit.

У меня есть XML-файл контекста бина, настроенный так

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xmlns:jpa="http://www.springframework.org/schema/data/jpa"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
        http://www.springframework.org/schema/data/jpa http://www.springframework.org/schema/data/jpa/spring-jpa-1.3.xsd
        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd">

    <context:property-placeholder location="file:src/main/webapp/WEB-INF/config/application-test.properties"
        ignore-resource-not-found="false" />

    <bean id="dataSource" class="com.jolbox.bonecp.BoneCPDataSource">
        <property name="driverClass" value="${db.driver}" />
        <property name="jdbcUrl" value="${db.url}" />
        <property name="username" value="${db.username}" />
        <property name="password" value="${db.password}" />
    </bean>
    <bean id="directorRepository" name="directorRepositoryMock" class="org.mockito.Mockito"
        factory-method="mock">
        <constructor-arg value="com.tvfreakz.repository.DirectorRepository" />
    </bean>

</beans>

Мой набор данных XML имеет следующий

<!DOCTYPE dataset SYSTEM "my-dataset.dtd">
<dataset>
    <director directorid="1" director="Ridley Scott" />
    <director directorid="2" director="James Cameron" />
    <director directorid="3" director="David Fincher" />
    <director directorid="4" director="Jean-Pierre Jeunet" />
</dataset>

Вот интерфейс репозитория

import org.springframework.data.jpa.repository.JpaRepository;
import org.springframework.stereotype.Repository;

import com.tvfreakz.model.entity.Director;

@Repository("directorRepository")
public interface DirectorRepository extends JpaRepository<Director, Long> {

  Director findByDirectorId(Long directorId);

}

И мой тестовый класс настроен так

import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertEquals;

import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.TestExecutionListeners;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.support.DependencyInjectionTestExecutionListener;
import org.springframework.test.context.support.DirtiesContextTestExecutionListener;
import org.springframework.test.context.transaction.TransactionalTestExecutionListener;

import com.github.springtestdbunit.DbUnitTestExecutionListener;
import com.github.springtestdbunit.annotation.DatabaseSetup;
import com.tvfreakz.model.entity.Director;

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"file:src/main/webapp/WEB-INF/config/testMVCContext.xml"})
@TestExecutionListeners({ DependencyInjectionTestExecutionListener.class,
  DirtiesContextTestExecutionListener.class,
  TransactionalTestExecutionListener.class,
  DbUnitTestExecutionListener.class })
@DatabaseSetup("testDataset.xml")
public class ITDirectorRepositoryTest {

  @Autowired
  private DirectorRepository repository;

  @Test
  public void testFindByDirectorIdWhenNoDirectorFound() {
    Director director = repository.findByDirectorId(10L);
    assertNull(director);
  }

  @Test
  public void testFindByDirectorIdWhendirectorIsFound() {
    Director director = repository.findByDirectorId(1L);
    assertEquals("Ridley Scott", director.getDirectorName());
  }

}

Я ожидал @DatabaseSetup аннотацию, чтобы взять мою конфигурацию и настроить все таблицы и вставки данных в мою тестовую базу данных, настроенную dataSource боб.

Это, похоже, ничего не делает (т.е. база данных остается пустой без таблиц или данных) и мой второй тест testFindByDirectorIdWhendirectorIsFound терпит неудачу, поскольку директор возвратился, является нулем. Что мне не хватает?

РЕДАКТИРОВАТЬ 1 ОК У меня есть некоторый прогресс, у меня теперь есть таблицы, но нет данных. Проблема была в том, что я не должен был использовать фиктивный репозиторий. Мне пришлось обновить свой контекстный XML до следующего. Итак, следующая проблема - попытаться понять, почему он не заполняет мои недавно созданные таблицы данными.

РЕДАКТИРОВАТЬ 2 JpaTransactionManager bean-компонент и управляемая аннотация добавлены в контекст

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xmlns:tx="http://www.springframework.org/schema/tx"
    xmlns:jpa="http://www.springframework.org/schema/data/jpa"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd
        http://www.springframework.org/schema/data/jpa http://www.springframework.org/schema/data/jpa/spring-jpa-1.3.xsd
        http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-4.0.xsd">

    <context:property-placeholder location="file:src/main/webapp/WEB-INF/config/application-test.properties"
        ignore-resource-not-found="false" />

<jpa:repositories base-package="com.tvfreakz.repository" />

<bean id="entityManagerFactory" class="org.springframework.orm.jpa.LocalContainerEntityManagerFactoryBean">
    <property name="persistenceUnitName" value="punit"/>
    <property name="dataSource" ref="dataSource"/>
    <property name="jpaVendorAdapter">
        <bean class="org.springframework.orm.jpa.vendor.HibernateJpaVendorAdapter">
            <property name="showSql" value="true"/>
        </bean>
    </property>
    <property name="jpaPropertyMap">
        <map>
            <entry key="hibernate.dialect" value="org.hibernate.dialect.MySQL5InnoDBDialect"/>
            <entry key="hibernate.hbm2ddl.auto" value="update"/>
            <entry key="hibernate.format_sql" value="true"/>
            <entry key="hibernate.show_sql" value="true"/>
        </map>
    </property>
</bean>

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

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

    <bean id="dataSource" class="com.jolbox.bonecp.BoneCPDataSource">
        <property name="driverClass" value="${db.driver}" />
        <property name="jdbcUrl" value="${db.url}" />
        <property name="username" value="${db.username}" />
        <property name="password" value="${db.password}" />
    </bean>

</beans>

EDIT 3 OK Мне удалось включить уровень DEBUG после включения slf4j в мои зависимости maven и создания файла log4j.properties с установленным уровнем DEBUG.

И я вижу это в журнале, который, возможно, дает подсказку?

14:47:05,670  INFO TestContextManager:242 - Could not instantiate TestExecutionListener [com.github.springtestdbunit.DbUnitTestExecutionListener]. Specify custom listener classes or make the default listener classes (and their required dependencies) available. Offending class: [org/dbunit/dataset/IDataSet]

Я считаю, что это относится к этой части моего тестового кода

@TestExecutionListeners({ DependencyInjectionTestExecutionListener.class,
  DirtiesContextTestExecutionListener.class,
  TransactionalTestExecutionListener.class,
  DbUnitTestExecutionListener.class })

1 ответ

Решение

Наряду с изменениями, внесенными в исходный пост, кажется, что в моем Maven pom.xml отсутствует зависимость DBUnit. После добавления следующего у меня теперь есть данные, спасибо Петри и Филу за вашу помощь!

    <dependency>
        <groupId>org.dbunit</groupId>
        <artifactId>dbunit</artifactId>
        <version>2.5.0</version>
    </dependency>
Другие вопросы по тегам