/* * Copyright 2012-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.boot.context.properties.bind; import java.time.LocalDate; import java.util.ArrayList; import java.util.Collections; import java.util.List; import org.junit.Before; import org.junit.Rule; import org.junit.Test; import org.junit.internal.matchers.ThrowableMessageMatcher; import org.junit.rules.ExpectedException; import org.mockito.Answers; import org.mockito.InOrder; import org.springframework.boot.context.properties.source.ConfigurationPropertyName; import org.springframework.boot.context.properties.source.ConfigurationPropertySource; import org.springframework.boot.context.properties.source.MockConfigurationPropertySource; import org.springframework.core.annotation.AnnotationUtils; import org.springframework.core.convert.ConversionFailedException; import org.springframework.core.env.StandardEnvironment; import org.springframework.format.annotation.DateTimeFormat; import org.springframework.test.context.support.TestPropertySourceUtils; import static org.assertj.core.api.Assertions.assertThat; import static org.hamcrest.Matchers.containsString; import static org.hamcrest.Matchers.instanceOf; import static org.mockito.ArgumentMatchers.any; import static org.mockito.ArgumentMatchers.eq; import static org.mockito.ArgumentMatchers.isA; import static org.mockito.Mockito.inOrder; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.withSettings; /** * Tests for {@link Binder}. * * @author Phillip Webb * @author Madhura Bhave */ public class BinderTests { @Rule public ExpectedException thrown = ExpectedException.none(); private List<ConfigurationPropertySource> sources = new ArrayList<>(); private Binder binder; @Before public void setup() { this.binder = new Binder(this.sources); } @Test public void createWhenSourcesIsNullShouldThrowException() throws Exception { this.thrown.expect(IllegalArgumentException.class); this.thrown.expectMessage("Sources must not be null"); new Binder((Iterable<ConfigurationPropertySource>) null); } @Test public void bindWhenNameIsNullShouldThrowException() throws Exception { this.thrown.expect(IllegalArgumentException.class); this.thrown.expectMessage("Name must not be null"); this.binder.bind((ConfigurationPropertyName) null, Bindable.of(String.class), BindHandler.DEFAULT); } @Test public void bindWhenTargetIsNullShouldThrowException() throws Exception { this.thrown.expect(IllegalArgumentException.class); this.thrown.expectMessage("Target must not be null"); this.binder.bind(ConfigurationPropertyName.of("foo"), null, BindHandler.DEFAULT); } @Test public void bindToValueWhenPropertyIsMissingShouldReturnUnbound() throws Exception { this.sources.add(new MockConfigurationPropertySource()); BindResult<String> result = this.binder.bind("foo", Bindable.of(String.class)); assertThat(result.isBound()).isFalse(); } @Test public void bindToValueShouldReturnPropertyValue() throws Exception { this.sources.add(new MockConfigurationPropertySource("foo", 123)); BindResult<Integer> result = this.binder.bind("foo", Bindable.of(Integer.class)); assertThat(result.get()).isEqualTo(123); } @Test public void bindToValueShouldReturnPropertyValueFromSecondSource() throws Exception { this.sources.add(new MockConfigurationPropertySource("foo", 123)); this.sources.add(new MockConfigurationPropertySource("bar", 234)); BindResult<Integer> result = this.binder.bind("bar", Bindable.of(Integer.class)); assertThat(result.get()).isEqualTo(234); } @Test public void bindToValueShouldReturnConvertedPropertyValue() throws Exception { this.sources.add(new MockConfigurationPropertySource("foo", "123")); BindResult<Integer> result = this.binder.bind("foo", Bindable.of(Integer.class)); assertThat(result.get()).isEqualTo(123); } @Test public void bindToValueWhenMultipleCandidatesShouldReturnFirst() throws Exception { this.sources.add(new MockConfigurationPropertySource("foo", 123)); this.sources.add(new MockConfigurationPropertySource("foo", 234)); BindResult<Integer> result = this.binder.bind("foo", Bindable.of(Integer.class)); assertThat(result.get()).isEqualTo(123); } @Test public void bindToValueWithPlaceholdersShouldResolve() throws Exception { StandardEnvironment environment = new StandardEnvironment(); TestPropertySourceUtils.addInlinedPropertiesToEnvironment(environment, "bar=23"); this.sources.add(new MockConfigurationPropertySource("foo", "1${bar}")); this.binder = new Binder(this.sources, new PropertySourcesPlaceholdersResolver(environment)); BindResult<Integer> result = this.binder.bind("foo", Bindable.of(Integer.class)); assertThat(result.get()).isEqualTo(123); } @Test public void bindToValueWithMissingPlaceholdersShouldThrowException() throws Exception { StandardEnvironment environment = new StandardEnvironment(); this.sources.add(new MockConfigurationPropertySource("foo", "${bar}")); this.binder = new Binder(this.sources, new PropertySourcesPlaceholdersResolver(environment)); this.thrown.expect(BindException.class); this.thrown.expectCause(ThrowableMessageMatcher.hasMessage(containsString( "Could not resolve placeholder 'bar' in value \"${bar}\""))); this.binder.bind("foo", Bindable.of(Integer.class)); } @Test public void bindToValueShouldTriggerOnSuccess() throws Exception { this.sources.add(new MockConfigurationPropertySource("foo", "1", "line1")); BindHandler handler = mock(BindHandler.class, withSettings().defaultAnswer(Answers.CALLS_REAL_METHODS)); Bindable<Integer> target = Bindable.of(Integer.class); this.binder.bind("foo", target, handler); InOrder ordered = inOrder(handler); ordered.verify(handler).onSuccess(eq(ConfigurationPropertyName.of("foo")), eq(target), any(), eq(1)); } @Test public void bindToJavaBeanShouldReturnPopulatedBean() throws Exception { this.sources.add(new MockConfigurationPropertySource("foo.value", "bar")); JavaBean result = this.binder.bind("foo", Bindable.of(JavaBean.class)).get(); assertThat(result.getValue()).isEqualTo("bar"); } @Test public void bindToJavaBeanWhenNonIterableShouldReturnPopulatedBean() throws Exception { MockConfigurationPropertySource source = new MockConfigurationPropertySource( "foo.value", "bar"); this.sources.add(source.nonIterable()); JavaBean result = this.binder.bind("foo", Bindable.of(JavaBean.class)).get(); assertThat(result.getValue()).isEqualTo("bar"); } @Test public void bindToJavaBeanShouldTriggerOnSuccess() throws Exception { this.sources .add(new MockConfigurationPropertySource("foo.value", "bar", "line1")); BindHandler handler = mock(BindHandler.class, withSettings().defaultAnswer(Answers.CALLS_REAL_METHODS)); Bindable<JavaBean> target = Bindable.of(JavaBean.class); this.binder.bind("foo", target, handler); InOrder inOrder = inOrder(handler); inOrder.verify(handler).onSuccess(eq(ConfigurationPropertyName.of("foo.value")), eq(Bindable.of(String.class)), any(), eq("bar")); inOrder.verify(handler).onSuccess(eq(ConfigurationPropertyName.of("foo")), eq(target), any(), isA(JavaBean.class)); } @Test public void bindWhenHasMalformedDateShouldThrowException() throws Exception { this.thrown.expectCause(instanceOf(ConversionFailedException.class)); this.sources.add(new MockConfigurationPropertySource("foo", "2014-04-01")); this.binder.bind("foo", Bindable.of(LocalDate.class)); } @Test public void bindWhenHasAnnotationsShouldChangeConvertedValue() throws Exception { this.sources.add(new MockConfigurationPropertySource("foo", "2014-04-01")); DateTimeFormat annotation = AnnotationUtils.synthesizeAnnotation( Collections.singletonMap("iso", DateTimeFormat.ISO.DATE), DateTimeFormat.class, null); LocalDate result = this.binder .bind("foo", Bindable.of(LocalDate.class).withAnnotations(annotation)) .get(); assertThat(result.toString()).isEqualTo("2014-04-01"); } public static class JavaBean { private String value; public String getValue() { return this.value; } public void setValue(String value) { this.value = value; } } public enum ExampleEnum { FOO_BAR, BAR_BAZ, BAZ_BOO } }