云原生核心技术 (12/12): 终章:使用 GitLab CI 将应用自动部署到 K8s (保姆级教程)

大家好,欢迎来到《云原生核心技术》系列的最终章!

我们一起走过了漫长而充实的旅程。从 Docker 的集装箱,到 K8s 这座自动化的数字港口;从部署单个 Pod,到构建复杂的有状态应用。现在,我们站在了实现全自动化的最后一站。

在上一篇中,我们理解了 CI/CD 的核心理念——那条连接开发与运维、通往高效与可靠的自动化高速公路。今天,我们不再纸上谈兵,而是要亲手铺设这条公路的每一块砖石。

我们将使用一个强大且整合度极高的工具——GitLab CI/CD,为我们第十篇中的 Spring Boot + MySQL + Redis 应用项目,打造一条从代码提交到自动部署至 Kubernetes 的完整流水线。

读完并实践本篇文章,你将获得一项在现代软件工程中极具价值的技能:将 DevOps 思想落地,实现应用的自动化、可靠化交付。

准备好,让我们开始最后的施工,为这个系列画上一个完美的句号!


第一步:环境准备与规划

在开始之前,请确保你已经拥有:

  1. 一个 GitLab 账号和项目:你可以使用 GitLab.com 的免费版,或者自建的 GitLab 实例。将我们第十篇的 Spring Boot 项目代码推送到这个 GitLab 仓库中。
  2. 一个运行中的 K8s 集群:继续使用我们的 Minikube 集群。minikube start
  3. 一个 Docker 镜像仓库:比如 Docker Hub。你需要一个可以推送镜像的账号。
  4. kubectlhelm 命令行工具:Helm 是 K8s 的包管理器,我们将用它来方便地安装 GitLab Runner。

我们的流水线规划如下:

  • test 阶段:运行 Maven 测试,确保代码质量。
  • build 阶段:使用 Docker 构建 Spring Boot 应用的镜像。
  • push 阶段:将构建好的镜像推送到 Docker Hub。
  • deploy 阶段:使用 kubectl 命令更新 K8s 中对应 Deployment 的镜像版本,触发滚动更新。

第二步:在 K8s 中安装 GitLab Runner

