/* * Copyright 2008-2017 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.jpa.repository.query; import static javax.persistence.TemporalType.*; import static org.hamcrest.Matchers.*; import static org.junit.Assert.*; import static org.mockito.ArgumentMatchers.*; import static org.mockito.Mockito.*; import java.lang.reflect.Method; import java.util.Date; import java.util.List; import java.util.Optional; import javax.persistence.Embeddable; import javax.persistence.Query; import javax.persistence.TemporalType; import org.junit.Before; import org.junit.Test; import org.junit.runner.RunWith; import org.mockito.Mock; import org.mockito.junit.MockitoJUnitRunner; import org.springframework.data.domain.Pageable; import org.springframework.data.domain.Sort; import org.springframework.data.jpa.repository.Temporal; import org.springframework.data.repository.query.Param; /** * Unit test for {@link ParameterBinder}. * * @author Oliver Gierke * @author Thomas Darimont */ @RunWith(MockitoJUnitRunner.class) public class ParameterBinderUnitTests { private Method valid; @Mock private Query query; private Method useIndexedParameters; private Method indexedParametersWithSort; @Before public void setUp() throws SecurityException, NoSuchMethodException { valid = SampleRepository.class.getMethod("valid", String.class); useIndexedParameters = SampleRepository.class.getMethod("useIndexedParameters", String.class); indexedParametersWithSort = SampleRepository.class.getMethod("indexedParameterWithSort", String.class, Sort.class); } static class User { } static interface SampleRepository { User useIndexedParameters(String lastname); User indexedParameterWithSort(String lastname, Sort sort); User valid(@Param("username") String username); User validWithPageable(@Param("username") String username, Pageable pageable); User validWithSort(@Param("username") String username, Sort sort); User validWithDefaultTemporalTypeParameter(@Temporal Date registerDate); User validWithCustomTemporalTypeParameter(@Temporal(TIMESTAMP) Date registerDate); User invalidWithTemporalTypeParameter(@Temporal String registerDate); List<User> validWithVarArgs(Integer... ids); User optionalParameter(Optional<String> name); } @Test(expected = IllegalArgumentException.class) public void rejectsToManyParameters() throws Exception { new ParameterBinder(new JpaParameters(valid), new Object[] { "foo", "bar" }); } @Test(expected = IllegalArgumentException.class) public void rejectsNullParameters() throws Exception { new ParameterBinder(new JpaParameters(valid), (Object[]) null); } @Test(expected = IllegalArgumentException.class) public void rejectsToLittleParameters() throws SecurityException, NoSuchMethodException { JpaParameters parameters = new JpaParameters(valid); new ParameterBinder(parameters); } @Test public void returnsPageableNoneIfNoPageableWasProvided() throws SecurityException, NoSuchMethodException { Method method = SampleRepository.class.getMethod("validWithPageable", String.class, Pageable.class); JpaParameters parameters = new JpaParameters(method); ParameterBinder binder = new ParameterBinder(parameters, new Object[] { "foo", null }); assertThat(binder.getPageable(), is(Pageable.unpaged())); } @Test public void bindWorksWithNullForSort() throws Exception { Method validWithSort = SampleRepository.class.getMethod("validWithSort", String.class, Sort.class); new ParameterBinder(new JpaParameters(validWithSort), new Object[] { "foo", null }).bind(query); verify(query).setParameter(eq(1), eq("foo")); } @Test public void bindWorksWithNullForPageable() throws Exception { Method validWithPageable = SampleRepository.class.getMethod("validWithPageable", String.class, Pageable.class); new ParameterBinder(new JpaParameters(validWithPageable), new Object[] { "foo", null }).bind(query); verify(query).setParameter(eq(1), eq("foo")); } @Test public void usesIndexedParametersIfNoParamAnnotationPresent() throws Exception { new ParameterBinder(new JpaParameters(useIndexedParameters), new Object[] { "foo" }).bind(query); verify(query).setParameter(eq(1), anyObject()); } @Test public void usesParameterNameIfAnnotated() throws Exception { when(query.setParameter(eq("username"), anyObject())).thenReturn(query); new ParameterBinder(new JpaParameters(valid), new Object[] { "foo" }) { @Override boolean hasNamedParameter(Query query) { return true; } }.bind(query); verify(query).setParameter(eq("username"), anyObject()); } @Test public void bindsEmbeddableCorrectly() throws Exception { Method method = getClass().getMethod("findByEmbeddable", SampleEmbeddable.class); JpaParameters parameters = new JpaParameters(method); SampleEmbeddable embeddable = new SampleEmbeddable(); new ParameterBinder(parameters, new Object[] { embeddable }).bind(query); verify(query).setParameter(1, embeddable); } @Test public void bindsSortForIndexedParameters() throws Exception { Sort sort = Sort.by("name"); ParameterBinder binder = new ParameterBinder(new JpaParameters(indexedParametersWithSort), new Object[] { "name", sort }); assertThat(binder.getSort(), is(sort)); } @Test // DATAJPA-107 public void shouldSetTemporalQueryParameterToDate() throws Exception { Method method = SampleRepository.class.getMethod("validWithDefaultTemporalTypeParameter", Date.class); JpaParameters parameters = new JpaParameters(method); Date date = new Date(); new ParameterBinder(parameters, new Object[] { date }).bind(query); verify(query).setParameter(eq(1), eq(date), eq(TemporalType.DATE)); } @Test // DATAJPA-107 public void shouldSetTemporalQueryParameterToTimestamp() throws Exception { Method method = SampleRepository.class.getMethod("validWithCustomTemporalTypeParameter", Date.class); JpaParameters parameters = new JpaParameters(method); Date date = new Date(); new ParameterBinder(parameters, new Object[] { date }).bind(query); verify(query).setParameter(eq(1), eq(date), eq(TemporalType.TIMESTAMP)); } @Test(expected = IllegalArgumentException.class) // DATAJPA-107 public void shouldThrowIllegalArgumentExceptionIfIsAnnotatedWithTemporalParamAndParameterTypeIsNotDate() throws Exception { Method method = SampleRepository.class.getMethod("invalidWithTemporalTypeParameter", String.class); JpaParameters parameters = new JpaParameters(method); new ParameterBinder(parameters, new Object[] { "foo", "" }); } @Test // DATAJPA-461 public void shouldAllowBindingOfVarArgsAsIs() throws Exception { Method method = SampleRepository.class.getMethod("validWithVarArgs", Integer[].class); JpaParameters parameters = new JpaParameters(method); Integer[] ids = new Integer[] { 1, 2, 3 }; new ParameterBinder(parameters, new Object[] { ids }).bind(query); verify(query).setParameter(eq(1), eq(ids)); } @Test // DATAJPA-809 public void unwrapsOptionalParameter() throws Exception { Method method = SampleRepository.class.getMethod("optionalParameter", Optional.class); JpaParameters parameters = new JpaParameters(method); new ParameterBinder(parameters, new Object[] { Optional.of("Foo") }).bind(query); verify(query).setParameter(eq(1), eq("Foo")); } public SampleEntity findByEmbeddable(SampleEmbeddable embeddable) { return null; } @SuppressWarnings("unused") static class SampleEntity { private SampleEmbeddable embeddable; } @Embeddable @SuppressWarnings("unused") public static class SampleEmbeddable { private String foo; private String bar; } }