This post is a continuation of our journey with self-hosted CI/CD agents. I encourage you to check part 1 if you want to see a different approach to that topic. In this post we will focus on Jenkins. It took a bit of time to install Jenkins on AKS. I encountered a few errors along the way. I will share my solution here so it might be helpful to others.
Overview
- Pros
- Highly configurable with plugins and extensions
- Active user base
- Provides APIs
- Cons
- Only community support
- Difficult to configure
- Difficult to debug
- Documentation could be improved
We will create and configure the following resources to show what Jenkins can offer:
- AKS cluster
- Azure Container Registry
- Promethes and Grafana to observe Jenkins agents
- Jenkins
- Github repo and connect it with Jenkins
Here is a link to GitHub repo with all files for reference . I created a script with the necessary commands to provision a basic setup. In this setup, we will use our own kubelet managed identity. I want to show you other possibilities for how we can create a cluster and connect other resources. We will assign AcrPull role to that identity by adding --attach-acr to az aks create command.
Video walkthrough
Create AKS cluster and ACR
- Run script:
1 2 3
# After running above script, if there were no errors, variables should be available in terminal. chmod +x aks.sh ./aks.sh 'add user id, for me, it is my email of AAD user'
- Get credentials to AKS and test connection:
1
2
az aks get-credentials --resource-group $resourceGroup --name $aksName
kubectl get nodes
- Create Jenkins namespace:
1
kubectl create namespace jenkins
- Create Service Principal with AcrPush role to ACR. We will use this credentials in Jenkins pipeline to push image. In previous posts we investigated login/password method and also managed identity.
1
az ad sp create-for-rbac -n jenkinsAcrAccess --role AcrPush --scope $acrId
- Save Service Principal credentials as Kubernetes Secret. Add your credentials to the command:
1
2
3
4
kubectl -n jenkins create secret generic acr-sp \
--from-literal=AZURE_CLIENT_ID=<appId> \
--from-literal=AZURE_CLIENT_SECRET=<password> \
--from-literal=AZURE_TENANT_ID=<tenant>
- Create ConfigMap from config.json file with information about container registry. We will use it in Jenkins pipeline to inform Kaniko about ACR. File the name with your ACR name (
echo $acrName
):
1
{ "credHelpers": { "<ACR name>.azurecr.io": "acr-env" } }
1
kubectl -n jenkins create configmap docker-config --from-file=config.json
Install Prometheus and Grafana
We will observe how many resources Jenkins and agent consume. We will use the prometheus-community Helm chart.
- Add Helm repository:
1 2
helm repo add prometheus-community https://prometheus-community.github.io/helm-charts helm repo update
- Install Helm chart in monitoring namespace:
1
helm install prometheus prometheus-community/kube-prometheus-stack --namespace monitoring --create-namespace
- Confirm that resources are running:
1
kubectl get all -n monitoring
- Expose Grafana and Prometheus in two tabs in terminal
1
kubectl port-forward svc/prometheus-grafana -n monitoring 4000:80
1
kubectl port-forward svc/prometheus-kube-prometheus-prometheus -n monitoring 4001:9090
- Log in to Grafana. Default login/password:
admin/prom-operator
1
open http://localhost:4000/login
We will come back here once we will install Jenkins.
Install Jenkins
We will install Jenkins using Helm chart.
- Add Helm repository:
1 2
helm repo add jenkinsci https://charts.jenkins.io helm repo update
- Create Service Account and Cluster Role:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77
# jenkins-sa.yaml --- apiVersion: v1 kind: ServiceAccount metadata: name: jenkins namespace: jenkins --- apiVersion: rbac.authorization.k8s.io/v1 kind: ClusterRole metadata: annotations: rbac.authorization.kubernetes.io/autoupdate: "true" labels: kubernetes.io/bootstrapping: rbac-defaults name: jenkins rules: - apiGroups: - '*' resources: - statefulsets - services - replicationcontrollers - replicasets - podtemplates - podsecuritypolicies - pods - pods/log - pods/exec - podpreset - poddisruptionbudget - persistentvolumes - persistentvolumeclaims - jobs - endpoints - deployments - deployments/scale - daemonsets - cronjobs - configmaps - namespaces - events - secrets verbs: - create - get - watch - delete - list - patch - update - apiGroups: - "" resources: - nodes verbs: - get - list - watch - update --- apiVersion: rbac.authorization.k8s.io/v1 kind: ClusterRoleBinding metadata: annotations: rbac.authorization.kubernetes.io/autoupdate: "true" labels: kubernetes.io/bootstrapping: rbac-defaults name: jenkins roleRef: apiGroup: rbac.authorization.k8s.io kind: ClusterRole name: jenkins subjects: - apiGroup: rbac.authorization.k8s.io kind: Group name: system:serviceaccounts:jenkins
1
kubectl apply -f jenkins-sa.yaml
- Create Storage Class with custom mount options:
1 2 3 4 5 6 7 8 9 10 11 12 13 14
# jenkins-storageclass.yaml kind: StorageClass apiVersion: storage.k8s.io/v1 metadata: name: jenkins-azurefile provisioner: file.csi.azure.com mountOptions: - uid=1000 - gid=1000 allowVolumeExpansion: true volumeBindingMode: Immediate reclaimPolicy: Delete parameters: skuName: Standard_LRS
1
kubectl apply -f jenkins-storageclass.yaml
4.Create Persistent Volume Claim using above Storage Class:
1
2
3
4
5
6
7
8
9
10
11
12
13
# jenkins-pvc.yaml
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
name: jenkins-pvc
namespace: jenkins
spec:
accessModes:
- ReadWriteMany
resources:
requests:
storage: 50Gi
storageClassName: jenkins-azurefile
1
kubectl apply -f jenkins-pvc.yaml
5.Create custom docker image with Jenkins and install plugins. Due to problems, I encountered installing Jenkins with default settings I had to create custom image. Customization of Jenkins is a complex task and each part needs a bit of tweaking. Link to custom image.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
# Dockerfile.jenkins
FROM jenkins/jenkins:2.375.2
USER root
RUN apt-get update && apt-get install -y lsb-release vim && \
curl -fsSLo /usr/share/keyrings/docker-archive-keyring.asc \
https://download.docker.com/linux/debian/gpg && \
echo "deb [arch=$(dpkg --print-architecture) \
signed-by=/usr/share/keyrings/docker-archive-keyring.asc] \
https://download.docker.com/linux/debian \
$(lsb_release -cs) stable" > /etc/apt/sources.list.d/docker.list && \
apt-get update && apt-get install -y docker-ce-cli
COPY ./plugins.txt .
USER jenkins
RUN jenkins-plugin-cli --plugins -f plugins.txt
1
2
3
4
5
6
7
8
9
10
11
#plugins.txt
kubernetes:1.31.3
workflow-job:1189.va_d37a_e9e4eda_
workflow-aggregator:581.v0c46fa_697ffd
git:5.0.0
git-client:4.0.0
github-branch-source:1696.v3a_7603564d04
configuration-as-code:1569.vb_72405b_80249
kubernetes-credentials-provider:0.22
job-dsl:1.81
credentials:1214.v1de940103927
6.Customize values.yaml file for Helm install:
Here you can find a template for that file.
I will not post my values.yaml file here since it has 427 lines. You can check it here.
We will focus only on specific options:
- resource requests and limits for pod
- prometheus
- PVC
- storageClass
- serviceAccount
- plugins (we will comment them out)
- backup (we will turn it off for demo)
1
helm install jenkins jenkinsci/jenkins --namespace jenkins -f jenkins-values.yaml
7.Confirm that pod is running. It may take upto 10 minutes to start with some container restarts along the way.
1
kubectl -n jenkins get pods
Configure Jenkins
- Get password for admin panel. Login:
admin
1
kubectl exec --namespace jenkins -it svc/jenkins -c jenkins -- /bin/cat /run/secrets/additional/chart-admin-password && echo
- Connect to Jenkins and log in:
1 2
# Open new tab in terminal kubectl -n jenkins port-forward jenkins-0 8080:8080
1
open http://localhost:8080/login
- Update plugins. Manage Jenkins > Manage Plugins > Updates
- Enable JGIT plugin. Manage Jenkins > Global Tool Configuration > Git , change to JGit
The reason we change to jgit is that default git plugin has problem with writing temporary credentials with proper permissions. This is solution to this error:
1
2
3
4
5
6
7
8
9
10
stdout:
stderr: @@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
@ WARNING: UNPROTECTED PRIVATE KEY FILE! @
@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@@
Permissions 0555 for '/var/jenkins_home/caches/git-0aa16db65c903d3ced737f801b217112@tmp/jenkins-gitclient-ssh2425211278542515051.key' are too open.
It is required that your private key files are NOT accessible by others.
This private key will be ignored.
Load key "/var/jenkins_home/caches/git-0aa16db65c903d3ced737f801b217112@tmp/jenkins-gitclient-ssh2425211278542515051.key": bad permissions
Permission denied (publickey).
fatal: Could not read from remote repository.
Set up connection with GitHub for private repository
- Generate SSH key and save it to a file.
1
ssh-keygen -t ed25519
- Copy the public key and add it Github repository
1
cat 'path to your file id_ed25519.pub'
- Create Kubernetes Secret with private key:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
# github-secret.yaml apiVersion: v1 kind: Secret metadata: name: jenkins-github-ssh namespace: jenkins labels: "jenkins.io/credentials-type": "basicSSHUserPrivateKey" annotations: "jenkins.io/credentials-description" : "ssh github.com:adamkielar/jenkins-runner" stringData: privateKey: | # Add private key. `cat 'path to your file id_ed25519`` -----BEGIN OPENSSH PRIVATE KEY----- -----END OPENSSH PRIVATE KEY----- username: # Add github username
1
kubectl apply -f github-secret.yaml
- Add github.com public key to know hosts in Manage Jenkins > Configure Global Security > Git Host Key Verification Configuration
1
ssh-keyscan github.com
Create New Job
We will create it using UI but we can also create it in configuration file.
Create Jenkinsfile with pipeline definition
Update file with ACR name and push changes to your repository.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
pipeline {
agent {
kubernetes {
defaultContainer 'kaniko'
yaml '''
kind: Pod
spec:
containers:
- name: kaniko
image: gcr.io/kaniko-project/executor:v1.9.0-debug
imagePullPolicy: Always
command:
- sleep
args:
- 99d
envFrom:
- secretRef:
name: acr-sp
volumeMounts:
- name: docker-config
mountPath: /kaniko/.docker/
volumes:
- name: docker-config
configMap:
name: docker-config
'''
}
}
stages {
stage('Kaniko Build and Push') {
steps {
sh '/kaniko/executor --dockerfile=${PWD}/Dockerfile -c ${PWD} --cache=true --destination=jenkinsacr9482.azurecr.io/jenkins-demo:v1.0.0'
}
}
stage('Deploy Application') {
agent {
kubernetes {
defaultContainer 'kubectl'
yaml '''
kind: Pod
spec:
containers:
- name: kubectl
image: quay.io/tfgco/kubectl
imagePullPolicy: Always
command:
- sleep
args:
- 99d
'''
}
}
steps {
sh 'kubectl apply -f /home/jenkins/agent/workspace/jenkins-on-aks_master/pod.yaml'
}
}
}
}
Build pipeline
- Scan repository to discover Jenkinsfile
- Build pipeline and check output
- Check Grafana
- Check ACR
- Check if pipeline finished with success
- Confirm that application pod is running
Known errors
Besides git plugin error that I describe above, I had also problems with:
- Problem with log in to Jenkins due to csrf:
In container running Jenkins change find this lines in /var/jenkins_home/config.xml
and change them as you can see below.
1
2
3
<useSecurity>false</useSecurity>
<!--<authorizationStrategy class="hudson.security.AuthorizationStrategy$Unsecured"/>
<securityRealm class="hudson.security.SecurityRealm$None"/>-->
- If you have problem installing Jenkins due to plugins errors, first install clean instance and then add plugins.
- If you have problems with mounting volumes to container, check if you mount volume with proper uid and gid set.