Ignore case when filtering out properties based on target name
Previously, PropertiesConfigurationFactory would only create a DefaultPropertyNamePatternsMatcher that ignored case if it was ignoring unknown fields. If the binding had a target name and unknown fields were not being ignored the matcher would consider the case when finding matches. This meant that SERVER_PORT would not being to ServerProperties.port as SERVER did not match the target name, server. This commit updates PropertiesConfigurationFactory to use a case-ignoring DefaultPropertyNamePatternsMatcher when binding with a target name. The tests have also been updated to test binding both with and without ignoring case and using either Properties or PropertySources. The above-described matching is only performed against PropertySources and the bug only occurred when using a target name, making it important to test all combinations. Closes gh-3745pull/3738/merge
parent
398d06e326
commit
fbe2e470f0
@ -0,0 +1,172 @@
|
|||||||
|
/*
|
||||||
|
* Copyright 2012-2015 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.Properties;
|
||||||
|
|
||||||
|
import javax.validation.constraints.NotNull;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
import org.junit.runners.Parameterized;
|
||||||
|
import org.junit.runners.Parameterized.Parameters;
|
||||||
|
import org.springframework.context.support.StaticMessageSource;
|
||||||
|
import org.springframework.core.env.MutablePropertySources;
|
||||||
|
import org.springframework.core.env.PropertiesPropertySource;
|
||||||
|
import org.springframework.core.io.ByteArrayResource;
|
||||||
|
import org.springframework.core.io.support.PropertiesLoaderUtils;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parameterized tests for {@link PropertiesConfigurationFactory}
|
||||||
|
*
|
||||||
|
* @author Dave Syer
|
||||||
|
* @author Andy Wilkinson
|
||||||
|
*/
|
||||||
|
@RunWith(Parameterized.class)
|
||||||
|
public class PropertiesConfigurationFactoryParameterizedTests {
|
||||||
|
|
||||||
|
private final boolean usePropertySource;
|
||||||
|
|
||||||
|
private String targetName;
|
||||||
|
|
||||||
|
private PropertiesConfigurationFactory<Foo> factory = new PropertiesConfigurationFactory<Foo>(
|
||||||
|
Foo.class);
|
||||||
|
|
||||||
|
@Parameters
|
||||||
|
public static Object[] parameters() {
|
||||||
|
return new Object[] { new Object[] { false, false },
|
||||||
|
new Object[] { false, true }, new Object[] { true, false },
|
||||||
|
new Object[] { true, true } };
|
||||||
|
}
|
||||||
|
|
||||||
|
public PropertiesConfigurationFactoryParameterizedTests(boolean ignoreUnknownFields,
|
||||||
|
boolean usePropertySource) {
|
||||||
|
this.factory.setIgnoreUnknownFields(ignoreUnknownFields);
|
||||||
|
this.usePropertySource = usePropertySource;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testValidPropertiesLoadsWithNoErrors() throws Exception {
|
||||||
|
Foo foo = createFoo("name: blah\nbar: blah");
|
||||||
|
assertEquals("blah", foo.bar);
|
||||||
|
assertEquals("blah", foo.name);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testValidPropertiesLoadsWithUpperCase() throws Exception {
|
||||||
|
Foo foo = createFoo("NAME: blah\nbar: blah");
|
||||||
|
assertEquals("blah", foo.bar);
|
||||||
|
assertEquals("blah", foo.name);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testUnderscore() throws Exception {
|
||||||
|
Foo foo = createFoo("spring_foo_baz: blah\nname: blah");
|
||||||
|
assertEquals("blah", foo.spring_foo_baz);
|
||||||
|
assertEquals("blah", foo.name);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBindToNamedTarget() throws Exception {
|
||||||
|
this.targetName = "foo";
|
||||||
|
Foo foo = createFoo("hi: hello\nfoo.name: foo\nfoo.bar: blah");
|
||||||
|
assertEquals("blah", foo.bar);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testBindToNamedTargetUppercaseUnderscores() throws Exception {
|
||||||
|
this.targetName = "foo";
|
||||||
|
Foo foo = createFoo("FOO_NAME: foo\nFOO_BAR: blah");
|
||||||
|
assertEquals("blah", foo.bar);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Foo createFoo(final String values) throws Exception {
|
||||||
|
setupFactory();
|
||||||
|
return bindFoo(values);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Foo bindFoo(final String values) throws Exception {
|
||||||
|
Properties properties = PropertiesLoaderUtils
|
||||||
|
.loadProperties(new ByteArrayResource(values.getBytes()));
|
||||||
|
if (this.usePropertySource) {
|
||||||
|
MutablePropertySources propertySources = new MutablePropertySources();
|
||||||
|
propertySources.addFirst(new PropertiesPropertySource("test", properties));
|
||||||
|
this.factory.setPropertySources(propertySources);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
this.factory.setProperties(properties);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.factory.afterPropertiesSet();
|
||||||
|
return this.factory.getObject();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void setupFactory() throws IOException {
|
||||||
|
this.factory.setTargetName(this.targetName);
|
||||||
|
this.factory.setMessageSource(new StaticMessageSource());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Foo needs to be public and to have setters for all properties
|
||||||
|
public static class Foo {
|
||||||
|
|
||||||
|
@NotNull
|
||||||
|
private String name;
|
||||||
|
|
||||||
|
private String bar;
|
||||||
|
|
||||||
|
private String spring_foo_baz;
|
||||||
|
|
||||||
|
private String fooBar;
|
||||||
|
|
||||||
|
public String getSpringFooBaz() {
|
||||||
|
return this.spring_foo_baz;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setSpringFooBaz(String spring_foo_baz) {
|
||||||
|
this.spring_foo_baz = spring_foo_baz;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return this.name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setName(String name) {
|
||||||
|
this.name = name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getBar() {
|
||||||
|
return this.bar;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setBar(String bar) {
|
||||||
|
this.bar = bar;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getFooBar() {
|
||||||
|
return this.fooBar;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setFooBar(String fooBar) {
|
||||||
|
this.fooBar = fooBar;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in New Issue