文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

SpringCloudGateway Nacos GitlabRunner全自动灰度服务怎么搭建

2023-07-06 03:19

关注

这篇文章主要介绍了SpringCloudGateway Nacos GitlabRunner全自动灰度服务怎么搭建的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇SpringCloudGateway Nacos GitlabRunner全自动灰度服务怎么搭建文章都会有所收获,下面我们一起来看看吧。

1 | 业务场景说明

要实现的业务场景:

2 | 具体实现方案

这里采用 SpringCloudGateway(SCG) + Nacos + GitlabRunner 来实现整个自动化的灰度发布。

下面分别从以上这三个组件来搭建。

2.1 | SCG

直接上代码,通过注释讲解。

@Component@Slf4jpublic class GrayLoadBalancerClientFilter implements GlobalFilter, Ordered {    @Resource    private LoadBalancerClientFactory clientFactory;    @Resource    private CustomProperty customProperty;    @Override    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {        URI url = exchange.getAttribute(ServerWebExchangeUtils.GATEWAY_REQUEST_URL_ATTR);        ServerWebExchangeUtils.addOriginalRequestUrl(exchange, url);        if (url == null || BizConstant.HTTP.equalsIgnoreCase(url.getScheme())) {            return chain.filter(exchange);        }        return doFilter(exchange, chain, url);    }    private Mono<Void> doFilter(ServerWebExchange exchange, GatewayFilterChain chain, URI url) {        return this.choose(exchange).doOnNext(res -> {            if (!res.hasServer()) {                throw NotFoundException.create(true, "Unable to find instance for ".concat(url.getHost()));            }            URI uri = exchange.getRequest().getURI();            String overrideScheme = null;            DelegatingServiceInstance delegatingServiceInstance = new DelegatingServiceInstance(res.getServer(), overrideScheme);            URI reqUrl = this.reconstructURI(delegatingServiceInstance, uri);            if (log.isDebugEnabled()) {                log.debug("GrayLoadBalancerClientFilter url chosen: {}", reqUrl.toString());            }            exchange.getAttributes().put(ServerWebExchangeUtils.GATEWAY_REQUEST_URL_ATTR, reqUrl);        }).then(chain.filter(exchange));    }    private URI reconstructURI(DelegatingServiceInstance delegatingServiceInstance, URI originalUri) {        return LoadBalancerUriTools.reconstructURI(delegatingServiceInstance, originalUri);    }    private Mono<Response<ServiceInstance>> choose(ServerWebExchange exchange) {        URI uri = exchange.getAttribute(ServerWebExchangeUtils.GATEWAY_REQUEST_URL_ATTR);        if (uri == null) {            throw new MMException("{} is null", ServerWebExchangeUtils.GATEWAY_REQUEST_URL_ATTR);        }        GrayLoadBalancer loadBalancer = new GrayLoadBalancer(clientFactory.getLazyProvider(uri.getHost(), ServiceInstanceListSupplier.class), uri.getHost(), customProperty);        return loadBalancer.choose(this.createRequest(exchange));    }    private Request createRequest(ServerWebExchange exchange) {        return new DefaultRequest(exchange.getRequest().getHeaders());    }    @Override    public int getOrder() {        return FILTER_ORDER_GRAY;    }}

NOTE

FILTER_ORDER_GRAY 是一个 int 常量,其值不能随意定义(如-1,0,1,2之类)。从下表可以看到,SCG 的 LoadBalancerClientFilter 执行顺序是 10100,那么 GrayLoadBalancerClientFilter 的执行顺序必须 > 10100 (否则自定义的 Filter 里就会有变量未被赋值), 这里假定 FILTER_ORDER_GRAY = 10110

SpringCloudGateway Nacos GitlabRunner全自动灰度服务怎么搭建

