Redian新闻
>
通过Jenkins构建CI/CD实现全链路灰度

通过Jenkins构建CI/CD实现全链路灰度

公众号新闻

点击上方“芋道源码”,选择“设为星标

管她前浪,还是后浪?

能浪的浪,才是好浪!

每天 10:33 更新文章,每天掉亿点点头发...

源码精品专栏

 
来源:阿里巴巴中间件

本文介绍通过 Jenkins 构建流水线的方式实现全链路灰度功能。

在发布过程中,为了整体稳定性,我们总是希望能够用小部分特定流量来验证下新发布应用是否正常。

即使新版本有问题,也能及时发现,控制影响面,保障了整体的稳定性。

01 整体架构

Aliware

我们以如下 Demo 为例:

为了保证稳定,我们约定如下上线流程:

其中,在灰度验证中,有几种不同的策略:

  • 直接使用线上小部分流量来测试(按照百分比放量)
  • 从线上按照特定规则选择流量(比如特定的 header、特定的 cookie 等)
  • 在客户端或浏览器上标识出流量是否灰度(比如通过 header 传递)

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://gitee.com/zhijiantianya/ruoyi-vue-pro
  • 视频教程:https://doc.iocoder.cn/video/

02 部署应用&创建泳道

Aliware

按照参考文档部署应用后,我们首先要区分线上流量和灰度流量。

创建泳道组,将整个链路涉及到的应用全选:

然后创建泳道组,将符合规则的应用划入 gray 泳道:

注:没有匹配的流量,会走到基线环境,也就是没有打标的应用节点上。

配置完成后,访问网关,如果不符合灰度规则,走基线环境:

如何符合灰度规则,走灰度环境:

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://gitee.com/zhijiantianya/yudao-cloud
  • 视频教程:https://doc.iocoder.cn/video/

03 Jenkins 流水线

Aliware

本文实践需要将源码打包后执行镜像推送,请确保 Jenkins 有权限推送到镜像仓库中。具体操作,请参见使用 kaniko 构建和推送容器镜像。

在 Jenkins 命名空间下使用生成的 config.json 文件创建名为 jenkins-docker-cfg 的 Secret。

kubectl create secret generic jenkins-docker-cfg -n jenkins --from-file=/root/.docker/config.json

1 在 Jenkins 中创建全链路灰度发布流水线

基于 Jenkins 实现自动化发布的流水线,通过该流水线可以使应用发布具备可灰度、可观测、可回滚的安全生产三板斧能力。

  1. 在 Jenkins 控制台左侧导航栏单击新建任务

  2. 输入任务名称 ,选择流水线 ,然后单击确定

  3. 在顶部菜单栏单击流水线 页签,在流水线 区域配置相关参数选择,输入脚本路径 ,然后单击保存

  • 定义 :选择 Pipeline script from SCM。
  • SCM :选择 Git。
  • Repository URL :输入 Git 仓库的 URL。
  • 脚本路径 :输入 Jenkinsfile。

您可以参考以下的文件填写好指定的参数,当然您也可以根据需求编写 Jenkinsfile ,并上传至 Git 的指定路径下(流水线中指定的脚本路径)。

