Commit 80754d5c authored by Stephane Nicoll's avatar Stephane Nicoll

Polish

Closes gh-3534
parent bf303b02
...@@ -71,7 +71,7 @@ ...@@ -71,7 +71,7 @@
* link:spring-boot-sample-profile[spring-boot-sample-profile] * link:spring-boot-sample-profile[spring-boot-sample-profile]
-- example showing Spring's `@profile` support -- example showing Spring's `@profile` support
* link:spring-boot-sample-property-validation[spring-boot-sample-property-validation] * link:spring-boot-sample-property-validation[spring-boot-sample-property-validation]
-- example showing the usage of @ConfigurationProperties` with a Spring's `Validator` -- example showing the usage of `@ConfigurationProperties` with a Spring `Validator`
* link:spring-boot-sample-parent-context[spring-boot-sample-parent-context] * link:spring-boot-sample-parent-context[spring-boot-sample-parent-context]
-- example showing an `ApplicationContext` with a parent -- example showing an `ApplicationContext` with a parent
* link:spring-boot-sample-aop[spring-boot-sample-aop] * link:spring-boot-sample-aop[spring-boot-sample-aop]
......
buildscript {
ext {
springBootVersion = '1.3.0.BUILD-SNAPSHOT'
}
repositories {
// NOTE: You should declare only repositories that you need here
mavenLocal()
mavenCentral()
maven { url "http://repo.spring.io/release" }
maven { url "http://repo.spring.io/milestone" }
maven { url "http://repo.spring.io/snapshot" }
}
dependencies {
classpath("org.springframework.boot:spring-boot-gradle-plugin:${springBootVersion}")
}
}
apply plugin: 'java'
apply plugin: 'eclipse'
apply plugin: 'idea'
apply plugin: 'spring-boot'
jar {
baseName = 'spring-boot-sample-property-validation'
version = '0.0.0'
}
run {
systemProperties = System.properties
}
repositories {
// NOTE: You should declare only repositories that you need here
mavenLocal()
mavenCentral()
maven { url "http://repo.spring.io/release" }
maven { url "http://repo.spring.io/milestone" }
maven { url "http://repo.spring.io/snapshot" }
}
dependencies {
compile("org.springframework.boot:spring-boot-starter")
compile("org.hibernate:hibernate-validator")
testCompile("org.springframework.boot:spring-boot-starter-test")
}
task wrapper(type: Wrapper) {
gradleVersion = '1.6'
}
...@@ -23,11 +23,13 @@ ...@@ -23,11 +23,13 @@
<groupId>org.springframework.boot</groupId> <groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId> <artifactId>spring-boot-starter</artifactId>
</dependency> </dependency>
<dependency> <dependency>
<groupId>org.hibernate</groupId> <groupId>org.springframework.boot</groupId>
<artifactId>hibernate-validator</artifactId> <artifactId>spring-boot-configuration-processor</artifactId>
<version>5.1.3.Final</version> <optional>true</optional>
</dependency> </dependency>
<dependency> <dependency>
<groupId>org.springframework.boot</groupId> <groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId> <artifactId>spring-boot-starter-test</artifactId>
......
...@@ -20,12 +20,18 @@ import org.springframework.boot.context.properties.ConfigurationProperties; ...@@ -20,12 +20,18 @@ import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component; import org.springframework.stereotype.Component;
@Component @Component
@ConfigurationProperties @ConfigurationProperties(prefix = "sample")
public class SampleProperties { public class SampleProperties {
/**
* Sample host.
*/
private String host; private String host;
private Integer port; /**
* Sample port.
*/
private Integer port = 8080;
public String getHost() { public String getHost() {
return host; return host;
......
...@@ -16,46 +16,29 @@ ...@@ -16,46 +16,29 @@
package sample.propertyvalidation; package sample.propertyvalidation;
import org.springframework.boot.context.properties.ConfigurationProperties; import java.util.regex.Pattern;
import org.springframework.core.annotation.AnnotationUtils;
import org.springframework.stereotype.Component;
import org.springframework.validation.Errors; import org.springframework.validation.Errors;
import org.springframework.validation.ValidationUtils; import org.springframework.validation.ValidationUtils;
import org.springframework.validation.Validator; import org.springframework.validation.Validator;
import java.util.HashSet; public class SamplePropertiesValidator implements Validator {
import java.util.Set;
import java.util.regex.Pattern;
@Component(value = "configurationPropertiesValidator")
public class ConfigurationPropertiesValidator implements Validator {
public static final String IP_REGEX = "^(?:[0-9]{1,3}\\.){3}[0-9]{1,3}$";
final Pattern pattern = Pattern.compile(IP_REGEX); final Pattern pattern = Pattern.compile("^(?:[0-9]{1,3}\\.){3}[0-9]{1,3}$");
private Set<String> validatedClasses = new HashSet<String>() {{
add(SampleProperties.class.getName());
}};
@Override @Override
public boolean supports(Class<?> aClass) { public boolean supports(Class<?> type) {
return AnnotationUtils.findAnnotation(aClass, ConfigurationProperties.class) != null; return type == SampleProperties.class;
} }
@Override @Override
public void validate(Object o, Errors errors) { public void validate(Object o, Errors errors) {
if(validatedClasses.contains(o.getClass().getName())) {
doValidation(o, errors);
}
}
private void doValidation(Object o, Errors errors) {
ValidationUtils.rejectIfEmpty(errors, "host", "host.empty"); ValidationUtils.rejectIfEmpty(errors, "host", "host.empty");
ValidationUtils.rejectIfEmpty(errors, "port", "port.empty"); ValidationUtils.rejectIfEmpty(errors, "port", "port.empty");
SampleProperties properties = (SampleProperties) o; SampleProperties properties = (SampleProperties) o;
if(!pattern.matcher(properties.getHost()).matches()) { if (properties.getHost() != null &&
!pattern.matcher(properties.getHost()).matches()) {
errors.rejectValue("host", "Invalid host"); errors.rejectValue("host", "Invalid host");
} }
} }
......
...@@ -18,26 +18,42 @@ package sample.propertyvalidation; ...@@ -18,26 +18,42 @@ package sample.propertyvalidation;
import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner; import org.springframework.boot.CommandLineRunner;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.boot.builder.SpringApplicationBuilder;
import org.springframework.boot.context.properties.EnableConfigurationProperties; import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Profile;
import org.springframework.stereotype.Service;
import org.springframework.validation.Validator;
@SpringBootApplication @SpringBootApplication
@EnableConfigurationProperties @EnableConfigurationProperties
public class SamplePropertyValidationApplication implements CommandLineRunner { public class SamplePropertyValidationApplication {
@Bean
public Validator configurationPropertiesValidator() {
return new SamplePropertiesValidator();
}
@Service
@Profile("app")
static class Startup implements CommandLineRunner {
@Autowired @Autowired
private SampleProperties properties; private SampleProperties properties;
@Override @Override
public void run(String... args) { public void run(String... args) {
System.out.println("host: " + this.properties.getHost()); System.out.println("=========================================");
System.out.println("port:" + this.properties.getPort()); System.out.println("Sample host: " + this.properties.getHost());
System.out.println("Sample port: " + this.properties.getPort());
System.out.println("=========================================");
}
} }
public static void main(String[] args) throws Exception { public static void main(String[] args) throws Exception {
SpringApplication.run(SamplePropertyValidationApplication.class, args); new SpringApplicationBuilder(SamplePropertyValidationApplication.class)
.profiles("app").run(args);
} }
} }
host: 192.168.0.1 sample.host=192.168.0.1
port: 8080 sample.port=7070
\ No newline at end of file \ No newline at end of file
...@@ -16,15 +16,15 @@ ...@@ -16,15 +16,15 @@
package sample.propertyvalidation; package sample.propertyvalidation;
import org.junit.After;
import org.junit.Rule;
import org.junit.Test; import org.junit.Test;
import org.junit.rules.ExpectedException;
import org.springframework.beans.factory.BeanCreationException; import org.springframework.beans.factory.BeanCreationException;
import org.springframework.boot.autoconfigure.web.ServerProperties; import org.springframework.boot.autoconfigure.web.ServerProperties;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.boot.test.EnvironmentTestUtils; import org.springframework.boot.test.EnvironmentTestUtils;
import org.springframework.context.annotation.AnnotationConfigApplicationContext; import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.validation.Validator;
import static org.junit.Assert.assertEquals; import static org.junit.Assert.assertEquals;
...@@ -32,54 +32,64 @@ import static org.junit.Assert.assertEquals; ...@@ -32,54 +32,64 @@ import static org.junit.Assert.assertEquals;
* Tests for {@link SamplePropertyValidationApplication}. * Tests for {@link SamplePropertyValidationApplication}.
* *
* @author Lucas Saldanha * @author Lucas Saldanha
* @author Stephane Nicoll
*/ */
public class SamplePropertyValidationApplicationTests { public class SamplePropertyValidationApplicationTests {
@Rule
public final ExpectedException thrown = ExpectedException.none();
private final AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); private final AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
@After
public void closeContext() {
context.close();
}
@Test @Test
public void testBindingValidProperties() { public void bindValidProperties() {
this.context.register(TestConfiguration.class); this.context.register(SamplePropertyValidationApplication.class);
EnvironmentTestUtils.addEnvironment(this.context, "host:192.168.0.1"); EnvironmentTestUtils.addEnvironment(this.context,
EnvironmentTestUtils.addEnvironment(this.context, "port:8080"); "sample.host:192.168.0.1", "sample.port:9090");
this.context.refresh(); this.context.refresh();
assertEquals(1, this.context.getBeanNamesForType(SampleProperties.class).length);
SampleProperties properties = this.context.getBean(SampleProperties.class); SampleProperties properties = this.context.getBean(SampleProperties.class);
assertEquals("192.168.0.1", properties.getHost()); assertEquals("192.168.0.1", properties.getHost());
assertEquals(8080, (int) properties.getPort()); assertEquals(Integer.valueOf(9090), properties.getPort());
} }
@Test(expected = BeanCreationException.class) @Test
public void testBindingInvalidProperties() { public void bindInvalidHost() {
this.context.register(TestConfiguration.class); this.context.register(SamplePropertyValidationApplication.class);
EnvironmentTestUtils.addEnvironment(this.context, "host:xxxxxx"); EnvironmentTestUtils.addEnvironment(this.context,
EnvironmentTestUtils.addEnvironment(this.context, "port:8080"); "sample.host:xxxxxx", "sample.port:9090");
thrown.expect(BeanCreationException.class);
thrown.expectMessage("xxxxxx");
this.context.refresh(); this.context.refresh();
} }
@Test @Test
public void testBindingValidPropertiesWithMultipleConfigurationPropertiesClasses() { public void bindNullHost() {
this.context.register(TestConfiguration.class); this.context.register(SamplePropertyValidationApplication.class);
this.context.register(ServerProperties.class);
EnvironmentTestUtils.addEnvironment(this.context, "host:192.168.0.1");
EnvironmentTestUtils.addEnvironment(this.context, "port:8080");
this.context.refresh();
assertEquals(1, this.context.getBeanNamesForType(SampleProperties.class).length); thrown.expect(BeanCreationException.class);
SampleProperties properties = this.context.getBean(SampleProperties.class); thrown.expectMessage("null");
assertEquals("192.168.0.1", properties.getHost()); thrown.expectMessage("host");
assertEquals(8080, (int) properties.getPort()); this.context.refresh();
} }
@Configuration @Test
@EnableConfigurationProperties(SampleProperties.class) public void validatorOnlyCalledOnSupportedClass() {
protected static class TestConfiguration { this.context.register(SamplePropertyValidationApplication.class);
this.context.register(ServerProperties.class); // our validator will not apply here
EnvironmentTestUtils.addEnvironment(this.context,
"sample.host:192.168.0.1", "sample.port:9090");
this.context.refresh();
@Bean SampleProperties properties = this.context.getBean(SampleProperties.class);
public Validator configurationPropertiesValidator() { assertEquals("192.168.0.1", properties.getHost());
return new ConfigurationPropertiesValidator(); assertEquals(Integer.valueOf(9090), properties.getPort());
}
} }
} }
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment