一 背景
在现代软件开发中,容器化和持续集成/持续部署(CI/CD)已成为关键的实践。Golang作为一种高效、可靠的编程语言,被广泛应用于开发云原生应用和微服务。Azure DevOps是一套功能强大的DevOps工具,提供了CI/CD流水线的构建、测试、部署和监控等功能。
本实战旨在演示如何使用Azure DevOps将Golang项目容器化,并通过CI/CD流水线自动化部署到Azure App Service服务中。Azure App Service是一种托管的云服务,可以轻松地托管和扩展Web应用程序,支持多种编程语言和框架。
通过将Golang项目容器化并使用Azure DevOps进行CI/CD,可以实现快速、可靠的部署和自动化测试,提高开发团队的效率和应用程序的质量。同时,利用Azure App Service的托管能力,可以简化部署和运维的工作,让开发者能够更专注于业务逻辑的开发。
先决条件
- 具有活动订阅的 Azure 帐户。 免费创建帐户。
- 一个 GitHub 帐户。 创建一个免费的 GitHub 帐户(如果没有)
- Azure DevOps 组织。 如果你没有组织,请创建一个组织。
- Azure 容器注册表。 创建 Azure 容器注册表(如果还没有)。
二 架构流程
data:image/s3,"s3://crabby-images/d9253/d925349816f9275d05772796e10108e9b32a352b" alt=""
三 实战
3.1 创建项目并托管源码
data:image/s3,"s3://crabby-images/1991b/1991bc99bb0f203529a3793152113814c5531462" alt=""
在Repos中生存链接信息。
密钥信息
shell
cd project
git init
git remote add origin https://183xxx@dev.azure.com/18329xxx/pipelines-go-docker/_git/pipelines-go-docker
git push -u origin --all
data:image/s3,"s3://crabby-images/de727/de72735528242418b81644c1416dba0135116b27" alt=""
3.2 创建Pipeline
3.2.1 Azure Container Registry准备
data:image/s3,"s3://crabby-images/08be2/08be2d453406ffc9c8aa46792eba05a83ef4f4bd" alt=""
data:image/s3,"s3://crabby-images/4a83c/4a83c1af571fb98618f664f0aaee1802c9fb8fbc" alt=""
记录后续用于访问azure container registry信息
shell
registry name: xuelcr
login server: xuelcxxx.io
username: xx
密码:m0lQgjxxxxxxxx+ACRDX89zc
推送镜像
shell
# 登陆registry
az login
az acr login --name xuelcr
# tag镜像
docker pull mcr.microsoft.com/mcr/hello-world
docker tag mcr.microsoft.com/mcr/hello-world xuelcr/samples/hello-world
# push
docker push xuelcr/samples/hello-world
3.2.2 App Service创建
创建容器类型的App Services
data:image/s3,"s3://crabby-images/dcaa5/dcaa51f3277adfbcfbd1d5be32497ce0f6609413" alt=""
data:image/s3,"s3://crabby-images/5afad/5afadbcf9ab50ac16044ea25fcfda3f21a38090d" alt=""
data:image/s3,"s3://crabby-images/1da51/1da51df9294e16c932b46a9ea2980c57e6924027" alt=""
在应用部署配置中,选择了容器类型,以及内置了azure registry 的环境变量
data:image/s3,"s3://crabby-images/b47c0/b47c06926f510d77bfc0f6d5d557a4f70d03af33" alt=""
3.2.3 Azure DevOps链接Azure Portal
- 链接azure 云,用于部署到app server
data:image/s3,"s3://crabby-images/89a69/89a69bb338d00dbadc721f1447f23f21a005fa9e" alt=""
链接Azure Cloud 控制器,build出来的镜像上传至Azure Container registry中,使用Azure Pipeline中的task进行镜像仓库登陆。
data:image/s3,"s3://crabby-images/efb8e/efb8e03d0b5ffc4b11f8ee98846b895afaf50a4c" alt=""
data:image/s3,"s3://crabby-images/ffe40/ffe40b4472af96d8fcc73134aa839e3d125a1237" alt=""
完成链接
data:image/s3,"s3://crabby-images/0f5da/0f5dad37f3c5daf01f685d65622693912a7896af" alt=""
data:image/s3,"s3://crabby-images/e8443/e84430cc4f410b55cf9221d1e98cc3d8ec90b80f" alt=""
- 连接 Azure 容器注册表
learn.microsoft.com/zh-cn/azure...
在项目的组织配置中添加服务链接
data:image/s3,"s3://crabby-images/27dca/27dcaf22963c6015b296e16daf7923b8d89ed2b0" alt=""
docker registry
data:image/s3,"s3://crabby-images/53692/53692e6a77f2c0df8f98eb68a10949ab4232c1ff" alt=""
配置Azure Container Registry信息
data:image/s3,"s3://crabby-images/586f6/586f6d6b649a73b568744273060adc7e8838f601" alt=""
创建服务链接
data:image/s3,"s3://crabby-images/3b756/3b756a01af99ce9291487c0ed4d79299c0784e2d" alt=""
3.2.4 创建Docker build Agent
由于在Agent中需要执行Docker build,需要一个self-host 类型的Agent来安装Docker,作为pipeline的执行器。 若要将托管服务标识与 Azure Pipelines 配合使用以将 Docker 映像发布到 Azure 容器注册表,我们必须在 Azure VM 上设置自己的自托管代理。
- 创建 VM
- 导航到 Azure 门户。
- 在左侧导航面板中选择"创建资源",然后选择"虚拟机"->"创建"。
- 选择你的订阅,然后选择用来创建容器注册表的资源组。
- 为虚拟机命名,并选择一个映像。
- 输入你的用户名和密码,然后选择"查看 + 创建"。
- 查看设置,查看完后选择"创建"。
- 完成部署后,选择"转到资源"。
data:image/s3,"s3://crabby-images/13783/13783fc332c1fc3dbdd09a1d8719e1fe00c8a7af" alt=""
- 创建代理池
- 在 Azure DevOps 项目中,选择齿轮图标
导航到"项目设置"。
- 选择"代理池",然后选择"添加池"。
- 选择"新建",然后从"池类型"下拉菜单中选择"自托管"。
- 为池指定名称,然后选中"授予对所有管道的访问权限"复选框。
data:image/s3,"s3://crabby-images/d34ff/d34ff7334eb6d0af088e8961e8aa0ee8ee2933a2" alt=""
- 完成操作后,选择"创建"。
- 现在选择刚刚创建的池,然后选择"新建代理"。
- 我们将使用此窗口中的说明在前面创建的 VM 中设置代理。 选择"复制"按钮将下载链接复制到剪贴板。
data:image/s3,"s3://crabby-images/091d9/091d94e5a5996ff655ff338f47ea1126fed98771" alt=""
shell
$ mkdir myagent && cd myagent
$ wget https://vstsagentpackage.azureedge.net/agent/3.230.0/vsts-agent-linux-x64-3.230.0.tar.gz
$ tar zxvf vsts-agent-linux-x64-3.230.0.tar.gz
$ ./config.sh
# 配置可以参考:https://learn.microsoft.com/en-us/azure/devops/pipelines/agents/linux-agent?view=azure-devops
# Azure Pipelines: https://dev.azure.com/{your-organization}
# server url:https://dev.azure.com/xxxxxx
# 使用PTA进行认证:6whxpaxxxxxxxxxxx
# Agent pool:docker
data:image/s3,"s3://crabby-images/66ed9/66ed95b9fa65f6a16c7d63dc32d0398a2120a41d" alt=""
至此自托管docker agent已准备就绪
- 设置托管标识
- 在 Azure 门户中,导航到之前创建的 VM。
- 在左侧导航面板中选择"标识",然后启用"系统分配的标识"。
- 完成后选择"保存",然后确认选择。
data:image/s3,"s3://crabby-images/4b899/4b89900fc16ede2f98a71d7b6292d361bb9c0b29" alt=""
- 选择"Azure 角色分配",然后选择"添加角色分配"。
- 从"范围"下拉菜单中选择"资源组"。
- 选择你的订阅和资源组,然后选择"AcrPush"角色。
- 重复步骤 5 和 6 以添加"AcrPull"角色。
data:image/s3,"s3://crabby-images/21f1b/21f1ba69cfbedf71e33eb6011c5ba386b6a529f0" alt=""
在Agent上安装docker
shell
curl -fsSL get.docker.com -o get-docker.sh
sudo sh get-docker.sh --mirror Aliyun
sudo groupadd docker
sudo usermod -aG docker
sudo chmod 666 /var/run/docker.sock
sudo systemctl restart docker
3.2.5 创建配置Pipeline
data:image/s3,"s3://crabby-images/8db3b/8db3b495fca719336183db884e8d887e448528d0" alt=""
shell
trigger:
- master
variables:
dockerRegistryServiceConnection: 'azure-container-registry'
imageRepository: 'xuelcr'
dockerfilePath: '**/Dockerfile'
tag: '$(Build.BuildId)'
stages:
- stage: Build
displayName: Build and publish stage
jobs:
- job: Build
displayName: Build job
pool:
name: 'docker'
steps:
- task: Docker@2
displayName: Build and publish image to Azure Container Registry
inputs:
command: buildAndPush
containerRegistry: $(dockerRegistryServiceConnection)
repository: $(imageRepository)
dockerfile: $(dockerfilePath)
tags: |
$(tag)
3.3 运行测试构建
提交代码后自动运行build pipeline,构建镜像并推送到镜像仓库
data:image/s3,"s3://crabby-images/5776b/5776bb1553f201736e5a7b2e84be4c9f2b6fc2b2" alt=""
完成job后,在容器注册表中查看镜像
遇到异常,排查:[learn.microsoft.com/en-us/answe...](https://link.juejin.cn?target=https%3A%2F%2Flearn.microsoft.com%2Fen-us%2Fanswers%2Fquestions%2F1434566%2Funable-to-load-information-to-the-grid-reason-((0) "https://learn.microsoft.com/en-us/answers/questions/1434566/unable-to-load-information-to-the-grid-reason-((0)")
data:image/s3,"s3://crabby-images/af23d/af23dec60e052bf223abac06deaf706099b84d28" alt=""
添加当前用户到存储库内
data:image/s3,"s3://crabby-images/b6611/b661130df751d1872878a707bff5d3aa4cf706fd" alt=""
添加完成后,可以正常看到存储库。
data:image/s3,"s3://crabby-images/de90a/de90aff1ccc47363f6eaf59c081ab2b1a9150d07" alt=""
3.4 创建部署配置
shell
trigger:
- master
variables:
dockerRegistryServiceConnection: 'azure-container-registry'
imageRepository: 'xuelcr'
dockerfilePath: '**/Dockerfile'
tag: '$(Build.BuildId)'
stages:
- stage: Build
displayName: Build and publish stage
jobs:
- job: Build
displayName: Build job
pool:
name: 'docker'
steps:
- task: Docker@2
displayName: Build and publish image to Azure Container Registry
inputs:
command: buildAndPush
containerRegistry: $(dockerRegistryServiceConnection)
repository: $(imageRepository)
dockerfile: $(dockerfilePath)
tags: |
$(tag)
- task: AzureRmWebAppDeployment@4
inputs:
ConnectionType: 'AzureRM'
azureSubscription: 'azure-global'
appType: 'webAppContainer'
WebAppName: 'golang-api-container'
DockerNamespace: 'xuelcr.azurecr.io'
DockerRepository: $(imageRepository)
DockerImageTag: $(tag)
data:image/s3,"s3://crabby-images/9fa68/9fa68c2100b4f32eadb170105bd12006943ce648" alt=""
更新应用配置的容器配置环境变量
data:image/s3,"s3://crabby-images/21f09/21f090a91ee60429012f2836b2f2fe6027b8babf" alt=""
可以在部署中心中,查看容器内部的日志,一边排查异常。
data:image/s3,"s3://crabby-images/9c00a/9c00affa8270a1717a0d5a13793161fd2c6dc39e" alt=""
通过浏览可以查看应用
data:image/s3,"s3://crabby-images/fc818/fc818cf68232ae3462de13673005334131fc18e3" alt=""
data:image/s3,"s3://crabby-images/2c0f2/2c0f25822ee715e62bf68f346cd677a86ec1913f" alt=""
修改代码进行提交
data:image/s3,"s3://crabby-images/63a65/63a6542184313a2c899335d8fcd3033b06542660" alt=""
查看最总部署完成。
data:image/s3,"s3://crabby-images/d9a5f/d9a5f1afcafadad9317c15ea655a7b0dfe238d4f" alt=""
四 注意实现
- 需要在Azure Devops中链接Azure Container registry和Azure cloud
- 需要配置self-host,并在内部安装运行docker,以便在构建的时候可以进行引用docker.sock
- 存储库目前没有权限,需要添加当前账号到IAM中
- 确保正确配置Azure DevOps环境:在Azure DevOps中创建项目,并设置好与Azure App Service的连接。确保在Azure DevOps中配置了正确的代理、凭据和权限,以便能够与Azure资源进行交互。
- 编写Dockerfile:创建一个Dockerfile来定义Golang项目的容器化配置。确保Dockerfile中包含了正确的基础镜像、依赖项安装、构建和运行命令。
- 配置CI/CD流水线:在Azure DevOps中创建CI/CD流水线,定义构建、测试和部署的步骤。确保流水线中包含了构建Golang项目、构建Docker镜像、推送镜像到容器注册表、部署到Azure App Service等必要的步骤。
- 设置触发器和触发条件:为CI/CD流水线设置触发器,例如当代码提交到特定分支时触发自动构建和部署。另外,可以设置触发条件,例如只有在通过了所有测试后才执行部署步骤。
- 配置环境变量和机密信息:在Azure DevOps中配置环境变量和机密信息,以便在流水线中使用。例如,可以配置容器注册表的凭据、Azure App Service的连接字符串等敏感信息。
- 运行单元测试和集成测试:确保在CI/CD流水线中包含运行Golang项目的单元测试和集成测试的步骤。这可以确保在部署到生产环境之前,项目的质量得到验证。
总结
在基于Azure DevOps的Golang项目容器化CI/CD实战中,关键要点包括配置正确的环境、编写Dockerfile、创建CI/CD流水线、设置触发器和条件、配置环境变量和机密信息、运行测试、监控日志、确保安全性和权限管理。遵循这些要点,可以实现高效、可靠的容器化部署和自动化测试,提高开发效率,保证应用程序质量。后期可以将其部署到容器集群中中