服务治理-条件路由
目的
各个项目之间相互调用的控制逻辑。通过这一特性可以实现服务提供者的灰度发布等操作。具体信息可以参照官网说明)
配置
对于版本2.7+之后的来说,dubbo服务注册粒度可选为:应用级别(application)、服务级别(service)这两个级别。所有在路由配置这一块来说,也是分为这两个粒度。具体的配置内容可以参考 官网
原理
简单的来说,条件路由
加载过程分为两个部分初始化
和实际作用阶段
。在初始化阶段中,从配置中心拉取该服务(该应用)配置的规则信息(如果有的话),进行装载;在项目启动完后,当有消费者调用服务提供者的时候,调用的过程经过如下步骤:
- 服务发现:从本地缓存、注册中心获取服务提供者信息,经过
路由(条件/标签)
的筛选得到合适的invoker(服务提供者)
信息; - 选取一个服务提供者:通过客户端选定的负载均衡策略来选取一个服务提供者;
- 调用服务;
初始化阶段
在项目启动的过程中,ServiceDiscoveryRegistryDirectory
调用buildChain()
来从配置中心获取到自己关联的服务配置的路由策略信息,在javadoc的介绍中,条件路由的具体实现已从2.6.6的ConditionRouter
变为ServiceRouter
;
从配置中心加载路由信息
// ListenableRouter
public ListenableRouter(URL url, String ruleKey) {
super(url); // 通过url信息获取ruleRepository(zk、apollo、nacos....)
this.setForce(false);
// ruleKey格式(servicepath + ":" + group + ":" + version): cn.com.dubbo.demo.provider.service.MyEmpService::
this.init(ruleKey);
}
private synchronized void init(String ruleKey) {
if (StringUtils.isEmpty(ruleKey)) {
return;
}
// key末尾条件常量:.condition-router
String routerKey = ruleKey + RULE_SUFFIX;
// 添加监听,及时获取到变化更新信息
this.getRuleRepository().addListener(routerKey, this);
// 从配置中心获取配置的路由信息(如果有的话)
// DefaultGovernanceRuleRepositoryImpl#getRule()
String rule = this.getRuleRepository().getRule(routerKey, DynamicConfiguration.DEFAULT_GROUP);
if (StringUtils.isNotEmpty(rule)) {
this.process(new ConfigChangedEvent(routerKey, DynamicConfiguration.DEFAULT_GROUP, rule));
}
}
// DefaultGovernanceRuleRepositoryImpl#getRule()
@Override
public String getRule(String key, String group, long timeout) throws IllegalStateException {
// 通过application.properties等系统配置文件获取到连接配置中心客户端并且初始化相关的信息
DynamicConfiguration dynamicConfiguration = getDynamicConfiguration();
if (dynamicConfiguration != null) {
// 获取配置的路有信息
return dynamicConfiguration.getConfig(key, group, timeout);
}
return null;
}
@Override
public String getConfig(String key, String group, long timeout) throws IllegalStateException {
return (String) iterateConfigOperation(configuration -> configuration.getConfig(key, group, timeout));
}
// func参数是一个函数,通过apply()触发func被调用
private Object iterateConfigOperation(Function<DynamicConfiguration, Object> func) {
Object value = null;
for (DynamicConfiguration configuration : configurations) {
// 触发 configuration -> configuration.getConfig(key, group, timeout)
value = func.apply(configuration);
if (value != null) {
break;
}
}
return value;
}
解析路由配置
@Override
public synchronized void process(ConfigChangedEvent event) {
if (logger.isDebugEnabled()) {
logger.debug("Notification of condition rule, change type is: " + event.getChangeType() +
", raw rule is:\n " + event.getContent());
}
if (event.getChangeType().equals(ConfigChangeType.DELETED)) {
routerRule = null;
conditionRouters = Collections.emptyList();
} else {
try {
routerRule = ConditionRuleParser.parse(event.getContent());
// 初始化 ConditionRouter,当后调用的时候,从ConditionRouter中获取到相关的路由策略来判断是否invokers是否符合
generateConditions(routerRule);
} catch (Exception e) {
logger.error("Failed to parse the raw condition rule and it will not take effect, please check " +
"if the condition rule matches with the template, the raw rule is:\n " + event.getContent(), e);
}
}
}
作用阶段
消费端调用过程
@Override
public Result invoke(final Invocation invocation) throws RpcException {
checkWhetherDestroyed();
// binding attachments into invocation.
// Map<String, Object> contextAttachments = RpcContext.getClientAttachment().getObjectAttachments();
// if (contextAttachments != null && contextAttachments.size() != 0) {
// ((RpcInvocation) invocation).addObjectAttachmentsIfAbsent(contextAttachments);
// }
// 获取可用的服务提供者列表
List<Invoker<T>> invokers = list(invocation);
LoadBalance loadbalance = initLoadBalance(invokers, invocation);
RpcUtils.attachInvocationIdIfAsync(getUrl(), invocation);
return doInvoke(invocation, invokers, loadbalance);
}
路由匹配过程
其中路由的作用也是在list()
中体现的,具体的路由匹配RouterChain.route()
如下:
public List<Invoker<T>> route(URL url, Invocation invocation) {
AddrCache<T> cache = this.cache.get();
if (cache == null) {
throw new RpcException(RpcException.ROUTER_CACHE_NOT_BUILD, "Failed to invoke the method "
+ invocation.getMethodName() + " in the service " + url.getServiceInterface()
+ ". address cache not build "
+ " on the consumer " + NetUtils.getLocalHost()
+ " using the dubbo version " + Version.getVersion()
+ ".");
}
BitList<Invoker<T>> finalBitListInvokers = new BitList<>(invokers, false);
for (StateRouter stateRouter : stateRouters) {
if (stateRouter.isEnable()) {
RouterCache<T> routerCache = cache.getCache().get(stateRouter.getName());
finalBitListInvokers = stateRouter.route(finalBitListInvokers, routerCache, url, invocation);
}
}
List<Invoker<T>> finalInvokers = new ArrayList<>(finalBitListInvokers.size());
for(Invoker<T> invoker: finalBitListInvokers) {
finalInvokers.add(invoker);
}
for (Router router : routers) {
finalInvokers = router.route(finalInvokers, url, invocation);
}
return finalInvokers;
}
有五个默认的router
,每一种类型的路由都有一个专属的handler来做相应的处理,对于条件路由来说,需要关注的就是ServiceRouter
的处理逻辑。
条件路由匹配
ConditionRouter.route()
,遍历所有的服务提供者信息,来匹配过滤已经设定好的路由规则得到符合条件的服务提供者信息。
@Override
public <T> List<Invoker<T>> route(List<Invoker<T>> invokers, URL url, Invocation invocation)
throws RpcException {
if (!enabled) {
return invokers;
}
if (CollectionUtils.isEmpty(invokers)) {
return invokers;
}
try {
if (!matchWhen(url, invocation)) {
return invokers;
}
List<Invoker<T>> result = new ArrayList<Invoker<T>>();
if (thenCondition == null) {
logger.warn("The current consumer in the service blacklist. consumer: " + NetUtils.getLocalHost() + ", service: " + url.getServiceKey());
return result;
}
for (Invoker<T> invoker : invokers) {
if (matchThen(invoker.getUrl(), url)) {
result.add(invoker);
}
}
if (!result.isEmpty()) {
return result;
} else if (this.isForce()) {
logger.warn("The route result is empty and force execute. consumer: " + NetUtils.getLocalHost() + ", service: " + url.getServiceKey() + ", router: " + url.getParameterAndDecoded(RULE_KEY));
return result;
}
} catch (Throwable t) {
logger.error("Failed to execute condition router rule: " + getUrl() + ", invokers: " + invokers + ", cause: " + t.getMessage(), t);
}
return invokers;
}
注意点
- 配置应用级别的路由时,必须要保证该应用是服务提供者,而且该服务必须要有消费者