Align precedence of @SpringBootTest properties with @TestPropertySource
This commit updates the precedence of properties configured using @SpringBootTest to align with @TestPropertySource. Properties configured using properties on @SpringBootTest are now added to the same property source as those configured using properties on @TestPropertySource so the precedence described in the javadoc of @TestPropertySource now applies in full. Additionally, if both @TestPropertySource properties and @SpringBootTest properties configure the same property, the value from @TestPropertySource will win. Closes gh-4828
This commit is contained in:
parent
5f5db1703c
commit
69b0829199
|
@ -19,10 +19,8 @@ package org.springframework.boot.test.context;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.LinkedHashMap;
|
|
||||||
import java.util.LinkedHashSet;
|
import java.util.LinkedHashSet;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
import org.springframework.beans.BeanUtils;
|
import org.springframework.beans.BeanUtils;
|
||||||
|
@ -36,7 +34,6 @@ import org.springframework.context.ConfigurableApplicationContext;
|
||||||
import org.springframework.core.SpringVersion;
|
import org.springframework.core.SpringVersion;
|
||||||
import org.springframework.core.annotation.AnnotatedElementUtils;
|
import org.springframework.core.annotation.AnnotatedElementUtils;
|
||||||
import org.springframework.core.env.ConfigurableEnvironment;
|
import org.springframework.core.env.ConfigurableEnvironment;
|
||||||
import org.springframework.core.env.MapPropertySource;
|
|
||||||
import org.springframework.core.env.StandardEnvironment;
|
import org.springframework.core.env.StandardEnvironment;
|
||||||
import org.springframework.test.context.ContextConfigurationAttributes;
|
import org.springframework.test.context.ContextConfigurationAttributes;
|
||||||
import org.springframework.test.context.ContextCustomizer;
|
import org.springframework.test.context.ContextCustomizer;
|
||||||
|
@ -90,8 +87,6 @@ public class SpringBootContextLoader extends AbstractContextLoader {
|
||||||
if (!ObjectUtils.isEmpty(config.getActiveProfiles())) {
|
if (!ObjectUtils.isEmpty(config.getActiveProfiles())) {
|
||||||
setActiveProfiles(environment, config.getActiveProfiles());
|
setActiveProfiles(environment, config.getActiveProfiles());
|
||||||
}
|
}
|
||||||
Map<String, Object> properties = getEnvironmentProperties(config);
|
|
||||||
addProperties(environment, properties);
|
|
||||||
application.setEnvironment(environment);
|
application.setEnvironment(environment);
|
||||||
List<ApplicationContextInitializer<?>> initializers = getInitializers(config,
|
List<ApplicationContextInitializer<?>> initializers = getInitializers(config,
|
||||||
application);
|
application);
|
||||||
|
@ -135,29 +130,19 @@ public class SpringBootContextLoader extends AbstractContextLoader {
|
||||||
+ StringUtils.arrayToCommaDelimitedString(profiles));
|
+ StringUtils.arrayToCommaDelimitedString(profiles));
|
||||||
}
|
}
|
||||||
|
|
||||||
protected Map<String, Object> getEnvironmentProperties(
|
protected String[] getInlinedProperties(MergedContextConfiguration config) {
|
||||||
MergedContextConfiguration config) {
|
ArrayList<String> properties = new ArrayList<String>();
|
||||||
Map<String, Object> properties = new LinkedHashMap<String, Object>();
|
|
||||||
// JMX bean names will clash if the same bean is used in multiple contexts
|
// JMX bean names will clash if the same bean is used in multiple contexts
|
||||||
disableJmx(properties);
|
disableJmx(properties);
|
||||||
properties.putAll(TestPropertySourceUtils
|
properties.addAll(Arrays.asList(config.getPropertySourceProperties()));
|
||||||
.convertInlinedPropertiesToMap(config.getPropertySourceProperties()));
|
|
||||||
if (!isEmbeddedWebEnvironment(config)) {
|
if (!isEmbeddedWebEnvironment(config)) {
|
||||||
properties.put("server.port", "-1");
|
properties.add("server.port=-1");
|
||||||
}
|
}
|
||||||
return properties;
|
return properties.toArray(new String[properties.size()]);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void disableJmx(Map<String, Object> properties) {
|
private void disableJmx(List<String> properties) {
|
||||||
properties.put("spring.jmx.enabled", "false");
|
properties.add("spring.jmx.enabled=false");
|
||||||
}
|
|
||||||
|
|
||||||
private void addProperties(ConfigurableEnvironment environment,
|
|
||||||
Map<String, Object> properties) {
|
|
||||||
// @IntegrationTest properties go before external configuration and after system
|
|
||||||
environment.getPropertySources().addAfter(
|
|
||||||
StandardEnvironment.SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME,
|
|
||||||
new MapPropertySource("integrationTest", properties));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private List<ApplicationContextInitializer<?>> getInitializers(
|
private List<ApplicationContextInitializer<?>> getInitializers(
|
||||||
|
@ -166,8 +151,8 @@ public class SpringBootContextLoader extends AbstractContextLoader {
|
||||||
for (ContextCustomizer contextCustomizer : config.getContextCustomizers()) {
|
for (ContextCustomizer contextCustomizer : config.getContextCustomizers()) {
|
||||||
initializers.add(new ContextCustomizerAdapter(contextCustomizer, config));
|
initializers.add(new ContextCustomizerAdapter(contextCustomizer, config));
|
||||||
}
|
}
|
||||||
initializers.add(new PropertySourceLocationsInitializer(
|
initializers.add(new TestPropertySourcesInitializer(
|
||||||
config.getPropertySourceLocations()));
|
config.getPropertySourceLocations(), getInlinedProperties(config)));
|
||||||
initializers.addAll(application.getInitializers());
|
initializers.addAll(application.getInitializers());
|
||||||
for (Class<? extends ApplicationContextInitializer<?>> initializerClass : config
|
for (Class<? extends ApplicationContextInitializer<?>> initializerClass : config
|
||||||
.getContextInitializerClasses()) {
|
.getContextInitializerClasses()) {
|
||||||
|
@ -258,21 +243,27 @@ public class SpringBootContextLoader extends AbstractContextLoader {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@link ApplicationContextInitializer} to setup test property source locations.
|
* {@link ApplicationContextInitializer} to set up test property sources.
|
||||||
*/
|
*/
|
||||||
private static class PropertySourceLocationsInitializer
|
private static class TestPropertySourcesInitializer
|
||||||
implements ApplicationContextInitializer<ConfigurableApplicationContext> {
|
implements ApplicationContextInitializer<ConfigurableApplicationContext> {
|
||||||
|
|
||||||
private final String[] propertySourceLocations;
|
private final String[] propertySourceLocations;
|
||||||
|
|
||||||
PropertySourceLocationsInitializer(String[] propertySourceLocations) {
|
private final String[] inlinedProperties;
|
||||||
|
|
||||||
|
TestPropertySourcesInitializer(String[] propertySourceLocations,
|
||||||
|
String[] inlinedProperties) {
|
||||||
this.propertySourceLocations = propertySourceLocations;
|
this.propertySourceLocations = propertySourceLocations;
|
||||||
|
this.inlinedProperties = inlinedProperties;
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void initialize(ConfigurableApplicationContext applicationContext) {
|
public void initialize(ConfigurableApplicationContext applicationContext) {
|
||||||
TestPropertySourceUtils.addPropertiesFilesToEnvironment(applicationContext,
|
TestPropertySourceUtils.addPropertiesFilesToEnvironment(applicationContext,
|
||||||
this.propertySourceLocations);
|
this.propertySourceLocations);
|
||||||
|
TestPropertySourceUtils.addInlinedPropertiesToEnvironment(applicationContext,
|
||||||
|
this.inlinedProperties);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -206,7 +206,9 @@ public class SpringBootTestContextBootstrapper extends DefaultTestContextBootstr
|
||||||
Class<?> testClass = mergedConfig.getTestClass();
|
Class<?> testClass = mergedConfig.getTestClass();
|
||||||
String[] properties = getProperties(testClass);
|
String[] properties = getProperties(testClass);
|
||||||
if (!ObjectUtils.isEmpty(properties)) {
|
if (!ObjectUtils.isEmpty(properties)) {
|
||||||
propertySourceProperties.addAll(Arrays.asList(properties));
|
// Added first so that inlined properties from @TestPropertySource take
|
||||||
|
// precedence
|
||||||
|
propertySourceProperties.addAll(0, Arrays.asList(properties));
|
||||||
}
|
}
|
||||||
if (getWebEnvironment(testClass) == WebEnvironment.RANDOM_PORT) {
|
if (getWebEnvironment(testClass) == WebEnvironment.RANDOM_PORT) {
|
||||||
propertySourceProperties.add("server.port=0");
|
propertySourceProperties.add("server.port=0");
|
||||||
|
|
|
@ -1,85 +0,0 @@
|
||||||
/*
|
|
||||||
* Copyright 2012-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.boot.test.context;
|
|
||||||
|
|
||||||
import javax.annotation.PostConstruct;
|
|
||||||
|
|
||||||
import org.junit.Test;
|
|
||||||
import org.junit.runner.RunWith;
|
|
||||||
|
|
||||||
import org.springframework.beans.factory.annotation.Autowired;
|
|
||||||
import org.springframework.beans.factory.annotation.Value;
|
|
||||||
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
|
|
||||||
import org.springframework.context.annotation.Bean;
|
|
||||||
import org.springframework.context.annotation.Configuration;
|
|
||||||
import org.springframework.context.support.PropertySourcesPlaceholderConfigurer;
|
|
||||||
import org.springframework.core.env.Environment;
|
|
||||||
import org.springframework.test.annotation.DirtiesContext;
|
|
||||||
import org.springframework.test.context.TestPropertySource;
|
|
||||||
import org.springframework.test.context.junit4.SpringRunner;
|
|
||||||
|
|
||||||
import static org.assertj.core.api.Assertions.assertThat;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Tests for {@link SpringBootTest} with {@link TestPropertySource} locations.
|
|
||||||
*
|
|
||||||
* @author Phillip Webb
|
|
||||||
*/
|
|
||||||
@RunWith(SpringRunner.class)
|
|
||||||
@DirtiesContext
|
|
||||||
@SpringBootTest(webEnvironment = WebEnvironment.NONE, properties = "value1=123")
|
|
||||||
@TestPropertySource(properties = "value2=456", locations = "classpath:/test-property-source-annotation.properties")
|
|
||||||
public class SpringBootTestPropertyLocationTests {
|
|
||||||
|
|
||||||
@Autowired
|
|
||||||
private Environment environment;
|
|
||||||
|
|
||||||
@Test
|
|
||||||
public void loadedProperties() throws Exception {
|
|
||||||
assertThat(this.environment.getProperty("value1")).isEqualTo("123");
|
|
||||||
assertThat(this.environment.getProperty("value2")).isEqualTo("456");
|
|
||||||
assertThat(this.environment.getProperty("annotation-referenced"))
|
|
||||||
.isEqualTo("fromfile");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Configuration
|
|
||||||
static class Config {
|
|
||||||
|
|
||||||
@Value("${value1}")
|
|
||||||
private String value1;
|
|
||||||
|
|
||||||
@Value("${value2}")
|
|
||||||
private String value2;
|
|
||||||
|
|
||||||
@Value("${annotation-referenced}")
|
|
||||||
private String annotationReferenced;
|
|
||||||
|
|
||||||
@PostConstruct
|
|
||||||
void checkValues() {
|
|
||||||
assertThat(this.value1).isEqualTo("123");
|
|
||||||
assertThat(this.value2).isEqualTo("456");
|
|
||||||
assertThat(this.annotationReferenced).isEqualTo("fromfile");
|
|
||||||
}
|
|
||||||
|
|
||||||
@Bean
|
|
||||||
public static PropertySourcesPlaceholderConfigurer propertyPlaceholder() {
|
|
||||||
return new PropertySourcesPlaceholderConfigurer();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
|
@ -0,0 +1,113 @@
|
||||||
|
/*
|
||||||
|
* Copyright 2012-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.boot.test.context;
|
||||||
|
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.junit.runner.RunWith;
|
||||||
|
|
||||||
|
import org.springframework.beans.factory.annotation.Autowired;
|
||||||
|
import org.springframework.beans.factory.annotation.Value;
|
||||||
|
import org.springframework.boot.test.context.SpringBootTest.WebEnvironment;
|
||||||
|
import org.springframework.context.annotation.Bean;
|
||||||
|
import org.springframework.context.annotation.Configuration;
|
||||||
|
import org.springframework.context.support.PropertySourcesPlaceholderConfigurer;
|
||||||
|
import org.springframework.test.annotation.DirtiesContext;
|
||||||
|
import org.springframework.test.context.TestPropertySource;
|
||||||
|
import org.springframework.test.context.junit4.SpringRunner;
|
||||||
|
|
||||||
|
import static org.assertj.core.api.Assertions.assertThat;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Tests for using {@link SpringBootTest} with {@link TestPropertySource}.
|
||||||
|
*
|
||||||
|
* @author Phillip Webb
|
||||||
|
* @author Andy Wilkinson
|
||||||
|
*/
|
||||||
|
@RunWith(SpringRunner.class)
|
||||||
|
@DirtiesContext
|
||||||
|
@SpringBootTest(webEnvironment = WebEnvironment.NONE, properties = {
|
||||||
|
"boot-test-inlined=foo", "b=boot-test-inlined", "c=boot-test-inlined" })
|
||||||
|
@TestPropertySource(properties = { "property-source-inlined=bar",
|
||||||
|
"a=property-source-inlined",
|
||||||
|
"c=property-source-inlined" }, locations = "classpath:/test-property-source-annotation.properties")
|
||||||
|
public class SpringBootTestWithTestPropertySourceTests {
|
||||||
|
|
||||||
|
@Autowired
|
||||||
|
private Config config;
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void propertyFromSpringBootTestProperties() {
|
||||||
|
assertThat(this.config.bootTestInlined).isEqualTo("foo");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void propertyFromTestPropertySourceProperties() {
|
||||||
|
assertThat(this.config.propertySourceInlined).isEqualTo("bar");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void propertyFromTestPropertySourceLocations() {
|
||||||
|
assertThat(this.config.propertySourceLocation).isEqualTo("baz");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void propertyFromPropertySourcePropertiesOverridesPropertyFromPropertySourceLocations() {
|
||||||
|
assertThat(this.config.propertySourceInlinedOverridesPropertySourceLocation)
|
||||||
|
.isEqualTo("property-source-inlined");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void propertyFromBootTestPropertiesOverridesPropertyFromPropertySourceLocations() {
|
||||||
|
assertThat(this.config.bootTestInlinedOverridesPropertySourceLocation)
|
||||||
|
.isEqualTo("boot-test-inlined");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void propertyFromPropertySourcePropertiesOverridesPropertyFromBootTestProperties() {
|
||||||
|
assertThat(this.config.propertySourceInlinedOverridesBootTestInlined)
|
||||||
|
.isEqualTo("property-source-inlined");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Configuration
|
||||||
|
static class Config {
|
||||||
|
|
||||||
|
@Value("${boot-test-inlined}")
|
||||||
|
private String bootTestInlined;
|
||||||
|
|
||||||
|
@Value("${property-source-inlined}")
|
||||||
|
private String propertySourceInlined;
|
||||||
|
|
||||||
|
@Value("${property-source-location}")
|
||||||
|
private String propertySourceLocation;
|
||||||
|
|
||||||
|
@Value("${a}")
|
||||||
|
private String propertySourceInlinedOverridesPropertySourceLocation;
|
||||||
|
|
||||||
|
@Value("${b}")
|
||||||
|
private String bootTestInlinedOverridesPropertySourceLocation;
|
||||||
|
|
||||||
|
@Value("${c}")
|
||||||
|
private String propertySourceInlinedOverridesBootTestInlined;
|
||||||
|
|
||||||
|
@Bean
|
||||||
|
public static PropertySourcesPlaceholderConfigurer propertyPlaceholder() {
|
||||||
|
return new PropertySourcesPlaceholderConfigurer();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -1 +1,3 @@
|
||||||
annotation-referenced=fromfile
|
property-source-location=baz
|
||||||
|
a=property-source-location
|
||||||
|
b=property-source-location
|
||||||
|
|
Loading…
Reference in New Issue