ZooKeeper配置中心:简化你的开发部署过程
ZooKeeper配置中心:简化你的开发部署过程
dong4j基于 Zookeeper 实现的一个配置中心
现在架构存在的问题
- 配置分散
比如修改日志配置需要修改 xxx-api, xxx-webgis, …, xxx-server. - 只有 dev, test, pro
本地开发时需要修改 dubbo.server.version, 提交时忘记改回来造成服务调用出错. - 配置类太多, 且有重复的配置类, 不好管理
- 非开发环境下应用配置在 war 包中,关键配置明文显示
- 修改配置后, 需要重新打包部署
- 关键配置随时可修改, 一不小心就会造成生产事故
**为了统一管理配置 **.
现将配置从 pom 中迁入到 xxx-common-config, 使用 maven filter 实现根据不同环境从 ${env}.yyyyy.properties 获取配置替换 application.properties
配置 (@占位符替换)
使用 <context:property-placeholder/>
配合 @Value("${}")
实现自动注入配置到配置类.
新增 local 环境用于本地开发
但是仍然解决不了敏感配置的安全, 配置不能统一管理等问题.
解决方案
使用配置中心, 统一管理配置, 将敏感配置隔离出来.
借助 ZooKeeper 我们实现的配置信息存储方案具有的优点如下:
- 简单。尽管前期搭建 ZooKeeper 服务器集群较为麻烦, 但是实现该方案后, 修改配置整个过程变得简单很多。用户只要修改配置, 无需进行其他任何操作,
配置自动生效。 - 可靠。ZooKeeper 服务集群具有无单点失效的特性, 使整个系统更加可靠。即使 ZooKeeper 集群中的一台机器失效, 也不会影响整体服务,
更不会影响分布式应用配置信息的更新。 - 实时。ZooKeeper 的数据更新通知机制, 可以在数据发生变化后, 立即通知给分布式应用程序, 具有很强的变化响应能力。
**能解决的问题 **
- 保证不同部署环境下应用配置的 **隔离性 **
- 保证非开发环境下应用配置的 **保密性 **
- 保证不同部署节点上同一应用配置的 **一致性 **
- 实现分布式环境下应用配置的 **可管理性 **
配置分类
现有配置
1 | # 环境变量 |
**来源 **
- 本地配置文件
- 数据库
- Redis
- Zookeeper
**读写频率 **
- 单次读取型
- dubbo.service.version
- Logback
- 多次读取型 (本地开发时动态切换环境)
- MongoDB
- Kafka
- Zookeeper
- JDBC
- Redis
- SSDB
- 动态读取型
- **字典数据 **
- waybill.upload
- 保险
- 支付
- 企查查
- 声网
- 上传地址
- 定位、话费、礼品兑换开关配置
- 商务端推送配置
- 兑吧配置
- 个推
- 中交配置
- 容联云配置
总体设计
系统架构
基础模型
- 用户通过配置中心修改 zk 节点配置
- client 监听到节点数据被修改, 获取 spring 容器配置类, 动态修改配置
序列图
Config center
功能:
- 统一管理配置
- 修改日志级别
- 修改日志采集率
4 查看应用状态 (是否在线, 上次部署时间等)
后期考虑增加的功能
- 应用下线邮件 (短信 / 微信) 提示
- kafka 操作, 查看信息
- dubbo 服务注册信息
Client
启动时读取配置,运行时根据配置调整行为
Zookeeper 配置节点设计
**只有 local 环境才有人员节点 **
只有 local 是本地开发配置, 因为开发需要, 有时会切换不同配置, 因此将配置分配到具体开发者身上, 这样修改一个配置时, 只会对某个开发者生效,
不影响其他人
**节点不能共用 **
意思是不能抽取出公共的配置, 因为修改了公共配置, 监听此节点的 client 都会修改配置.
模块说明
Client 端
Client 端暂时放在 trace 模块中, 方便开发测试
配置中心 admin
1 | . |
技术选型
Curator(馆长, 管理员)
替代 Zkclient 的另一个简单强大的 Zookeeper 客户端, (Curator)馆长与 (Zookeeper) 动物园, 天生一对 🤣🤣
Curator 包含了几个包:
- curator-framework:对 zookeeper 的底层 api 的一些封装
- curator-client:提供一些客户端的操作,例如重试策略等
- curator-recipes:封装了一些高级特性,如:Cache 事件监听、选举、分布式锁、分布式计数器、分布式 Barrier 等
Spring Boot
用于开发配置中心的 Web 端框架, 实现快速开发, 简单部署.
开发配置管理解决方案
clone 代码到本地 (git 并没有管理 xxx-common-config 中的 application.yml)
maven profile 选择 local(默认)
执行 xxx-common-config 的 pullConfig() 方法, 用于拉取配置
完成后会在 resource 目录下生成 application.yml 文件, 用于编译时替换占位符 (这里是考虑使用 @Bean 注入 jdbc, redis, kafka, mongodb 等对象,
但是 dubbo 和 logback 注入有问题, 以后会优化这里)1
env: local
zookeeper 配置
zookeeper:
connect:
list: 192.168.2.8:2181
dubbo 配置
dubbo:
service:
version: 5.2.8
…
1 | 其他配置这会根据 maven profile , 启动 app 时动态注入. |
这样可以避免切换环境后还要手动修改配置
开发时切换配置
可能有这样的需求, 我们开始在 local 环境开发, test 环境有出现 bug, 为了复现 bug, 我们可能会切换到 test 环境.
这个的解决方案:
我们不需要重启应用, 直接在 Web 端切换这个开发者的配置即可.
原理:
切换功能只会将 test 的配置替换到当前开发者节点下的 local 配置, 然后自动调用 zk 监听机制, 动态修改配置即可.
Dev Test Prod 环境部署解决方案
当需要部署到非 local 环境的服务器时, 需要执行 xxx-common-config 中的 pullDeployConfig() 方法
pullConfig() 与 pullDeployConfig() 的区别:
因开发需要切换环境时, 调用 pullConfig() 后不会覆盖 application.yml 中自定义配置.
因为部署时, 并不需要本地自定义配置, 因此 pullDeployConfig() 会将原来的 application.yml 重命名为 application.yml.local( 不会被打包到 war
中), 然后拉取配置生成 application.yml
第三方公司配置解决方案
第三方公司配置由本公司统一管理, 只需要在 configs 节点下增加一个子节点, 比如 chengtong, 然后再添加配置节点
打包时, 切换到 chengtong 分支, maven 选择 chengtong profile, 如果需要个性化配置, 在本地新建 application.yml 配置即可, 默认会以本地优先 (
只针对第三方公司)
如果是 xxx 公司, 以 dev,test,prod 环境打包时, 则会忽略本地配置文件.
原因是开发时拉取配置, 个性化配置后 (修改 dubbo service version, 修改日志输出等), 不小心提交了本地配置, 打包时不影响, 依然是以 zk 配置为主.
Todo list
- 迁移项目中的配置文件及类到
xxx-common-config
中- 使用 @Value(“${…}”) 替换 xml 的 bean 配置
- 整理 xxx-common-config 中的配置映射到 zk 的节点
- 实现一键创建 zk 节点
- shell 命令或者 Go 实现
- 从数据源(zk、mysql、redis)读取配置信息写入 properties
- 实现 pullConfig() 方法
- 合并本地配置, 切换环境时, 哪些配置不需要覆盖
- 实现 pullDeployConfig() 方法
- 文件重命名
- 重新生成 properties .yml
- 实现 pullConfig() 方法
- mvn 打包前,读取配置替换 xml 中的占位符(dubbo、logstash)
- 删除多余环境配置, 只保留 application.yml
- 修改 maven 的 profiles
- 修改 maven 的 build filter
- 从 Java 对象中映射到 Env 环境配置类中
- 启动时读取 zk(mysql, redis) 配置 (不包括 jdbc, redis, logback..)
- 在实例化 bean 之前, 将配置动态注入到 Spring Environment 对象中
- 启动完成后添加 listener, 监听配置变化
- 实现 callback, 动态修改配置