@Slf4jpublic class GrayLoadBalancer implements ReactorServiceInstanceLoadBalancer {    private ObjectProvider<ServiceInstanceListSupplier> serviceInstanceListSupplierProvider;    private String serviceId;    private CustomProperty customProperty;    public GrayLoadBalancer(ObjectProvider<ServiceInstanceListSupplier> serviceInstanceListSupplierProvider, String serviceId, CustomProperty customProperty) {        this.serviceId = serviceId;        this.serviceInstanceListSupplierProvider = serviceInstanceListSupplierProvider;        this.customProperty = customProperty;    }    @Override    public Mono<Response<ServiceInstance>> choose(Request request) {        HttpHeaders headers = (HttpHeaders) request.getContext();        if (this.serviceInstanceListSupplierProvider != null) {            ServiceInstanceListSupplier supplier = this.serviceInstanceListSupplierProvider.getIfAvailable(NoopServiceInstanceListSupplier::new);            return supplier.get().next().map(item -> getInstanceResponse(item, headers));        }        return null;    }    private Response<ServiceInstance> getInstanceResponse(List<ServiceInstance> instances, HttpHeaders headers) {        if (instances.isEmpty()) {            return getServiceInstanceEmptyResponse();        }        return getServiceInstanceResponseByUidsOrGrayTag(instances, headers);    }        private Response<ServiceInstance> getServiceInstanceResponseByUidsOrGrayTag(List<ServiceInstance> instances, HttpHeaders headers) {        List<ServiceInstance> grayInstances = new ArrayList<>();        List<ServiceInstance> normalInstances = new ArrayList<>();        for (ServiceInstance instance : instances) {            Map<String, String> metadata = instance.getMetadata();            // nacos元数据包含“gray-tag”的key值,且value="true",则判定为灰度实例            String isGrayInstance = metadata.get(BizConstant.GRAY_TAG);            if (BizConstant.TRUE.equals(isGrayInstance)) {                grayInstances.add(instance);            } else {                normalInstances.add(instance);            }        }        //没有灰度服务,直接返回        if (grayInstances.isEmpty()) {            return new DefaultResponse(chooseOneInstance(normalInstances));        }        //有灰度服务,判断是否需要灰度        if (checkIfNeedGray(headers)) {            log.info("gray service of {} will be called", this.serviceId);            return new DefaultResponse(chooseOneInstance(grayInstances));        }        return new DefaultResponse(chooseOneInstance(normalInstances));    }        private ServiceInstance chooseOneInstance(List<ServiceInstance> serviceInstances) {        // strategy 1:可用的里面随机选择一个        int size = serviceInstances.size();        if (size == 1) {            return serviceInstances.get(0);        }        Random rand = new Random();        int random = rand.nextInt(size);        return serviceInstances.get(random);    }        private boolean checkIfNeedGray(HttpHeaders headers) {        String grayTag = headers.getFirst(BizConstant.GRAY_TAG);        if (grayTag != null) {            if (BizConstant.TRUE.equalsIgnoreCase(grayTag)) {                // todo 可扩展点:目前是只判断header里是否有BizConstant.GRAY_TAG的kv不为空且v="true",后面v可以改为版本号                return true;            }        }        String uid = headers.getFirst(BizConstant.UID);        if (uid != null && customProperty.getGraySetting().getGrayUids().contains(uid)) {            return true;        }        return false;    }    private Response<ServiceInstance> getServiceInstanceEmptyResponse() {        log.warn("No servers available for service: " + this.serviceId);        return new EmptyResponse();    }}
@Component@Slf4jpublic class Https2HttpFilter implements GlobalFilter, Ordered {    @Override    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {        ServerHttpRequest request = exchange.getRequest();        URI originalUri = request.getURI();        ServerHttpRequest.Builder mutate = request.mutate();        String forwardUri = request.getURI().toString();        if (forwardUri != null && forwardUri.startsWith(BizConstant.HTTPS)) {            try {                URI mutatedUri = new URI(BizConstant.HTTP,                        originalUri.getUserInfo(),                        originalUri.getHost(),                        originalUri.getPort(),                        originalUri.getPath(),                        originalUri.getQuery(),                        originalUri.getFragment());                mutate.uri(mutatedUri);            } catch (Exception e) {                log.error(e.getMessage());                throw new MMException("Https related error");            }        }        ServerHttpRequest build = mutate.build();        return chain.filter(exchange.mutate().request(build).build());    }    @Override    public int getOrder() {        return FILTER_ORDER_HTTPS_2_HTTP;    }}

NOTE

FILTER_ORDER_HTTPS_2_HTTP 是一个 int 常量,需要满足 LoadBalancerClientFilter 的执行顺序(10100) < FILTER_ORDER_HTTPS_2_HTTP < FILTER_ORDER_GRAY (10110)。这里可以假定 FILTER_ORDER_HTTPS_2_HTTP = 10105。之所以需要加一个Https2HttpFilter 过滤器,是因为如果 https 请求直接进入到 GrayLoadBalancerClientFilter 会报 NotSslRecordException 证书错误。

2.2 | Nacos

Nacos 主要做一件事情:通过 metadata 维护灰度服务。

SpringCloudGateway Nacos GitlabRunner全自动灰度服务怎么搭建

从上图可以看出,metadata 里 gray-tag=true 的实例即为灰度服务的实例。

通过 webUI 的编辑按钮可以实时的新增修改 metadata。

那么,如何在代码侧配置呢?

可以直接在bootstrap.yml添加以下字段:

spring:  cloud:    nacos:      discovery:        metadata:          # 如果${gray}变量不存在,则gray-tag=false          gray-tag: ${gray:false}

2.3 | GitlabRunner

gitlab-runner 主要是 kube_deploy.yml 和 .gitlab-ci.yml 的一个联动配置

apiVersion: apps/v1kind: Deploymentmetadata:  name: ccc-deploy  namespace: cccspec:  template:    spec:      containers:       - env:          - name: gray            value: "gray-tag" # 这里的gray-tag值 将会在在.gitlab-ci.yml的脚本中被替换
...stages:  - k8s-deployk8s-deploy-gray-service:  stage: k8s-deploy  script:    - echo "=============== 开始 k8s 部署任务 ==============="    - sed -i "s/gray-tag/true/g" kube_deploy.yml # 这    - kubectl apply -f kube_deploy.yml  only:    - /^tag_gray_.*$/k8s-deploy-normal-service:  stage: k8s-deploy  script:    - echo "=============== 开始 k8s 部署任务 ==============="    - sed -i "s/gray-tag/false/g" kube_deploy.yml # 这里替换 gray-tag 为 false    - kubectl apply -f kube_deploy.yml  only:    - /^tag_normal_.*$/ ...

此时,当打了一个以 tag_gray_ 开头的 tag 之后,kube_deploy.yml里的gray-tag就会被替换成 true,那么,nacos 的元数据上就会有一个gray-tag=true的标签,就会走灰度服务的发布流程。同理,以 tag_normal_ 开头的 tag,就会走正常服务的发布流程。

把这段脚本嵌入到 pipeline 之后,就可以通过 tag 的方式,自动化部署灰度/正常服务了。

3 | 后续 TODO

目前实现的是后端服务的灰度发布,一个完整的灰度,还包含了前端应用的灰度,后续会就前端的灰度发布再做一次整理。

4 | 使用版本说明

实战依赖版本

GroupSpring CloudSpring CloudSpring CloudSpring Cloud Alibaba NacosSpring Cloud Alibaba Nacos
ComponentHoxton.SR3GatewayLoadBalancerConfigDiscovery
Version-2.2.2.RELEASE2.2.2.RELEASE2.2.5.RELEASE2.2.5.RELEASE

需要注意的

在 Spring Cloud 全家桶中,最初的网关使用的是 Netflix 的 Zuul 1x 版本,但是由于其性能问题,Spring Cloud 在苦等 Zuul 2x 版本未果的情况下,推出了自家的网关产品,取名叫 Spring Cloud Gateway (以下简称 SCG),基于Webflux,通过底层封装Netty,实现异步IO,大大地提示了性能。

Zuul 1x 版本

本质上就是一个同步Servlet,采用多线程阻塞模型进行请求转发。简单讲,每来一个请求,Servlet容器要为该请求分配一个线程专门负责处理这个请求,直到响应返回客户端这个线程才会被释放返回容器线程池。如果后台服务调用比较耗时,那么这个线程就会被阻塞,阻塞期间线程资源被占用,不能干其它事情。我们知道Servlet容器线程池的大小是有限制的,当前端请求量大,而后台慢服务比较多时,很容易耗尽容器线程池内的线程,造成容器无法接受新的请求。且不支持任何长连接,如websocket

NOTE 由于两个网关的底层架构不一致,负载均衡的逻辑也完全不一致,本文只探讨 Spring Cloud Gateway 配合 Nacos 来实现灰度发布( Spring Cloud Zuul 网关的灰度发布不展开)。

关于“SpringCloudGateway Nacos GitlabRunner全自动灰度服务怎么搭建”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“SpringCloudGateway Nacos GitlabRunner全自动灰度服务怎么搭建”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网行业资讯频道。

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