在之前的文章之中我们解析了一下 Spring Cloud 只需要在 RestTemplate 类型的 Spring Bean 上面添加一个 @LoadBalanced 注解,并且通过 RestTemplate 以 http://serviceId/访问路径 就可以访问真实服务的地址 http://lolcahost:端口/访问路径。同样的 Nepxion Discovery 需要实现服务的灰度发布,也对 Spring Cloud 的负载均衡进行了扩展。基于 Nepxion Discovery 版本为:6.3.5-SNAPSHOT。
1、Nacos 负载均衡实现
Spring Cloud 对于负载均衡默认并没有对 Nacos 的实现,那么我们就来看一下 Nacos 是如何实现负载均衡的。具体实现类是 NacosRule,在使用 nacos 作为注册中心时,并没有使用这个策略。
public class NacosRule extends AbstractLoadBalancerRule {
private static final Logger LOGGER = LoggerFactory.getLogger(NacosRule.class);
@Autowired
private NacosDiscoveryProperties nacosDiscoveryProperties;
@Autowired
private NacosServiceManager nacosServiceManager;
@Override
public Server choose(Object key) {
try {
String clusterName = this.nacosDiscoveryProperties.getClusterName();
String group = this.nacosDiscoveryProperties.getGroup();
DynamicServerListLoadBalancer loadBalancer = (DynamicServerListLoadBalancer) getLoadBalancer();
String name = loadBalancer.getName();
NamingService namingService = nacosServiceManager
.getNamingService(nacosDiscoveryProperties.getNacosProperties());
List<Instance> instances = namingService.selectInstances(name, group, true);
if (CollectionUtils.isEmpty(instances)) {
LOGGER.warn("no instance in service {}", name);
return null;
}
List<Instance> instancesToChoose = instances;
if (StringUtils.isNotBlank(clusterName)) {
List<Instance> sameClusterInstances = instances.stream()
.filter(instance -> Objects.equals(clusterName,
instance.getClusterName()))
.collect(Collectors.toList());
if (!CollectionUtils.isEmpty(sameClusterInstances)) {
instancesToChoose = sameClusterInstances;
}
else {
LOGGER.warn(
"A cross-cluster call occurs,name = {}, clusterName = {}, instance = {}",
name, clusterName, instances);
}
}
Instance instance = ExtendBalancer.getHostByRandomWeight2(instancesToChoose);
return new NacosServer(instance);
}
catch (Exception e) {
LOGGER.warn("NacosRule error", e);
return null;
}
}
@Override
public void initWithNiwsConfig(IClientConfig iClientConfig) {
}
}
这个对象里面注入了 NacosDiscoveryProperties 配置属性以及 NacosServiceManager 服务管理器。首先通过服务组(没有配置就是 DEFAULT),以及服务的名称获取到服务实例的列表。如果服务的配置信息里面包含服务的集群信息(clusterName),就会过滤服务实例列表,如果列表里面有当前群集的信息。如果过滤后的列表不为空就使用同集群的服务,否则还是使用未过滤之前的服务实例列表。最后通过权重随机算法获取到一个服务信息。
在Nacos 的 Spring Cloud 自动配置依赖模块 spring-cloud-starter-alibaba-nacos-discovery 里面自定义了 RibbonClient 客户端相关的服务配置。
RibbonNacosAutoConfiguration.java
@Configuration(proxyBeanMethods = false)
@EnableConfigurationProperties
@ConditionalOnBean(SpringClientFactory.class)
@ConditionalOnRibbonNacos
@ConditionalOnNacosDiscoveryEnabled
@AutoConfigureAfter(RibbonAutoConfiguration.class)
@RibbonClients(defaultConfiguration = NacosRibbonClientConfiguration.class)
public class RibbonNacosAutoConfiguration {
}
@Configuration(proxyBeanMethods = false)
@ConditionalOnRibbonNacos
public class NacosRibbonClientConfiguration {
@Autowired
private PropertiesFactory propertiesFactory;
@Bean
@ConditionalOnMissingBean
public ServerList<?> ribbonServerList(IClientConfig config,
NacosDiscoveryProperties nacosDiscoveryProperties) {
if (this.propertiesFactory.isSet(ServerList.class, config.getClientName())) {
ServerList serverList = this.propertiesFactory.get(ServerList.class, config,
config.getClientName());
return serverList;
}
NacosServerList serverList = new NacosServerList(nacosDiscoveryProperties);
serverList.initWithNiwsConfig(config);
return serverList;
}
@Bean
@ConditionalOnMissingBean
public NacosServerIntrospector nacosServerIntrospector() {
return new NacosServerIntrospector();
}
}
在这里它使用了的是基于 Nacos 注册中心,NacosServerList实现 ServerList 接口对 Nacos 里面服务信息列表的获取。同时在 Spring 容器中添加了 ServerIntrospector 的服务自省实现类 NacosServerIntrospector 可以获取到这个服务是否安全的(基于 https 协议) 以及服务信息的元数据。
2、Nepxion Discovery 的扩展
Nepxion Discovery 对负载均衡的扩展其实就是对 IRule 接口实现的扩展。这里其实就服务灰度发布的真正实现。DiscoveryClient 根据服务 ID 获取到这个服务所有暴露的可用的服务列表,通过负载均衡里面的策略选择一个合适的服务进行调用。最终就调用到Nepxion Discovery 对负载均衡的扩展其实就是对 IRule 接口实现的扩展
下面就是Nepxion Discovery 对 IRule 这个策略接口的实现:

