Overview #
The following tutorial is based on the previous post “Azure DevOps Pipeline: Build Container Image and push the Image to Azure Container Registry”.
AKS Prerequisites #
Create AKS Cluster #
I’m using the following AKS cluster, for more details, check out my post about AKS Kubernetes clusters.
# Create Kubernetes cluster
az aks create \
--resource-group container-playground \
--name aks-cluster-1 \
--location northeurope \
--kubernetes-version 1.30.0 \
--node-count 1 \
--node-vm-size Standard_DS2_v2 \
--node-osdisk-size 30
Configure Kubectl #
# Configure Kubectl to use the AKS cluster
az aks get-credentials \
--resource-group container-playground \
--name aks-cluster-1
Create a Namespace #
Create a namespace for the deployment:
# Create namespace
kubectl create ns dev
Azure DevOps #
Adopt the Repository #
Overview #
The repository file and folderstructure should look like this:
├── azure-pipelines.yml
├── Dockerfile
├── HTML
│ └── index.html
├── kube-manifests
│ └── 01-deployment-and-loadbalancer-service.yml
└── README.md
GitHub Repository: https://github.com/jueklu/azure-devops-aks
Deployment and LoadBalancer Manifest #
Create a manifest for the deployment and loadbalancer service:
kube-manifests/01-deployment-and-loadbalancer-service.yml
apiVersion: apps/v1
kind: Deployment
metadata:
name: container-deployment
labels:
app: apache-container
spec:
replicas: 1
selector:
matchLabels:
app: apache-container
template:
metadata:
labels:
app: apache-container
spec:
containers:
- name: apache-container
image: jkwregistry.azurecr.io/azuredevopscontainerexample
ports:
- containerPort: 80
---
apiVersion: v1
kind: Service
metadata:
name: container-loadbalancer
labels:
app: apache-container
spec:
type: LoadBalancer
selector:
app: apache-container
ports:
- port: 80
targetPort: 80
Adopt Deployment Pipeline #
Add a task that publishes the build artifacts:
azure-pipelines.yml
# Docker
# Build and push an image to Azure Container Registry
# https://docs.microsoft.com/azure/devops/pipelines/languages/docker
trigger:
- main
resources:
- repo: self
variables:
# Container registry service connection established during pipeline creation
dockerRegistryServiceConnection: '875a58dd-registry-service-connection'
imageRepository: 'azuredevopscontainerexample'
containerRegistry: 'jkwregistry.azurecr.io'
dockerfilePath: '$(Build.SourcesDirectory)/Dockerfile'
tag: '$(Build.BuildId)'
# Agent VM image name
vmImageName: 'ubuntu-latest'
stages:
- stage: Build
displayName: Build and push stage
jobs:
- job: Build
displayName: Build
pool:
vmImage: $(vmImageName)
steps:
- task: Docker@2
displayName: Build and push an image to container registry
inputs:
command: buildAndPush
repository: $(imageRepository)
dockerfile: $(dockerfilePath)
containerRegistry: $(dockerRegistryServiceConnection)
tags: |
$(tag)
# Add Publish Build Artifacts Task
- task: PublishBuildArtifacts@1
inputs:
PathtoPublish: '$(System.DefaultWorkingDirectory)/kube-manifests'
ArtifactName: 'kube-manifests'
publishLocation: 'Container'
AKS Service Connection #
Create a service connection for the AKS cluster:
-
Go to: Project Settings > (Pipelines) “Service connections”
-
Click “New service connection”
-
Select “Kubernetes” (login again)
data:image/s3,"s3://crabby-images/0da57/0da57c392f7f3ced9c082ba567015d8ca824260f" alt=""
-
Select “Authentication method” x “Azure Subscription”
-
Select your Azure Subscription
-
Select your AKS cluster
-
Select the previously created “dev” namespace
-
Define a service connection name, like
aks-cluster-1 dev
-
Click “Save”
data:image/s3,"s3://crabby-images/b04b2/b04b28401d7d7e2cbc667fcab00eb406fa671219" alt=""
Organization Settings #
Open the dashboard of your organization:
https://dev.azure.com/jueklu/
- Go to: “Organization settings”
data:image/s3,"s3://crabby-images/f61e1/f61e1827d945f24bd7153d6064c739ced01abdcb" alt=""
-
Select (Pipelines) “Settings”
-
Disable “Disable creation of classic release pipelines”
data:image/s3,"s3://crabby-images/8e9ef/8e9ef1f55d62a0412ef8e6ea21fc5dbf571b3e76" alt=""
Release Pipeline #
Create Release Pipeline #
-
Go to: “Pipelines” > “Releases”
-
Click “New pipeline”
data:image/s3,"s3://crabby-images/844e9/844e9a19f2e43a61a912ed41f64c6cc74b46f2e6" alt=""
- Select template type “Empty job”
data:image/s3,"s3://crabby-images/1e744/1e744a2f2c677d41e67bbe57e74c3cba1eac6304" alt=""
-
Define a stage name like
Dev
-
Click “Add an artifact”
data:image/s3,"s3://crabby-images/3c90e/3c90ed0fcb2b592b083f96cc8d9b0b3aace48928" alt=""
-
Select source type “Build”
-
Select your project and repository
-
Click “Add”
data:image/s3,"s3://crabby-images/46bfa/46bfad224c4ffa14a219dc1876e33d6775b091c5" alt=""
- Add a “Continuous deployment trigger”
data:image/s3,"s3://crabby-images/49323/493233525498b721c5dfc7cf2d6ecf680fb3e502" alt=""
- Set continuois deployment trigger to “Enabled”
data:image/s3,"s3://crabby-images/84ab0/84ab00cf0825f1fc9cdaabd286f4027c712a5e1c" alt=""
-
Select the “Tasks” tab
-
Click “Agent job”
-
Set “Agent pool” to “Azure Pipelines”
-
Set “Agent Specification” to “ubuntu-latest”
data:image/s3,"s3://crabby-images/84e29/84e291935b1d3a25c1d390527c27f0bf8269c049" alt=""
-
Click “+” to add a task
-
Select “Deploy to Kubernetes” > “Add”
data:image/s3,"s3://crabby-images/8db45/8db4522ff36c13e88212d21edb4f347bef4e5798" alt=""
-
Define a display name like
image pull secret
-
Set action to “create secret”
-
Select service connection type “Kubernetes Service Connection”
-
Select your Kubernetes service connection
aks-cluster-1 dev
-
Set “Type of secret” to “dockerRegistry”
-
Define a secret name like
dev-secret
-
Select your “Docker registry service connection”
jkwregistry
data:image/s3,"s3://crabby-images/94598/94598b00356aab0d5bc3ea49449cad9632fa1f27" alt=""
-
Click “+” to add a task
-
Select “Deploy to Kubernetes” > “Add”
data:image/s3,"s3://crabby-images/42213/42213e5c42570f238f6d35a3ceb49c95d0e7a00d" alt=""
-
Define a display name like
deploy dev
-
Set action to “deploy”
-
Select service connection type “Kubernetes Service Connection”
-
Select your Kubernetes service connection
aks-cluster-1 dev
-
Define namespace
dev
-
Set “Manifests” to
$(System.DefaultWorkingDirectory)/_azure-devops-container-example/kube-manifests/01-deployment-and-loadbalancer-service.yml
-
Set “Containers” to
jkwregistry.azurecr.io/azuredevopscontainerexample:$(Build.BuildId)
-
Set “ImagePullSecrets” to
dev-secret
-
Click “Save”
data:image/s3,"s3://crabby-images/7c2fc/7c2fcda5e60e0bc7ebe6a272721bc4b7c889a8bf" alt=""
- Click “OK”
data:image/s3,"s3://crabby-images/b8590/b8590641ee434979377371cd83d58918193af771" alt=""
Change & Push Repository #
Adopt the index.html
file and push it into the repository.
Verify the Release Pipeline #
data:image/s3,"s3://crabby-images/406a5/406a5042770cbbc74943949363f6322dfc033e70" alt=""
data:image/s3,"s3://crabby-images/fc2d0/fc2d03081141c4508b86cba96afd23ffb7192dfb" alt=""
Verify the Deployment with Kubectl #
# List pods in "dev" namespace
get pods -n dev
# Shell output:
NAME READY STATUS RESTARTS AGE
container-deployment-c85569b74-g4lqc 1/1 Running 0 31s
# List services in "dev" namespace
kubectl get svc -n dev
# Shell output:
NAME TYPE CLUSTER-IP EXTERNAL-IP PORT(S) AGE
container-loadbalancer LoadBalancer 10.0.114.205 172.205.94.30 80:31470/TCP 2m29s
Test the deployment with curl:
# Curl the loadbalancer external IP
curl 172.205.94.30:80
# Shell output:
<!DOCTYPE html>
<html>
<head>
<title>jklug.work</title>
</head>
<body>
<h1>Azure DevOps Example</h1>
<p>Build Container 2</p>
</body>
Delete Resources #
Delete AKS Cluster #
# Delete AKS cluster
az aks delete \
--resource-group container-playground \
--name aks-cluster-1 \
--yes \
--no-wait
Delete Azure Container Registry #
# Delete the Azure container registry
az acr delete --name jkwregistry --resource-group container-playground --yes
Delete Resource Group #
# Delete the Azure resource group
az group delete --name container-playground --yes --no-wait
Delete Azure DevOps Project #
-
Open the project dashboard: https://dev.azure.com/jueklu/Project1
-
Select “Project settings”
-
Select (General) “Overview” and scroll down to “Delete project”
-
Click “Delete”
Delete Azure DevOps Organization #
-
Open the organization dashboard: https://dev.azure.com/jueklu
-
Select “Organization settings”
-
Select (General) “Overview” and scroll down to “Delete organization”
-
Click “Delete”