/*
* Copyright 2011-2016 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.springframework.data.repository.core.support;
import static org.assertj.core.api.Assertions.*;
import static org.hamcrest.Matchers.*;
import static org.junit.Assume.*;
import static org.mockito.Mockito.*;
import java.io.Serializable;
import java.lang.reflect.Method;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.Set;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.Future;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.rules.ExpectedException;
import org.junit.runner.RunWith;
import org.mockito.Mock;
import org.mockito.Mockito;
import org.mockito.junit.MockitoJUnitRunner;
import org.springframework.beans.factory.support.DefaultListableBeanFactory;
import org.springframework.core.SpringVersion;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.data.projection.ProjectionFactory;
import org.springframework.data.repository.PagingAndSortingRepository;
import org.springframework.data.repository.Repository;
import org.springframework.data.repository.RepositoryDefinition;
import org.springframework.data.repository.core.EntityInformation;
import org.springframework.data.repository.core.NamedQueries;
import org.springframework.data.repository.core.RepositoryInformation;
import org.springframework.data.repository.core.RepositoryMetadata;
import org.springframework.data.repository.query.RepositoryQuery;
import org.springframework.data.repository.sample.User;
import org.springframework.data.util.Version;
import org.springframework.scheduling.annotation.Async;
import org.springframework.scheduling.annotation.AsyncAnnotationBeanPostProcessor;
import org.springframework.test.util.ReflectionTestUtils;
import org.springframework.transaction.interceptor.TransactionalProxy;
import org.springframework.util.ClassUtils;
import org.springframework.util.concurrent.ListenableFuture;
/**
* Unit tests for {@link RepositoryFactorySupport}.
*
* @author Oliver Gierke
* @author Mark Paluch
*/
@RunWith(MockitoJUnitRunner.class)
public class RepositoryFactorySupportUnitTests {
static final Version SPRING_VERSION = Version.parse(SpringVersion.getVersion());
static final Version FOUR_DOT_TWO = new Version(4, 2);
public @Rule ExpectedException exception = ExpectedException.none();
DummyRepositoryFactory factory;
@Mock PagingAndSortingRepository<Object, Object> backingRepo;
@Mock ObjectRepositoryCustom customImplementation;
@Mock MyQueryCreationListener listener;
@Mock PlainQueryCreationListener otherListener;
@Before
public void setUp() {
factory = new DummyRepositoryFactory(backingRepo);
}
@Test
public void invokesCustomQueryCreationListenerForSpecialRepositoryQueryOnly() throws Exception {
Mockito.reset(factory.strategy);
when(factory.strategy.resolveQuery(Mockito.any(Method.class), Mockito.any(RepositoryMetadata.class),
Mockito.any(ProjectionFactory.class), Mockito.any(NamedQueries.class))).thenReturn(factory.queryOne,
factory.queryTwo);
factory.addQueryCreationListener(listener);
factory.addQueryCreationListener(otherListener);
factory.getRepository(ObjectRepository.class);
verify(listener, times(1)).onCreation(Mockito.any(MyRepositoryQuery.class));
verify(otherListener, times(2)).onCreation(Mockito.any(RepositoryQuery.class));
}
@Test
public void routesCallToRedeclaredMethodIntoTarget() {
ObjectRepository repository = factory.getRepository(ObjectRepository.class);
repository.save(repository);
verify(backingRepo, times(1)).save(Mockito.any(Object.class));
}
@Test
public void invokesCustomMethodIfItRedeclaresACRUDOne() {
ObjectRepository repository = factory.getRepository(ObjectRepository.class, customImplementation);
repository.findById(1);
verify(customImplementation, times(1)).findById(1);
verify(backingRepo, times(0)).findById(1);
}
@Test
public void createsRepositoryInstanceWithCustomIntermediateRepository() {
CustomRepository repository = factory.getRepository(CustomRepository.class);
Pageable pageable = PageRequest.of(0, 10);
repository.findAll(pageable);
verify(backingRepo, times(1)).findAll(pageable);
}
@Test
@SuppressWarnings("unchecked")
public void createsProxyForAnnotatedRepository() {
Class<?> repositoryInterface = AnnotatedRepository.class;
Class<? extends Repository<?, ?>> foo = (Class<? extends Repository<?, ?>>) repositoryInterface;
assertThat(factory.getRepository(foo)).isNotNull();
}
@Test // DATACMNS-341
public void usesDefaultClassLoaderIfNullConfigured() {
factory.setBeanClassLoader(null);
assertThat(ReflectionTestUtils.getField(factory, "classLoader")).isEqualTo(ClassUtils.getDefaultClassLoader());
}
@Test // DATACMNS-489
public void wrapsExecutionResultIntoFutureIfConfigured() throws Exception {
final Object reference = new Object();
when(factory.queryOne.execute(Mockito.any(Object[].class))).then(invocation -> {
Thread.sleep(500);
return reference;
});
ConvertingRepository repository = factory.getRepository(ConvertingRepository.class);
AsyncAnnotationBeanPostProcessor processor = new AsyncAnnotationBeanPostProcessor();
processor.setBeanFactory(new DefaultListableBeanFactory());
repository = (ConvertingRepository) processor.postProcessAfterInitialization(repository, null);
Future<Object> future = repository.findByFirstname("Foo");
assertThat(future.isDone()).isFalse();
while (!future.isDone()) {
Thread.sleep(300);
}
assertThat(future.get()).isEqualTo(reference);
verify(factory.queryOne, times(1)).execute(Mockito.any(Object[].class));
}
@Test // DATACMNS-509
public void convertsWithSameElementType() {
List<String> names = Collections.singletonList("Dave");
when(factory.queryOne.execute(Mockito.any(Object[].class))).thenReturn(names);
ConvertingRepository repository = factory.getRepository(ConvertingRepository.class);
Set<String> result = repository.convertListToStringSet();
assertThat(result).hasSize(1);
assertThat(result.iterator().next()).isEqualTo("Dave");
}
@Test // DATACMNS-509
public void convertsCollectionToOtherCollectionWithElementSuperType() {
List<String> names = Collections.singletonList("Dave");
when(factory.queryOne.execute(Mockito.any(Object[].class))).thenReturn(names);
ConvertingRepository repository = factory.getRepository(ConvertingRepository.class);
Set<Object> result = repository.convertListToObjectSet();
assertThat(result).hasSize(1);
assertThat(result.iterator().next()).isEqualTo("Dave");
}
@Test // DATACMNS-656
public void rejectsNullRepositoryProxyPostProcessor() {
exception.expect(IllegalArgumentException.class);
exception.expectMessage(RepositoryProxyPostProcessor.class.getSimpleName());
factory.addRepositoryProxyPostProcessor(null);
}
@Test // DATACMNS-715, SPR-13109
public void addsTransactionProxyInterfaceIfAvailable() throws Exception {
assertThat(factory.getRepository(SimpleRepository.class)).isInstanceOf(TransactionalProxy.class);
}
@Test // DATACMNS-714
public void wrapsExecutionResultIntoCompletableFutureIfConfigured() throws Exception {
assumeThat(SPRING_VERSION.isGreaterThanOrEqualTo(FOUR_DOT_TWO), is(true));
User reference = new User();
expect(prepareConvertingRepository(reference).findOneByFirstname("Foo"), reference);
}
@Test // DATACMNS-714
public void wrapsExecutionResultIntoListenableFutureIfConfigured() throws Exception {
assumeThat(SPRING_VERSION.isGreaterThanOrEqualTo(FOUR_DOT_TWO), is(true));
User reference = new User();
expect(prepareConvertingRepository(reference).findOneByLastname("Foo"), reference);
}
@Test // DATACMNS-714
public void wrapsExecutionResultIntoCompletableFutureWithEntityCollectionIfConfigured() throws Exception {
assumeThat(SPRING_VERSION.isGreaterThanOrEqualTo(FOUR_DOT_TWO), is(true));
List<User> reference = Collections.singletonList(new User());
expect(prepareConvertingRepository(reference).readAllByFirstname("Foo"), reference);
}
@Test // DATACMNS-714
public void wrapsExecutionResultIntoListenableFutureWithEntityCollectionIfConfigured() throws Exception {
List<User> reference = Collections.singletonList(new User());
expect(prepareConvertingRepository(reference).readAllByLastname("Foo"), reference);
}
@Test // DATACMNS-763
@SuppressWarnings("rawtypes")
public void rejectsRepositoryBaseClassWithInvalidConstructor() {
RepositoryInformation information = mock(RepositoryInformation.class);
doReturn(CustomRepositoryBaseClass.class).when(information).getRepositoryBaseClass();
EntityInformation entityInformation = mock(EntityInformation.class);
exception.expect(IllegalStateException.class);
exception.expectMessage(entityInformation.getClass().getName());
exception.expectMessage(String.class.getName());
factory.getTargetRepositoryViaReflection(information, entityInformation, "Foo");
}
@Test
public void callsStaticMethodOnInterface() {
ObjectRepository repository = factory.getRepository(ObjectRepository.class, customImplementation);
assertThat(repository.staticMethodDelegate()).isEqualTo("OK");
verifyZeroInteractions(customImplementation);
verifyZeroInteractions(backingRepo);
}
private ConvertingRepository prepareConvertingRepository(final Object expectedValue) {
when(factory.queryOne.execute(Mockito.any(Object[].class))).then(invocation -> {
Thread.sleep(200);
return expectedValue;
});
AsyncAnnotationBeanPostProcessor processor = new AsyncAnnotationBeanPostProcessor();
processor.setBeanFactory(new DefaultListableBeanFactory());
return (ConvertingRepository) processor
.postProcessAfterInitialization(factory.getRepository(ConvertingRepository.class), null);
}
private void expect(Future<?> future, Object value) throws Exception {
assertThat(future.isDone()).isFalse();
while (!future.isDone()) {
Thread.sleep(50);
}
assertThat(future.get()).isEqualTo(value);
verify(factory.queryOne, times(1)).execute(Mockito.any(Object[].class));
}
interface SimpleRepository extends Repository<Object, Serializable> {}
interface ObjectRepository extends Repository<Object, Object>, ObjectRepositoryCustom {
Object findByClass(Class<?> clazz);
Object findByFoo();
Object save(Object entity);
static String staticMethod() {
return "OK";
}
default String staticMethodDelegate() {
return staticMethod();
}
}
interface ObjectRepositoryCustom {
Object findById(Object id);
}
interface PlainQueryCreationListener extends QueryCreationListener<RepositoryQuery> {
}
interface MyQueryCreationListener extends QueryCreationListener<MyRepositoryQuery> {
}
interface MyRepositoryQuery extends RepositoryQuery {
}
interface ReadOnlyRepository<T, ID extends Serializable> extends Repository<T, ID> {
Optional<T> findById(ID id);
Iterable<T> findAll();
Page<T> findAll(Pageable pageable);
List<T> findAll(Sort sort);
boolean existsById(ID id);
long count();
}
interface CustomRepository extends ReadOnlyRepository<Object, Long> {
}
@RepositoryDefinition(domainClass = Object.class, idClass = Long.class)
interface AnnotatedRepository {
}
interface ConvertingRepository extends Repository<Object, Long> {
Set<String> convertListToStringSet();
Set<Object> convertListToObjectSet();
@Async
Future<Object> findByFirstname(String firstname);
// DATACMNS-714
@Async
CompletableFuture<User> findOneByFirstname(String firstname);
// DATACMNS-714
@Async
CompletableFuture<List<User>> readAllByFirstname(String firstname);
// DATACMNS-714
@Async
ListenableFuture<User> findOneByLastname(String lastname);
// DATACMNS-714
@Async
ListenableFuture<List<User>> readAllByLastname(String lastname);
}
static class CustomRepositoryBaseClass {
public CustomRepositoryBaseClass(EntityInformation<?, ?> information) {}
}
}