Как создать экземпляр интерфейса CrudRepository во время тестирования в Spring?

У меня есть приложение Spring и в нем я не использую конфигурацию xml, только Java Config. Все в порядке, но при попытке проверить это я столкнулся с проблемами с включением автопроводки компонентов в тестах. Итак, начнем. У меня есть интерфейс:

@Repository
public interface ArticleRepository extends CrudRepository<Page, Long> {
    Article findByLink(String name);
    void delete(Page page);
}

И компонент/сервис:

@Service
public class ArticleServiceImpl implements ArticleService {
    @Autowired
    private ArticleRepository articleRepository;
...
}

Я не хочу использовать конфигурации xml, поэтому для своих тестов я пытаюсь протестировать ArticleServiceImpl, используя только конфигурацию Java. Итак, для целей теста я сделал:

@Configuration
@ComponentScan(basePackages = {"com.example.core", "com.example.repository"})
public class PagesTestConfiguration {


@Bean
public ArticleRepository articleRepository() {
       // (1) What to return ?
}

@Bean
public ArticleServiceImpl articleServiceImpl() {
    ArticleServiceImpl articleServiceImpl = new ArticleServiceImpl();
    articleServiceImpl.setArticleRepository(articleRepository());
    return articleServiceImpl;
}

}

В articleServiceImpl() мне нужно поместить экземпляр articleRepository(), но это интерфейс. Как создать новый объект с новым ключевым словом? Возможно ли это без создания класса конфигурации xml и включения Autowiring? Можно ли включить автоматическое подключение при использовании только JavaConfigurations во время тестирования?

Ответ 1

Это то, что я нашел, - это минимальная настройка для теста контроллера spring, для которого требуется конфигурация репозитория с автосогласованным JPA (с использованием spring -boot 1.2 со встроенным spring 4.1.4.RELEASE, DbUnit 2.4.8).

Тест выполняется против встроенной базы данных HSQL, которая автоматически заполняется файлом данных xml при запуске теста.

Класс тестирования:

@RunWith( SpringJUnit4ClassRunner.class )
@ContextConfiguration( classes = { TestController.class,
                                   RepoFactory4Test.class } )
@TestExecutionListeners( { DependencyInjectionTestExecutionListener.class,
                           DirtiesContextTestExecutionListener.class,
                           TransactionDbUnitTestExecutionListener.class } )
@DatabaseSetup( "classpath:FillTestData.xml" )
@DatabaseTearDown( "classpath:DbClean.xml" )
public class ControllerWithRepositoryTest
{
    @Autowired
    private TestController myClassUnderTest;

    @Test
    public void test()
    {
        Iterable<EUser> list = myClassUnderTest.findAll();

        if ( list == null || !list.iterator().hasNext() )
        {
            Assert.fail( "No users found" );
        }
        else
        {
            for ( EUser eUser : list )
            {
                System.out.println( "Found user: " + eUser );
            }
        }
    }

    @Component
    static class TestController
    {
        @Autowired
        private UserRepository myUserRepo;

        /**
         * @return
         */
        public Iterable<EUser> findAll()
        {
            return myUserRepo.findAll();
        }
    }
}

Примечания:

  • @ContextConfiguration аннотация, которая включает только встроенный TestController и класс конфигурации JPA RepoFactory4Test.

  • Аннотации @TestExecutionListeners необходимы для того, чтобы последующие аннотации @DatabaseSetup и @DatabaseTearDown имели эффект

Связанный класс конфигурации:

@Configuration
@EnableJpaRepositories( basePackageClasses = UserRepository.class )
public class RepoFactory4Test
{
    @Bean
    public DataSource dataSource()
    {
        EmbeddedDatabaseBuilder builder = new EmbeddedDatabaseBuilder();
        return builder.setType( EmbeddedDatabaseType.HSQL ).build();
    }