#!groovy
pipeline {
    // 定义本次构建使用哪个标签的构建环境,本示例中为 “slave-pipeline”
    agent{
        node{
          label 'slave-pipeline'
        }
    }
    //常量参数,初始确定后一般不需更改
    environment{
        IMAGE = sh(returnStdout: true,script: 'echo registry.$image_region.aliyuncs.com/$image_namespace/$image_reponame:$image_tag').trim()
        BRANCH =  sh(returnStdout: true,script: 'echo $branch').trim()
    }
    options {
        //保持构建的最大个数
        buildDiscarder(logRotator(numToKeepStr: '10'))
    }
    parameters {
        string(name: 'image_region'defaultValue: 'cn-shanghai')
        string(name: 'image_namespace'defaultValue: 'yizhan')
        string(name: 'image_reponame'defaultValue: 'spring-cloud-a')
        string(name: 'image_tag'defaultValue: 'gray')
        string(name: 'branch'defaultValue: 'master')
        string(name: 'number_of_pods'defaultValue: '2')
    }
    //pipeline的各个阶段场景
    stages {
        stage('代码打包') {
            steps{
                container("maven") {
                    echo "镜像构建......"
                    sh "cd A && mvn clean package"
                }
            }
        }
        stage('镜像构建及发布'){
          steps{
              container("kaniko") {
                  sh "kaniko -f `pwd`/A/Dockerfile -c `pwd`/A --destination=${IMAGE} --skip-tls-verify"
              }
          }
        }
        stage('灰度部署') {
            steps{
                container('kubectl') {
                    echo "灰度部署......"
                    sh "cd A && sed -i -E \"s/${env.image_reponame}:.+/${env.image_reponame}:${env.image_tag}/\" A-gray-deployment.yaml"
                    sh "cd A && sed -i -E \"s/replicas:.+/replicas: ${env.number_of_pods}/\" A-gray-deployment.yaml"
                    sh "kubectl apply -f A/A-gray-deployment.yaml -n default"
                }
            }
        }
        stage('结束灰度') {
            input {
                message "请确认是否全量发布"
                ok "确认"
                parameters {
                    string(name: 'continue'defaultValue: 'true'description: 'true为全量发布,其他为回滚')
                }
            }
            steps{
                script {
                    env.continue = sh (script: 'echo ${continue}'returnStdout: true).trim()
                    if (env.continue.equals('true')) {
                        container('kubectl') {
                            echo "全量发布......"
                            sh "cd A && sed -i -E \"s/${env.image_reponame}:.+/${env.image_reponame}:${env.image_tag}/\" A-deployment.yaml"
                            sh "cd A && sed -i -E \"s/replicas:.+/replicas: ${env.number_of_pods}/\" A-deployment.yaml"
                            sh "kubectl apply -f A/A-deployment.yaml -n default"
                        }
                    } else {
                        echo '回滚'
                    }
                    container('kubectl') {
                        sh "kubectl delete -f A/A-gray-deployment.yaml -n default"
                    }
                }
            }
        }
    }
}

2 构建 Jenkins 流水线

  1. 在 Jenkins 控制台单击流水线右侧的图标。

  2. 单击流水线的开始构建

说明:第一次构建因为需要从 Git 仓库拉取配置并初始化流水线,所以可能会报错,再次执行 Build with Parameters,生成相关的参数,填写相关的参数,再次执行构建。

查看部署状态,代码打包,镜像构建及发布,灰度部署 阶段都已经完成,结束灰度 阶段等待确认。

  • 如果验证结果符合预期,则执行全量发布,请参见后文的全量发布应用。
  • 如果验证结果不符合预期时,则执行回滚,请参见后文的回滚应用。

3 结果验证

  1. 登录容器服务控制台,在控制台左侧导航栏中,单击集群

  2. 集群列表 页面中,单击目标集群名称或者目标集群右侧操作 列下的详情

  3. 在集群管理页面左侧导航栏选择工作负载 > 无状态

  4. 无状态 应用列表页面,spring-cloud-a-gray应用已经自动创建,并且它的镜像已经替换为spring-cloud-a:gray版本。

  1. 在集群管理页面左侧导航栏选择网络 > 服务 ,选择设置的命名空间,单击zuul-slb 服务的外部端点 ,查看真实的调用情况。
  • 不带灰度 Header 进行调用,发现路由到 A 的正常节点。

    • Curl 命令:
curl http://182.92.XX.XX/A/a
- 执行结果如下:
A[10.4.XX.XX] -> B[10.4.XX.XX] -> C[10.4.XX.XX]%
  • 带上符合条件的参数进行访问,路由到 A 的灰度节点中。

    • Curl 命令:
curl http://182.92.XX.XX/A/a?name=xiaoming
- 执行结果如下:
Agray[10.4.XX.XX] -> B[10.4.XX.XX] -> C[10.4.XX.XX]%
  1. 登录 MSE 治理中心控制台,在应用详情页面,可以看到灰度流量已经进入到灰度的 Pod 中。

4 全量发布应用

结果验证通过之后,确认全量发布。

  1. 在 Jenkins 控制台中,单击目标流水线名称。

  2. 单击需要全量发布的阶段,在请确认是否全量发布 对话框中输入 true,然后单击确认

  1. 在容器服务控制台,发现 spring-cloud-a-gray 应用已经被删除,并且 spring-cloud-a 应用的镜像已经替换为 spring-cloud-a:gray 版本。
  1. 在 MSE治理中心控制台,发现灰度流量已经消失。

5 回滚应用

如果发现验证结果不符合预期时,则回滚应用。

  1. 在 Jenkins 控制台中,单击目标流水线名称。

  2. 单击需要全量发布的阶段,在请确认是否全量发布 对话框中输入 false,然后单击确认

  1. 在容器服务控制台,发现 spring-cloud-a-gray 应用已经被删除,并且 spring-cloud-a 应用的镜像仍然是老版本。
  1. 在 MSE 治理中心控制台,发现灰度流量已经消失。

04 总结

Aliware

