Nacos 配置中心源码 | 京东物流技术团队
data:image/s3,"s3://crabby-images/dd570/dd5704ced2677b5fdd424d23928c0b8868fb2d66" alt="Nacos 配置中心源码 | 京东物流技术团队"
客户端
入口
在引入配置中心 maven 依赖的 jar 文件中找到 spring-cloud-starter-alibaba-nacos-config-2.2.5.RELEASE.jar!/META-INF/spring.factories
,在该配置文件找到 NacosConfigBootstrapConfiguration 配置类,该类是 nacos 配置中心的入口类,类中注册了三个 bean。
data:image/s3,"s3://crabby-images/dac7e/dac7ef721e2c65cec9b99f84a9561d5f1f8fe9a5" alt=""
data:image/s3,"s3://crabby-images/120fb/120fb61928c2e5e84110e6280b856f5d1949970f" alt=""
NacosConfigProperties:属性配置类,对应配置文件中 spring.cloud.nacos.config 前缀的属性。
NacosConfigManager:管理 NacosConfigProperties 和 ConfigService。
NacosPropertySourceLocator:加载配置中心配置信息。
NacosConfigManager
在 NacosConfigManager 构造方法中,调用了 createConfigService 方法,该方法通过工厂类调用 ConfigService 实现类的构造方法创建 ConfigService 实例。
data:image/s3,"s3://crabby-images/64880/6488075114bb9575bc4187f2ab68ea48e61bd74e" alt=""
data:image/s3,"s3://crabby-images/74653/74653c8ee56d244d2b128f26bf9c776e41d5da08" alt=""
在 ConfigService 的实现类 NacosConfigService 的构造方法中会初始化 this.agent = new MetricsHttpAgent(new ServerHttpAgent(properties));,该 agent 是用来像服务端发送请求的代理。
data:image/s3,"s3://crabby-images/29a72/29a72ccb0d57c723aef8434d3b6c4202a27448d0" alt=""
data:image/s3,"s3://crabby-images/12d7c/12d7c6d06b7c0acba0aeea90839cff2763abebd9" alt=""
ServerHttpAgent 类中 NacosRestTemplate 属性是发送远程调用的工具类,会调用 HttpMethod.GET 方法调用服务端 rest 请求。
data:image/s3,"s3://crabby-images/3962b/3962bb41ba60c17e87c463ce63ed91192bda67f9" alt=""
在回到 NacosConfigService#NacosConfigService 的方法中 this.worker = new ClientWorker(this.agent, this.configFilterChainManager, properties); 该属性是客户端工作线程类,在类的内部有两个线程池:
1. 只有一个线程的线程池 this.executor = Executors.newScheduledThreadPool(1, new ThreadFactory()
用来执行定时任务,每隔 10ms 执行一次 checkConfigInfo(); 方法,按照每 3000 个配置项为一批次捞取待轮询的 cacheData 实例,将其包装成为一个 LongPollingTask 提交进入第二个线程池 executorService 处理。
data:image/s3,"s3://crabby-images/5f226/5f22647b77e4fc20242e602f0cbf15e74a718b3b" alt=""
data:image/s3,"s3://crabby-images/e909f/e909f6c09cce4a008c5e7e2e8660972ae2dba010" alt=""
2.线程数等于处理器个数的线程池,用来执行 ClientWorker.LongPollingRunnable#LongPollingRunnable#run,cacheMap 中缓存着需要刷新的配置,将 cacheMap 中数量以 3000 分一个组,分别创建一个 LongPollingRunnable 用来监听配置更新,在 LongPollingRunnable#run 方法中调用 checkLocalConfig(cacheData); 检查本地的配置,容错的处理;调用 checkUpdateDataIds(cacheDatas, inInitializingCacheList); 方法是向 nacos 服务端 发送一个长连接超时事件 30s,返回有更新的 dataids;调用 getServerConfig(dataId, group, tenant, 3000L); 方法是根据返回有变化的 dataids 调用服务端配置中心接口获取配置属性,并更新本地快照;调用 checkListenerMd5();方式,对有变化的配置添加监听处理;最后继续调用 executorService.execute(this); 方法轮询处理。
data:image/s3,"s3://crabby-images/173f8/173f80e5564c141148e75db6ca953a9296b7354d" alt=""
data:image/s3,"s3://crabby-images/1de8f/1de8ffb9d369d69f8d2810783d452d1bb87833ff" alt=""
data:image/s3,"s3://crabby-images/8ca3a/8ca3a87d0be5c1e831b3ba004e06eab56c3b16cc" alt=""
CacheData#checkListenerMd5
data:image/s3,"s3://crabby-images/21ee8/21ee8e8f474d918a76a2513a58e18970be93724d" alt=""
data:image/s3,"s3://crabby-images/20d6d/20d6dd07762bd4dcf4d7890ef89f83176616703c" alt=""
在 listener.receiveConfigInfo(contentTmp); 方法中会调用到 AbstractSharedListener#receiveConfigInfo 方法,会发布 RefreshEvent 事件。
data:image/s3,"s3://crabby-images/21a9b/21a9bbb958b8bc33ef17c475abc3172d64e7d0d8" alt=""
data:image/s3,"s3://crabby-images/37519/37519bccf8f9726c73043855060d65bfd04e1e30" alt=""
对应的事件监听器为:RefreshEventListener, Spring Cloud 实现的,在该监听器里更新配置和刷新容器中标记了 @RefreshScope 的配置,在 onApplicationEvent 方法中监听 2 个事件,ApplicationReadyEvent(spring boot 事件,表示 application 应该初始化完成)、RefreshEvent。
data:image/s3,"s3://crabby-images/aeb96/aeb968eb5e6d7c1064e95a8a7bbc23bd5cd473a3" alt=""
data:image/s3,"s3://crabby-images/cb590/cb5900429e7b8fa5df58960d644386b6aeb0cfd0" alt=""
RefreshEvent:this.handle((RefreshEvent)event);处理该事件,用来刷新容器中标记了 @RefreshScope 注解的配置,org.springframework.cloud.context.refresh.ContextRefresher#refresh
data:image/s3,"s3://crabby-images/1e4e2/1e4e2bb0a1ddc9f917438a2b1986efe2d6587d7b" alt=""
data:image/s3,"s3://crabby-images/efb1a/efb1ac8ef713116fc246919ec181da6f3bb7dc69" alt=""
refreshEnvironment();中 extract(this.context.getEnvironment().getPropertySources()) 抽取除系统变量外的其他变量;addConfigFilesToEnvironment();把原有的 environment 里面的参数放到一个新建的 spring context 容器下重新加载,完事之后关闭新容器,这里就是获取参数的新值;
data:image/s3,"s3://crabby-images/fa5bf/fa5bf96667e7975c3131d7598efe9eb17a113d6a" alt=""
data:image/s3,"s3://crabby-images/2d8c8/2d8c864e3810bd801919f038dc6ce763f00d470c" alt=""
changes(before,extract(this.context.getEnvironment().getPropertySources())) 获取新的参数值,并和之前得进行比较找出改变得参数值。
data:image/s3,"s3://crabby-images/f50d9/f50d9e1c1f7b0ed808663f78e5164cf17689f6e4" alt=""
this.context.publishEvent(new EnvironmentChangeEvent(this.context, keys)); 发布环境变更事件,并带上改变得参数值。
回到 ContextRefresher#refresh 方法,看下 this.scope.refreshAll(); 刷新标记 @RefreshScope 注解的 bean。
data:image/s3,"s3://crabby-images/42e73/42e734a830b13f19b8d19f53e63ce08311db17f1" alt=""
data:image/s3,"s3://crabby-images/23411/23411a8b1d600a9667669e83197032f661b9d855" alt=""
super.destroy(); 方法,清楚 scope 里面的缓存,下次就会重新从 BeanFactory 获取一个新的实例会使用新的配置。
this.context.publishEvent(new RefreshScopeRefreshedEvent()); 方法发布事件。
服务端
DumpService
DumpService 类是一个抽象类负责从存储中查询配置保存到磁盘上,它有两个子类,EmbeddedDumpService 嵌入式存储(DERBY)、ExternalDumpService 扩展数存储。
data:image/s3,"s3://crabby-images/6cc7b/6cc7b57989577f52f6fc95276935b3c8090faac7" alt=""
ExternalDumpService 实现类的 init 方法上 @PostConstruct 注解,在 spring 构建 bean 的过程中会执行带有 @PostConstruct 的初始化方法。
data:image/s3,"s3://crabby-images/49060/490602580a669fc9c1583eec564f1788ae89c599" alt=""
调用到抽象父类 DumpService#dumpOperate 的方法,调用到 dumpConfigInfo 方法,dumpConfigInfo 方法会判断是全量更新,还是追加更新。
data:image/s3,"s3://crabby-images/5a5a1/5a5a12554e029e3c85f90206063ba98e3d2835ed" alt=""
如果 isAllDump 为 true 会走全量更新,会进行判断是否有快速更新配置、是否存在心跳检查文件、最后检查时间是否小于 6 小时,上述判断都满足就不走全量更新,否则走全量更新。
data:image/s3,"s3://crabby-images/39048/390480b8d8a0cedc85cf6c652f44a81f08dad62e" alt=""
dumpAllProcessor.process(new DumpAllTask());将数据库中的所有 configInfo 配置信息查询出来,写入服务器端磁盘缓存。
data:image/s3,"s3://crabby-images/e329a/e329ae4688bf07f48c7be2f9f717038ddbd30660" alt=""
persistService.findConfigMaxId(); 查询数据库中最大的主键,用于分页处理。
data:image/s3,"s3://crabby-images/ad2e2/ad2e2351c624645c0f1cc9d4cd0a5a8c109138e9" alt=""
persistService.findAllConfigInfoFragment(lastMaxId, PAGE_SIZE); 从数据库中分页查询数据,每次查询 1000 条。
data:image/s3,"s3://crabby-images/67a28/67a2845c1d01974e566ec526055836cb9224d688" alt=""
data:image/s3,"s3://crabby-images/205b9/205b977852446842281b4e16c5c836d497ea36b6" alt=""
ConfigCacheService.dump(cf.getDataId(), cf.getGroup(), cf.getTenant(), cf.getContent(), cf.getLastModified(),cf.getType()); 写入磁盘
data:image/s3,"s3://crabby-images/c1026/c1026d3411cabd3b5e1a084042866a1f69a2af90" alt=""
保存到文件中
data:image/s3,"s3://crabby-images/5b25d/5b25d419d4820d427e38f0cb40c236607d25f70e" alt=""
updateMd5(groupKey, md5, lastModifiedTs); 缓存配置信息的 MD5 到内存中,并发布 LocalDataChangeEvent 事件。
data:image/s3,"s3://crabby-images/63388/63388b9df1ea9579146f5c1dc3a657e9f7d0838a" alt=""
事件监听器会在 NotifyCenter.registerSubscriber 调用。
data:image/s3,"s3://crabby-images/65e83/65e832c0e06e90a3c5aad06c5f54f81d23e40bdb" alt=""
获取配置
HttpMethod.GET /nacos/v1/cs/configs 获取服务端配置接口,ConfigController#getConfig。
data:image/s3,"s3://crabby-images/0d8f9/0d8f9e0745a6b6ef5dab0c6abceb2e3000f0cb4d" alt=""
在 getConfig 中调用了 inner.doGetConfig(request, response, dataId, group, tenant, tag, clientIp);
在 doGetConfig 方法中会调用 DiskUtil.targetBetaFile(dataId, group, tenant);方法,从本地磁盘上获取,不是从 mysql 中拉取,如果直接修改 mysql 数据不会生效的,需要发布 ConfigDataChangeEvent 事件,触发更新。
data:image/s3,"s3://crabby-images/b25f2/b25f2ba262a4d0acb87456f19d6dd1dfbdf02018" alt=""
监听配置
HttpMethod.POST 请求调用 /nacos/v1/cs/configs/listener 轮询接口调用长连接。
data:image/s3,"s3://crabby-images/397ec/397eced6885cbceff51784356d8ac33426146ef6" alt=""
longPollingService.addLongPollingClient(request, response, clientMd5Map, probeRequestSize); 长连接轮询处理。
data:image/s3,"s3://crabby-images/d9eeb/d9eeb620c202bbf94060df55f1e1c3da1373938b" alt=""
SwitchService.getSwitchInteger(SwitchService.FIXED_DELAY_TIME, 500); 最多处理 29.5s 需要保留 0.5s 来响应客户端,避免超时。
data:image/s3,"s3://crabby-images/4edfc/4edfc34d5a1f9ebef8096d409fee6f72f7f86f61" alt=""
MD5Util.compareMd5(req, rsp, clientMd5Map); 比较客户端的 md5 与当前服务端的是否一致,不一致返回到 changedGroups。
data:image/s3,"s3://crabby-images/ac7c0/ac7c0124d69c43d64dc853ba4d3a64c2eda13677" alt=""
有不一致数据直接响应 generateResponse(req, rsp, changedGroups);
data:image/s3,"s3://crabby-images/0795b/0795ba28f5fca31a371ea78330417dc5e16fd7ed" alt=""
线程池执行长连接任务 ConfigExecutor.executeLongPolling。
data:image/s3,"s3://crabby-images/c1fa9/c1fa99667c8fe55f8d39bce26656544733a74c2a" alt=""
LongPollingService.ClientLongPolling#run 长轮询。
data:image/s3,"s3://crabby-images/d4350/d4350dd4d96d30bcfc938140804166bab3cd1ca8" alt=""
ConfigExecutor.scheduleLongPolling 延迟 29.5s 执行
data:image/s3,"s3://crabby-images/cc6b3/cc6b37d2c571a26dff4c0bf42313bdcb3ae302fb" alt=""
延迟执行先删除队列中自己的任务 allSubs.remove(ClientLongPolling.this);
data:image/s3,"s3://crabby-images/99f75/99f757869bb7385fc58c1b909c5cc7c646d6c441" alt=""
allSubs.add(this); 添加到队列
data:image/s3,"s3://crabby-images/b60ee/b60ee32762e1459707437a2b33df7ed811279567" alt=""
inner.doPollingConfig(request, response, clientMd5Map, probeModify.length());
data:image/s3,"s3://crabby-images/7a3a0/7a3a0a10621462bdb701afa17ffa301157e31fdb" alt=""
MD5Util.compareMd5(request, response, clientMd5Map); 和当前配置比较,返回有变更的配置
nacos 管理端变更配置
HttpMethod.POST /nacos/v1/cs/configs
data:image/s3,"s3://crabby-images/c8140/c814076c38b88c5c7c187a70048afe8977d02633" alt=""
persistService.insertOrUpdate(srcIp, srcUser, configInfo, time, configAdvanceInfo, true); 持节化信息到数据库。
data:image/s3,"s3://crabby-images/53660/536600bfb865cbf2b3caa6bbaba592720d20754e" alt=""
data:image/s3,"s3://crabby-images/6b6f0/6b6f0840b1fa144d33fd25a259831c71baab6d43" alt=""
data:image/s3,"s3://crabby-images/b5305/b5305949d912ba5ee7a56c83ce243643dfe8ad8d" alt=""
data:image/s3,"s3://crabby-images/c4b28/c4b2819122c1b5bb8248a2332d22eeffda4f3237" alt=""
回到 ConfigController#publishConfig 看下 ConfigChangePublisher.notifyConfigChange 方法,触发 ConfigDataChangeEvent 事件。
data:image/s3,"s3://crabby-images/1799b/1799bcdea2e2e50df1b1e0b63b08608255f04ab7" alt=""
data:image/s3,"s3://crabby-images/b53bc/b53bc5c2815f870f9731d07a4eadd6bba0a235df" alt=""
ConfigDataChangeEvent 事件监听。
data:image/s3,"s3://crabby-images/1f909/1f909ca841b23c2623837f75f4f06846f3317477" alt=""
ConfigExecutor.executeAsyncNotify(new AsyncTask(nacosAsyncRestTemplate, queue)); 同步其他节点。
data:image/s3,"s3://crabby-images/74790/747907f5d5a0373696b81f5335edf567a9af90b1" alt=""
还有 LongPollingService 初始化的时候订阅了 LocalDataChangeEvent 事件,也会监听到。
data:image/s3,"s3://crabby-images/b0d67/b0d6775b64559e23d1724f61766d9a324c953eb0" alt=""
ConfigExecutor.executeLongPolling(new DataChangeTask(evt.groupKey, evt.isBeta, evt.betaIps));
看下 LongPollingService.DataChangeTask#run,push 模式, 遍历 allSubs 把变化的 key 响应客户端。clientSub.sendResponse(Arrays.asList(groupKey));
data:image/s3,"s3://crabby-images/1c694/1c6940737cdcefb82286beedf09d0ace5ad1a446" alt=""
作者:京东物流 张士欣
来源:京东云开发者社区 自猿其说 Tech 转载请注明来源
版权声明: 本文为 InfoQ 作者【京东科技开发者】的原创文章。
原文链接:【http://xie.infoq.cn/article/8dedcb45886d00e04f23387b7】。文章转载请联系作者。
评论