GitLab Runner 是执行 .gitlab-ci.yml 文件中定义任务的代理程序。最云原生的方式,就是把它直接安装在我们的 K8s 集群里。

  1. 获取 Runner 注册信息

    • 打开你的 GitLab 项目,在左侧导航栏找到 Settings -> CI/CD
    • 展开 Runners 部分。你会看到一个 URL (https://gitlab.com/) 和一个注册令牌 (registration token),类似 GR13489...。把这两项复制下来,后面会用到。
  2. 使用 Helm 安装 GitLab Runner

    • 添加 GitLab Helm 仓库:
      helm repo add gitlab https://charts.gitlab.io
      helm repo update
      
    • 创建一个 values.yaml 配置文件,用于定制我们的 Runner。
      # values.yaml
      gitlabUrl: "https://gitlab.com/"  # 填入你复制的 GitLab URL
      runnerRegistrationToken: "GR13489..." # 填入你复制的注册令牌# Runner 的配置
      runners:# Runner 的标签,后面在 .gitlab-ci.yml 中会用它来选择 Runnertags: "kubernetes,minikube"
      
    • 执行 Helm 安装命令:
      helm install --namespace gitlab-runner --create-namespace gitlab-runner \
      -f values.yaml \
      gitlab/gitlab-runner
      
    • 验证安装:稍等片刻,然后检查 Pod 是否在 gitlab-runner 命名空间中成功运行。
      kubectl get pods -n gitlab-runner
      # 你应该能看到一个名为 gitlab-runner-gitlab-runner-xxxx 的 Pod 正在运行
      
    • 回到 GitLab 项目的 Runner 设置页面,你应该能看到一个新的 Runner 已经注册成功并显示为绿色可用状态。

第三步:配置 GitLab CI/CD 变量

为了安全,我们不能把敏感信息(如 Docker Hub 密码、K8s 凭证)直接写在 .gitlab-ci.yml 文件里。我们需要将它们存储在 GitLab 的 CI/CD 变量中。

进入 Settings -> CI/CD,展开 Variables 部分。添加以下变量:

  1. Docker Hub 凭证

    • DOCKER_USER: 你的 Docker Hub 用户名。
    • DOCKER_PASS: 你的 Docker Hub 密码或访问令牌 (Access Token)。(建议设置为 Masked 类型)
  2. K8s 集群凭证 (kubeconfig)

    • 我们需要让 GitLab Runner 能够访问我们的 Minikube 集群。
    • 在你的本地电脑上,找到 ~/.kube/config 文件,并将其内容完整复制下来。
    • 在 GitLab 变量设置中,创建一个新变量:
      • Key: KUBE_CONFIG
      • Type: File (这很重要,它会将变量内容作为文件提供给 Runner)
      • Value: 粘贴你刚才复制的 kubeconfig 全部内容。

第四步:编写 .gitlab-ci.yml,定义流水线

这是最核心的一步。在你的 Spring Boot 项目根目录下,创建 .gitlab-ci.yml 文件,内容如下:

# .gitlab-ci.yml# 定义流水线的各个阶段,任务会按此顺序执行
stages:- test- build- push- deploy# 定义一些全局变量
variables:# Maven 镜像,用于测试阶段MAVEN_IMAGE: maven:3.8-openjdk-17# Docker 镜像名称。$CI_PROJECT_NAME 是 GitLab 预定义变量,代表项目名IMAGE_NAME: $CI_REGISTRY_USER/$CI_PROJECT_NAME# 镜像标签。$CI_COMMIT_SHORT_SHA 是提交的短哈希值,确保每次构建的标签唯一IMAGE_TAG: $CI_COMMIT_SHORT_SHA# --- 阶段一:测试 ---
maven_test:stage: testimage: $MAVEN_IMAGEscript:- echo "Running Maven tests..."- ./mvnw testtags:- kubernetes # 指定使用我们安装在 K8s 里的 Runner# --- 阶段二:构建 Docker 镜像 ---
docker_build:stage: build# 使用 Docker-in-Docker (dind) 镜像,以便在容器内执行 Docker 命令image: docker:20.10.16services:- docker:20.10.16-dindscript:- echo "Building Docker image..."- docker build -t $IMAGE_NAME:$IMAGE_TAG .# 保存镜像为 tar 文件,以便在后续阶段使用- docker save $IMAGE_NAME:$IMAGE_TAG > image.tar# artifacts 用于在不同阶段之间传递文件artifacts:paths:- image.tartags:- kubernetes# --- 阶段三:推送 Docker 镜像 ---
docker_push:stage: pushimage: docker:20.10.16services:- docker:20.10.16-dind# 需要上一个阶段构建的镜像文件needs:- docker_buildscript:- echo "Pushing Docker image..."# 加载镜像文件- docker load < image.tar# 登录 Docker Hub,使用我们之前定义的 CI/CD 变量- echo $DOCKER_PASS | docker login -u $DOCKER_USER --password-stdin- docker push $IMAGE_NAME:$IMAGE_TAGtags:- kubernetes# --- 阶段四:部署到 Kubernetes ---
deploy_to_k8s:stage: deploy# 使用一个包含 kubectl 的镜像image:name: bitnami/kubectl:latestentrypoint: [""]needs:- docker_pushscript:- echo "Deploying to Kubernetes..."# 配置 kubectl,让它使用我们通过变量传入的 KUBE_CONFIG- export KUBECONFIG=$KUBE_CONFIG# 使用 kubectl set image 命令来更新 Deployment 的镜像,触发滚动更新# 将 'springboot-app-deployment' 替换为你在 K8s 中实际的 Deployment 名称- kubectl set image deployment/springboot-app-deployment springboot-app=$IMAGE_NAME:$IMAGE_TAG- echo "Deployment updated successfully!"tags:- kubernetes

请注意

  • springboot-app-deployment 和容器名 springboot-app 替换成你在第十篇中使用的实际名称。
  • $CI_REGISTRY_USER 默认是你的 GitLab 用户名,你也可以硬编码为你的 Docker Hub 用户名。

第五步:触发流水线,见证自动化魔法!

现在,一切准备就绪。

  1. 提交并推送代码:将包含 .gitlab-ci.yml 的项目推送到你的 GitLab 仓库。

    git add .
    git commit -m "feat: Add GitLab CI/CD pipeline for automated deployment"
    git push
    
  2. 观察流水线

    • 立刻打开你的 GitLab 项目页面,进入 CI/CD -> Pipelines
    • 你会看到一个新的流水线被触发了,状态为 running
    • 点击这个流水线,你可以看到我们定义的四个阶段 test, build, push, deploy 正在依次执行。你可以点击每一个 job 查看详细的实时日志输出。
  3. 验证部署

    • deploy 阶段成功完成后,打开你的终端。
    • 检查你的 Deployment,你会看到它的镜像已经被更新为最新的版本(带有新的 commit 哈希标签)。
      kubectl describe deployment springboot-app-deployment
      
    • 检查 Pod,你会看到 K8s 已经完成了滚动更新,新的 Pod 正在运行。
      kubectl get pods
      

现在,尝试修改一下 Spring Boot 代码中的某个字符串,然后再次 git push。你将再次看到整个自动化流程被触发,几分钟后,你的线上应用就被无缝地更新了!


总结与展望:你的云原生之旅,才刚刚开始

恭喜你!你已经成功地走完了从 Docker 到 Kubernetes 再到自动化 CI/CD 的全过程!

回顾这12篇的旅程,我们从最基础的概念开始,一步一个脚印,最终搭建起了一套现代、高效、自动化的云原生应用交付系统。你不再是一个简单的代码编写者或服务器管理者,而是一个能够掌控从代码到生产完整生命周期的现代工程师。

但这并不是终点,而是一个全新的起点。 云原生的世界浩瀚无垠,还有更多激动人心的领域等待你去探索:

  • 服务网格 (Service Mesh): 使用 Istio, Linkerd 来实现更高级的流量管理、可观测性和安全性。
  • 可观测性 (Observability): 深入学习 Prometheus, Grafana, Jaeger,打造全方位的监控、日志和追踪系统。
  • Serverless/FaaS: 探索 Knative, OpenFaaS,让你的应用真正做到按需运行,极致弹性。
  • GitOps: 学习 ArgoCD, Flux,将 Git 作为管理基础设施和应用的唯一真实来源。

希望这个系列能为你打开一扇通往云原生世界的大门。保持好奇,持续学习,愿你在这片广阔的数字海洋中,乘风破浪,扬帆远航!

感谢你的一路相伴,我们的系列到此结束。江湖再见!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/news/909201.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

DEVICENET转MODBUS TCP网关连接ABB机器人配置案例

在工业自动化场景中&#xff0c;DeviceNet和Modbus TCP是两种常见的通信协议。DeviceNet通常用于连接现场设备&#xff08;如传感器、执行器等&#xff09;&#xff0c;而Modbus TCP则广泛应用于以太网环境下的远程监控和数据采集。当需要将基于DeviceNet协议的ABB机器人集成到…

达梦数据库单机部署dmhs同步复制(dm8->kafka)

本文讨论了达梦数据实时同步软件DMHS的相关内容&#xff0c;包括概念总结、环境模拟及部署实现从达梦数据库到Kafka队列的同步复制。关键要点包括&#xff1a; 1.DMHS系统概述&#xff1a; 达梦公司推出的异构环境高性能数据库实时同步系统&#xff0c;可应用于应急、容灾等多…

爬虫+动态代理助力 AI 训练数据采集

文章目录 引言新手之选&#xff1a;网页抓取API可靠之选&#xff1a;动态住宅代理总结 引言 近年来&#xff0c;AI 技术飞速发展&#xff0c;很多朋友都投身于 AI 模型的训练。然而&#xff0c;相较于模型的获取&#xff0c;高质量的数据往往更加难以收集。一方面&#xff0…

OpenEuler服务器警告邮件自动化发送:原理、配置与安全实践

OpenEuler服务器警告邮件自动化发送&#xff1a;原理、配置与安全实践 在服务器的运维管理过程中&#xff0c;及时感知系统异常状态至关重要。当OpenEuler系统运行时&#xff0c;将服务器的警告信息实时推送至邮箱&#xff0c;能帮助运维人员快速响应潜在问题&#xff0c;保障…

使用vite-plugin-html在 HTML 文件中动态注入数据,如元数据、环境变量、标题

vite-plugin-html 是一个用于 Vite 构建工具的插件&#xff0c;它可以帮助你在构建过程中动态注入一些 HTML 内容&#xff0c;比如标题、元数据、环境变量等。通过使用这个插件&#xff0c;你可以根据项目的配置和环境变量自动生成带有动态内容的 HTML 文件&#xff0c;适用于 …

学习笔记087——Java接口和抽象类的区别和使用

文章目录 1、主要区别2、使用场景2.1 使用接口的情况&#xff1a;2.1 使用抽象类的情况&#xff1a; 3、Java 8及以后的接口增强4、设计建议 1、主要区别 特性接口(Interface)抽象类(Abstract Class)定义方式使用interface关键字使用abstract class关键字方法实现Java 8前不能…

Squid 代理服务器实战:解决动态 IP 访问第三方接口的生产级方案

前言&#xff1a;动态IP场景下的业务痛点与解决方案 在企业开发场景中&#xff0c;经常会遇到这样的需求&#xff1a;第三方服务&#xff08;如API接口、云平台服务&#xff09;要求将访问源IP加入白名单以保障安全。然而&#xff0c;企业办公网络通常采用动态IP分配&#xff0…

React中子传父组件通信操作指南

文章目录 为什么需要子传父通信&#xff1f;方法一&#xff1a;回调函数&#xff08;最常用&#xff09;基础示例实际场景&#xff1a;待办事项列表 方法二&#xff1a;使用useRef传递引用方法三&#xff1a;Context API&#xff08;跨层级通信&#xff09;方法四&#xff1a;自…

【android bluetooth 框架分析 04】【bt-framework 层详解 5】【AbstractionLayer介绍】

1. AbstractionLayer 介绍 我们在阅读 native 和 java 层 蓝牙服务代码时&#xff0c;会发现很多 AbstractionLayer.xxxxx 的字段。 这些字段 虽然很容易理解是干什么的。 但是 大家有没有考虑过&#xff0c; 为啥要专门定义一个类来存放他们。 这样设计的意义是什么&#xff…

AI大模型从0到1记录学习 大模型技术之机器学习 day27-day60

机器学习概述 机器学习&#xff08;Machine Learning, ML&#xff09;主要研究计算机系统对于特定任务的性能&#xff0c;逐步进行改善的算法和统计模型。通过输入海量训练数据对模型进行训练&#xff0c;使模型掌握数据所蕴含的潜在规律&#xff0c;进而对新输入的数据进行准确…

c/c++ 汇编码中的.cfi 指令有什么用途?

author: hjjdebug date: 2025年 06月 12日 星期四 14:24:40 CST descrip: c/c 汇编码中的.cfi 指令有什么用途? 文章目录 1. 几个简写词.2. 看一个简单的测试代码:3. 生成汇编代码:4. 分析.cfi 指令5. 小结: 1. 几个简写词. cfi(call frame info) 调用帧信息, 名词. 描述的是…

ArcGIS Pro 3.4 二次开发 - 任务

环境:ArcGIS Pro SDK 3.4 + .NET 8 文章目录 任务1 任务1.1 检索项目中的所有任务项1.2 打开任务文件 - .esriTasks 文件1.3 打开项目任务项1.4 关闭任务项1.5 导出任务项1.6 获取任务信息 - 从 TaskProjectItem1.7 获取任务信息 - 从 .esriTasks 文件1.8 在任务文件中打开特定…

vscode如何修改终端的默认配置

问题困扰&#xff1a; 每次打开都是 powershell, 因为每次要是用 git bash, 所以每次手动切换很麻烦。 要将默认终端设置为 Git Bash&#xff0c;可以通过以下步骤完成。以下是详细的操作方法&#xff1a; 步骤 1&#xff1a;打开终端设置 在 Visual Studio Code 的菜单栏中…

kafka快速入门与知识汇总

​ kafka快速入门与知识汇总 一、前言 kafka是一款消息中间件&#xff0c;可以用于传输消息和日志收集、监控项目状况。与其类似的技术栈有rocketmq、rabbitmq等&#xff0c;但这些技术栈大多应用在一些简单的消息传输平台&#xff0c;而kafka则因其对大量数据的高性能处理在…

设计模式——观察者设计模式(行为型)

摘要 本文详细介绍了观察者设计模式&#xff0c;包括其定义、结构、实现方式、适用场景以及实战示例。通过代码示例展示了如何在Spring框架下实现观察者模式&#xff0c;以及如何通过该模式实现状态变化通知。同时&#xff0c;对比了观察者模式与消息中间件在设计理念、耦合程…

uniapp 页面栈一定深度后,回首页导航到新页面的解决方案

uniapp 页面栈一定深度后&#xff0c;回首页导航到新页面的解决方案 uniapp 页面导航解决方案 在 uniapp 中&#xff0c;要实现先弹出页面栈回到首页&#xff0c;然后再跳转到指定页面。 /*** description 后台选择链接专用跳转*/ interface Link {path: string;name?: stri…

数据结构 散列表 学习 2025年6月12日15:30:48

数据结构 散列表 哈希表(Hash Table): 通过哈希函数将键&#xff08;key&#xff09;映射到存储位置&#xff0c;从而实现快速的插入、删除和查找操作。 哈希表是现代编程中最重要的数据结构之一&#xff0c;几乎所有编程语言都提供了内置实现。 计数 #include <stdio.h&g…

数据结构之LinkedList

系列文章目录 数据结构之ArrayList-CSDN博客 目录 系列文章目录 前言 一、模拟实现链表 1. 遍历链表 2. 插入节点 3. 删除节点 4. 清空链表 二、链表的常见操作 1. 反转链表 2. 返回链表的中间节点 3. 链表倒数第 k 个节点 4. 合并两个有序链表 5. 分割链表 6. 判…

DC3靶机渗透

1. 靶机介绍 主要的内容有 sql 注入漏洞、joomla 框架漏洞、ssh 攻击、shell 反弹、提权 信息收集(ip、端口、目录、指纹信息)--->利用漏洞--->反弹---->提权 2. 信息收集 2.1. 扫描存活 ip 192.168.220.134 2.2. 端口扫描 nmap -T4 -A -p- 192.168.220.134 …

C# 线程交互

一、为什么要进行线程交互 在C#中&#xff0c;线程交互通常涉及到多个线程之间的数据共享和同步。‌. 一、全局变量 在C#中&#xff0c;全局变量是指在程序的任何地方都可以访问的变量。通常&#xff0c;全局变量是在类的外部定义的&#xff0c;或者在所有方法之外定义的。全…