Merge branch '1.1.x'
This commit is contained in:
commit
cc51296397
|
|
@ -16,6 +16,7 @@
|
||||||
|
|
||||||
package org.springframework.boot.bind;
|
package org.springframework.boot.bind;
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collection;
|
import java.util.Collection;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
@ -24,12 +25,14 @@ import java.util.concurrent.ConcurrentHashMap;
|
||||||
import org.springframework.beans.MutablePropertyValues;
|
import org.springframework.beans.MutablePropertyValues;
|
||||||
import org.springframework.beans.PropertyValue;
|
import org.springframework.beans.PropertyValue;
|
||||||
import org.springframework.beans.PropertyValues;
|
import org.springframework.beans.PropertyValues;
|
||||||
|
import org.springframework.core.env.CompositePropertySource;
|
||||||
import org.springframework.core.env.EnumerablePropertySource;
|
import org.springframework.core.env.EnumerablePropertySource;
|
||||||
import org.springframework.core.env.PropertySource;
|
import org.springframework.core.env.PropertySource;
|
||||||
import org.springframework.core.env.PropertySources;
|
import org.springframework.core.env.PropertySources;
|
||||||
import org.springframework.core.env.PropertySourcesPropertyResolver;
|
import org.springframework.core.env.PropertySourcesPropertyResolver;
|
||||||
import org.springframework.core.env.StandardEnvironment;
|
import org.springframework.core.env.StandardEnvironment;
|
||||||
import org.springframework.util.PatternMatchUtils;
|
import org.springframework.util.PatternMatchUtils;
|
||||||
|
import org.springframework.util.ReflectionUtils;
|
||||||
import org.springframework.validation.DataBinder;
|
import org.springframework.validation.DataBinder;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
@ -73,50 +76,77 @@ public class PropertySourcesPropertyValues implements PropertyValues {
|
||||||
.toArray(new String[0]);
|
.toArray(new String[0]);
|
||||||
String[] exacts = names == null ? new String[0] : names.toArray(new String[0]);
|
String[] exacts = names == null ? new String[0] : names.toArray(new String[0]);
|
||||||
for (PropertySource<?> source : propertySources) {
|
for (PropertySource<?> source : propertySources) {
|
||||||
if (source instanceof EnumerablePropertySource) {
|
processPropertSource(source, resolver, includes, exacts);
|
||||||
EnumerablePropertySource<?> enumerable = (EnumerablePropertySource<?>) source;
|
}
|
||||||
if (enumerable.getPropertyNames().length > 0) {
|
}
|
||||||
for (String propertyName : enumerable.getPropertyNames()) {
|
|
||||||
if (this.NON_ENUMERABLE_ENUMERABLES.contains(source.getName())
|
private void processPropertSource(PropertySource<?> source,
|
||||||
&& !PatternMatchUtils.simpleMatch(includes, propertyName)) {
|
PropertySourcesPropertyResolver resolver, String[] includes, String[] exacts) {
|
||||||
continue;
|
if (source instanceof EnumerablePropertySource) {
|
||||||
}
|
EnumerablePropertySource<?> enumerable = (EnumerablePropertySource<?>) source;
|
||||||
Object value = source.getProperty(propertyName);
|
if (enumerable.getPropertyNames().length > 0) {
|
||||||
try {
|
for (String propertyName : enumerable.getPropertyNames()) {
|
||||||
value = resolver.getProperty(propertyName);
|
if (this.NON_ENUMERABLE_ENUMERABLES.contains(source.getName())
|
||||||
}
|
&& !PatternMatchUtils.simpleMatch(includes, propertyName)) {
|
||||||
catch (RuntimeException ex) {
|
continue;
|
||||||
// Probably could not resolve placeholders, ignore it here
|
}
|
||||||
}
|
Object value = source.getProperty(propertyName);
|
||||||
if (!this.propertyValues.containsKey(propertyName)) {
|
try {
|
||||||
this.propertyValues.put(propertyName, new PropertyValue(
|
value = resolver.getProperty(propertyName);
|
||||||
propertyName, value));
|
}
|
||||||
}
|
catch (RuntimeException ex) {
|
||||||
|
// Probably could not resolve placeholders, ignore it here
|
||||||
|
}
|
||||||
|
if (!this.propertyValues.containsKey(propertyName)) {
|
||||||
|
this.propertyValues.put(propertyName, new PropertyValue(
|
||||||
|
propertyName, value));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else {
|
}
|
||||||
// We can only do exact matches for non-enumerable property names, but
|
else if (source instanceof CompositePropertySource) {
|
||||||
// that's better than nothing...
|
CompositePropertySource composite = (CompositePropertySource) source;
|
||||||
for (String propertyName : exacts) {
|
for (PropertySource<?> nested : extractSources(composite)) {
|
||||||
Object value;
|
processPropertSource(nested, resolver, includes, exacts);
|
||||||
value = resolver.getProperty(propertyName);
|
}
|
||||||
if (value != null && !this.propertyValues.containsKey(propertyName)) {
|
}
|
||||||
this.propertyValues.put(propertyName, new PropertyValue(
|
else {
|
||||||
propertyName, value));
|
// We can only do exact matches for non-enumerable property names, but
|
||||||
continue;
|
// that's better than nothing...
|
||||||
}
|
for (String propertyName : exacts) {
|
||||||
value = source.getProperty(propertyName.toUpperCase());
|
Object value;
|
||||||
if (value != null && !this.propertyValues.containsKey(propertyName)) {
|
value = resolver.getProperty(propertyName);
|
||||||
this.propertyValues.put(propertyName, new PropertyValue(
|
if (value != null && !this.propertyValues.containsKey(propertyName)) {
|
||||||
propertyName, value));
|
this.propertyValues.put(propertyName, new PropertyValue(propertyName,
|
||||||
continue;
|
value));
|
||||||
}
|
continue;
|
||||||
|
}
|
||||||
|
value = source.getProperty(propertyName.toUpperCase());
|
||||||
|
if (value != null && !this.propertyValues.containsKey(propertyName)) {
|
||||||
|
this.propertyValues.put(propertyName, new PropertyValue(propertyName,
|
||||||
|
value));
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private Collection<PropertySource<?>> extractSources(CompositePropertySource composite) {
|
||||||
|
Field field = ReflectionUtils.findField(CompositePropertySource.class,
|
||||||
|
"propertySources");
|
||||||
|
field.setAccessible(true);
|
||||||
|
try {
|
||||||
|
@SuppressWarnings("unchecked")
|
||||||
|
Collection<PropertySource<?>> collection = (Collection<PropertySource<?>>) field
|
||||||
|
.get(composite);
|
||||||
|
return collection;
|
||||||
|
}
|
||||||
|
catch (Exception e) {
|
||||||
|
throw new IllegalStateException(
|
||||||
|
"Cannot extract property sources from composite", e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public PropertyValue[] getPropertyValues() {
|
public PropertyValue[] getPropertyValues() {
|
||||||
Collection<PropertyValue> values = this.propertyValues.values();
|
Collection<PropertyValue> values = this.propertyValues.values();
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,125 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-2013 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.bind;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.util.Collections;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.springframework.context.support.StaticMessageSource;
|
||||||
|
import org.springframework.core.env.CompositePropertySource;
|
||||||
|
import org.springframework.core.env.MapPropertySource;
|
||||||
|
import org.springframework.core.env.MutablePropertySources;
|
||||||
|
import org.springframework.core.io.ByteArrayResource;
|
||||||
|
import org.springframework.core.io.support.PropertiesLoaderUtils;
|
||||||
|
import org.springframework.validation.Validator;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for {@link PropertiesConfigurationFactory}.
|
||||||
|
*
|
||||||
|
* @author Dave Syer
|
||||||
|
*/
|
||||||
|
public class PropertiesConfigurationFactoryMapTests {
|
||||||
|
|
||||||
|
private PropertiesConfigurationFactory<Foo> factory;
|
||||||
|
|
||||||
|
private Validator validator;
|
||||||
|
|
||||||
|
private boolean ignoreUnknownFields = true;
|
||||||
|
|
||||||
|
private String targetName = null;
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testValidPropertiesLoadsWithNoErrors() throws Exception {
|
||||||
|
Foo foo = createFoo("map.name: blah\nmap.bar: blah");
|
||||||
|
assertEquals("blah", foo.map.get("bar"));
|
||||||
|
assertEquals("blah", foo.map.get("name"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBindToNamedTarget() throws Exception {
|
||||||
|
this.targetName = "foo";
|
||||||
|
Foo foo = createFoo("hi: hello\nfoo.map.name: foo\nfoo.map.bar: blah");
|
||||||
|
assertEquals("blah", foo.map.get("bar"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBindFromPropertySource() throws Exception {
|
||||||
|
this.targetName = "foo";
|
||||||
|
setupFactory();
|
||||||
|
MutablePropertySources sources = new MutablePropertySources();
|
||||||
|
sources.addFirst(new MapPropertySource("map", Collections.singletonMap(
|
||||||
|
"foo.map.name", (Object) "blah")));
|
||||||
|
this.factory.setPropertySources(sources);
|
||||||
|
this.factory.afterPropertiesSet();
|
||||||
|
Foo foo = this.factory.getObject();
|
||||||
|
assertEquals("blah", foo.map.get("name"));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBindFromCompositePropertySource() throws Exception {
|
||||||
|
this.targetName = "foo";
|
||||||
|
setupFactory();
|
||||||
|
MutablePropertySources sources = new MutablePropertySources();
|
||||||
|
CompositePropertySource composite = new CompositePropertySource("composite");
|
||||||
|
composite.addPropertySource(new MapPropertySource("map", Collections
|
||||||
|
.singletonMap("foo.map.name", (Object) "blah")));
|
||||||
|
sources.addFirst(composite);
|
||||||
|
this.factory.setPropertySources(sources);
|
||||||
|
this.factory.afterPropertiesSet();
|
||||||
|
Foo foo = this.factory.getObject();
|
||||||
|
assertEquals("blah", foo.map.get("name"));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Foo createFoo(final String values) throws Exception {
|
||||||
|
setupFactory();
|
||||||
|
return bindFoo(values);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Foo bindFoo(final String values) throws Exception {
|
||||||
|
this.factory.setProperties(PropertiesLoaderUtils
|
||||||
|
.loadProperties(new ByteArrayResource(values.getBytes())));
|
||||||
|
this.factory.afterPropertiesSet();
|
||||||
|
return this.factory.getObject();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setupFactory() throws IOException {
|
||||||
|
this.factory = new PropertiesConfigurationFactory<Foo>(Foo.class);
|
||||||
|
this.factory.setValidator(this.validator);
|
||||||
|
this.factory.setTargetName(this.targetName);
|
||||||
|
this.factory.setIgnoreUnknownFields(this.ignoreUnknownFields);
|
||||||
|
this.factory.setMessageSource(new StaticMessageSource());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Foo needs to be public and to have setters for all properties
|
||||||
|
public static class Foo {
|
||||||
|
private Map<String, Object> map = new HashMap<String, Object>();
|
||||||
|
|
||||||
|
public Map<String, Object> getMap() {
|
||||||
|
return this.map;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setMap(Map<String, Object> map) {
|
||||||
|
this.map = map;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
@ -20,6 +20,7 @@ import java.util.Collections;
|
||||||
|
|
||||||
import org.junit.Before;
|
import org.junit.Before;
|
||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
import org.springframework.core.env.CompositePropertySource;
|
||||||
import org.springframework.core.env.MapPropertySource;
|
import org.springframework.core.env.MapPropertySource;
|
||||||
import org.springframework.core.env.MutablePropertySources;
|
import org.springframework.core.env.MutablePropertySources;
|
||||||
import org.springframework.core.env.PropertySource;
|
import org.springframework.core.env.PropertySource;
|
||||||
|
|
@ -66,6 +67,17 @@ public class PropertySourcesPropertyValuesTests {
|
||||||
assertEquals("bar", propertyValues.getPropertyValue("foo").getValue());
|
assertEquals("bar", propertyValues.getPropertyValue("foo").getValue());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCompositeValue() {
|
||||||
|
PropertySource<?> map = this.propertySources.get("map");
|
||||||
|
CompositePropertySource composite = new CompositePropertySource("composite");
|
||||||
|
composite.addPropertySource(map);
|
||||||
|
this.propertySources.replace("map", composite);
|
||||||
|
PropertySourcesPropertyValues propertyValues = new PropertySourcesPropertyValues(
|
||||||
|
this.propertySources);
|
||||||
|
assertEquals("bar", propertyValues.getPropertyValue("foo").getValue());
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testEnumeratedValue() {
|
public void testEnumeratedValue() {
|
||||||
PropertySourcesPropertyValues propertyValues = new PropertySourcesPropertyValues(
|
PropertySourcesPropertyValues propertyValues = new PropertySourcesPropertyValues(
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue