一个注解实现SpringBoot应用的动态配置,配置热重载最简洁的方案。
- ❤️ 无侵入,完全兼容SpringBoot原生的配置获取方式(@Value / @ConfigurationProperties)
- ⚡ 超轻量,超快响应, 不依赖SpringBoot核心库以外的任何三方库
- 😀 极易使用, 只提供一个简单的注解: @DynamicConfig;一个事件:ConfigurationChangedEvent
- ☸ 在K8S集群中和K8S ConfigMap完美结合的SpringBoot/SpringCloud应用的动态配置方式
- 不需要SpringCloud ConfigServer配置中心服务
- 不需要SpringCloud依赖,不需要@RefreshScope注解,不会重建Spring Bean
- 不需要配置中心服务器
- 不需要学习额外的注解和SDK API
注意:Spring Boot 2.4以下版本请使用 1.0.8 版本
Maven
<dependency>
<groupId>top.code2life</groupId>
<artifactId>spring-boot-dynamic-config</artifactId>
<version>1.0.9</version>
</dependency>
Gradle
implementation 'top.code2life:spring-boot-dynamic-config:1.0.9'
使用方法1: 在包含@Value成员的类上添加 @DynamicConfig。
import lombok.Data;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import top.code2life.config.DynamicConfig;
import java.util.Set;
@Data
@Component
@DynamicConfig // add annotation here !
public class DynamicFeatures {
@Value("${dynamic-test-plain:default}")
private String plainValue;
@Value("#{@featureGate.convert('${dynamic-feature-conf}')}")
private Set<String> someBetaFeatureConfig;
// @DynamicConfig // adding annotation here also works!
@Value("#{@testComponent.transform(${dynamic.transform-a:20}, ${dynamic.transform-b:10})} ")
private double transformBySpEL;
public double transform(double t1, double t2) {
return t1 / t2;
}
}
// file: application-profile.yml
// ============================
// dynamic-test-plain: someVal # kebab-case is recommended
// dynamicFeatureConf: a,b,c # camelCase compatible
// dynamic:
// transform-a: 100
// transform-b: 10
使用方法2: 在有 @ConfigurationProperties 注解的类上添加 @DynamicConfig。
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;
import top.code2life.config.DynamicConfig;
import java.util.Map;
@Data
@DynamicConfig // add annotation here !
@Configuration
@ConfigurationProperties(prefix = "my-prop")
public class TestConfigurationProperties {
private String str;
private Double doubleVal;
private Map<String, Object> mapVal;
}
// file: application-another-profile.yml
// ============================
// my-prop: # or myProp, relax binding supported
// str: someVal
// double-val: 100.0
// mapVal:
// k: v
java -jar your-spring-boot-app.jar --spring.config.location=/path/to/config
在Dynamic Config的1.0.9版本支持了Spring Boot 2.4之后版本的 'spring.config.import' 特性,并且增强了原生的 configtree 特性!
# config.import could be used with config.location TOGETHER
java -jar your-spring-boot-app.jar --spring.config.import=/path/to/configA.yaml,/path/to/configB.yaml
# or use the config tree feature
java -jar your-spring-boot-app.jar --spring.config.import=configtree:/path/to/conf-dir/
启动后配置路径下的任何文件修改(/path/to/config/application-xxx.yml)都会在相关联的注有@DynamicConfig的Spring Bean里立即生效 ,getter方法可以直接获取到最新配置值。
Spring Boot 2.4之后支持了导入配置文件或目录的参数,比如'--spring.config.import=configtree:/path/to/conf-dir/' 这个启动参数会让Spring Boot递归查找所有子目录和文件,加载到Spring Environment中作为Property,Key就是文件相对路径名:
比如配置文件是: 'path/to/conf-dir/module-a/file-b.yaml', 则代码中可以用 'module-a.file-b.yaml' 取到文件的内容,但类型只有String,并没有加载成完整的PropertySource。
Dynamic Config 1.0.9 版本增强了这个特性,如果查找到带后缀的配置文件,会额外加载到Spring Environment中成为独立的、可热重载的PropertySource,使用方式如下。
举个例子,比如spring.config.import目录是 '/path/to/conf-dir', 子目录中有这个配置文件 '/path/to/conf-dir/module-a/file-b.yaml'
# file: /path/to/conf-dir/module-a/file-b.yaml
prop-c-in-file: example-value
prop-obj-key-infile:
fieldA: value
fieldB: value
代码中无需做任何变更,只需要加上文件路径的前缀即可,比如 module-a/file-b.yaml => module-a.file-b。
@Value("${module-a.file-b.prop-c-in-file}")
@DynamicConfig
private String loadedByConfigTree
// or
@DynamicConfig
@ConfigurationProperties(prefix = "module-a.file-b.prop-obj-key-infile")
public class PropsLoadedByConfigTree {
// ...
}
- 以代码的方式管理配置,Everything as Code;
- Git是维护配置信息的最佳版本控制系统,绝大多数应用,不需要配置管理中心这样的系统;
- 配置由持续集成系统自动化部署到开发/产线环境,而不是登陆到某个系统,手动输入配置值;
- Git Ops的方式做自动化运维,更符合DevOps的思想,在应用服务数量非常多的时候,也更具备伸缩性。
- 使用 --spring.config.location 参数启动时,初始化 DynamicConfigPropertiesWatcher 这个Bean;
- 与此同时,初始化 DynamicConfigBeanPostProcessor 这个BeanPostProcessor,用来处理 @DynamicConfig;
- DynamicConfigBeanPostProcessor 收集所有带有 @DynamicConfig 注解的Bean的元数据,包括Bean的名称、实例、@Value成员变量等等;
- DynamicConfigPropertiesWatcher 开始监听 spring.config.location 目录下所有的文件变动,对于变化的Yaml/Properties,生成PropertySource动态替换Environment Bean中的PropertySource;
- DynamicConfigPropertiesWatcher 在ApplicationContext中发布配置变动的Event:ConfigurationChangedEvent;
- DynamicConfigBeanPostProcessor 订阅了上述事件,计算变动的文件,对哪些Key造成了差异
- 对于每个有差异的Property Key, DynamicConfigBeanPostProcessor 比对记录的Bean元数据,找到相关联的Bean
- 对于这次变动相关联的Bean, 调用反射方法,或 ConfigurationPropertiesBindingPostProcessor 的API,绑定到当前Bean的成员变量中,实现配置动态生效
任何SpringBoot/SpringCloud应用都可以使用这个库,只要依赖的SpringBoot版本在SpringBoot 2.0以上即可。
- √ SpringBoot 2.4.x, 2.5.x, 2.6.x, 2.7.x, 3.0.0 and Above (1.0.9以上版本)
- √ SpringBoot 2.3.x (1.0.8及以下版本)
- √ SpringBoot 2.2.x (1.0.8及以下版本)
- √ SpringBoot 2.1.x (1.0.8及以下版本)
- √ SpringBoot 2.0.x (1.0.8及以下版本)
- X SpringBoot 1.5.x 不支持
注意:
- SpringBoot 2.0.x中不能使用JUnit5,只能使用JUnit4,因此在Spring 2.0.x版本跑本仓库里的单元测试,需要切换到JUnit4
- spring-boot-dynamic-config仅包含spring-boot库的编译依赖,不依赖任何其他三方库
Spring Boot Dynamic Config is Open Source software released under the Apache 2.0 license.