Hello friends!!! Once again lets us discuss one of the important topics for spring boot is customizing spring boot autoconfiguration. As all we know that Freedom of choice is an awesome thing. If you like PIZZA so you have many choices to select your favorite PIZZA like PAN PIZZA, PIZZA with cheese bust, pizza with different different tops yummy :). Now friends please come to our discussion, here we’re going to look at two ways to influence auto-configuration: explicit configuration overrides and fine-grained configuration with properties.
Spring Boot auto-configuration attempts to automatically configure your Spring application based on the jar dependencies that you have added. If you can get what you need without it then why would you do extra work, writing and maintaining extra configuration code, but sometimes there are many cases Spring Boot auto configuration is not good enough. For example one of the cases is when you’re applying security to your application. Security in your application is one of major concern so it is not single fit for all because there are decisions around application security that Spring Boot has no business making for you even though Spring Boot provides auto configuration for some basic spring security things.
Just adding Spring Security Starter to add spring security auto configuration to the application. In gradle add following line to build.gradle file.
That’s it! Rebuild your application and run it. It’s now a secure web application! The security starter adds Spring Security to the application’s classpath. With Spring Security on the classpath, auto-configuration kicks in and a very basic Spring Security setup is created. When open this application browser basic authentication is needed then user name is “user” and password is printed in logs.
Basic default configuration probably is not fit for your application because unlike authentication page and password is printed in the logs. That is why you will prefer customized security configuration in the application. For overriding default spring boot auto configuration just writing explicit XML based or Java based configuration in the application this means writing a configuration class that extends WebSecurityConfigurerAdapter.
Spring Boot allows you to externalize your configuration so you can work with the same application code in different environments. You can use properties files, YAML files, environment variables, Java system properties, JNDI and command-line arguments to externalize configuration. Spring Boot offers over 300 properties for auto configuration of beans in the application.
Let’s see one of property when you start the application one banner is printed at log screen. So you can disable that banner by setting property spring.main.show-banner to false.
There are, in fact, several ways to set properties for a Spring Boot application. Spring Boot uses a very particular PropertySource order that is designed to allow sensible overriding of values. Properties are considered in the following order:
As for the application.properties and application.yml files, they can reside in any of four locations:
- Externally, in a /config subdirectory of the directory from which the application is run
- Externally, in the directory from which the application is run
- Internally, in a package named “config”
- Internally, at the root of the classpath
Again, this list is in order of precedence. That is, an application.properties file in a /config subdirectory will override the same properties set in an application.properties file in the application’s classpath.
Also, I’ve found that if you have both application.properties and application.yml side by side at the same level of precedence, properties in application.yml will override those in application.properties.
Auto-configuration
Spring Boot auto-configuration attempts to automatically configure your Spring application based on the jar dependencies that you have added. You should only ever add one @EnableAutoConfiguration annotation.
Gradually replacing auto-configuration
Auto-configuration is noninvasive, at any point you can start to define your own configuration to replace specific parts of the auto-configuration. For example, if you add your own DataSource bean, the default embedded database support will back away.
Disabling specific auto-configuration
1. DISABLING TEMPLATE CACHING
Thymeleaf templates are cached by default that is why changes is never replicate unless you restart the application. You can disable Thymeleaf template caching by setting spring.thymeleaf.cache to false.
Using command-line argument:
$ java -jar myapp-0.0.1-SNAPSHOT.jar --spring.thymeleaf.cache=false
Using application.yml file
spring:
thymeleaf:
cache: false
You’ll want to make sure that this application.yml file doesn’t follow the application into production.
Using environment variable:
$ export spring_thymeleaf_cache=false
Others template caching can be turned off for Spring Boot’s other supported template options by setting these properties:
- spring.freemarker.cache (Freemarker)
- spring.groovy.template.cache (Groovy templates)
- spring.velocity.cache (Velocity)
2. CONFIGURING THE EMBEDDED SERVER
By default embedded server for spring boot application is tomcat with port 8080. It is fine in case of single application running but it will be become problem in case run multiple applications simultaneously. If all of the applications try to start a Tomcat server on the same port, there’ll be port collisions starting with the second application.
So to prevent these collisions of port we need to do is set the server.port property.
If this is a one-time change, it’s easy enough to do this as a command-line argument:
$ java -jar myapp-0.0.1-SNAPSHOT.jar --server.port=8181
For permanent port change use application.yml:
server:
port: 8000
3. CONFIGURING LOGGING
By default, Spring Boot configures logging via Logback (http://logback.qos.ch) to log to the console at INFO level.
For Maven builds, you can exclude Logback by excluding the default logging starter transitively resolved by the root starter dependency:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
<exclusions>
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-logging</artifactId>
</exclusion>
</exclusions>
</dependency>
In Gradle, it’s easiest to place the exclusion under the configurations section:
configurations {
all*.exclude group:'org.springframework.boot',
module:'spring-boot-starter-logging'
}
With the default logging starter excluded, you can now include the starter for the logging implementation you’d rather use. With a Maven build you can add Log4j like this:
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-log4j</artifactId>
</dependency>
In a Gradle build you can add Log4j like this:
compile("org.springframework.boot:spring-boot-starter-log4j")
For full control over the logging configuration, you can create a logback.xml file at the root of the classpath (in src/main/resources). Here’s an example of a simple logback.xml file you might use:
<configuration>
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>
%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n
</pattern>
</encoder>
</appender>
<logger name="root" level="INFO"/>
<root level="INFO">
<appender-ref ref="STDOUT" />
</root>
</configuration>
To set the logging levels, you create properties that are prefixed with
logging.level, followed by the name of the logger for which you want to set the logging level.
application.yml:
logging:
level:
root: WARN
org:
springframework:
security: DEBUG
Now suppose that you want to write the log entries to a file named
MyApp.log at
/opt/logs/. The
logging.path and
logging.file properties can help with that:
logging:
path: /opt/logs/
file: MyApp.log
level:
root: WARN
org:
springframework:
security: DEBUG
By default, the log files will rotate once they hit 10 megabytes in size.
application.properties:
logging.path=/opt/logs/
logging.file=MyApp.log
logging.level.root=WARN
logging.level.root.org.springframework.security=DEBUG
You can also change name of logger configuration file name i.e. you can change name of file
logback.xml to
log-config.xml by setting the property
logging.config in property file or
YML file.
logging:
config:
classpath:log-config.xml
4. CONFIGURING A DATA SOURCE
application.yml: if you’re using a MySQL database, your
application.yml file might look like this
spring:
datasource:
url: jdbc:mysql://localhost/dojdb
username: root
password: root
Specify driver name for database with property
spring.datasource
.driver-class-name property as follows
spring:
datasource:
url: jdbc:mysql://localhost/dojdb
username: root
password: root
driver-class-name: com.mysql.jdbc.Driver
Using JNDI for DataSource:
By using property
spring.datasource.jndi-name
spring:
datasource:
jndi-name: java:/comp/env/jdbc/dojDBDS
Type-safe Configuration Properties
If you are working with multiple properties or your data is hierarchical in nature then sometimes it may be problem in configuration properties. Spring Boot provides an alternative method of working with properties that allows strongly typed beans to govern and validate the configuration of your application.
For example:
@Component
@ConfigurationProperties(prefix="database")
public class DatabaseSettings {
private String dbname;
private String dburl;
// ... getters and setters
}
The
@EnableConfigurationProperties annotation is automatically applied to your project so that any beans annotated with
@ConfigurationProperties will be configured from the Environment properties. The
@ConfigurationProperties annotation won’t work unless you’ve enabled it by adding
@EnableConfigurationProperties in one of your Spring configuration classes. This style of configuration works particularly well with the
SpringApplication external
YAML configuration:
application.yml
database:
dbname: dojdb
dburl: jdbc:mysql//192.168.1.1:3309/dojdb
# additional configuration as required
To work with
@ConfigurationProperties beans you can just inject them in the same way as any other bean.
@Service
public class UserService {
@Autowired
private DatabaseSettings connection;
//...
@PostConstruct
public void openConnection() {
Server server = new Server();
this.connection.configure(server);
}
}
It is also possible to shortcut the registration of
@ConfigurationProperties bean definitions by simply listing the properties classes directly in the
@EnableConfigurationProperties annotation:
@Configuration
@EnableConfigurationProperties(DatabaseSettings.class)
public class DBConfiguration {
}
Using @ConfigurationProperties for outside beans:
We can use
@ConfigurationProperties for beans which either outside from the application or not in your control. Let‘s see below
@ConfigurationProperties(prefix = "foo")
@Bean
public FooComponent fooComponent() {
...
}
Any property defined with the foo prefix will be mapped onto that
FooComponent bean in a similar manner as the
DatabaseSettings example above.
Note: Biding Property in Spring Boot
It’s also worth noting that Spring Boot’s property resolver is clever enough to treat camel-cased properties as interchangeable with similarly named properties with hyphens or underscores. In other words, a property named
database.dbName is equivalent to both
database.db_name and
database.db-name. And also
DATABASE_DB_NAME as environment variable in OS.
@ConfigurationProperties Validation
Spring Boot will attempt to validate external configuration, by default using JSR-303 (if it is on the classpath). You can simply add
JSR-303 javax.validation constraint annotations to your @ConfigurationProperties class:
@Component
@ConfigurationProperties(prefix="database")
public class DatabaseSettings {
@NotNull
private String dbName;
@NotNull
private String dbUrl;
// ... getters and setters
}
Configuring with profiles
When applications are deployed to different runtime environments, there are usually some configuration details that will differ. The details of a database connection, for instance, are likely different in a development environment than in a quality assurance environment, and different still in a production environment.
@Configuration
@Profile("production")
public class ProductionConfiguration {
// ...
}
The
@Profile annotation used here requires that the “
production” profile be active at runtime for this configuration to be applied. If the “
production” profile isn’t active, this configuration will be ignored and applied any default auto configuration.
Activate profiles:
Profiles can be activated by setting the
spring.profiles.active property using any of the means available for setting any other configuration property. For example, you could activate the “
production” profile by running the application at the
command line like this:
$ java -jar MyApp-0.0.1-SNAPSHOT.jar -- spring.profiles.active=production
Or you can add the
spring.profiles.active property to
application.yml:
spring:
profiles:
active: production
Profile-specific properties:
If you’re using
application.properties to express configuration properties, you can provide profile-specific properties by creating additional properties files named with the pattern “
application-{profile}.properties”. Profile-specific properties are loaded from the same locations as standard
application.properties, with profile-specific files always overriding the non-specific ones irrespective of whether the profile-specific files are inside or outside your packaged jar.
For Example the
development configuration would be in a file named
application-development.properties and contain properties for verbose, console written logging:
logging.level.root=DEBUG
But for
production,
application-production.properties would configure logging to be at WARN level and higher and to write to a log file:
logging.path=/opt/logs/
logging.file=MyApp.log
logging.level.root=WARN
Multi-profile YAML documents:
You can specify multiple profile-specific YAML documents in a single file by using a
spring.profiles key to indicate when the document applies. For example:
logging:
level:
root: INFO
---
spring:
profiles: development
logging:
level:
root: DEBUG
---
spring:
profiles: production
logging:
path: /opt/
file: MyApp.log
level:
root: WARN
As you can see, this
application.yml file is divided into three sections by a set of triple hyphens (---). The second and third sections each specify a value for
spring .profiles. This property indicates which profile each section’s properties apply to.
The properties defined in the middle section apply to development because it sets
spring.profiles to “
development”. Similarly, the last section has
spring.profiles set to “
production”, making it applicable when the “
production” profile is active.
The first section, on the other hand, doesn’t specify a value for
spring.profiles. Therefore, its properties are common to all profiles or are defaults if the active profile doesn’t otherwise have the properties set.
Customizing error pages
Spring Boot offers this “
whitelabel” error page by default as part of auto-configuration. The default error handler that’s auto-configured by Spring Boot looks for a view whose name is “
error”. The easiest way to customize the error page is to create a custom view that will resolve for a view named “
error”.
Ultimately this depends on the view resolvers in place when the error view is being resolved. This includes
- Any bean that implements Spring’s View interface and has a bean ID of “error” (resolved by Spring’s BeanNameViewResolver)
- A Thymeleaf template named “error.html” if Thymeleaf is configured
- A FreeMarker template named “error.ftl” if FreeMarker is configured
- A Velocity template named “error.vm” if Velocity is configured
- A JSP template named “error.jsp” if using JSP views
Summary
Spring Boot Auto configuration do almost configuration for your application. When autoconfiguration doesn’t fit your needs, Spring Boot allows you to override and fine-tune the configuration it provides.
Happy Spring Boot Learning!!!