Creating a cluster with kubeadm

    The kubeadm tool is good if you need:

    • A simple way for you to try out Kubernetes, possibly for the first time.
    • A way for existing users to automate setting up a cluster and test their application.
    • A building block in other ecosystem and/or installer tools with a larger scope.

    You can install and use kubeadm on various machines: your laptop, a set of cloud servers, a Raspberry Pi, and more. Whether you’re deploying into the cloud or on-premises, you can integrate kubeadm into provisioning systems such as Ansible or Terraform.

    To follow this guide, you need:

    • One or more machines running a deb/rpm-compatible Linux OS; for example: Ubuntu or CentOS.
    • 2 GiB or more of RAM per machine—any less leaves little room for your apps.
    • At least 2 CPUs on the machine that you use as a control-plane node.
    • Full network connectivity among all machines in the cluster. You can use either a public or a private network.

    You also need to use a version of kubeadm that can deploy the version of Kubernetes that you want to use in your new cluster.

    applies to kubeadm as well as to Kubernetes overall. Check that policy to learn about what versions of Kubernetes and kubeadm are supported. This page is written for Kubernetes v1.27.

    The kubeadm tool’s overall feature state is General Availability (GA). Some sub-features are still under active development. The implementation of creating the cluster may change slightly as the tool evolves, but the overall implementation should be pretty stable.

    Note: Any commands under kubeadm alpha are, by definition, supported on an alpha level.

    Objectives

    • Install a single control-plane Kubernetes cluster
    • Install a Pod network on the cluster so that your Pods can talk to each other

    Install a and kubeadm on all the hosts. For detailed instructions and other prerequisites, see Installing kubeadm.

    Note:

    If you have already installed kubeadm, run apt-get update && apt-get upgrade or yum update to get the latest version of kubeadm.

    When you upgrade, the kubelet restarts every few seconds as it waits in a crashloop for kubeadm to tell it what to do. This crashloop is expected and normal. After you initialize your control-plane, the kubelet runs normally.

    Preparing the required container images

    This step is optional and only applies in case you wish kubeadm init and kubeadm join to not download the default container images which are hosted at registry.k8s.io.

    Kubeadm has commands that can help you pre-pull the required images when creating a cluster without an internet connection on its nodes. See Running kubeadm without an internet connection for more details.

    Kubeadm allows you to use a custom image repository for the required images. See for more details.

    Initializing your control-plane node

    The control-plane node is the machine where the control plane components run, including (the cluster database) and the API Server (which the command line tool communicates with).

    1. (Recommended) If you have plans to upgrade this single control-plane kubeadm cluster to high availability you should specify the --control-plane-endpoint to set the shared endpoint for all control-plane nodes. Such an endpoint can be either a DNS name or an IP address of a load-balancer.
    2. Choose a Pod network add-on, and verify whether it requires any arguments to be passed to kubeadm init. Depending on which third-party provider you choose, you might need to set the --pod-network-cidr to a provider-specific value. See Installing a Pod network add-on.
    3. (Optional) kubeadm tries to detect the container runtime by using a list of well known endpoints. To use different container runtime or if there are more than one installed on the provisioned node, specify the --cri-socket argument to kubeadm. See .

    To initialize the control-plane node run:

    Considerations about apiserver-advertise-address and ControlPlaneEndpoint

    While --apiserver-advertise-address can be used to set the advertise address for this particular control-plane node’s API server, --control-plane-endpoint can be used to set the shared endpoint for all control-plane nodes.

    --control-plane-endpoint allows both IP addresses and DNS names that can map to IP addresses. Please contact your network administrator to evaluate possible solutions with respect to such mapping.

    Here is an example mapping:

    1. 192.168.0.102 cluster-endpoint

    Where 192.168.0.102 is the IP address of this node and cluster-endpoint is a custom DNS name that maps to this IP. This will allow you to pass --control-plane-endpoint=cluster-endpoint to kubeadm init and pass the same DNS name to kubeadm join. Later you can modify cluster-endpoint to point to the address of your load-balancer in an high availability scenario.

    Turning a single control plane cluster created without --control-plane-endpoint into a highly available cluster is not supported by kubeadm.

    More information

    For more information about kubeadm init arguments, see the kubeadm reference guide.

    To configure kubeadm init with a configuration file see .

    To customize control plane components, including optional IPv6 assignment to liveness probe for control plane components and etcd server, provide extra arguments to each component as documented in custom arguments.

    To reconfigure a cluster that has already been created see .

    To run kubeadm init again, you must first tear down the cluster.

    If you join a node with a different architecture to your cluster, make sure that your deployed DaemonSets have container image support for this architecture.

    kubeadm init first runs a series of prechecks to ensure that the machine is ready to run Kubernetes. These prechecks expose warnings and exit on errors. kubeadm init then downloads and installs the cluster control plane components. This may take several minutes. After it finishes you should see:

    1. Your Kubernetes control-plane has initialized successfully!
    2. To start using your cluster, you need to run the following as a regular user:
    3. mkdir -p $HOME/.kube
    4. sudo cp -i /etc/kubernetes/admin.conf $HOME/.kube/config
    5. You should now deploy a Pod network to the cluster.
    6. Run "kubectl apply -f [podnetwork].yaml" with one of the options listed at:
    7. /docs/concepts/cluster-administration/addons/
    8. You can now join any number of machines by running the following on each node
    9. as root:
    10. kubeadm join <control-plane-host>:<control-plane-port> --token <token> --discovery-token-ca-cert-hash sha256:<hash>

    To make kubectl work for your non-root user, run these commands, which are also part of the kubeadm init output:

    1. mkdir -p $HOME/.kube
    2. sudo cp -i /etc/kubernetes/admin.conf $HOME/.kube/config
    3. sudo chown $(id -u):$(id -g) $HOME/.kube/config

    Alternatively, if you are the root user, you can run:

    1. export KUBECONFIG=/etc/kubernetes/admin.conf

    Warning: Kubeadm signs the certificate in the admin.conf to have Subject: O = system:masters, CN = kubernetes-admin. system:masters is a break-glass, super user group that bypasses the authorization layer (e.g. RBAC). Do not share the admin.conf file with anyone and instead grant users custom permissions by generating them a kubeconfig file using the kubeadm kubeconfig user command. For more details see .

    Make a record of the kubeadm join command that kubeadm init outputs. You need this command to join nodes to your cluster.

    Installing a Pod network add-on

    Caution:

    This section contains important information about networking setup and deployment order. Read all of this advice carefully before proceeding.

    You must deploy a Container Network Interface (CNI) based Pod network add-on so that your Pods can communicate with each other. Cluster DNS (CoreDNS) will not start up before a network is installed.

    • Take care that your Pod network must not overlap with any of the host networks: you are likely to see problems if there is any overlap. (If you find a collision between your network plugin’s preferred Pod network and some of your host networks, you should think of a suitable CIDR block to use instead, then use that during kubeadm init with --pod-network-cidr and as a replacement in your network plugin’s YAML).

    • By default, kubeadm sets up your cluster to use and enforce use of (role based access control). Make sure that your Pod network plugin supports RBAC, and so do any manifests that you use to deploy it.

    • If you want to use IPv6—either dual-stack, or single-stack IPv6 only networking—for your cluster, make sure that your Pod network plugin supports IPv6. IPv6 support was added to CNI in v0.6.0.

    Note: Kubeadm should be CNI agnostic and the validation of CNI providers is out of the scope of our current e2e testing. If you find an issue related to a CNI plugin you should log a ticket in its respective issue tracker instead of the kubeadm or kubernetes issue trackers.

    Several external projects provide Kubernetes Pod networks using CNI, some of which also support .

    See a list of add-ons that implement the Kubernetes networking model.

    You can install a Pod network add-on with the following command on the control-plane node or a node that has the kubeconfig credentials:

    1. kubectl apply -f <add-on.yaml>

    You can install only one Pod network per cluster.

    Once a Pod network has been installed, you can confirm that it is working by checking that the CoreDNS Pod is Running in the output of kubectl get pods --all-namespaces. And once the CoreDNS Pod is up and running, you can continue by joining your nodes.

    If your network is not working or CoreDNS is not in the Running state, check out the for kubeadm.

    By default, kubeadm enables the admission controller that restricts what labels can be self-applied by kubelets on node registration. The admission controller documentation covers what labels are permitted to be used with the kubelet --node-labels option. The node-role.kubernetes.io/control-plane label is such a restricted label and kubeadm manually applies it using a privileged client after a node has been created. To do that manually you can do the same by using kubectl label and ensure it is using a privileged kubeconfig such as the kubeadm managed /etc/kubernetes/admin.conf.

    Control plane node isolation

    By default, your cluster will not schedule Pods on the control plane nodes for security reasons. If you want to be able to schedule Pods on the control plane nodes, for example for a single machine Kubernetes cluster, run:

    1. kubectl taint nodes --all node-role.kubernetes.io/control-plane-

    The output will look something like:

    This will remove the node-role.kubernetes.io/control-plane:NoSchedule taint from any nodes that have it, including the control plane nodes, meaning that the scheduler will then be able to schedule Pods everywhere.

    Joining your nodes

    The nodes are where your workloads (containers and Pods, etc) run. To add new nodes to your cluster do the following for each machine:

    • SSH to the machine

    • Become root (e.g. sudo su -)

    • Install a runtime if needed

    • Run the command that was output by kubeadm init. For example:

      1. kubeadm join --token <token> <control-plane-host>:<control-plane-port> --discovery-token-ca-cert-hash sha256:<hash>

    If you do not have the token, you can get it by running the following command on the control-plane node:

    1. kubeadm token list

    The output is similar to this:

    1. TOKEN TTL EXPIRES USAGES DESCRIPTION EXTRA GROUPS
    2. 8ewj1p.9r9hcjoqgajrj4gi 23h 2018-06-12T02:51:28Z authentication, The default bootstrap system:
    3. signing token generated by bootstrappers:
    4. 'kubeadm init'. kubeadm:
    5. default-node-token

    By default, tokens expire after 24 hours. If you are joining a node to the cluster after the current token has expired, you can create a new token by running the following command on the control-plane node:

    1. kubeadm token create

    The output is similar to this:

    1. 5didvk.d09sbcov8ph2amjw

    If you don’t have the value of , you can get it by running the following command chain on the control-plane node:

    1. openssl dgst -sha256 -hex | sed 's/^.* //'

    The output is similar to:

    Note: To specify an IPv6 tuple for <control-plane-host>:<control-plane-port>, IPv6 address must be enclosed in square brackets, for example: [2001:db8::101]:2073.

    The output should look something like:

    1. [preflight] Running pre-flight checks
    2. ... (log output of join workflow) ...
    3. Node join complete:
    4. * Certificate signing request sent to control-plane and response
    5. received.
    6. * Kubelet informed of new secure connection details.
    7. Run 'kubectl get nodes' on control-plane to see this machine join.

    A few seconds later, you should notice this node in the output from kubectl get nodes when run on the control-plane node.

    Note: As the cluster nodes are usually initialized sequentially, the CoreDNS Pods are likely to all run on the first control-plane node. To provide higher availability, please rebalance the CoreDNS Pods with kubectl -n kube-system rollout restart deployment coredns after at least one new node is joined.

    (Optional) Controlling your cluster from machines other than the control-plane node

    1. scp root@<control-plane-host>:/etc/kubernetes/admin.conf .
    2. kubectl --kubeconfig ./admin.conf get nodes

    Note:

    The example above assumes SSH access is enabled for root. If that is not the case, you can copy the admin.conf file to be accessible by some other user and scp using that other user instead.

    The admin.conf file gives the user superuser privileges over the cluster. This file should be used sparingly. For normal users, it’s recommended to generate an unique credential to which you grant privileges. You can do this with the kubeadm alpha kubeconfig user --client-name <CN> command. That command will print out a KubeConfig file to STDOUT which you should save to a file and distribute to your user. After that, grant privileges by using kubectl create (cluster)rolebinding.

    (Optional) Proxying API Server to localhost

    If you want to connect to the API Server from outside the cluster you can use kubectl proxy:

    1. scp root@<control-plane-host>:/etc/kubernetes/admin.conf .
    2. kubectl --kubeconfig ./admin.conf proxy

    You can now access the API Server locally at http://localhost:8001/api/v1

    Clean up

    If you used disposable servers for your cluster, for testing, you can switch those off and do no further clean up. You can use kubectl config delete-cluster to delete your local references to the cluster.

    However, if you want to deprovision your cluster more cleanly, you should first drain the node and make sure that the node is empty, then deconfigure the node.

    Remove the node

    Talking to the control-plane node with the appropriate credentials, run:

    1. kubectl drain <node name> --delete-emptydir-data --force --ignore-daemonsets

    Before removing the node, reset the state installed by kubeadm:

    1. kubeadm reset

    The reset process does not reset or clean up iptables rules or IPVS tables. If you wish to reset iptables, you must do so manually:

    1. iptables -F && iptables -t nat -F && iptables -t mangle -F && iptables -X

    If you want to reset the IPVS tables, you must run the following command:

    Now remove the node:

    1. kubectl delete node <node name>

    If you wish to start over, run kubeadm init or kubeadm join with the appropriate arguments.

    You can use kubeadm reset on the control plane host to trigger a best-effort clean up.

    See the reference documentation for more information about this subcommand and its options.

    • Verify that your cluster is running properly with
    • SeeUpgrading kubeadm clusters for details about upgrading your cluster using kubeadm.
    • Learn about advanced kubeadm usage in the
    • Learn more about Kubernetes concepts and .
    • See the Cluster Networking page for a bigger list of Pod network add-ons.
    • See the to explore other add-ons, including tools for logging, monitoring, network policy, visualization & control of your Kubernetes cluster.
    • Configure how your cluster handles logs for cluster events and from applications running in Pods. See Logging Architecture for an overview of what is involved.

    Feedback

    Version skew policy

    While kubeadm allows version skew against some components that it manages, it is recommended that you match the kubeadm version with the versions of the control plane components, kube-proxy and kubelet.

    kubeadm’s skew against the Kubernetes version

    kubeadm can be used with Kubernetes components that are the same version as kubeadm or one version older. The Kubernetes version can be specified to kubeadm by using the --kubernetes-version flag of kubeadm init or the field when using --config. This option will control the versions of kube-apiserver, kube-controller-manager, kube-scheduler and kube-proxy.

    Example:

    • kubeadm is at 1.27
    • kubernetesVersion must be at 1.27 or 1.26

    kubeadm’s skew against the kubelet

    Similarly to the Kubernetes version, kubeadm can be used with a kubelet version that is the same version as kubeadm or one version older.

    Example:

    • kubeadm is at 1.27
    • kubelet on the host must be at 1.27 or 1.26

    kubeadm’s skew against kubeadm

    There are certain limitations on how kubeadm commands can operate on existing nodes or whole clusters managed by kubeadm.

    If new nodes are joined to the cluster, the kubeadm binary used for kubeadm join must match the last version of kubeadm used to either create the cluster with kubeadm init or to upgrade the same node with kubeadm upgrade. Similar rules apply to the rest of the kubeadm commands with the exception of kubeadm upgrade.

    Example for kubeadm join:

    • kubeadm version 1.27 was used to create a cluster with kubeadm init
    • Joining nodes must use a kubeadm binary that is at version 1.27

    Nodes that are being upgraded must use a version of kubeadm that is the same MINOR version or one MINOR version newer than the version of kubeadm used for managing the node.

    Example for :

    • kubeadm version 1.26 was used to create or upgrade the node
    • The version of kubeadm used for upgrading the node must be at 1.26 or 1.27

    To learn more about the version skew between the different Kubernetes component see the Version Skew Policy.

    Cluster resilience

    The cluster created here has a single control-plane node, with a single etcd database running on it. This means that if the control-plane node fails, your cluster may lose data and may need to be recreated from scratch.

    Workarounds:

    • Use multiple control-plane nodes. You can read to pick a cluster topology that provides high-availability.

    kubeadm deb/rpm packages and binaries are built for amd64, arm (32-bit), arm64, ppc64le, and s390x following the multi-platform proposal.

    Multiplatform container images for the control plane and addons are also supported since v1.12.

    Troubleshooting

    If you are running into difficulties with kubeadm, please consult our troubleshooting docs.