基于consul与ribbon灰度的问题

实现方案描述

所有服务都会注册到consul上,灰度实例注册到consul上时打上一个tag,有请求打过来时根据配置的规则判断是否需要进行灰度处理,若需要灰度处理,则进行负载均衡时根据tag进行过滤,获取指定灰度tag的实例进行流量转发。

遇到的问题

当使用灰度实例进行压测时影响到了正常实例,导致流量分布不均,个别实例CPU负载较高,其他实例CPU负载下降明显。

问题定位

默认IRule实现com.netflix.loadbalancer.ZoneAvoidanceRule的父类com.netflix.loadbalancer.PredicateBasedRule的choose方法

/**
 * Get a server by calling {@link AbstractServerPredicate#chooseRandomlyAfterFiltering(java.util.List, Object)}.
 * The performance for this method is O(n) where n is number of servers to be filtered.
 */
@Override
public Server choose(Object key) {
    ILoadBalancer lb = getLoadBalancer();
    Optional<Server> server = getPredicate().chooseRoundRobinAfterFiltering(lb.getAllServers(), key);
    if (server.isPresent()) {
        return server.get();
    } else {
        return null;
    }       
}

从com.netflix.loadbalancer.AbstractServerPredicate获取server实例

/**
  * Choose a server in a round robin fashion after the predicate filters a given list of servers and load balancer key. 
  */
public Optional<Server> chooseRoundRobinAfterFiltering(List<Server> servers, Object loadBalancerKey) {
    List<Server> eligible = getEligibleServers(servers, loadBalancerKey);
    if (eligible.size() == 0) {
        return Optional.absent();
    }
    // 根据下标获取服务
    return Optional.of(eligible.get(incrementAndGetModulo(eligible.size())));
}


/**
 * Referenced from RoundRobinRule
 * Inspired by the implementation of {@link AtomicInteger#incrementAndGet()}.
 *
 * @param modulo The modulo to bound the value of the counter.
 * @return The next value.
 */
private int incrementAndGetModulo(int modulo) {
    for (;;) {
        int current = nextIndex.get();
        int next = (current + 1) % modulo;
        if (nextIndex.compareAndSet(current, next) && current < modulo)
            return current;
    }
}

 错误主要发生在incrementAndGetModulo这个方法。同一个服务会使用同一个负载均衡器,无论它们注册到consul时是否带有灰度标签,正清情况下(无灰度请求)incrementAndGetModulo会按部就班的进行递增、取模,获取实例下标,形参modulo就是过滤后可选服务集合的数量;当引入灰度实例后modulo参数不一致,如正常实例50,灰度实例1,那么modulo分别为50和1,而且它们同属一个服务,共用一个loadbalance,会操作同一个nextIndex,所以灰度流量打乱了正常请求的计数器,导致了流量分布不均。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

吊儿郎当当

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值