    @Bean
    public EntityManagerFactory entityManagerFactory()
    {
        HibernateJpaVendorAdapter vendorAdapter = new HibernateJpaVendorAdapter();
        vendorAdapter.setGenerateDdl( true );

        LocalContainerEntityManagerFactoryBean factory = new LocalContainerEntityManagerFactoryBean();
        factory.setJpaVendorAdapter( vendorAdapter );
        factory.setPackagesToScan( EUser.class.getPackage().getName() );
        factory.setDataSource( dataSource() );
        factory.afterPropertiesSet();

        return factory.getObject();
    }

    @Bean
    public PlatformTransactionManager transactionManager()
    {
        JpaTransactionManager txManager = new JpaTransactionManager();
        txManager.setEntityManagerFactory( entityManagerFactory() );
        return txManager;
    }
}

UserRepository - это простой интерфейс:

public interface UserRepository extends CrudRepository<EUser, Long>
{
}   

EUser - это простой аннотированный класс @Entity:

@Entity
@Table(name = "user")
public class EUser
{
    @Id
    @Column(name = "id")
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Max( value=Integer.MAX_VALUE )
    private Long myId;

    @Column(name = "email")
    @Size(max=64)
    @NotNull
    private String myEmail;

    ...
}

FillTestData.xml:

<?xml version="1.0" encoding="UTF-8"?>
<dataset>
    <user id="1"
          email="[email protected]"
          ...
    />
</dataset>

DbClean.xml:

<?xml version="1.0" encoding="UTF-8"?>
<dataset>
    <user />
</dataset>

Ответ 2

Если вы используете Spring Boot, вы можете немного упростить эти подходы, добавив @SpringBootTest для загрузки в свой ApplicationContext. Это позволяет вам автоматически подключаться к хранилищам данных Spring. Обязательно добавьте @RunWith(SpringRunner.class) чтобы подобрали аннотации, относящиеся к весне:

@RunWith(SpringRunner.class)
@SpringBootTest
public class OrphanManagementTest {

  @Autowired
  private UserRepository userRepository;

  @Test
  public void saveTest() {
    User user = new User("Tom");
    userRepository.save(user);
    Assert.assertNotNull(userRepository.findOne("Tom"));
  }
}

Вы можете прочитать больше о тестировании в весенней загрузке в их документах.

Ответ 3

Вы не можете использовать репозитории в своем классе конфигурации, потому что из классов конфигурации он находит все свои репозитории с помощью @EnableJpaRepositories.

  • Измените конфигурацию Java на:
@Configuration
@EnableWebMvc
@EnableTransactionManagement
@ComponentScan("com.example")
@EnableJpaRepositories(basePackages={"com.example.jpa.repositories"})//Path of your CRUD repositories package
@PropertySource("classpath:application.properties")
public class JPAConfiguration {
  //Includes jpaProperties(), jpaVendorAdapter(), transactionManager(), entityManagerFactory(), localContainerEntityManagerFactoryBean()
  //and dataSource()  
}
  1. Если у вас много классов реализации репозитория, создайте отдельный класс, как показано ниже
@Service
public class RepositoryImpl {
   @Autowired
   private UserRepositoryImpl userService;
}
  1. В вашем контроллере Autowire to RepositoryImpl и оттуда вы можете получить доступ ко всем классам реализации репозитория.
@Autowired
RepositoryImpl repository;

Использование:

repository.getUserService() findUserByUserName (Username);.

Удалить @Repository Annotation в ArticleRepository и ArticleServiceImpl должен реализовать ArticleRepository, а не ArticleService.

Ответ 4

Что вам нужно сделать:

  • удалите @Repository из ArticleRepository

  • добавить @EnableJpaRepositories в PagesTestConfiguration.java

    @Configuration
    @ComponentScan(basePackages = {"com.example.core"}) // are you sure you wanna scan all the packages?
    @EnableJapRepository(basePackageClasses = ArticleRepository.class) // assuming you have all the spring data repo in the same package.
    public class PagesTestConfiguration {
    
    @Bean
    public ArticleServiceImpl articleServiceImpl() {
        ArticleServiceImpl articleServiceImpl = new ArticleServiceImpl();
        return articleServiceImpl;
    }
    }