Spring Boot Configuration

When Do You Need Configuration Files?

In the context of using the Spring Boot framework, the need for configuration files often arises in the following scenarios:

  1. Database configurations differ between local or company VM setups and formal production environments.
  2. When testing API integration internally, the actual API endpoint may be inaccessible, requiring dynamic adjustment of API URL display.
  3. A single API server serves multiple clients, each with unique business requirements within the same functionality. Configuration files are essential to manage these differences.
  4. Managing numerous custom configuration files can become cumbersome. Centralizing them in a dedicated class is a preferred approach.

Setting Properties at Program Startup

By using the spring.profiles.active property, you can switch configuration files based on different environments. This can be configured in application.properties as follows:

Using properties format:

spring.profiles.active=local

Using YAML format:

spring:
  profiles:
    active: local

Alternatively, you can specify it through JVM startup parameters:

java -jar your-application.jar -DAPI_ENV=xxxxx

Then, reference it in your program as follows:

spring.profiles.active=${API_ENV}

Or in YAML:

spring:
  profiles:
    active: ${API_ENV}

You can also use @Value("${spring.profiles.active}") to inject a single configuration value.

Determining Which Bean to Activate Based on Configuration Values - @ConditionalOnProperty

@ConditionalOnProperty is a Spring annotation used to determine whether a specific bean should be activated during application startup.

Consider an interface CustomerAPIService which served as connecting with microservice or real world api. After that, there are two implementing classes, MockCustomerAPIServiceImpl and CustomerAPIServiceImpl. By using @ConditionalOnProperty, you can dynamically select the implementation based on the value of the mock.enable property:

interface CustomerAPIService {

}

@ConditionalOnProperty(name="mock.enable", havingValue = "true", matchIfMissing = false)
@Service
class MockCustomerAPIServiceImpl implements CustomerAPIService {

}

@ConditionalOnProperty(name="mock.enable", havingValue = "false")
@Service
class CustomerAPIServiceImpl implements CustomerAPIService {

}

This way, when mock.enable is set to true, Bean B will be created. When mock.enable is set to false or not specified, Bean C will be created.

Organizing Custom Configuration Files - @ConfigurationProperties

When dealing with multiple custom properties that need to be managed, you can use @ConfigurationProperties to consolidate related settings into one class:

@ConfigurationProperties(prefix = "myapp")
public class MyAppProperties {

    private String a;
    private String b;

    // Getter and Setter methods
}

Then, configure these properties in application.properties:

myapp.a=valueA
myapp.b=valueB

Inject the MyAppProperties into your Spring Boot application:

@Service
public class MyService {

    private final MyAppProperties myAppProperties;

    @Autowired
    public MyService(MyAppProperties myAppProperties) {
        this.myAppProperties = myAppProperties;
    }

    // Use the configuration values from myAppProperties
}

This approach provides a clearer way to manage and use multiple custom configuration files.

references