使用Azure Bicep,開發一個從GitHub持續佈署到Azure Container Apps的Web站台
程式碼簽入GitHub之後,啟動GitHub Action流程,編譯並部署程式到Azure Container Apps,是開發系統時常見的功能需求。本篇範例協助開發人員使用GitHub與Azure Bicep,逐步完成必要的設計和實作。
- 範例下載:SleepZone.zip
操作步驟
1.註冊並登入Azure Portal。於首頁左上角的選單裡,點擊資源群組後,進入資源群組頁面。
data:image/s3,"s3://crabby-images/fc0c4/fc0c46bc7aaac57b9177a1a1777388cbaeccb542" alt="01.建立ResourceGroup01.png"
2.於資源群組頁面,點擊建立按鈕,依照頁面提示建立一個ResourceGroup,並命名為sleep-zone-group。
data:image/s3,"s3://crabby-images/76105/7610546520677893442928575e9956c7f35b4257" alt="01.建立ResourceGroup02.png"
data:image/s3,"s3://crabby-images/4014e/4014e17ae94206d0bd704e26bf108891cda0f5fd" alt="01.建立ResourceGroup03.png"
3.於ResourceGroup頁面的屬性頁籤,取得「資源識別碼」。
data:image/s3,"s3://crabby-images/a92c7/a92c72fa1765a64952d4a60e1d66afc54f3f124d" alt="01.建立ResourceGroup04.png"
data:image/s3,"s3://crabby-images/befb2/befb2b60f9c0e044f5d1010a3cc16a288a9e815a" alt="01.建立ResourceGroup05.png"
4.回到Azure Portal。於右上角的選單裡,點擊Cloud Shell按鈕,依頁面提示開啟Cloud Shell視窗。
data:image/s3,"s3://crabby-images/eeb1e/eeb1eb38845432bf6a04de4a5f6c19ed139ab097" alt="02.建立Application01.png"
data:image/s3,"s3://crabby-images/92464/9246413eded2be522c2f998919c21dac2add7c4b" alt="02.建立Application02.png"
data:image/s3,"s3://crabby-images/ef2d3/ef2d381ab9100be6f086435774b08af18fab83b3" alt="02.建立Application03.png"
5.在Cloud Shell視窗,切換至Bash並執行下列指令,以取得部署使用的「服務主體憑證」。該指令會建立名為sleep-zone-app-contributor的應用程式註冊,並授權它為sleep-zone-group資源群組的參與者(Contributor)角色。
az ad sp create-for-rbac \
--name "sleep-zone-app-contributor" \
--role "Contributor" \
--scopes xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx \
--sdk-auth
- 服務主體名稱:--name "sleep-zone-app-contributor",可自訂,限制使用英文小寫與「-」。
- 服務主體授權角色:--role "Contributor"。
- 服務主體授權範圍:--scopes xxxxxxxxxxxxxxx。(xxxxx填入先前取得的資源識別碼)
data:image/s3,"s3://crabby-images/77860/778604a4f2840674321ac1b39666f5a94f7c2e41" alt="02.建立Application04.png"
data:image/s3,"s3://crabby-images/3f3f9/3f3f925acb9383a26b5602010b8f495f5f7bafc8" alt="02.建立Application05.png"
6.註冊並登入GitHub Dashboard。點擊首頁左上角的New按鈕,依照頁面提示建立一個Repository,並命名為SleepZone。
data:image/s3,"s3://crabby-images/970e6/970e6409a373b98840b48f4bb986bdf33452a0ed" alt="03.建立Repository01.png"
data:image/s3,"s3://crabby-images/b48cd/b48cd8e2eaf0289c363894bf6d774ddf7b625960" alt="03.建立Repository02.png"
7.於Repository頁面,點擊右上角的Setting按鈕進入Setting頁面,並選擇左側選單裡Secrets and variables的Action頁籤。於Action頁籤,點選New repository secret按鈕,依照頁面提示建立一個名為AZURE_CREDENTIALS的Actions Secret,Secret內容則是填入先前取得的「服務主體憑證」。
data:image/s3,"s3://crabby-images/82f9e/82f9ee9a162857ca9c51f35f846b6b83fcadc3aa" alt="03.建立Repository03.png"
data:image/s3,"s3://crabby-images/76a55/76a559369d678703c8d317535d88eca0c7a24590" alt="03.建立Repository04.png"
data:image/s3,"s3://crabby-images/9772d/9772d075aec2cf9d3fd069ed03165400b692cfc6" alt="03.建立Repository05.png"
8.Clone SleepZone Repository到本機Repository資料夾,於本機Repository資料夾裡建立src資料夾。開啟命令提示字元進入src資料夾,輸入下列指令,用以安裝MDP.WebApp範本、並且建立一個名為WebApplication1的Web站台。
// 建立Web站台
dotnet new install MDP.WebApp
dotnet new MDP.WebApp -n WebApplication1
data:image/s3,"s3://crabby-images/96281/9628160f4405e65f1e6cd56bf23b05c5151ddf87" alt="04.建立WebApplication01.png"
data:image/s3,"s3://crabby-images/83f76/83f76b33ea3bc344767f7ab824a1cac180f06d6d" alt="04.建立WebApplication02.png"
9.於本機Repository資料夾裡,進入src/WebApplication1資料夾,並加入Dockerfile。
FROM mcr.microsoft.com/dotnet/aspnet:6.0 AS base
WORKDIR /app
EXPOSE 80
EXPOSE 443
FROM mcr.microsoft.com/dotnet/sdk:7.0 AS build
WORKDIR /repo
COPY . .
RUN dotnet publish "src/WebApplication1/WebApplication1.csproj" -c Release -o /app/publish /p:UseAppHost=false
FROM base AS final
WORKDIR /app
COPY --from=build /app/publish .
ENTRYPOINT ["dotnet", "WebApplication1.dll"]
data:image/s3,"s3://crabby-images/e7300/e73005c96dbf098abf36cdd96962c929c7cad030" alt="04.建立WebApplication03.png"
10.於本機Repository資料夾裡,建立.github\workflows資料夾,並加入azure-build-deployment.yml、azure-build-deployment.bicep。
// azure-build-deployment.yml
name: azure-build-deployment
on:
push:
branches:
- main
env:
RESOURCE_GROUP_NAME: sleep-zone-group
CONTAINER_APPS_NAME: sleep-zone-app
DOCKER_FILE_PATH: ./src/WebApplication1/Dockerfile
jobs:
deploy:
runs-on: ubuntu-latest
steps:
- name: Login Azure CLI
uses: azure/login@v1
with:
creds: ${{ secrets.AZURE_CREDENTIALS }}
- name: Checkout GitHub Repository
uses: actions/checkout@v4
- name: Deploy AzureContainerRegistry by MDP
uses: MDPOps/azure-container-registry-deploy@v1
with:
resourceGroupName: ${{ env.RESOURCE_GROUP_NAME }}
dockerFilePath: ${{ env.DOCKER_FILE_PATH }}
dockerImageName: ${{ env.CONTAINER_APPS_NAME }}
id: azure-container-registry-deploy
- name: Deploy AzureResourceManager by MDP
uses: MDPOps/azure-resource-manager-deploy@v1
with:
resourceGroupName: ${{ env.RESOURCE_GROUP_NAME }}
parameters: >-
containerAppName=${{ env.CONTAINER_APPS_NAME }}
containerRegistryName=${{ steps.azure-container-registry-deploy.outputs.containerRegistryName }}
containerRegistryCredentials=${{ toJson(steps.azure-container-registry-deploy.outputs.containerRegistryCredentials) }}
id: azure-resource-manager-deploy
- Git分支名稱:main,要特別注意Repository裡的分支是 master or main。
- 資源群組名稱:RESOURCE_GROUP_NAME: sleep-zone-group,Azure資源群組名稱。
- 容器應用名稱:CONTAINER_APPS_NAME: sleep-zone-app,可自訂,限制使用英文小寫與「-」。
- Dockerfile路徑:DOCKER_FILE_PATH: ./src/WebApplication1/Dockerfile,路徑區分大小寫,相對於Repository資料夾。
// azure-build-deployment.bicep
// Inputs
@description('ContainerApp Name')
param containerAppName string
@description('ContainerRegistry Name')
param containerRegistryName string
@description('ContainerRegistry Credentials')
@secure()
param containerRegistryCredentials object
// LogAnalyticsWorkspace
module logAnalyticsWorkspace 'modules/logAnalyticsWorkspace.bicep' = {
name: 'logAnalyticsWorkspace'
params: {
}
}
// ContainerAppsEnvironment
module containerAppsEnvironment 'modules/containerAppsEnvironment.bicep' = {
name: 'containerAppsEnvironment'
params: {
logAnalyticsWorkspaceName: logAnalyticsWorkspace.outputs.name
}
}
// ContainerApp
module containerApp 'modules/containerApp.bicep' = {
name: 'containerApp'
params: {
containerAppName: containerAppName
containerAppsEnvironmentName: containerAppsEnvironment.outputs.name
containerRegistryName: containerRegistryName
containerRegistryCredentials: containerRegistryCredentials
}
}
data:image/s3,"s3://crabby-images/ba7df/ba7dfac4a5811de6ac32238ccae140760c97dca6" alt="04.建立WebApplication04.png"
11.簽入並推送本機Repository的變更,到遠端GitHub主機之後。回到GitHub Dashboard,進入SleepZone Repository頁面。可以於Actions頁籤裡,可以看到GitHub Action流程已經觸發執行並成功完成部署。(需時約五分鐘)
data:image/s3,"s3://crabby-images/70bb0/70bb0399136d61bd85b6c270f2f217e23eec9920" alt="05.執行結果01.png"
12.回到Azure Portal。於首頁左上角的選單裡,點擊容器應用程式後,進入容器應用程式頁面,可以看到剛剛建立的sleep-zone-app容器應用程式。
data:image/s3,"s3://crabby-images/1691c/1691c7f1efcbb54522b49cbb2a6ddfb8342673b4" alt="05.執行結果02.png"
data:image/s3,"s3://crabby-images/8900e/8900eb14a35cba57c00e89d9a3b097b4c73f7f92" alt="05.執行結果03.png"
13.於sleep-zone-app容器應用程式的概觀頁面,點擊「應用程式 URL」鏈結,在開啟的Browser視窗內,可以看到由WebApplication1程式碼所提供的Hello World。
data:image/s3,"s3://crabby-images/62be3/62be3ec6ddedd325c3937c1dcd24500df95275b7" alt="05.執行結果04.png"
data:image/s3,"s3://crabby-images/76a8d/76a8d7a1abb89b12d71106e5345cec80cf086939" alt="05.執行結果05.png"
14.完成上述步驟之後,每次推送程式碼到GitHub,都會啟動GitHub Action流程,編譯並部署程式到Azure Container Apps。
data:image/s3,"s3://crabby-images/24f63/24f63fff2b2cf00b0136f7cc774476b8560eef95" alt="05.執行結果06.png"
能以更簡潔的文字與程式碼,傳達出程式設計背後的精神。
真正做到「以形寫神」的境界。