Externalized configuration in Spring Boot
Indrek Ots
by Indrek Ots
3 min read

Categories

  • articles

Tags

  • spring
  • spring boot
  • configuration
  • java

Externalized configuration is essential if you wish to deploy the same deployable unit to different environments without modifying the source code. I’m certain you have aspects in your program that should change based on the execution environment. For starters, common parameters that need adjusting are the database connection strings or some cache specific values. Spring Boot provides several easy methods of externalizing configuration.

Properties file

In a barebones Spring Boot application you have an application.properties file in your classpath which is typically in the resources folder. Spring Boot reads properties from this file without you having to configure anything. Application code can access these values using Spring’s Environment or have them be injected into beans via the @Value annotation. This works great in development but when you start to deploy your application to other environments (e.g. test, production) you’ll soon find out that you need to have different property values per environment. How can I do that with Spring Boot? I’m glad you asked, because that’s what we’ll look at next.

Profile-specific properties file

In addition to the default application.properties file we talked about, you can create profile-specific properties files. Profiles in Spring allow you to configure certain parts of your application to suit the needs of different environments. For example, your database connection string might be different in your test environment compared to your dev settings. To create a profile-specific properties file you need to follow the following naming convention application-{profile}.properties.

Let’s look at a trivial example. You have an application.porperties file with the following content.

name=My Spring application
greeting=Hello

Alongside you also have an application-test.properties with one property defined.

name=My Spring application in test

When you run your Spring Boot application, by default, property values from application.properties will be picked up. But when you start Spring Boot with -Dspring.profiles.active=test, you instruct Spring to look for properties form application-test.properties. I intentionally did not define a value for greeting in application-test.properties. When running with the test profile, Spring will first look into application-test.properties and for values which are not found there, it will use applicaiton.properties. Essentially profile-specific properties override non-specific ones.

Properties files outside of your packaged jar

Up until now we’ve looked how to handle properties files that are packaged inside the jar. But what if my settings are provided by a configuration management system (e.g. Ansible, Puppet, Chef) after the jar file has been packaged? Rather than looking into the classpath, this would require the application to read the file from the file system.

A naive approach would be to pass the JVM an argument pointing to your external configuration file (e.g. -Dconfig.location=/path/to/external.properties) and inside your application code read the properties from the specified file. This definitely works but Spring Boot offers a much simpler solution. Place your application.properties outside of your packaged jar file. Spring Boot will look in the same directory for properties and the values can be accessed as if the file was inside the classpath. The same applies for profile-specific properties files.

Evaluation order

Properties files that are placed outside of your packaged jar override the ones inside your jar. Profile-specific files always take precedence. Properties are considered in the following order:

  • Profile-specific application properties outside of your packaged jar
  • Profile-specific application properties packaged inside your jar
  • Application properties outside of your packaged jar
  • Application properties packaged inside your jar

Final words

You learned how to use configuration files with Spring Boot. As you can see, the framework takes care of externalized configuration for you. The developer’s job is to place the property values in correct files. I strongly encourage you to read more about configuration in Spring Boot. You’ll find that there are more ways of externalizing configuration than you saw in this post.