nacos+ribbon自定义ab测试路由策略

原理:通过请求头埋入指定服务的metadata标识,扩展ribbon的choose策略。

一定要注意使用hystrix的隔离策略
划重点!!!一定要信号量隔离。
因为一旦使用线程隔离(线程池存在复用),会导致InheritThreadLocal(只会在线程init时拷贝父线程的ThreadLocal值)失效。

传送门

  • 下载源码编译

应用集成

maven依赖

<dependency>
  <groupId>io.jmnarloch</groupId> 
  <artifactId>ribbon-discovery-filter-spring-cloud-starter</artifactId>
  <version>2.1.0</version>
</dependency>
  • 应用配置
#AB测试用,网关会根据该参数路由
spring.cloud.nacos.discovery.metadata.launcher=A
  • 拦截器
public class AbTestingFilter extends OncePerRequestFilter {
    private final static String SWITCH_KEY = "launcher",
            SWITCH_HEAD_KEY = "switchTag",
            DEFAULT_ENV = "B";//默认B为线上测试环境
    @Value("${spring.cloud.nacos.discovery.metadata.launcher}")
    private String launcherTag;

    @Override
    protected void doFilterInternal(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, FilterChain filterChain) throws ServletException, IOException {
        String switchTag = httpServletRequest.getHeader(SWITCH_HEAD_KEY);
        log.info("head switch tag = {}", switchTag);
        try {
            //不能调度,因为调用链路出现了不一致
            if (!launcherTag.equals(switchTag)) {
                //返回错误即可
                return;
            }
            // 开启ribbon的灰度开关(线程级别)
            RibbonFilterContextHolder.getCurrentContext().add(SWITCH_KEY, switchTag);
            filterChain.doFilter(httpServletRequest, httpServletResponse);
        } finally {
            if (DEFAULT_ENV.equals(switchTag))
                RibbonFilterContextHolder.getCurrentContext().remove(SWITCH_KEY);
        }
    }
}
  • feign conf
    增加请求头,自定义全局的feign interceptor
    @Bean
    public RequestInterceptor newRequestInterceptor(@Value("${spring.cloud.nacos.discovery.metadata.launcher:A}") String tag) {
        return requestTemplate -> requestTemplate.header("switchTag", tag);
    }

zuul网关集成

则参考如下代码

maven依赖同上

拦截器

public class TestingAbFilter extends ZuulFilter {

    @Override
    public String filterType() {
        return FilterConstants.PRE_TYPE;
    }

    @Override
    public int filterOrder() {
        return -1;
    }

    @Override
    public boolean shouldFilter() {
        return true;
    }

    @Override
    public Object run() throws ZuulException {
        RequestContext ctx = RequestContext.getCurrentContext();
        HttpServletRequest request = ctx.getRequest();
        //是否需要做AB切换
        if (request.getParameter("foo") != null) {
            //重写请求头新增AB开关透传到下游服务
            ctx.addZuulRequestHeader("switchTag", "A");
            RibbonFilterContextHolder.getCurrentContext().add("launcher", "A");
        } else {
            RibbonFilterContextHolder.getCurrentContext().add("launcher", "B");
        }
        return null;
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值