In this article, we will see how to install and get a WordPress site running on Azure Kubernetes Services,
- WordPress (To host the Website)
- Nginx (For Load balancing )
- Cert Manager ( Create and Issue let’s encrypt Certificates)
- Kubeapps ( Bitnami Packages)
Let’s create a resource group az-pr-uaen-web-rg and create an Azure Kubernetes Cluster in the specified resource group
data:image/s3,"s3://crabby-images/9134e/9134e38d88e05c5c58ade079f559b619256af196" alt=""
Choose an appropriate SKU and leave the default autoscale
data:image/s3,"s3://crabby-images/6709d/6709d66231035ea83ba8d360ecaaf712d692432b" alt=""
Leave things default
data:image/s3,"s3://crabby-images/ec831/ec83134860ecc4f8b67249332cf842e970e51b70" alt=""
Choose a security group for Kubernetes Administration
data:image/s3,"s3://crabby-images/ef8da/ef8dacbcc2fd80cf51d8e8cd850f358b564809b3" alt=""
Leave kubenet networking
data:image/s3,"s3://crabby-images/db62b/db62bc86bafc2885dc18d37b386d563835251552" alt=""
not using calico or container registry in this scenario.
data:image/s3,"s3://crabby-images/b26b1/b26b1aa46c62e5e88766a569dc75e4d2965b7297" alt=""
leave the secret store CSI driver disabled by default
data:image/s3,"s3://crabby-images/79374/79374ea12b76b0eab88165b16fec22d7fdeb6d51" alt=""
Azure Kubernetes Cluster is created .
data:image/s3,"s3://crabby-images/9106a/9106a9ac4f59daf897e113d28ac33a23408a6c39" alt=""
data:image/s3,"s3://crabby-images/f04fc/f04fc4edb46b857b247dcd227392dbbbe596541c" alt=""
Let’s install Microsoft CLI and Chocolatey on the local machine.
Chocolatey Software | Installing Chocolatey
data:image/s3,"s3://crabby-images/4555c/4555c16f6581590a05a6b90a498893a5e3f0cabc" alt=""
Install Kubernetes Helm using Choco
choco install kubernetes-helm
data:image/s3,"s3://crabby-images/dc877/dc8771f294d274431db4a7618aedc525defbdbb6" alt=""
Install Bitnami repo using helm
helm repo add bitnami https://charts.bitnami.com/bitnami
data:image/s3,"s3://crabby-images/8dcdd/8dcdd2f25851bbacc8eb670d64940b18514cc72f" alt=""
Download kubectl.exe to run kubectl commands
curl -LO "https://dl.k8s.io/release/v1.24.0/bin/windows/amd64/kubectl.exe"
data:image/s3,"s3://crabby-images/178f2/178f200158880c2d7587120c55ae4bf848d56422" alt=""
Download and install kubelogin if you want to authenticate to AKS using Azure AD
data:image/s3,"s3://crabby-images/685e1/685e155dc9e345fcd72d1e6d9e4006c24d0746d3" alt=""
Login to aks cluster
az aks get-credentials --resource-group az-pr-uaen-web-rg --name az-aks-azure365pro --admin
data:image/s3,"s3://crabby-images/40aab/40aabcde0956b575e215072cf14e8a3325fd4fb5" alt=""
if you have multiple subscriptions
az login Get the subscription ID you are working on az account list -o table --all az account set --subscription e08f63cf-xxxx-44e3-ba41-xxxxxxx az aks get-credentials --resource-group az-pr-uaen-web-rg --name az-aks-azure365pro --admin kubectl get namespaces
To list all namespaces
data:image/s3,"s3://crabby-images/5cbb0/5cbb0c49291961387e86f4db86ed903a60267ff2" alt=""
Add Bitnami repo
create kubeapps namespace
Install bitnami kubeapps
helm repo add bitnami https://charts.bitnami.com/bitnami kubectl create namespace az-kubeapps helm install kubeapps bitnami/kubeapps -n az-kubeapps
data:image/s3,"s3://crabby-images/4ef40/4ef40e641aed66246643411699d5abed7cb39aa3" alt=""
Now kubeapps is installed – We will configure things to access them using an nginx load balancer.
data:image/s3,"s3://crabby-images/12b00/12b00a0b4ff85f50bfc7372ac6afb5a980b2a6f1" alt=""
kubectl create serviceaccount kubeapps-operator kubectl create clusterrolebinding kubeapps-operator --clusterrole=cluster-admin --serviceaccount=default:kubeapps-operator kubectl get secret $(kubectl get serviceaccount kubeapps-operator -o jsonpath='{.secrets[].name}') -o jsonpath='{.data.token}' -o go-template='{{.data.token | base64decode}}'
if you run on Windows 10
‘base64decode}}” is not recognized as an internal or external command, operable program, or batch file.
switch to Powershell
.\kubectl.exe get secret $(.\kubectl.exe get serviceaccount kubeapps-operator -o jsonpath='{.secrets[].name}') -o jsonpath='{.data.token}' -o go-template='{{.data.token | base64decode}}'
A secret token has to be generated like below to get inside kubeapps , Copy the token to a safe location
data:image/s3,"s3://crabby-images/19eda/19eda73c46ba123260bf26350dab3b81d410d832" alt=""
To update helm repo
helm repo update
Install Nginx controller / Please note we are utilizing the native Nginx controller here.
kubectl create namespace az-nginx helm install az-nginx bitnami/nginx-ingress-controller -n az-nginx
data:image/s3,"s3://crabby-images/8f94a/8f94a1d11cb6aad0974f8b44cf9350fd410e791b" alt=""
data:image/s3,"s3://crabby-images/0f48f/0f48fe46f4b938d026ba7c267236ebab53bcb551" alt=""
Now Nginx is available with an external IP
kubectl get svc -n az-nginx kubectl get svc --all-namespaces
data:image/s3,"s3://crabby-images/e6f85/e6f85016c1ffb91a8ca9e1ebee186f8027b436c6" alt=""
Expose kubeapps at port 80 just to access for now. later we can apply SSL
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: kubeapps
namespace: kubeapps
spec:
ingressClassName: nginx
rules:
– host: kubeapps.azure365pro.com
http:
paths:
– pathType: Prefix
backend:
service:
name: kubeapps
port:
number: 80
path: /
kubectl apply -f .\kubeapps-ingress.yaml –namespace az-kubeapps
data:image/s3,"s3://crabby-images/4b880/4b8804de67dbd619b61f9dad41eace35605f101b" alt=""
Deleting bad ingress rules – For Examples only
kubectl get all,nodes,ing -A -o wide
data:image/s3,"s3://crabby-images/09fe8/09fe8f00e10935ac73aea4ad9e7d0c32526033a3" alt=""
To list Helm installations
data:image/s3,"s3://crabby-images/84ebf/84ebfb9b70d2040542323c9a77a4d08f74fa52bf" alt=""
To uninstall Helm installations – For examples only
helm uninstall az-nginx
data:image/s3,"s3://crabby-images/3b8c2/3b8c2f68f460f9f7c157bc0aaa7627ceb9ea64d6" alt=""
data:image/s3,"s3://crabby-images/bc098/bc09808106f1d0c39957179da4dff1be0d76387e" alt=""
data:image/s3,"s3://crabby-images/86bfd/86bfd3fc98b9beb9f678a12b29f585c4c0d7b7d8" alt=""
data:image/s3,"s3://crabby-images/98269/98269425f4ce576e1e291b33cd869c7333c8d4e9" alt=""
Choosing to install with Cluster IP as we will use the nginx IP to publish and Expose using lets encrypt SSL .
data:image/s3,"s3://crabby-images/9e25c/9e25cbfb5a04461f399c17a47ae5718a698949b1" alt=""
data:image/s3,"s3://crabby-images/15f8b/15f8b8c440bcfc13e53b5cebe2c21cc5609db3fb" alt=""
ingress without SSL – YAML samples
data:image/s3,"s3://crabby-images/6742f/6742f5c6befc52d9e2166405d49d62018a263259" alt=""
Install cert-manager with InstallCRDs true
data:image/s3,"s3://crabby-images/2f223/2f2235c0237b3edb967bf346837f94c57ba99533" alt=""
issuer.yaml with staging lets encrypt . Once you confirm things are working as expected. you can get the live URL updated.
apiVersion: cert-manager.io/v1
kind: ClusterIssuer
metadata:
name: letsencrypt-staging
spec:
acme:
# Update with real email
email: info@azure365pro.com
server: https://acme-staging-v02.api.letsencrypt.org/directory
# To use on production , use the following line instead:
#server: https://acme-v02.api.letsencrypt.org/directory
privateKeySecretRef:
name: issuer-letsencrypt-staging
solvers:
– http01:
ingress:
class: nginx
Created issuer
Listing Secrets
kubectl apply -f issuer.yaml -n az-cert
kubectl get secrets -n az-cert
data:image/s3,"s3://crabby-images/026f9/026f97f03c178d5b705d2dca228fe8c07a4e5461" alt=""
cert.yaml for cluster issuer – secret and WordPress should be in the same namespace inorder to work seamlessly.
apiVersion: cert-manager.io/v1
kind: Certificate
metadata:
name: az-wp-1
spec:
secretName: virtualpetals-tls
dnsNames:
– www.virtualpetals.com
issuerRef:
name: letsencrypt-staging
kind: ClusterIssuer
kubectl apply -f cert.yaml -n az-wp
data:image/s3,"s3://crabby-images/3f611/3f611008f071d3c2569461fd3a799ddde6e8829f" alt=""
Applying TLS rules on nginx
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: az-wp
namespace: az-wp
spec:
ingressClassName: nginx
tls:
– hosts:
– www.virtualpetals.com
secretName: virtualpetals-tls
rules:
– host: www.virtualpetals.com
http:
paths:
– path: /
pathType: Prefix
backend:
service:
name: az-wp-1-wordpress
port:
number: 80
kubectl apply -f wp-ingress.yaml -n az-wp
data:image/s3,"s3://crabby-images/b9bbb/b9bbbd821bd70cd642d327af7b11fe79d9ffc99a" alt=""
To describe ingress using ing
data:image/s3,"s3://crabby-images/9409e/9409e6de04bec3592852156981768c44bbec1d8a" alt=""
To check current YAML Config
data:image/s3,"s3://crabby-images/8507d/8507df761bc007b1b4fb9f55c2b260ccf05c7e25" alt=""
As we are using a staging issuer, It shows staging. Once we feel SSL is getting assigned and staging getting issued.
We are good to convert to production
data:image/s3,"s3://crabby-images/3a382/3a382a56d1cbeb964b98c07d98481e4cc8490367" alt=""
Good to know – To list and delete files from a running container
kubectl exec az-wp-1-wordpress-6cb985c596-84j8j -n az-wp -- ls /bitnami/wordpress/wp-content/plugins/really-simple-ssl kubectl exec az-wp-1-wordpress-6cb985c596-84j8j -n az-wp -- sh -c 'rm -rf /bitnami/wordpress/wp-content/plugins/really-simple-ssl/*'
Getting the production URL live and disabling the staging URL
apiVersion: cert-manager.io/v1
kind: ClusterIssuer
metadata:
name: letsencrypt-production
spec:
acme:
# Update with real email
email: info@azure365pro.com
#server: https://acme-staging-v02.api.letsencrypt.org/directory
# To use on production , use the following line instead:
server: https://acme-v02.api.letsencrypt.org/directory
privateKeySecretRef:
name: issuer-letsencrypt-production
solvers:
– http01:
ingress:
class: nginx
kubectl apply -f issuer-prod.yaml -n az-cert
data:image/s3,"s3://crabby-images/40174/40174a864ef293f646067e14b11f8a3758e77abb" alt=""
apiVersion: cert-manager.io/v1
kind: Certificate
metadata:
name: az-wp-0
namespace: az-wp
spec:
secretName: virtualpetals-tls-pr
dnsNames:
– www.virtualpetals.com
– virtualpetals.com
issuerRef:
name: letsencrypt-production
kind: ClusterIssuer
data:image/s3,"s3://crabby-images/48caa/48caaec5e54c70e610d8e67113a8545a6f6c6e96" alt=""
To increase body size in ingress for bulk uploads for example.
apiVersion: networking.k8s.io/v1
kind: Ingress
metadata:
name: az-wp
namespace: az-wp
annotations:
nginx.ingress.kubernetes.io/proxy-body-size: 400m
spec:
ingressClassName: nginx
tls:
– hosts:
– www.virtualpetals.com
– virtualpetals.com
secretName: virtualpetals-tls-pr
rules:
– host: “www.virtualpetals.com”
http:
paths:
– path: /
pathType: Prefix
backend:
service:
name: az-wp-1-wordpress
port:
number: 80
– host: “virtualpetals.com”
http:
paths:
– path: /
pathType: Prefix
backend:
service:
name: az-wp-1-wordpress
port:
number: 80
data:image/s3,"s3://crabby-images/fcb0a/fcb0ad5fd9e5c1571d854f50da111f13478b372d" alt=""
Now website is up using Azure Kubernetes Services.
data:image/s3,"s3://crabby-images/6c95f/6c95fe3d3c31d49fdf32d06bda2e270041721975" alt=""
apiVersion: cert-manager.io/v1
kind: Certificate
metadata:
name: az-wp-1
spec:
secretName: kubeapps.azure365pro.com
dnsNames:
– kubeapps.azure365pro.com
issuerRef:
name: letsencrypt-production
kind: ClusterIssuer
kubectl apply -f cert-prod-kubeapps.yaml -n az-kubeapps
data:image/s3,"s3://crabby-images/0ad4b/0ad4b121d6f99340fdc8f4ca9c14f10336d2378a" alt=""
data:image/s3,"s3://crabby-images/b782b/b782b06c7e21f92dc3e3f1885bae2da185ba8492" alt=""
Now you can see Kubeapps is live with SSL / WordPress is live with SSL / Nginx is configured with Cert Manager.