- Demo environment via GitHub Actions
- Included compontents
- Demonstrations
This demo uses GitHub Actions to orchestrate deployment of infrastructure, application build and applicatio deployment.
- Terraform is used to deploy AKS cluster (including monitoring, policy engine and AAD integration), Application Gateway (WAF), PostgreSQL as a Service, Key Vault, Service Bus, Event Hub and other infrastructure components. In main.tf check what variables are needed and store them in your GitHub secrets to fit your environment
- AKS and its components (AAD Pod Identity, Application Gateway Ingress controller, Key Vault FlexVolume) are configured for ManagedIdentity (no Service Principal)
- Helm is used to deploy base AKS cluster components such as Ingress controller for App Gw, DAPR, KEDA, Grafana, Prometheus
- Azure Container Registry is used to build and package applications
- DNS record cloud.tomaskubica.net and *.cloud.tomaskubica.net is preconfigured with CNAME pointing to App Gw public IP - only thing that is done outside of Actions. You need to change this to fit your environment
Currently covered
- Azure Kubernetes Service
- Azure Container Registry
- AKS AAD integration with RBAC
- Azure Application Gateway Web Application Firewall as Ingress
- Azure Monitor for Containers including Prometheus scrapper
- Azure Database for PostgreSQL
- Azure Blob Storage
- Azure Service Bus
- Grafana with Azure Monitor datasource
- Prometheus
- Windows nodes
- DAPR
- State (with Cosmos DB)
- Pub/sub (With Service Bus)
- Trigger (Service Bus) with KEDA scaling
- Connector (Blob Storage)
- Secrets API (Key Vault)
- Service-to-service communication
- OpenTelemetry export to Application Insights
- KEDA
- AAD Pod Identity
- Key Vault FlexVolume
- Linkerd
- Linkerd telemetry to Application Insights
- Flagger
- NGINX ingress
- Monitoring -> diagnostic logs
- Azure Monitor for Containers (non-AKS cluster monitoring)
- Istio
- OpenTelemetry app demo
Planned
- DAPR API authentication
- Virtual Nodes (after vnet limitations are resolved)
- RUDR and/or Crossplane
- Azure Policy with Gatekeeper v3
- Osiris
- PSQL managed identity access
- Hybrid -> Azure Arc for Kubernetes
- GitOps
- Flagger
- with Linker
- with Istio
Todo application is deployed with traffic generator so we can see some telemetry information in monitoring tools.
Todo application in default namespace is not instrumented with application insights natively in code, but codeless attach is used to still get distributed tracing and telemetry to Application Insights - check appid-blabla workspace in Azure.
Application dapr-demo is instrumented for distributed tracing via DAPR OpenTelemetry export to Application Insights.
Application does not include Prometheus telemetry, but myappspa is built on nginx container and sidecar adapter has been used to expose NGINX telemetry as Prometheus telemetry. This is gathered to non/persistent Prometheus instance http://prometheus.cloud.tomaskubica.net. Eg. see telemetry with this PromQL:
nginx_http_requests_total { app="myapp-spa" }
Also Azure Monitor is configured to gather Prometheus telemetry from applications and store it in Azure. Open Logs and see metrics from myappspa using followign Kusto query:
InsightsMetrics
| where Namespace == "prometheus"
| where Name == "nginx_http_requests_total"
| extend podName = tostring(parse_json(Tags).pod_name)
| summarize metrics = percentile(Val,95) by bin(TimeGenerated, 5m), podName
| order by TimeGenerated asc
| render timechart
Azure Monitor allows for rich visualizations (Workbooks), deep realtime analytics and machine learning, Alerting capabilities including complex workflows with Logic Apps or analyzing data in Azure Sentinel (SIEM).
Visualizations can be also done in Grafana for both backends - Prometheus and Azure Monitor. Open Grafana at http://grafana.cloud.tomaskubica.net (login is tomas/Azure12345678) and see:
- AKS cluster dashboard (based on data from Azure Monitor)
- Prometheus telemetry for myappspa (based on data from Prometheus backend)
- Prometheus telemetry for myappspa (based on data from Azure Monitor)
- DAPR dashboards
- NGINX Controller dashboard
Logs from containers are in following table:
ContainerLog
Audit logs from Kubernetes API, Application Gateway, PostgreSQL and other sources.
AzureDiagnostics
| where Category == "kube-apiserver"
AzureDiagnostics
| where Category == "cluster-autoscaler"
AzureDiagnostics
| where Category == "kube-audit"
AzureDiagnostics
| where Category == "kube-controller-manager"
AzureDiagnostics
| where Category == "PostgreSQLLogs"
AzureDiagnostics
| where Category == "ApplicationGatewayAccessLog"
There is Kubernetes cluster deployed in VM K3s configured with Azure Monitor for Containers agent. Got to Azure Monitor -> Containers and select clusters from all environments (Azure and Non-Azure).
Various demo apps are sending tracing data to Azure Monitor (Application Insights) instances:
- appin-opentelemetry-demo - Python app with OpenTelemetry SDK directly reporting to Azure Monitor
- appin-linkerd-demo - Python app with OpenCensus SDK enhanced by traces from Linkerd service mesh all connected via Collector
- appin-demo - Java-based app with no instrumentation in code using codeless attach
- appin-dapr-demo - traces from DAPR
Access todo application at http://cloud.tomaskubica.net (Ingress via Application Gateway)
Check monitoring - Application Insights, codeless attach, Prometheus, Grafana, Azure Monitor.
FlexVolume is used to pass PostgreSQL secrets from Key Vault
DAPR and KEDA components are deployed in dapr and keda namespaces. KEDA is configured to leverage AAD Pod Identity for authentication to Service Bus. DAPR currently does not offer this for Service Bus component, but AAD pod identity is used for accessing Key Vault secrets.
Store and retrieve state
kubectl exec -ti nodea-0 -n dapr -- bash /home/user/write.sh
kubectl exec -ti nodea-1 -n dapr -- bash /home/user/read.sh
DAPR is configured to enable messaging between services using Service Bus backend. Connect to nodea-0 and use curl to send message to DAPR. Deployment subscribeorders will receive message. Also look into Service Bus in portal to see subscriber has been created in orders topic.
kubectl exec -ti nodea-0 -n dapr -- bash /home/user/createorder.sh
kubectl logs -l app=subscribeorders -n dapr -c container
Service bindingservicebus is configured with binding to DAPR events from Service Bus queue binding. There is KEDA scaling bounded to this queue so you should not see any Pods running. Go to nodea to generate 20 messages and whats Pods being created to deal with load.
kubectl exec -ti nodea-0 -n dapr -- python /home/user/sendMessagesToServiceBus.py
DAPR provides Blob Storage output binding for nodea. Check it out.
kubectl exec -ti nodea-0 -n dapr -- bash /home/user/blobout.sh myOutFile.json
DAPR Secrets API is configured to point to Azure Key Vault. Use DAPR API to read secrets.
kubectl exec -ti nodea-0 -n dapr -- bash /home/user/getsecret.sh
Service cart comes with /add API call of type POST. You can use DAPR sidecar to call other services.
kubectl exec -ti nodea-0 -n dapr -- bash /home/user/add.sh
Grafana demo install has some Dashboards defined connected to Prometheus. Login is tomas/Azure12345678. Find it at http://grafana.cloud.tomaskubica.net
Telemetry and logs are also gathered to Azure Monitor.
Basic IIS instance is accessible at iis.cloud.tomaskubica.net and runs in windows namespace.
Bookstore sample application is deployed in osm-demo namespace based on official OSM demo. Demo steps are prepared as flaggs in Helm Chart.
helm upgrade -i osm-bookstore ./helm/bookstore -n osm-demo \
--set namespace=osm-demo \
--set thief=false \
--set weight.v1=100 \
--set weight.v2=0
Use port-forwarding to access services.
# Bookstore-v1 on 9001
kubectl port-forward $(kubectl get pods --selector app=bookstore-v1 -n osm-demo --no-headers | grep 'Running' | awk '{print $1}') -n osm-demo 9001:80 &
# Bookstore-v2 on 9002
kubectl port-forward $(kubectl get pods --selector app=bookstore-v1 -n osm-demo --no-headers | grep 'Running' | awk '{print $1}') -n osm-demo 9002:80 &
# Bookthief on 9010
kubectl port-forward $(kubectl get pods --selector app=bookthief -n osm-demo --no-headers | grep 'Running' | awk '{print $1}') -n osm-demo 9010:80 &
# Bookbuyer on 9020
kubectl port-forward $(kubectl get pods --selector app=bookbuyer -n osm-demo --no-headers | grep 'Running' | awk '{print $1}') -n osm-demo 9020:80 &
# Bookwarehouse on 9030
kubectl port-forward $(kubectl get pods --selector app=bookwarehouse -n osm-demo --no-headers | grep 'Running' | awk '{print $1}') -n osm-demo 9030:80 &
# Grafana on 9040
kubectl port-forward $(kubectl get pods --selector app=osm-grafana -n osm-system --no-headers | grep 'Running' | awk '{print $1}') -n osm-system 9040:3000 &
# Jaeger on 9041
kubectl port-forward $(kubectl get pods --selector app=jaeger -n osm-system --no-headers | grep 'Running' | awk '{print $1}') -n osm-system 9041:16686 &
# Prometheus on 9042
kubectl port-forward $(kubectl get pods --selector app=osm-prometheus -n osm-system --no-headers | grep 'Running' | awk '{print $1}') -n osm-system 9042:7070 &
# Stop forwarding
killall kubectl
Demonstrate:
- Bookstore counters get incremented because TrafficTarget is configured for Bookbuyer to talk to Bookstore (note specific headers etc.)
- Bookbuyer UI shows incrementing counters
- Bookthief counters are not incrementing as default policy is deny all and traffic has not been permitted
- Show TrafficTarget CRD
Upgrade Helm chart and switch flag to allow traffic from Bookthief to Bookbuyer.
helm upgrade -i osm-bookstore ./helm/bookstore -n osm-demo \
--set namespace=osm-demo \
--set thief=true \
--set weight.v1=100 \
--set weight.v2=0
Demonstrate:
- Show TrafficTarget CRD
- Show Bookthief counter being incremented
Upgrade Helm Chart to send 10% of traffic to v2.
helm upgrade -i osm-bookstore ./helm/bookstore -n osm-demo \
--set namespace=osm-demo \
--set thief=true \
--set weight.v1=90 \
--set weight.v2=10
Demonstrate:
- Show Bookbuyer counter to see v2 being slowly incremented
Todo application is deployed in linkerd-demo namespace and can be accessed at http://linkerd.cloud.tomaskubica.net
Traffic generator is deployed so we can see some traffic in monitoring.
Checkout Linkerd dashboard (including Grafana) at http://linkerd.i.cloud.tomaskubica.net Login is tomas/Azure12345678.
Checkout Application Insights for OpenCensus instrumented application and how Linkerd enhanced this data.
Test retry policy. Access service with no retry policy (failRate means % of requests that will cause crash). You will likely see failed responses.
kubectl exec client-0 -c container -n linkerd-demo -- curl -vs -m 30 retry-service-noprofile?failRate=50
Try accessing service with retry policy. Even if container fails Linkerd should hold your connection and retry.
kubectl exec client-0 -c container -n linkerd-demo -- curl -vs -m 30 retry-service?failRate=50
Canary - there are v1 and v2 deployments and services (myweb-service-v1 and myweb-service-v2). Also there is production service myweb-service with TrafficSplit configured for 90% of traffic going to v1. Test it.
kubectl exec client-0 -c container -n linkerd-demo -- bash -c 'for x in {0..30}; do curl -s myweb-service; echo; done'
Note this can be also used to inject errors for purpose of ressiiency testing (simply create "faulty" Pod returning whatever you want and use traffic split to send 5% of requests to it).
Linkerd may help balancing gRPC communication. As HTTP/2 leads to long-lived sessions and multiplexing of paralel requests within single session therefore limiting entropy on L4 making balancing uneffective, Linkerd acting as gRPC proxy can fix this.
kubectl logs -l app=grpc-client-nosm -n linkerd-demo # Should not be balanced
kubectl logs -l app=grpc-client-withsm -c client -n linkerd-demo # Should be balanced
linkerd -n linkerd-demo edges deployment
linkerd -n linkerd-demo tap deploy | grep tls=true
In this demo I currenlty focus on features not supported by Linkerd. Note that all Linkerd functionality is in some form available in Istio also. Also note decision is not only about features as some architecture decisions between two project are very different. Linkerd focus on maximum efficiency as still more lightweight and performing, but missing some feature (althogh with aggresive roadmap in this space). Istio is not governed by CNCF and is much more feature rich, yet more complex and with more overhead (although recent steps to move away from too distributed architecture to more monolithic approach shows a lot of improvements).
When target service starts failing:
- clients retries are increasing load on already troubled service making it harder for it to recover and get up to speed
- users experience long latencies and even failed operations after waiting
In some cases it is better to declare service unavailable in order to:
- get service some breathing room to recover
- inform users feature is not available rather than having them experience waiting and failures
First let's see service with Istio retry.
Attempt request to service with 90% probability of returning 503 error.
# Open log stream in different window
kubectl logs -l app=retry -c retry-backend -f
kubectl exec client-0 -c container -- curl -vs -m 30 "retry-service?failRate=90&mode=busy"
In logs we should see multiple failed attempts as Istio retries until we get 200 response.
Now let's try service with circuit breaker configured. Try to access it multiple times and at some point it will just quickly respond with 503 (and no new logs will be visible at target as Istio does not send traffic). Wait for a minute and try again, when circuit is switched again.
# Open log stream in different window
kubectl logs -l app=retry-breaker -n istio-demo -c retry-backend-breaker -f
kubectl exec -it client-0 -c container -n istio-demo -- bash
curl -vs -m 30 "retry-service-breaker?failRate=90&mode=busy"
Similar to Linkerd Istio can use percentage to route traffic to different service versions or inject faults.
Istio also offers header-based routing eg. for A/B testing.
kubectl exec client-0 -c container -n istio-demo -- curl -s myweb-service
kubectl exec client-0 -c container -n istio-demo -- curl -s --cookie "usertype=tester" myweb-service
You can use Kubernetes Network Policy to filter egress traffic based on L4 rules. With Istio you can manage this based on FQDNs and also apply additional Istio policies such as retry or circuit breaker. Optionaly egress traffic can be handled from pod sidecar directly or centralized via egress gateway (eg. to get control of source IP).
Demo cluster is configured to not allow access to external services not explicitly defined. From following two only httpbin.org is configured.
kubectl exec client-0 -c container -n istio-demo -- curl -sv https://www.tomaskubica.cz
kubectl exec client-0 -c container -n istio-demo -- curl -sv http://www.tomaskubica.cz
kubectl exec client-0 -c container -n istio-demo -- curl -sv http://httpbin.org/ip
TBD
Traditionaly you would use Kubernetes Ingress to handle this. Using Istio Gateway gives advantage of using advanced routing, traffic split and other capabilities of Istio on entrance to cluster in consistent way.
TBD
https://kiali.istio.cloud.tomaskubica.cz https://jaeger.istio.cloud.tomaskubica.cz https://grafana.istio.cloud.tomaskubica.cz https://prometheus.istio.cloud.tomaskubica.cz
istioctl dashboard envoy client-0
Flagger supports automated canary releasing with NGINX Ingress, Linkerd Service Mesh, Istio Service Mesh and others.
- Supports canary, green/blue and A/B testing (header based routing)
- Easy to use, very lightweight (no sidecars)
- Does not solve backend service-to-service canary via internal path (must go via Ingress)
Application is exposed at http://canary.nginx.i.cloud.tomaskubica.net. You should see v1 message and blue background and load-balanced response from multiple instances. Keep window open so traffic is generated.
Should you need to restart this scenario, you can do this:
helm delete canary -n canary
helm upgrade -i canary ./helm/canary -n canary -f ./helm/canary/values.yaml
Check status of Canary release (should be in initialized phase)
kubectl describe canary canary-nginx -n canary
In separate window start generating traffic.
while true; do curl http://canary.nginx.i.cloud.tomaskubica.net/version; done
Deploy new version, watch Flagger orchestrating process. Application exposes latency metrics which are evaluated by Flagger. Since latency is bellow 500ms we should see Flagger progressing and finaly moving all users to new version.
helm upgrade -i canary ./helm/canary -n canary --reuse-values \
--set ingress.canary.imagetag="3\.2\.1"
kubectl describe canary canary-nginx -n canary
We will now try next version, but this time simulate long latency. In new window run this:
while true; do curl http://canary.nginx.i.cloud.tomaskubica.net/delay/3; done
Roll new version. Because some calls get routed to delay function metrics for canary version are likely to go worse over time and Flagger should halt advancement and rollback.
helm upgrade -i canary ./helm/canary -n canary --reuse-values \
--set ingress.canary.imagetag="3\.2\.2"
kubectl describe canary canary-nginx -n canary
In previous example percentage of connections was routed to new release. With A/B scenario we will not do this randomly, but rather decide based on header or cookie. As an example only internal or beta users will get routed to new version to test it out before complete rollout.
Check status of Canary release (should be in initialized phase)
kubectl describe canary ab-nginx -n canary
In separate window start generating traffic.
while true; do curl http://ab.nginx.i.cloud.tomaskubica.net/version; done
Deploy new version, watch Flagger orchestrating process. In another window start curl using beta tester header to reach new version when deployed.
while true; do curl -H 'X-Canary: insider' http://ab.nginx.i.cloud.tomaskubica.net/version; done
Application exposes latency metrics which are evaluated by Flagger. Since latency is bellow 500ms we should see Flagger progressing and finaly moving all users to new version.
helm upgrade -i canary ./helm/canary -n canary --reuse-values --set ingress.ab.imagetag="3\.2\.1"
kubectl describe canary ab-nginx -n canary
We will now try next version, but this time simulate long latency. In new window run this:
while true; do curl -H 'X-Canary: insider' http://ab.nginx.i.cloud.tomaskubica.net/delay/3; done
Roll new version. Because some calls get routed to delay function metrics for canary version are likely to go worse over time and Flagger should halt advancement and rollback.
helm upgrade -i canary ./helm/canary -n canary --reuse-values --set ingress.ab.imagetag="3\.2\.2"
kubectl describe canary ab-nginx -n canary
- Supports canary, green/blue
- Does not support A/B testing (header based routing)
- Dependency on Linkerd (sidecars), more overhead
- Supports internal service-to-service traffic
TBD
- Supports canary, green/blue, A/B testing (header based routing) and mirroring
- Dependency on Istio (sidecars), significant overhead
- Supports internal service-to-service traffic
TBD
Work in progress
AKS resource group is configured with Azure Policy (ARM is using connector for Gatekeeper v3) with exceptions for most namespaces except namespace "policy". 6 policies are configured:
- No public IP on load balancer
- Must include label release-type
- Ingress must be HTTPS (no HTTP)
- Only container repos with azurecr.io in name are allowed (no Docker Hub)
- Must include limits on resources, but never more than 200m and 128Mi
- No privileged containers
Go to Azure Security Center to see found issues in our AKS cluster and Azure Container Registry vulnerabilities in images (powered by Qualys).
Check AAD integration for Kubernetes users
rm ~/.kube/config
az aks get-credentials -n tomasdemoaks-test -g aks-test
kubectl get nodes # login as admin@tomaskubicaoffice.onmicrosoft.com
This demo is extensively using concept of Managed Identity so applications can securely access secrets and resources without need to handover passwords.
AAD Pod Identity is used to bring managed identity to AKS on per-Pod basis with AzureIdentity and AzureIdentityBinding CRDs. This how to get short-lived token and use it to access Key Vault.
# Get Pod name
nodepod=$(kubectl get pods -l app=myapp-node -o name | head -n 1)
# Get Key Vault name
terraform output keyvault_name
# Use managed identity
kubectl exec -ti $nodepod -- bash
# Get token
curl -s "http://169.254.169.254/metadata/identity/oauth2/token?resource=https://vault.azure.net"
# Parse and store token
export token=$(curl -s http://169.254.169.254/metadata/identity/oauth2/token?resource=https://vault.azure.net | jq -r '.access_token')
# Access Key Vault with token
keyvault=vault-demo-e25wbosx
curl -H "Authorization: Bearer ${token}" https://$keyvault.vault.azure.net/secrets/psql-jdbc?api-version=7.0
Azure Key Vault FlexVolume is used to pass Key Vault secrets as files. On backend it is using Managed Identity, but application does not need to ask for token, secret is mapped to its file system directly.
# Get Pod name
todopod=$(kubectl get pods -l app=myapp-todo -o name | head -n 1)
kubectl exec -ti $todopod -- bash -c 'ls -lah /keyvault; cat /keyvault/psql-jdbc'
Some components deployed in this demo also access Azure resources via Managed Identity:
- Azure Monitor
- FlexVolume
- AAD Pod Identity (it needs to talk to Azure and is using managed identity for that)
- Kubernetes itself
- KEDA autoscaler
- Application Gateway Ingress Controller
Some components are using standard passwords/connection_strings until managed identity is supported. This includes:
- Some DAPR components
- PostgreSQL connection
Demonstration of deploying Azure services in any Kubernetes environment.
Environment ready, description TBD
Environment ready, description TBD
Environment ready, description TBD