在微服务治理架构中,全链路灰度功能能提供虚拟泳道,极大的方便了测试、发布时的快速验证,能够帮助 DevOPs 提升线上稳定性。

阿里云微服务引擎(MSE)能够给您带来全生命周期的、全方位的微服务治理能力,保障您的线上稳定性、提升开发、运维效率。

相关链接:

参考文档:

https://github.com/aliyun/alibabacloud-microservice-demo/blob/master/mse-simple-demo/helm/mse-simple-demo/README.md

示例代码仓库地址:

https://gitee.com/mse-group/alibabacloud-microservice-demo/tree/master/mse-simple-demo

容器服务控制台

https://cs.console.aliyun.com/#/k8s/cluster/list

MSE治理中心控制台

https://mse.console.aliyun.com/#/overview

使用 kaniko 构建和推送容器镜像:

https://help.aliyun.com/document_detail/106712.htm



欢迎加入我的知识星球,一起探讨架构,交流源码。加入方式,长按下方二维码噢

已在知识星球更新源码解析如下:

最近更新《芋道 SpringBoot 2.X 入门》系列,已经 101 余篇,覆盖了 MyBatis、Redis、MongoDB、ES、分库分表、读写分离、SpringMVC、Webflux、权限、WebSocket、Dubbo、RabbitMQ、RocketMQ、Kafka、性能测试等等内容。

提供近 3W 行代码的 SpringBoot 示例,以及超 4W 行代码的电商微服务项目。

获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。

文章有帮助的话,在看,转发吧。

谢谢支持哟 (*^__^*)

微信扫码关注该文公众号作者

戳这里提交新闻线索和高质量文章给我们。
相关阅读
华为3D实景地图,30分钟构建超精细数字世界,达到厘米级四川观察:以互联网品牌全链路服务,探索“新闻+政务服务商务”运营新模式转帖《为你而来》----(文)ZYWAdobe 2022 发布会,Ps 实现全自动做图了?!bāo huā shēng?bō huā shēng?全球首富马斯克发推炫耀:ACCD授予我荣誉学位!快来看看ACCD这所院校到底牛在哪?!《留学鬼话备忘录》信了这些鬼话,实现全拒不是梦。特斯拉上海工厂停产,微信灰度测试划线功能,阿里云复盘服务中断事件,台积电3nm下周量产,这就是今天的其他大新闻!孩子王上市一周年:单季净利润持续改善 已实现全渠道数字化REAL ID实施日延长至2025年!冬季“三重流行病”传播来到高点!纽约急诊室爆满急急急!想短时间实现全家移居欧洲?这个项目最快 | 公司专栏灰度要约回购背后有何盘算?DCG能否安然得物云原生全链路追踪Trace2.0-采集篇德国计划出台“绿卡”美移民局最新报告:2022财年中国人获得35,306张职业移民绿卡,未来几个月I-140可实现全面加急虽在异乡非异客 且把他乡作故乡 下Linux 6.2内核合并了新的Zstd实现专访丨民政部副部长唐承沛:推动实现全体老年人享有基本养老服务自由西欧4国7天5晚4-7人小众私家包团丨法国巴黎进出、比利时布鲁塞尔、荷兰阿姆斯特丹、卢森堡、巴黎自由行 CDGCDG7VEPSON LCD Projector H840B with wall mount急急急!想短时间实现全家移居欧洲?这个项目最快!一文看懂分布式链路监控系统巴色会(三)----读者来信在德国,你算富人吗?自从项目用了灰度发布,睡觉真香!核心交换机链路聚合、冗余、堆叠、热备份BLACKPINK杂志拍摄幕后花絮曝光,Jennie、Lisa再次挑战造型尺度?说实话,Jenkins 真得牛逼!只是你不会用而已好消息,Real ID实施日期推迟至2025年!吉普JEEP SPIRIT线下专柜799!!!三折不到入手美国大牌JEEP情侣款三合一冲锋衣~[模型] John Jenkins Designs 出品 1/30比例 I型装甲弹药运输车美媒:中国又实现全球首创,歼20指挥攻击11可组织无人机集群空战好消息,纽约Real ID实施日期推迟至2025年!如何利用EB-5双递交实现全家快速安全合法登陆美国李家超:中央已同意香港可以与内地逐步有序实现全面“通关”
logo
联系我们隐私协议©2024 redian.news
Redian新闻
Redian.news刊载任何文章,不代表同意其说法或描述,仅为提供更多信息,也不构成任何建议。文章信息的合法性及真实性由其作者负责,与Redian.news及其运营公司无关。欢迎投稿,如发现稿件侵权,或作者不愿在本网发表文章,请版权拥有者通知本网处理。