旧路由规则

在 Dubbo 2.6.x 版本以及更早的版本中配置路由规则

路由规则 1 决定一次 dubbo 服务调用的目标服务器,分为条件路由规则和脚本路由规则,并且支持可扩展 2

写入路由规则

向注册中心写入路由规则的操作通常由监控中心或治理中心的页面完成

  1. RegistryFactory registryFactory = ExtensionLoader.getExtensionLoader(RegistryFactory.class).getAdaptiveExtension();
  2. Registry registry = registryFactory.getRegistry(URL.valueOf("zookeeper://10.20.153.10:2181"));
  3. registry.register(URL.valueOf("route://0.0.0.0/com.foo.BarService?category=routers&dynamic=false&rule=" + URL.encode("host = 10.20.153.10 => host = 10.20.153.11")));

其中:

  • route:// 表示路由规则的类型,支持条件路由规则和脚本路由规则,可扩展,必填
  • 0.0.0.0 表示对所有 IP 地址生效,如果只想对某个 IP 的生效,请填入具体 IP,必填
  • com.foo.BarService 表示只对指定服务生效,必填
  • group=foo 对指定服务的指定group生效,不填表示对未配置group的指定服务生效
  • version=1.0对指定服务的指定version生效,不填表示对未配置version的指定服务生效
  • category=routers 表示该数据为动态配置类型,必填
  • dynamic=false 表示该数据为持久数据,当注册方退出时,数据依然保存在注册中心,必填
  • enabled=true 覆盖规则是否生效,可不填,缺省生效。
  • force=false 当路由结果为空时,是否强制执行,如果不强制执行,路由结果为空的路由规则将自动失效,可不填,缺省为 false
  • runtime=false 是否在每次调用时执行路由规则,否则只在提供者地址列表变更时预先执行并缓存结果,调用时直接从缓存中获取路由结果。如果用了参数路由,必须设为 true,需要注意设置会影响调用的性能,可不填,缺省为 false
  • priority=1 路由规则的优先级,用于排序,优先级越大越靠前执行,可不填,缺省为 0
  • rule=URL.encode("host = 10.20.153.10 => host = 10.20.153.11") 表示路由规则的内容,必填

条件路由规则

基于条件表达式的路由规则,如:host = 10.20.153.10 => host = 10.20.153.11

规则:

  • => 之前的为消费者匹配条件,所有参数和消费者的 URL 进行对比,当消费者满足匹配条件时,对该消费者执行后面的过滤规则。
  • => 之后为提供者地址列表的过滤条件,所有参数和提供者的 URL 进行对比,消费者最终只拿到过滤后的地址列表。
  • 如果匹配条件为空,表示对所有消费方应用,如:=> host != 10.20.153.11
  • 如果过滤条件为空,表示禁止访问,如:host = 10.20.153.10 =>

表达式:

参数支持:

  • 服务调用信息,如:method, argument 等,暂不支持参数路由
  • URL 本身的字段,如:protocol, host, port 等
  • 以及 URL 上的所有参数,如:application, organization 等

条件支持:

  • 等号 = 表示”匹配”,如:host = 10.20.153.10
  • 不等号 != 表示”不匹配”,如:host != 10.20.153.10

值支持:

  • 以逗号 , 分隔多个值,如:host != 10.20.153.10,10.20.153.11
  • 以星号 * 结尾,表示通配,如:host != 10.20.*
  • 以美元符 $ 开头,表示引用消费者参数,如:host = $host

示例:

  1. 排除预发布机:

    1. => host != 172.22.3.91
  2. 白名单 3

    1. host != 10.20.153.10,10.20.153.11 =>
  3. 黑名单:

    1. host = 10.20.153.10,10.20.153.11 =>
  4. 服务寄宿在应用上,只暴露一部分的机器,防止整个集群挂掉:

    1. => host = 172.22.3.1*,172.22.3.2*
  5. 为重要应用提供额外的机器:

    1. application != kylin => host != 172.22.3.95,172.22.3.96
  6. 读写分离:

    1. method = find*,list*,get*,is* => host = 172.22.3.94,172.22.3.95,172.22.3.96
    2. method != find*,list*,get*,is* => host = 172.22.3.97,172.22.3.98
  7. 前后台分离:

    1. application = bops => host = 172.22.3.91,172.22.3.92,172.22.3.93
    2. application != bops => host = 172.22.3.94,172.22.3.95,172.22.3.96
  8. 隔离不同机房网段:

    1. host != 172.22.3.* => host != 172.22.3.*
  9. 提供者与消费者部署在同集群内,本机只访问本机的服务:

    1. => host = $host

