Configure Kubernetes Control Plane Cluster
1. 🧭 Scope Legend
Use these scope markers throughout this page and the wider Kubernetes CI/CD documentation.
2. 🧩 Control Plane Inputs
kubeadm init. This value must never persist.https://pkgs.k8s.io/core:/stable:/v1.35/deb/sudo kubeadm init \
--control-plane-endpoint "ac-cicd-api.aspireclan.com:443" \
--upload-certs \
--pod-network-cidr=10.244.0.0/16 \
--service-cidr=10.96.0.0/12 \
--v=53. 🖥️ VM Specs
CPU: 2 vCPU
RAM: 8 GB
Disk: 100 GBClone the base Ubuntu 24 minimal template.
4. 🧱 Prepare the Base Template VM
Login to the VM and complete the base OS preparation.
lsblksudo growpart /dev/sda 3
sudo pvresize /dev/sda3
sudo lvextend -l +100%FREE /dev/ubuntu-vg/ubuntu-lv
sudo resize2fs /dev/ubuntu-vg/ubuntu-lv
df -hHDD resize reference: Resize Ubuntu VM HDD
sudo apt update && sudo apt upgrade -y
sudo apt autoremove
sudo apt update && sudo apt upgrade -yswapon --show
sudo modprobe overlay
sudo modprobe br_netfiltercat <<EOF | sudo tee /etc/modules-load.d/k8s.conf
overlay
br_netfilter
EOFcat <<EOF | sudo tee /etc/sysctl.d/k8s.conf
net.bridge.bridge-nf-call-iptables = 1
net.bridge.bridge-nf-call-ip6tables = 1
net.ipv4.ip_forward = 1
EOFsudo sysctl --systemsudo apt install -y containerd
sudo mkdir -p /etc/containerd
containerd config default | sudo tee /etc/containerd/config.toml >/dev/null
sudo nano /etc/containerd/config.tomlOnly change these kinds of values:
trueregistry.k8s.io/pause:3.10.1sudo systemctl daemon-reload
sudo systemctl enable containerd
sudo systemctl restart containerdsudo ufw disable
sudo systemctl disable ufwsudo mkdir -p /etc/apt/keyrings
curl -fsSL https://pkgs.k8s.io/core:/stable:/v1.35/deb/Release.key | sudo gpg --dearmor -o /etc/apt/keyrings/kubernetes-apt-keyring.gpg
echo "deb [signed-by=/etc/apt/keyrings/kubernetes-apt-keyring.gpg] https://pkgs.k8s.io/core:/stable:/v1.35/deb/ /" | sudo tee /etc/apt/sources.list.d/kubernetes.listsudo apt update
sudo apt install -y kubelet kubeadm kubectl
sudo apt-mark hold kubelet kubeadm kubectl
sudo systemctl enable kubeletsudo kubeadm config images pull
sudo systemctl status containerd --no-pager -l
sudo systemctl status kubelet --no-pager -l
swapon --showcat <<EOF | sudo tee /etc/crictl.yaml
runtime-endpoint: unix:///run/containerd/containerd.sock
image-endpoint: unix:///run/containerd/containerd.sock
timeout: 10
debug: false
EOFsudo crictl infosudo swapoff -a
sudo sed -i.bak '/ swap / s/^/#/' /etc/fstab
swapon --show5. 🧹 Clean the VM Before Templating
sudo apt autoremove -y
sudo apt clean
sudo journalctl --vacuum-time=3d
sudo truncate -s 0 /etc/machine-id
sudo rm -f /var/lib/dbus/machine-id
sudo ln -s /etc/machine-id /var/lib/dbus/machine-idOptional cleanup:
sudo rm -rf /tmp/*
sudo rm -rf /var/tmp/*Then shut it down:
sudo shutdown -h now6. 📀 Create Template
tmpl-ac-cicd-cp-00Create the template only after all Kubernetes base packages and cleanup steps are complete.
7. 🧬 Clone Template for Production Control Plane VM
sudo hostnamectl set-hostname ac-cicd-cp-01
hostnamectl8. 🚀 Bootstrap Primary Control Plane (ac-cicd-cp-01)
Clone the template and configure the primary control plane VM.
ac-cicd-cp-01192.168.8.62Complete static IP and VM network configuration using: Configure VMs
sudo rebootswapon --showIt should show nothing. If it shows anything, run:
sudo swapoff -a
sudo sed -i.bak '/ swap / s/^/#/' /etc/fstab
swapon --showsudo kubeadm init \
--control-plane-endpoint "ac-cicd-api.aspireclan.com:443" \
--upload-certs \
--pod-network-cidr=10.244.0.0/16 \
--service-cidr=10.96.0.0/12 \
--v=5Securely store the output of kubeadm init.
mkdir -p $HOME/.kube
sudo cp -i /etc/kubernetes/admin.conf $HOME/.kube/config
sudo chown $(id -u):$(id -g) $HOME/.kube/configcurl -LO https://raw.githubusercontent.com/projectcalico/calico/v3.28.2/manifests/calico.yaml
sudo nano calico.yamlUpdate the Calico manifest to align with the pod CIDR.
- name: CALICO_IPV4POOL_IPIP
value: "Always"
- name: CALICO_IPV4POOL_VXLAN
value: "Never"
- name: CALICO_IPV4POOL_CIDR
value: "10.244.0.0/16"kubectl apply -f calico.yaml
kubectl get nodes -o wide
kubectl get pods -n kube-system -o wide
kubectl get ippool default-ipv4-ippool -o yaml
kubectl -n kube-system get cm kubeadm-config -o yamlkubectl get nodes
kubectl -n kube-system edit configmap corednsAdd this block above the existing .:53 block in the Corefile:
aspireclan.com:53 {
forward . 192.168.8.4
cache 30
}kubectl rollout restart deployment coredns -n kube-system
kubectl rollout status deployment coredns -n kube-systemcat <<'EOF' | kubectl apply -f -
apiVersion: v1
kind: Pod
metadata:
name: net-test
namespace: default
spec:
restartPolicy: Never
tolerations:
- key: "node-role.kubernetes.io/control-plane"
operator: "Exists"
effect: "NoSchedule"
containers:
- name: net-test
image: curlimages/curl:8.10.1
command: ["sh", "-c"]
args:
- |
nslookup harbor.aspireclan.com || true
echo
curl -vk --connect-timeout 10 https://harbor.aspireclan.com/v2/ || true
EOFkubectl get pod net-test -n default -o widekubectl logs net-test -n defaultkubectl delete pod net-test -n default --ignore-not-foundThe Harbor check should return unauthorized 401.
9. 🧯 Enable Firewall on ac-cicd-cp-01 (Post-Bootstrap)
sudo ufw allow 6443/tcp
sudo ufw allow 2379:2380/tcp
sudo ufw allow 10250/tcp
sudo ufw allow 30000:32767/tcp
sudo ufw allow 179/tcp
sudo ufw allow proto 4 from any to any
sudo ufw allow from 192.168.8.0/22
sudo ufw enable
sudo ufw status verbose10. 🧬 Join Additional Control Planes
Clone the Template and create CP-02 & CP-03 control plane VMs.
Repeat the following on the remaining control plane VMs.
ac-cicd-cp-02192.168.8.63ac-cicd-cp-03192.168.8.64Complete static IP and VM network configuration using: Configure VMs
sudo reboot
swapon --showIt should show nothing. If it shows anything, run:
sudo swapoff -a
sudo sed -i.bak '/ swap / s/^/#/' /etc/fstab
swapon --showRun the control plane join command below on each additional control plane node:
sudo kubeadm join ac-cicd-api.aspireclan.com:443 --token aaa.bbbbbbbbbbbbbbbb --discovery-token-ca-cert-hash sha256:xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx --control-plane --certificate-key yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyyymkdir -p $HOME/.kube
sudo cp -i /etc/kubernetes/admin.conf $HOME/.kube/config
sudo chown $(id -u):$(id -g) $HOME/.kube/configVerify the cluster after each join:
kubectl get nodes
kubectl get nodes -o wide
kubectl get pods -n kube-system -o wide
kubectl get pods -A -o wide
kubectl get endpoints kubernetes -n default11. 🧯 Enable Firewall on Additional Control Planes
sudo ufw allow 6443/tcp
sudo ufw allow 2379:2380/tcp
sudo ufw allow 10250/tcp
sudo ufw allow 30000:32767/tcp
sudo ufw allow 179/tcp
sudo ufw allow proto 4 from any to any
sudo ufw allow from 192.168.8.0/22
sudo ufw enable
sudo ufw status verbose