下面我们看一下扩展策略接口的作用:
| 实现类 | 策略描述 | 实现说明 |
|---|---|---|
PredicateBasedRuleDecorator |
抽象类,继承PredicateBasedRule并且重写了 IRule#choose,支持 version权重和 region 权重 |
分别从 header 里面和远程配置中心获取version权重和 region 权重。优先级:header > 配置中心 |
DiscoveryEnabledBaseRule |
继承PredicateBasedRuleDecorator,复合判断 server 的可用性选择 server 以及通过灰度参数选择 server |
使用 ZoneAvoidancePredicate 和 DiscoveryEnabledBasePredicate 来判断是否选择某个 server,前一个判断判定一个 zone 的运行性能是否可用,剔除不可用的 zone(的所有 server),DiscoveryEnabledBasePredicate用于过滤掉不满足灰度条件的 server |
ZoneAvoidanceRuleDecorator |
继承 ZoneAvoidanceRule并且重写了 IRule#choose,支持 version权重和 region 权重 |
分别从 header 里面和远程配置中心获取version权重和 region 权重。优先级:header > 配置中心 |
DiscoveryEnabledZoneAvoidanceRule |
继承ZoneAvoidanceRuleDecorator,复合判断 server 的可用性选择 server以及通过灰度参数选择 server |
使用 AvailabilityPredicate 和 DiscoveryEnabledZoneAvoidancePredicate 来判断是否选择某个 server,前一个判断判定一个 Server 是否可用,剔除不可用的 Server,DiscoveryEnabledBasePredicate用于过滤掉不满足灰度条件的 server |
在Nepxion Discovery 框架针对 Robbin 的负载均衡配置有 StrategyLoadBalanceConfiguration 和 PluginLoadBalanceConfiguration 这两个配置类。
StrategyLoadBalanceConfiguration.java
@Configuration
@AutoConfigureBefore(RibbonClientConfiguration.class)
public class StrategyLoadBalanceConfiguration {
@Autowired
private ConfigurableEnvironment environment;
@RibbonClientName
private String serviceId = "client";
@Autowired
private PropertiesFactory propertiesFactory;
@Autowired
private PluginAdapter pluginAdapter;
@Autowired(required = false)
private DiscoveryEnabledAdapter discoveryEnabledAdapter;
@Bean
public IRule ribbonRule(IClientConfig config) {
if (this.propertiesFactory.isSet(IRule.class, serviceId)) {
return this.propertiesFactory.get(IRule.class, config, serviceId);
}
boolean zoneAvoidanceRuleEnabled = environment.getProperty(StrategyConstant.SPRING_APPLICATION_STRATEGY_ZONE_AVOIDANCE_RULE_ENABLED, Boolean.class, Boolean.TRUE);
if (zoneAvoidanceRuleEnabled) {
DiscoveryEnabledZoneAvoidanceRule discoveryEnabledRule = new DiscoveryEnabledZoneAvoidanceRule();
discoveryEnabledRule.initWithNiwsConfig(config);
DiscoveryEnabledZoneAvoidancePredicate discoveryEnabledPredicate = discoveryEnabledRule.getDiscoveryEnabledPredicate();
discoveryEnabledPredicate.setPluginAdapter(pluginAdapter);
discoveryEnabledPredicate.setDiscoveryEnabledAdapter(discoveryEnabledAdapter);
return discoveryEnabledRule;
} else {
DiscoveryEnabledBaseRule discoveryEnabledRule = new DiscoveryEnabledBaseRule();
DiscoveryEnabledBasePredicate discoveryEnabledPredicate = discoveryEnabledRule.getDiscoveryEnabledPredicate();
discoveryEnabledPredicate.setPluginAdapter(pluginAdapter);
discoveryEnabledPredicate.setDiscoveryEnabledAdapter(discoveryEnabledAdapter);
return discoveryEnabledRule;
}
}
}
下面是PluginLoadBalanceConfiguration 配置类:
PluginLoadBalanceConfiguration
@Configuration
@AutoConfigureAfter(RibbonClientConfiguration.class)
public class PluginLoadBalanceConfiguration {
@RibbonClientName
private String serviceId = "client";
@Autowired
private PropertiesFactory propertiesFactory;
@Autowired
private LoadBalanceListenerExecutor loadBalanceListenerExecutor;
@Bean
@ConditionalOnMissingBean
public IRule ribbonRule(IClientConfig config) {
if (this.propertiesFactory.isSet(IRule.class, serviceId)) {
return this.propertiesFactory.get(IRule.class, config, serviceId);
}
ZoneAvoidanceRuleDecorator rule = new ZoneAvoidanceRuleDecorator();
rule.initWithNiwsConfig(config);
return rule;
}
@Bean
public ILoadBalancer ribbonLoadBalancer(IClientConfig config, ServerList<Server> serverList, ServerListFilter<Server> serverListFilter, IRule rule, IPing ping, ServerListUpdater serverListUpdater) {
if (this.propertiesFactory.isSet(ILoadBalancer.class, serviceId)) {
return this.propertiesFactory.get(ILoadBalancer.class, config, serviceId);
}
ZoneAwareLoadBalancer<?> loadBalancer = new ZoneAwareLoadBalancer<>(config, rule, ping, serverList, serverListFilter, serverListUpdater);
loadBalanceListenerExecutor.setLoadBalancer(loadBalancer);
return loadBalancer;
}
}
PluginLoadBalanceConfiguration 的类上面的配置是@AutoConfigureAfter(RibbonClientConfiguration.class),所以它的初始化顺序在 RibbonClientConfiguration 这个Spring Cloud Ribbon 原始配置类之后,而StrategyLoadBalanceConfiguration 类上面的配置是 @AutoConfigureBefore(RibbonClientConfiguration.class),所以它初始化顺序在RibbonClientConfiguration 这个Spring Cloud Ribbon 原始配置类之前。那么三个配置类的初始化顺序为:
StrategyLoadBalanceConfiguration ===> RibbonClientConfiguration ===> PluginLoadBalanceConfiguration
并且这三个配置类里面都配置了IRule 这个负载均衡策略,所以生效的就是 StrategyLoadBalanceConfiguration 里面配置的这个策略。默认采用 DiscoveryEnabledZoneAvoidanceRule 这个策略。PluginLoadBalanceConfiguration 和 RibbonClientConfiguration 都配置了 ILoadBalancer 这个类型的 bean。虽然 RibbonClientConfiguration 的初始化顺序先于PluginLoadBalanceConfiguration ,但是 RibbonClientConfiguration 配置的 ILoadBalancer 上面标注了 @ConditionalOnMissingBean 注解。这样 PluginLoadBalanceConfiguration里面的定义ILoadBalancer 这个类型的 Spring bean 就会生效,而RibbonClientConfiguration里面定义的ILoadBalancer 这个类型的 Spring bean 就不会生效。
3、负载均衡对灰度发布的支持
从上面的分析中我们可以得出结论,在使用 Nepxion Discovery 进行服务灰度发布的时候进行服务选择会使用 DiscoveryEnabledZoneAvoidanceRule这个策略。首先这个策略扩展了Server choose(Object key) 方法,也就是分别从 header 里面和远程配置中心获取version权重和 region 权重。并且优先级:header > 配置中心。如果配置了就会通过 AbstractMapWeightRandomLoadBalance 的两个实现类 StrategyMapWeightRandomLoadBalance 与 RuleMapWeightRandomLoadBalance 这两个负载均衡器进行服务权重选择。
当没有设置version权重和 region 权重时,DiscoveryEnabledZoneAvoidanceRule 就会通过 DiscoveryEnabledZoneAvoidancePredicate#apply 进行服务的选择。其实它最终会调用到 DefaultDiscoveryEnabledAdapter#apply 方法。
DefaultDiscoveryEnabledAdapter#apply
public class DefaultDiscoveryEnabledAdapter implements DiscoveryEnabledAdapter {
@Override
public boolean apply(Server server) {
boolean enabled = applyEnvironment(server);
if (!enabled) {
return false;
}
enabled = applyRegion(server);
if (!enabled) {
return false;
}
enabled = applyAddress(server);
if (!enabled) {
return false;
}
enabled = applyIdBlacklist(server);
if (!enabled) {
return false;
}
enabled = applyAddressBlacklist(server);
if (!enabled) {
return false;
}
enabled = applyVersion(server);
if (!enabled) {
return false;
}
return applyStrategy(server);
}
......
}
从上面的参数我们就可以看到灰度发布支持的纬度包含:
- 版本权重
- Region 权重
- 环境
- Region
- 地址
- ID 黑名单
- 地址黑名单
- 版本
- 自定义策略
我们再来对比一下 Nepxion Discovery 官网 上面支持灰度的定义的参数,是不是和我们上面提到的维度基本相同。

以上就是 Nepxion Discovery 对 Spring Cloud 负载均衡扩展实现灰度发布的实现。