脚本路由规则

脚本路由规则 4 支持 JDK 脚本引擎的所有脚本,比如:javascript, jruby, groovy 等,通过 type=javascript 参数设置脚本类型,缺省为 javascript。

  1. "script://0.0.0.0/com.foo.BarService?category=routers&dynamic=false&rule=" + URL.encode("(function route(invokers) { ... } (invokers))")

基于脚本引擎的路由规则,如:

  1. function route(invokers) {
  2. var result = new java.util.ArrayList(invokers.size());
  3. for (i = 0; i < invokers.size(); i ++) {
  4. if ("10.20.153.10".equals(invokers.get(i).getUrl().getHost())) {
  5. result.add(invokers.get(i));
  6. }
  7. }
  8. return result;
  9. } (invokers)); // 表示立即执行方法

标签路由规则

标签路由规则 5 ,当应用选择装配标签路由(TagRouter)之后,一次 dubbo 调用能够根据请求携带的 tag 标签智能地选择对应 tag 的服务提供者进行调用。

服务提供者

  1. 给应用装配标签路由器:
  1. @Bean
  2. public ApplicationConfig applicationConfig() {
  3. ApplicationConfig applicationConfig = new ApplicationConfig();
  4. applicationConfig.setName("provider-book");
  5. applicationConfig.setQosEnable(false);
  6. // instruct tag router
  7. Map<String,String> parameters = new HashMap<>();
  8. parameters.put(Constants.ROUTER_KEY, "tag");
  9. applicationConfig.setParameters(parameters);
  10. return applicationConfig;
  11. }
  1. 给应用设置具体的标签:
  1. @Bean
  2. public ProviderConfig providerConfig(){
  3. ProviderConfig providerConfig = new ProviderConfig();
  4. providerConfig.setTag("red");
  5. return providerConfig;
  6. }

应用未装配 tag 属性或服务提供者未设置 tag 属性,都将被认为是默认的应用,这些默认应用将会在调用无法匹配 provider 时当作降级方案。

服务消费者

  1. RpcContext.getContext().setAttachment(Constants.REQUEST_TAG_KEY,"red");

请求标签的作用域为每一次 invocation,使用 attachment 来传递请求标签,注意保存在 attachment 中的值将会在一次完整的远程调用中持续传递,得益于这样的特性,我们只需要在起始调用时,通过一行代码的设置,达到标签的持续传递。

目前仅仅支持 hardcoding 的方式设置 requestTag。注意到 RpcContext 是线程绑定的,优雅的使用 TagRouter 特性,建议通过 servlet 过滤器(在 web 环境下),或者定制的 SPI 过滤器设置 requestTag。

规则描述

  1. request.tag=red 时优先选择 tag=red 的 provider。若集群中不存在与请求标记对应的服务,可以降级请求 tag=null 的 provider,即默认 provider。

  2. request.tag=null 时,只会匹配 tag=null 的 provider。即使集群中存在可用的服务,若 tag 不匹配就无法调用,这与规则1不同,携带标签的请求可以降级访问到无标签的服务,但不携带标签/携带其他种类标签的请求永远无法访问到其他标签的服务。


  1. 2.2.0 以上版本支持 ↩︎

  2. 路由规则扩展点:路由扩展 ↩︎

  3. 注意:一个服务只能有一条白名单规则,否则两条规则交叉,就都被筛选掉了 ↩︎

  4. 注意:脚本没有沙箱约束,可执行任意代码,存在后门风险 ↩︎

  5. 该特性在 2.7.0 以上版本支持 ↩︎

最后修改 September 21, 2021: Bug fix miss mialbox (#953) (57cf51b)